UNPKG

matrix-react-sdk

Version:
308 lines (270 loc) 32.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = exports.PhoneNumber = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _languageHandler = require("../../../../languageHandler"); var _MatrixClientPeg = require("../../../../MatrixClientPeg"); var sdk = _interopRequireWildcard(require("../../../../index")); var _Modal = _interopRequireDefault(require("../../../../Modal")); var _AddThreepid = _interopRequireDefault(require("../../../../AddThreepid")); var _replaceableComponent = require("../../../../utils/replaceableComponent"); var _dec, _class, _class2, _temp; /* TODO: Improve the UX for everything in here. This is a copy/paste of EmailAddresses, mostly. */ // TODO: Combine EmailAddresses and PhoneNumbers to be 3pid agnostic class PhoneNumber extends _react.default.Component { constructor(props) { super(props); (0, _defineProperty2.default)(this, "onRevokeClick", e => { e.stopPropagation(); e.preventDefault(); this.changeBinding({ bind: false, label: "revoke", errorTitle: (0, _languageHandler._t)("Unable to revoke sharing for phone number") }); }); (0, _defineProperty2.default)(this, "onShareClick", e => { e.stopPropagation(); e.preventDefault(); this.changeBinding({ bind: true, label: "share", errorTitle: (0, _languageHandler._t)("Unable to share phone number") }); }); (0, _defineProperty2.default)(this, "onVerificationCodeChange", e => { this.setState({ verificationCode: e.target.value }); }); (0, _defineProperty2.default)(this, "onContinueClick", async e => { e.stopPropagation(); e.preventDefault(); this.setState({ continueDisabled: true }); const token = this.state.verificationCode; try { await this.state.addTask.haveMsisdnToken(token); this.setState({ addTask: null, continueDisabled: false, verifying: false, verifyError: null, verificationCode: "" }); } catch (err) { this.setState({ continueDisabled: false }); if (err.errcode !== 'M_THREEPID_AUTH_FAILED') { const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); console.error("Unable to verify phone number: " + err); _Modal.default.createTrackedDialog('Unable to verify phone number', '', ErrorDialog, { title: (0, _languageHandler._t)("Unable to verify phone number."), description: err && err.message ? err.message : (0, _languageHandler._t)("Operation failed") }); } else { this.setState({ verifyError: (0, _languageHandler._t)("Incorrect verification code") }); } } }); const { bound } = props.msisdn; this.state = { verifying: false, verificationCode: "", addTask: null, continueDisabled: false, bound }; } // TODO: [REACT-WARNING] Replace with appropriate lifecycle event UNSAFE_componentWillReceiveProps(nextProps) { // eslint-disable-line camelcase const { bound } = nextProps.msisdn; this.setState({ bound }); } async changeBinding({ bind, label, errorTitle }) { if (!(await _MatrixClientPeg.MatrixClientPeg.get().doesServerSupportSeparateAddAndBind())) { return this.changeBindingTangledAddBind({ bind, label, errorTitle }); } const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); const { medium, address } = this.props.msisdn; try { if (bind) { const task = new _AddThreepid.default(); this.setState({ verifying: true, continueDisabled: true, addTask: task }); // XXX: Sydent will accept a number without country code if you add // a leading plus sign to a number in E.164 format (which the 3PID // address is), but this goes against the spec. // See https://github.com/matrix-org/matrix-doc/issues/2222 await task.bindMsisdn(null, `+${address}`); this.setState({ continueDisabled: false }); } else { await _MatrixClientPeg.MatrixClientPeg.get().unbindThreePid(medium, address); } this.setState({ bound: bind }); } catch (err) { console.error(`Unable to ${label} phone number ${address} ${err}`); this.setState({ verifying: false, continueDisabled: false, addTask: null }); _Modal.default.createTrackedDialog(`Unable to ${label} phone number`, '', ErrorDialog, { title: errorTitle, description: err && err.message ? err.message : (0, _languageHandler._t)("Operation failed") }); } } async changeBindingTangledAddBind({ bind, label, errorTitle }) { const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); const { medium, address } = this.props.msisdn; const task = new _AddThreepid.default(); this.setState({ verifying: true, continueDisabled: true, addTask: task }); try { await _MatrixClientPeg.MatrixClientPeg.get().deleteThreePid(medium, address); // XXX: Sydent will accept a number without country code if you add // a leading plus sign to a number in E.164 format (which the 3PID // address is), but this goes against the spec. // See https://github.com/matrix-org/matrix-doc/issues/2222 if (bind) { await task.bindMsisdn(null, `+${address}`); } else { await task.addMsisdn(null, `+${address}`); } this.setState({ continueDisabled: false, bound: bind }); } catch (err) { console.error(`Unable to ${label} phone number ${address} ${err}`); this.setState({ verifying: false, continueDisabled: false, addTask: null }); _Modal.default.createTrackedDialog(`Unable to ${label} phone number`, '', ErrorDialog, { title: errorTitle, description: err && err.message ? err.message : (0, _languageHandler._t)("Operation failed") }); } } render() { const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); const Field = sdk.getComponent('elements.Field'); const { address } = this.props.msisdn; const { verifying, bound } = this.state; let status; if (verifying) { status = /*#__PURE__*/_react.default.createElement("span", { className: "mx_ExistingPhoneNumber_verification" }, /*#__PURE__*/_react.default.createElement("span", null, (0, _languageHandler._t)("Please enter verification code sent via text."), /*#__PURE__*/_react.default.createElement("br", null), this.state.verifyError), /*#__PURE__*/_react.default.createElement("form", { onSubmit: this.onContinueClick, autoComplete: "off", noValidate: true }, /*#__PURE__*/_react.default.createElement(Field, { type: "text", label: (0, _languageHandler._t)("Verification code"), autoComplete: "off", disabled: this.state.continueDisabled, value: this.state.verificationCode, onChange: this.onVerificationCodeChange }))); } else if (bound) { status = /*#__PURE__*/_react.default.createElement(AccessibleButton, { className: "mx_ExistingPhoneNumber_confirmBtn", kind: "danger_sm", onClick: this.onRevokeClick }, (0, _languageHandler._t)("Revoke")); } else { status = /*#__PURE__*/_react.default.createElement(AccessibleButton, { className: "mx_ExistingPhoneNumber_confirmBtn", kind: "primary_sm", onClick: this.onShareClick }, (0, _languageHandler._t)("Share")); } return /*#__PURE__*/_react.default.createElement("div", { className: "mx_ExistingPhoneNumber" }, /*#__PURE__*/_react.default.createElement("span", { className: "mx_ExistingPhoneNumber_address" }, "+", address), status); } } exports.PhoneNumber = PhoneNumber; (0, _defineProperty2.default)(PhoneNumber, "propTypes", { msisdn: _propTypes.default.object.isRequired }); let PhoneNumbers = (_dec = (0, _replaceableComponent.replaceableComponent)("views.settings.discovery.PhoneNumbers"), _dec(_class = (_temp = _class2 = class PhoneNumbers extends _react.default.Component { render() { let content; if (this.props.msisdns.length > 0) { content = this.props.msisdns.map(e => { return /*#__PURE__*/_react.default.createElement(PhoneNumber, { msisdn: e, key: e.address }); }); } else { content = /*#__PURE__*/_react.default.createElement("span", { className: "mx_SettingsTab_subsectionText" }, (0, _languageHandler._t)("Discovery options will appear once you have added a phone number above.")); } return /*#__PURE__*/_react.default.createElement("div", { className: "mx_PhoneNumbers" }, content); } }, (0, _defineProperty2.default)(_class2, "propTypes", { msisdns: _propTypes.default.array.isRequired }), _temp)) || _class); exports.default = PhoneNumbers; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/views/settings/discovery/PhoneNumbers.js"],"names":["PhoneNumber","React","Component","constructor","props","e","stopPropagation","preventDefault","changeBinding","bind","label","errorTitle","setState","verificationCode","target","value","continueDisabled","token","state","addTask","haveMsisdnToken","verifying","verifyError","err","errcode","ErrorDialog","sdk","getComponent","console","error","Modal","createTrackedDialog","title","description","message","bound","msisdn","UNSAFE_componentWillReceiveProps","nextProps","MatrixClientPeg","get","doesServerSupportSeparateAddAndBind","changeBindingTangledAddBind","medium","address","task","AddThreepid","bindMsisdn","unbindThreePid","deleteThreePid","addMsisdn","render","AccessibleButton","Field","status","onContinueClick","onVerificationCodeChange","onRevokeClick","onShareClick","PropTypes","object","isRequired","PhoneNumbers","content","msisdns","length","map","array"],"mappings":"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","sourcesContent":["/*\nCopyright 2019 New Vector Ltd\nCopyright 2019 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React from 'react';\nimport PropTypes from 'prop-types';\n\nimport { _t } from \"../../../../languageHandler\";\nimport {MatrixClientPeg} from \"../../../../MatrixClientPeg\";\nimport * as sdk from '../../../../index';\nimport Modal from '../../../../Modal';\nimport AddThreepid from '../../../../AddThreepid';\nimport {replaceableComponent} from \"../../../../utils/replaceableComponent\";\n\n/*\nTODO: Improve the UX for everything in here.\nThis is a copy/paste of EmailAddresses, mostly.\n */\n\n// TODO: Combine EmailAddresses and PhoneNumbers to be 3pid agnostic\n\nexport class PhoneNumber extends React.Component {\n    static propTypes = {\n        msisdn: PropTypes.object.isRequired,\n    };\n\n    constructor(props) {\n        super(props);\n\n        const { bound } = props.msisdn;\n\n        this.state = {\n            verifying: false,\n            verificationCode: \"\",\n            addTask: null,\n            continueDisabled: false,\n            bound,\n        };\n    }\n\n    // TODO: [REACT-WARNING] Replace with appropriate lifecycle event\n    UNSAFE_componentWillReceiveProps(nextProps) { // eslint-disable-line camelcase\n        const { bound } = nextProps.msisdn;\n        this.setState({ bound });\n    }\n\n    async changeBinding({ bind, label, errorTitle }) {\n        if (!await MatrixClientPeg.get().doesServerSupportSeparateAddAndBind()) {\n            return this.changeBindingTangledAddBind({ bind, label, errorTitle });\n        }\n\n        const ErrorDialog = sdk.getComponent(\"dialogs.ErrorDialog\");\n        const { medium, address } = this.props.msisdn;\n\n        try {\n            if (bind) {\n                const task = new AddThreepid();\n                this.setState({\n                    verifying: true,\n                    continueDisabled: true,\n                    addTask: task,\n                });\n                // XXX: Sydent will accept a number without country code if you add\n                // a leading plus sign to a number in E.164 format (which the 3PID\n                // address is), but this goes against the spec.\n                // See https://github.com/matrix-org/matrix-doc/issues/2222\n                await task.bindMsisdn(null, `+${address}`);\n                this.setState({\n                    continueDisabled: false,\n                });\n            } else {\n                await MatrixClientPeg.get().unbindThreePid(medium, address);\n            }\n            this.setState({ bound: bind });\n        } catch (err) {\n            console.error(`Unable to ${label} phone number ${address} ${err}`);\n            this.setState({\n                verifying: false,\n                continueDisabled: false,\n                addTask: null,\n            });\n            Modal.createTrackedDialog(`Unable to ${label} phone number`, '', ErrorDialog, {\n                title: errorTitle,\n                description: ((err && err.message) ? err.message : _t(\"Operation failed\")),\n            });\n        }\n    }\n\n    async changeBindingTangledAddBind({ bind, label, errorTitle }) {\n        const ErrorDialog = sdk.getComponent(\"dialogs.ErrorDialog\");\n        const { medium, address } = this.props.msisdn;\n\n        const task = new AddThreepid();\n        this.setState({\n            verifying: true,\n            continueDisabled: true,\n            addTask: task,\n        });\n\n        try {\n            await MatrixClientPeg.get().deleteThreePid(medium, address);\n            // XXX: Sydent will accept a number without country code if you add\n            // a leading plus sign to a number in E.164 format (which the 3PID\n            // address is), but this goes against the spec.\n            // See https://github.com/matrix-org/matrix-doc/issues/2222\n            if (bind) {\n                await task.bindMsisdn(null, `+${address}`);\n            } else {\n                await task.addMsisdn(null, `+${address}`);\n            }\n            this.setState({\n                continueDisabled: false,\n                bound: bind,\n            });\n        } catch (err) {\n            console.error(`Unable to ${label} phone number ${address} ${err}`);\n            this.setState({\n                verifying: false,\n                continueDisabled: false,\n                addTask: null,\n            });\n            Modal.createTrackedDialog(`Unable to ${label} phone number`, '', ErrorDialog, {\n                title: errorTitle,\n                description: ((err && err.message) ? err.message : _t(\"Operation failed\")),\n            });\n        }\n    }\n\n    onRevokeClick = (e) => {\n        e.stopPropagation();\n        e.preventDefault();\n        this.changeBinding({\n            bind: false,\n            label: \"revoke\",\n            errorTitle: _t(\"Unable to revoke sharing for phone number\"),\n        });\n    }\n\n    onShareClick = (e) => {\n        e.stopPropagation();\n        e.preventDefault();\n        this.changeBinding({\n            bind: true,\n            label: \"share\",\n            errorTitle: _t(\"Unable to share phone number\"),\n        });\n    }\n\n    onVerificationCodeChange = (e) => {\n        this.setState({\n            verificationCode: e.target.value,\n        });\n    }\n\n    onContinueClick = async (e) => {\n        e.stopPropagation();\n        e.preventDefault();\n\n        this.setState({ continueDisabled: true });\n        const token = this.state.verificationCode;\n        try {\n            await this.state.addTask.haveMsisdnToken(token);\n            this.setState({\n                addTask: null,\n                continueDisabled: false,\n                verifying: false,\n                verifyError: null,\n                verificationCode: \"\",\n            });\n        } catch (err) {\n            this.setState({ continueDisabled: false });\n            if (err.errcode !== 'M_THREEPID_AUTH_FAILED') {\n                const ErrorDialog = sdk.getComponent(\"dialogs.ErrorDialog\");\n                console.error(\"Unable to verify phone number: \" + err);\n                Modal.createTrackedDialog('Unable to verify phone number', '', ErrorDialog, {\n                    title: _t(\"Unable to verify phone number.\"),\n                    description: ((err && err.message) ? err.message : _t(\"Operation failed\")),\n                });\n            } else {\n                this.setState({verifyError: _t(\"Incorrect verification code\")});\n            }\n        }\n    }\n\n    render() {\n        const AccessibleButton = sdk.getComponent('elements.AccessibleButton');\n        const Field = sdk.getComponent('elements.Field');\n        const { address } = this.props.msisdn;\n        const { verifying, bound } = this.state;\n\n        let status;\n        if (verifying) {\n            status = <span className=\"mx_ExistingPhoneNumber_verification\">\n                <span>\n                    {_t(\"Please enter verification code sent via text.\")}\n                    <br />\n                    {this.state.verifyError}\n                </span>\n                <form onSubmit={this.onContinueClick} autoComplete=\"off\" noValidate={true}>\n                    <Field\n                        type=\"text\"\n                        label={_t(\"Verification code\")}\n                        autoComplete=\"off\"\n                        disabled={this.state.continueDisabled}\n                        value={this.state.verificationCode}\n                        onChange={this.onVerificationCodeChange}\n                    />\n                </form>\n            </span>;\n        } else if (bound) {\n            status = <AccessibleButton\n                className=\"mx_ExistingPhoneNumber_confirmBtn\"\n                kind=\"danger_sm\"\n                onClick={this.onRevokeClick}\n            >\n                {_t(\"Revoke\")}\n            </AccessibleButton>;\n        } else {\n            status = <AccessibleButton\n                className=\"mx_ExistingPhoneNumber_confirmBtn\"\n                kind=\"primary_sm\"\n                onClick={this.onShareClick}\n            >\n                {_t(\"Share\")}\n            </AccessibleButton>;\n        }\n\n        return (\n            <div className=\"mx_ExistingPhoneNumber\">\n                <span className=\"mx_ExistingPhoneNumber_address\">+{address}</span>\n                {status}\n            </div>\n        );\n    }\n}\n\n@replaceableComponent(\"views.settings.discovery.PhoneNumbers\")\nexport default class PhoneNumbers extends React.Component {\n    static propTypes = {\n        msisdns: PropTypes.array.isRequired,\n    }\n\n    render() {\n        let content;\n        if (this.props.msisdns.length > 0) {\n            content = this.props.msisdns.map((e) => {\n                return <PhoneNumber msisdn={e} key={e.address} />;\n            });\n        } else {\n            content = <span className=\"mx_SettingsTab_subsectionText\">\n                {_t(\"Discovery options will appear once you have added a phone number above.\")}\n            </span>;\n        }\n\n        return (\n            <div className=\"mx_PhoneNumbers\">\n                {content}\n            </div>\n        );\n    }\n}\n"]}