Merge pull request #756 from matrix-org/dbkr/add_msisdn
Support adding phone numbers in UserSettingspull/21833/head
commit
a8d85ca2ad
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2016 OpenMarket Ltd
|
Copyright 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -51,11 +52,36 @@ class AddThreepid {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to add a msisdn threepid. This will trigger a side-effect of
|
||||||
|
* sending a test message to the provided phone number.
|
||||||
|
* @param {string} phoneCountry The ISO 2 letter code of the country to resolve phoneNumber in
|
||||||
|
* @param {string} phoneNumber The national or international formatted phone number to add
|
||||||
|
* @param {boolean} bind If True, bind this phone number to this mxid on the Identity Server
|
||||||
|
* @return {Promise} Resolves when the text message has been sent. Then call haveMsisdnToken().
|
||||||
|
*/
|
||||||
|
addMsisdn(phoneCountry, phoneNumber, bind) {
|
||||||
|
this.bind = bind;
|
||||||
|
return MatrixClientPeg.get().requestAdd3pidMsisdnToken(
|
||||||
|
phoneCountry, phoneNumber, this.clientSecret, 1,
|
||||||
|
).then((res) => {
|
||||||
|
this.sessionId = res.sid;
|
||||||
|
return res;
|
||||||
|
}, function(err) {
|
||||||
|
if (err.errcode == 'M_THREEPID_IN_USE') {
|
||||||
|
err.message = "This phone number is already in use";
|
||||||
|
} else if (err.httpStatus) {
|
||||||
|
err.message = err.message + ` (Status ${err.httpStatus})`;
|
||||||
|
}
|
||||||
|
throw err;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the email link has been clicked by attempting to add the threepid
|
* Checks if the email link has been clicked by attempting to add the threepid
|
||||||
* @return {Promise} Resolves if the password was reset. Rejects with an object
|
* @return {Promise} Resolves if the email address was added. Rejects with an object
|
||||||
* with a "message" property which contains a human-readable message detailing why
|
* with a "message" property which contains a human-readable message detailing why
|
||||||
* the reset failed, e.g. "There is no mapped matrix user ID for the given email address".
|
* the request failed.
|
||||||
*/
|
*/
|
||||||
checkEmailLinkClicked() {
|
checkEmailLinkClicked() {
|
||||||
var identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
var identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
||||||
|
@ -73,6 +99,29 @@ class AddThreepid {
|
||||||
throw err;
|
throw err;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes a phone number verification code as entered by the user and validates
|
||||||
|
* it with the ID server, then if successful, adds the phone number.
|
||||||
|
* @return {Promise} Resolves if the phone number was added. Rejects with an object
|
||||||
|
* with a "message" property which contains a human-readable message detailing why
|
||||||
|
* the request failed.
|
||||||
|
*/
|
||||||
|
haveMsisdnToken(token) {
|
||||||
|
return MatrixClientPeg.get().submitMsisdnToken(
|
||||||
|
this.sessionId, this.clientSecret, token,
|
||||||
|
).then((result) => {
|
||||||
|
if (result.errcode) {
|
||||||
|
throw result;
|
||||||
|
}
|
||||||
|
const identityServerDomain = MatrixClientPeg.get().idBaseUrl.split("://")[1];
|
||||||
|
return MatrixClientPeg.get().addThreePid({
|
||||||
|
sid: this.sessionId,
|
||||||
|
client_secret: this.clientSecret,
|
||||||
|
id_server: identityServerDomain
|
||||||
|
}, this.bind);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = AddThreepid;
|
module.exports = AddThreepid;
|
||||||
|
|
|
@ -229,6 +229,8 @@ import views$rooms$TopUnreadMessagesBar from './components/views/rooms/TopUnread
|
||||||
views$rooms$TopUnreadMessagesBar && (module.exports.components['views.rooms.TopUnreadMessagesBar'] = views$rooms$TopUnreadMessagesBar);
|
views$rooms$TopUnreadMessagesBar && (module.exports.components['views.rooms.TopUnreadMessagesBar'] = views$rooms$TopUnreadMessagesBar);
|
||||||
import views$rooms$UserTile from './components/views/rooms/UserTile';
|
import views$rooms$UserTile from './components/views/rooms/UserTile';
|
||||||
views$rooms$UserTile && (module.exports.components['views.rooms.UserTile'] = views$rooms$UserTile);
|
views$rooms$UserTile && (module.exports.components['views.rooms.UserTile'] = views$rooms$UserTile);
|
||||||
|
import views$settings$AddPhoneNumber from './components/views/settings/AddPhoneNumber';
|
||||||
|
views$settings$AddPhoneNumber && (module.exports.components['views.settings.AddPhoneNumber'] = views$settings$AddPhoneNumber);
|
||||||
import views$settings$ChangeAvatar from './components/views/settings/ChangeAvatar';
|
import views$settings$ChangeAvatar from './components/views/settings/ChangeAvatar';
|
||||||
views$settings$ChangeAvatar && (module.exports.components['views.settings.ChangeAvatar'] = views$settings$ChangeAvatar);
|
views$settings$ChangeAvatar && (module.exports.components['views.settings.ChangeAvatar'] = views$settings$ChangeAvatar);
|
||||||
import views$settings$ChangeDisplayName from './components/views/settings/ChangeDisplayName';
|
import views$settings$ChangeDisplayName from './components/views/settings/ChangeDisplayName';
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
Copyright 2015, 2016 OpenMarket Ltd
|
Copyright 2015, 2016 OpenMarket Ltd
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -139,6 +140,7 @@ module.exports = React.createClass({
|
||||||
|
|
||||||
componentWillMount: function() {
|
componentWillMount: function() {
|
||||||
this._unmounted = false;
|
this._unmounted = false;
|
||||||
|
this._addThreepid = null;
|
||||||
|
|
||||||
if (PlatformPeg.get()) {
|
if (PlatformPeg.get()) {
|
||||||
q().then(() => {
|
q().then(() => {
|
||||||
|
@ -321,12 +323,16 @@ module.exports = React.createClass({
|
||||||
UserSettingsStore.setEnableNotifications(event.target.checked);
|
UserSettingsStore.setEnableNotifications(event.target.checked);
|
||||||
},
|
},
|
||||||
|
|
||||||
onAddThreepidClicked: function(value, shouldSubmit) {
|
_onAddEmailEditFinished: function(value, shouldSubmit) {
|
||||||
if (!shouldSubmit) return;
|
if (!shouldSubmit) return;
|
||||||
|
this._addEmail();
|
||||||
|
},
|
||||||
|
|
||||||
|
_addEmail: function() {
|
||||||
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
var QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
var QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
|
||||||
|
|
||||||
var email_address = this.refs.add_threepid_input.value;
|
var email_address = this.refs.add_email_input.value;
|
||||||
if (!Email.looksValid(email_address)) {
|
if (!Email.looksValid(email_address)) {
|
||||||
Modal.createDialog(ErrorDialog, {
|
Modal.createDialog(ErrorDialog, {
|
||||||
title: "Invalid Email Address",
|
title: "Invalid Email Address",
|
||||||
|
@ -334,10 +340,10 @@ module.exports = React.createClass({
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.add_threepid = new AddThreepid();
|
this._addThreepid = new AddThreepid();
|
||||||
// we always bind emails when registering, so let's do the
|
// we always bind emails when registering, so let's do the
|
||||||
// same here.
|
// same here.
|
||||||
this.add_threepid.addEmailAddress(email_address, true).done(() => {
|
this._addThreepid.addEmailAddress(email_address, true).done(() => {
|
||||||
Modal.createDialog(QuestionDialog, {
|
Modal.createDialog(QuestionDialog, {
|
||||||
title: "Verification Pending",
|
title: "Verification Pending",
|
||||||
description: "Please check your email and click on the link it contains. Once this is done, click continue.",
|
description: "Please check your email and click on the link it contains. Once this is done, click continue.",
|
||||||
|
@ -352,7 +358,7 @@ module.exports = React.createClass({
|
||||||
description: "Unable to add email address"
|
description: "Unable to add email address"
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
ReactDOM.findDOMNode(this.refs.add_threepid_input).blur();
|
ReactDOM.findDOMNode(this.refs.add_email_input).blur();
|
||||||
this.setState({email_add_pending: true});
|
this.setState({email_add_pending: true});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -391,8 +397,8 @@ module.exports = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
verifyEmailAddress: function() {
|
verifyEmailAddress: function() {
|
||||||
this.add_threepid.checkEmailLinkClicked().done(() => {
|
this._addThreepid.checkEmailLinkClicked().done(() => {
|
||||||
this.add_threepid = undefined;
|
this._addThreepid = null;
|
||||||
this.setState({
|
this.setState({
|
||||||
phase: "UserSettings.LOADING",
|
phase: "UserSettings.LOADING",
|
||||||
});
|
});
|
||||||
|
@ -811,30 +817,35 @@ module.exports = React.createClass({
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
var addThreepidSection;
|
let addEmailSection;
|
||||||
if (this.state.email_add_pending) {
|
if (this.state.email_add_pending) {
|
||||||
addThreepidSection = <Loader />;
|
addEmailSection = <Loader key="_email_add_spinner" />;
|
||||||
} else if (!MatrixClientPeg.get().isGuest()) {
|
} else if (!MatrixClientPeg.get().isGuest()) {
|
||||||
addThreepidSection = (
|
addEmailSection = (
|
||||||
<div className="mx_UserSettings_profileTableRow" key="new">
|
<div className="mx_UserSettings_profileTableRow" key="_newEmail">
|
||||||
<div className="mx_UserSettings_profileLabelCell">
|
<div className="mx_UserSettings_profileLabelCell">
|
||||||
</div>
|
</div>
|
||||||
<div className="mx_UserSettings_profileInputCell">
|
<div className="mx_UserSettings_profileInputCell">
|
||||||
<EditableText
|
<EditableText
|
||||||
ref="add_threepid_input"
|
ref="add_email_input"
|
||||||
className="mx_UserSettings_editable"
|
className="mx_UserSettings_editable"
|
||||||
placeholderClassName="mx_UserSettings_threepidPlaceholder"
|
placeholderClassName="mx_UserSettings_threepidPlaceholder"
|
||||||
placeholder={ "Add email address" }
|
placeholder={ "Add email address" }
|
||||||
blurToCancel={ false }
|
blurToCancel={ false }
|
||||||
onValueChanged={ this.onAddThreepidClicked } />
|
onValueChanged={ this._onAddEmailEditFinished } />
|
||||||
</div>
|
</div>
|
||||||
<div className="mx_UserSettings_threepidButton mx_filterFlipColor">
|
<div className="mx_UserSettings_threepidButton mx_filterFlipColor">
|
||||||
<img src="img/plus.svg" width="14" height="14" alt="Add" onClick={ this.onAddThreepidClicked.bind(this, undefined, true) }/>
|
<img src="img/plus.svg" width="14" height="14" alt="Add" onClick={this._addEmail} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
threepidsSection.push(addThreepidSection);
|
const AddPhoneNumber = sdk.getComponent('views.settings.AddPhoneNumber');
|
||||||
|
const addMsisdnSection = (
|
||||||
|
<AddPhoneNumber key="_addMsisdn" onThreepidAdded={this._refreshFromServer} />
|
||||||
|
);
|
||||||
|
threepidsSection.push(addEmailSection);
|
||||||
|
threepidsSection.push(addMsisdnSection);
|
||||||
|
|
||||||
var accountJsx;
|
var accountJsx;
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,172 @@
|
||||||
|
/*
|
||||||
|
Copyright 2017 Vector Creations Ltd
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
import sdk from '../../../index';
|
||||||
|
import AddThreepid from '../../../AddThreepid';
|
||||||
|
import WithMatrixClient from '../../../wrappers/WithMatrixClient';
|
||||||
|
import Modal from '../../../Modal';
|
||||||
|
|
||||||
|
|
||||||
|
export default WithMatrixClient(React.createClass({
|
||||||
|
displayName: 'AddPhoneNumber',
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
matrixClient: React.PropTypes.object.isRequired,
|
||||||
|
onThreepidAdded: React.PropTypes.func,
|
||||||
|
},
|
||||||
|
|
||||||
|
getInitialState: function() {
|
||||||
|
return {
|
||||||
|
busy: false,
|
||||||
|
phoneCountry: null,
|
||||||
|
phoneNumber: "",
|
||||||
|
msisdn_add_pending: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillMount: function() {
|
||||||
|
this._addThreepid = null;
|
||||||
|
this._addMsisdnInput = null;
|
||||||
|
this._unmounted = false;
|
||||||
|
},
|
||||||
|
|
||||||
|
componentWillUnmount: function() {
|
||||||
|
this._unmounted = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
_onPhoneCountryChange: function(phoneCountry) {
|
||||||
|
this.setState({ phoneCountry: phoneCountry });
|
||||||
|
},
|
||||||
|
|
||||||
|
_onPhoneNumberChange: function(ev) {
|
||||||
|
this.setState({ phoneNumber: ev.target.value });
|
||||||
|
},
|
||||||
|
|
||||||
|
_onAddMsisdnEditFinished: function(value, shouldSubmit) {
|
||||||
|
if (!shouldSubmit) return;
|
||||||
|
this._addMsisdn();
|
||||||
|
},
|
||||||
|
|
||||||
|
_onAddMsisdnSubmit: function(ev) {
|
||||||
|
ev.preventDefault();
|
||||||
|
this._addMsisdn();
|
||||||
|
},
|
||||||
|
|
||||||
|
_collectAddMsisdnInput: function(e) {
|
||||||
|
this._addMsisdnInput = e;
|
||||||
|
},
|
||||||
|
|
||||||
|
_addMsisdn: function() {
|
||||||
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
||||||
|
|
||||||
|
this._addThreepid = new AddThreepid();
|
||||||
|
// we always bind phone numbers when registering, so let's do the
|
||||||
|
// same here.
|
||||||
|
this._addThreepid.addMsisdn(this.state.phoneCountry, this.state.phoneNumber, true).then((resp) => {
|
||||||
|
this._promptForMsisdnVerificationCode(resp.msisdn);
|
||||||
|
}).catch((err) => {
|
||||||
|
console.error("Unable to add phone number: " + err);
|
||||||
|
let msg = err.message;
|
||||||
|
Modal.createDialog(ErrorDialog, {
|
||||||
|
title: "Error",
|
||||||
|
description: msg,
|
||||||
|
});
|
||||||
|
}).finally(() => {
|
||||||
|
if (this._unmounted) return;
|
||||||
|
this.setState({msisdn_add_pending: false});
|
||||||
|
}).done();
|
||||||
|
this._addMsisdnInput.blur();
|
||||||
|
this.setState({msisdn_add_pending: true});
|
||||||
|
},
|
||||||
|
|
||||||
|
_promptForMsisdnVerificationCode:function (msisdn, err) {
|
||||||
|
if (this._unmounted) return;
|
||||||
|
const TextInputDialog = sdk.getComponent("dialogs.TextInputDialog");
|
||||||
|
let msgElements = [
|
||||||
|
<div key="_static" >A text message has been sent to +{msisdn}.
|
||||||
|
Please enter the verification code it contains</div>
|
||||||
|
];
|
||||||
|
if (err) {
|
||||||
|
let msg = err.error;
|
||||||
|
if (err.errcode == 'M_THREEPID_AUTH_FAILED') {
|
||||||
|
msg = "Incorrect verification code";
|
||||||
|
}
|
||||||
|
msgElements.push(<div key="_error" className="error">{msg}</div>);
|
||||||
|
}
|
||||||
|
Modal.createDialog(TextInputDialog, {
|
||||||
|
title: "Enter Code",
|
||||||
|
description: <div>{msgElements}</div>,
|
||||||
|
button: "Submit",
|
||||||
|
onFinished: (should_verify, token) => {
|
||||||
|
if (!should_verify) {
|
||||||
|
this._addThreepid = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this._unmounted) return;
|
||||||
|
this.setState({msisdn_add_pending: true});
|
||||||
|
this._addThreepid.haveMsisdnToken(token).then(() => {
|
||||||
|
this._addThreepid = null;
|
||||||
|
this.setState({phoneNumber: ''});
|
||||||
|
if (this.props.onThreepidAdded) this.props.onThreepidAdded();
|
||||||
|
}).catch((err) => {
|
||||||
|
this._promptForMsisdnVerificationCode(msisdn, err);
|
||||||
|
}).finally(() => {
|
||||||
|
if (this._unmounted) return;
|
||||||
|
this.setState({msisdn_add_pending: false});
|
||||||
|
}).done();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
const Loader = sdk.getComponent("elements.Spinner");
|
||||||
|
if (this.state.msisdn_add_pending) {
|
||||||
|
return <Loader />;
|
||||||
|
} else if (this.props.matrixClient.isGuest()) {
|
||||||
|
return <div />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const CountryDropdown = sdk.getComponent('views.login.CountryDropdown');
|
||||||
|
// XXX: This CSS relies on the CSS surrounding it in UserSettings as its in
|
||||||
|
// a tabular format to align the submit buttons
|
||||||
|
return (
|
||||||
|
<form className="mx_UserSettings_profileTableRow" onSubmit={this._onAddMsisdnSubmit}>
|
||||||
|
<div className="mx_UserSettings_profileLabelCell">
|
||||||
|
</div>
|
||||||
|
<div className="mx_UserSettings_profileInputCell">
|
||||||
|
<div className="mx_Login_phoneSection">
|
||||||
|
<CountryDropdown onOptionChange={this._onPhoneCountryChange}
|
||||||
|
className="mx_Login_phoneCountry"
|
||||||
|
value={this.state.phoneCountry}
|
||||||
|
/>
|
||||||
|
<input type="text"
|
||||||
|
ref={this._collectAddMsisdnInput}
|
||||||
|
className="mx_UserSettings_phoneNumberField"
|
||||||
|
placeholder="Add phone number"
|
||||||
|
value={this.state.phoneNumber}
|
||||||
|
onChange={this._onPhoneNumberChange}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mx_UserSettings_threepidButton mx_filterFlipColor">
|
||||||
|
<input type="image" value="Add" src="img/plus.svg" width="14" height="14" />
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}))
|
Loading…
Reference in New Issue