Change what AddressTile takes to be Objects

Rather than just passing in a list of strings. This paves the way
for passing in display names & avatars of looked-up threepids.
This commit is contained in:
David Baker 2017-01-24 18:23:34 +00:00
parent 5091bab657
commit 6c263c1c89
3 changed files with 109 additions and 69 deletions
src/components/views

View file

@ -164,7 +164,13 @@ module.exports = React.createClass({
queryList = this._userList.filter((user) => { queryList = this._userList.filter((user) => {
return this._matches(query, user); return this._matches(query, user);
}).map((user) => { }).map((user) => {
return user.userId; return {
addressType: 'mx',
address: user.userId,
displayName: user.displayName,
avatarMxc: user.avatarUrl,
isKnown: true,
}
}); });
// If the query isn't a user we know about, but is a // If the query isn't a user we know about, but is a
@ -172,7 +178,11 @@ module.exports = React.createClass({
if (queryList.length == 0) { if (queryList.length == 0) {
const addrType = Invite.getAddressType(query); const addrType = Invite.getAddressType(query);
if (addrType !== null) { if (addrType !== null) {
queryList.push(query); queryList.push({
addressType: addrType,
address: query,
isKnown: false,
});
} }
} }
} }
@ -204,7 +214,7 @@ module.exports = React.createClass({
onSelected: function(index) { onSelected: function(index) {
var inviteList = this.state.inviteList.slice(); var inviteList = this.state.inviteList.slice();
inviteList.push(this.state.queryList[index].userId); inviteList.push(this.state.queryList[index]);
this.setState({ this.setState({
inviteList: inviteList, inviteList: inviteList,
queryList: [], queryList: [],
@ -239,10 +249,14 @@ module.exports = React.createClass({
return; return;
} }
const addrTexts = addrs.map((addr) => {
return addr.address;
});
if (this.props.roomId) { if (this.props.roomId) {
// Invite new user to a room // Invite new user to a room
var self = this; var self = this;
Invite.inviteMultipleToRoom(this.props.roomId, addrs) Invite.inviteMultipleToRoom(this.props.roomId, addrTexts)
.then(function(addrs) { .then(function(addrs) {
var room = MatrixClientPeg.get().getRoom(self.props.roomId); var room = MatrixClientPeg.get().getRoom(self.props.roomId);
return self._showAnyInviteErrors(addrs, room); return self._showAnyInviteErrors(addrs, room);
@ -257,9 +271,9 @@ module.exports = React.createClass({
return null; return null;
}) })
.done(); .done();
} else if (this._isDmChat(addrs)) { } else if (this._isDmChat(addrTexts)) {
// Start the DM chat // Start the DM chat
createRoom({dmUserId: addrs[0]}) createRoom({dmUserId: addrTexts[0]})
.catch(function(err) { .catch(function(err) {
console.error(err.stack); console.error(err.stack);
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
@ -276,7 +290,7 @@ module.exports = React.createClass({
var room; var room;
createRoom().then(function(roomId) { createRoom().then(function(roomId) {
room = MatrixClientPeg.get().getRoom(roomId); room = MatrixClientPeg.get().getRoom(roomId);
return Invite.inviteMultipleToRoom(roomId, addrs); return Invite.inviteMultipleToRoom(roomId, addrTexts);
}) })
.then(function(addrs) { .then(function(addrs) {
return self._showAnyInviteErrors(addrs, room); return self._showAnyInviteErrors(addrs, room);
@ -294,7 +308,7 @@ module.exports = React.createClass({
} }
// Close - this will happen before the above, as that is async // Close - this will happen before the above, as that is async
this.props.onFinished(true, addrs); this.props.onFinished(true, addrTexts);
}, },
_updateUserList: new rate_limited_func(function() { _updateUserList: new rate_limited_func(function() {
@ -345,7 +359,10 @@ module.exports = React.createClass({
_isOnInviteList: function(uid) { _isOnInviteList: function(uid) {
for (let i = 0; i < this.state.inviteList.length; i++) { for (let i = 0; i < this.state.inviteList.length; i++) {
if (this.state.inviteList[i].toLowerCase() === uid) { if (
this.state.inviteList[i].addressType == 'mx' &&
this.state.inviteList[i].address.toLowerCase() === uid
) {
return true; return true;
} }
} }
@ -380,24 +397,35 @@ module.exports = React.createClass({
}, },
_addInputToList: function() { _addInputToList: function() {
const addrType = Invite.getAddressType(this.refs.textinput.value); const addressText = this.refs.textinput.value.trim();
if (addrType !== null) { const addrType = Invite.getAddressType(addressText);
const inviteList = this.state.inviteList.slice(); const addrObj = {
inviteList.push(this.refs.textinput.value.trim()); addressType: addrType,
this.setState({ address: addressText,
inviteList: inviteList, isKnown: false,
queryList: [], };
}); if (addrType == null) {
return inviteList; } else if (addrType == 'mx') {
} else { const user = MatrixClientPeg.get().getUser(addrObj.address);
this.setState({ error: true }); if (user) {
return null; addrObj.displayName = user.displayName;
addrObj.avatarMxc = user.avatarUrl;
addrObj.isKnown = true;
}
} }
const inviteList = this.state.inviteList.slice();
inviteList.push(addrObj);
this.setState({
inviteList: inviteList,
queryList: [],
});
return inviteList;
}, },
render: function() { render: function() {
var TintableSvg = sdk.getComponent("elements.TintableSvg"); const TintableSvg = sdk.getComponent("elements.TintableSvg");
var AddressSelector = sdk.getComponent("elements.AddressSelector"); const AddressSelector = sdk.getComponent("elements.AddressSelector");
this.scrollElement = null; this.scrollElement = null;
var query = []; var query = [];

View file

@ -16,18 +16,19 @@ limitations under the License.
'use strict'; 'use strict';
var React = require("react"); const React = require("react");
var sdk = require("../../../index"); const sdk = require("../../../index");
var classNames = require('classnames'); const classNames = require('classnames');
const InviteAddressType = require("./AddressTile");
module.exports = React.createClass({ export default React.createClass({
displayName: 'AddressSelector', displayName: 'AddressSelector',
propTypes: { propTypes: {
onSelected: React.PropTypes.func.isRequired, onSelected: React.PropTypes.func.isRequired,
// List of strings: the addresses to display // List of the addresses to display
addressList: React.PropTypes.array.isRequired, addressList: React.PropTypes.arrayOf(InviteAddressType).isRequired,
truncateAt: React.PropTypes.number.isRequired, truncateAt: React.PropTypes.number.isRequired,
selected: React.PropTypes.number, selected: React.PropTypes.number,

View file

@ -23,16 +23,33 @@ var Invite = require("../../../Invite");
var MatrixClientPeg = require("../../../MatrixClientPeg"); var MatrixClientPeg = require("../../../MatrixClientPeg");
var Avatar = require('../../../Avatar'); var Avatar = require('../../../Avatar');
module.exports = React.createClass({ // React PropType definition for an object describing
// an address that can be invited to a room (which
// could be a third party identifier or a matrix ID)
// along with some additional information about the
// address / target.
export const InviteAddressType = React.PropTypes.shape({
addressType: React.PropTypes.oneOf([
'mx', 'email'
]).isRequired,
address: React.PropTypes.string.isRequired,
displayName: React.PropTypes.string,
avatarMxc: React.PropTypes.string,
// true if the address is known to be a valid address (eg. is a real
// user we've seen) or false otherwise (eg. is just an address the
// user has entered)
isKnown: React.PropTypes.bool,
});
export default React.createClass({
displayName: 'AddressTile', displayName: 'AddressTile',
propTypes: { propTypes: {
address: React.PropTypes.string.isRequired, address: InviteAddressType.isRequired,
canDismiss: React.PropTypes.bool, canDismiss: React.PropTypes.bool,
onDismissed: React.PropTypes.func, onDismissed: React.PropTypes.func,
justified: React.PropTypes.bool, justified: React.PropTypes.bool,
networkName: React.PropTypes.string,
networkUrl: React.PropTypes.string,
}, },
getDefaultProps: function() { getDefaultProps: function() {
@ -40,35 +57,26 @@ module.exports = React.createClass({
canDismiss: false, canDismiss: false,
onDismissed: function() {}, // NOP onDismissed: function() {}, // NOP
justified: false, justified: false,
networkName: "",
networkUrl: "",
}; };
}, },
render: function() { render: function() {
var userId, name, imgUrl, email; const address = this.props.address;
var BaseAvatar = sdk.getComponent('avatars.BaseAvatar'); const name = address.displayName || address.address;
var TintableSvg = sdk.getComponent("elements.TintableSvg");
// Check if the addr is a valid type let imgUrl;
var addrType = Invite.getAddressType(this.props.address); if (address.avatarMxc) {
if (addrType === "mx") { imgUrl = MatrixClientPeg.get().mxcUrlToHttp(
let user = MatrixClientPeg.get().getUser(this.props.address); address.avatarMxc, 25, 25, 'crop'
if (user) { );
userId = user.userId; }
name = user.rawDisplayName || userId;
imgUrl = Avatar.avatarUrlForUser(user, 25, 25, "crop"); if (address.addressType === "mx") {
} else { if (!imgUrl) imgUrl = 'img/icon-mx-user.svg';
name=this.props.address; } else if (address.addressType === 'email') {
imgUrl = "img/icon-mx-user.svg"; if (!imgUrl) imgUrl = 'img/icon-email-user.svg';
}
} else if (addrType === "email") {
email = this.props.address;
name="email";
imgUrl = "img/icon-email-user.svg";
} else { } else {
name="Unknown"; if (!imgUrl) imgUrl = "img/avatar-error.svg";
imgUrl = "img/avatar-error.svg";
} }
// Removing networks for now as they're not really supported // Removing networks for now as they're not really supported
@ -83,15 +91,18 @@ module.exports = React.createClass({
} }
*/ */
var info; const BaseAvatar = sdk.getComponent('avatars.BaseAvatar');
var error = false; const TintableSvg = sdk.getComponent("elements.TintableSvg");
if (addrType === "mx" && userId) {
var nameClasses = classNames({ let info;
let error = false;
if (address.addressType === "mx" && address.isKnown) {
const nameClasses = classNames({
"mx_AddressTile_name": true, "mx_AddressTile_name": true,
"mx_AddressTile_justified": this.props.justified, "mx_AddressTile_justified": this.props.justified,
}); });
var idClasses = classNames({ const idClasses = classNames({
"mx_AddressTile_id": true, "mx_AddressTile_id": true,
"mx_AddressTile_justified": this.props.justified, "mx_AddressTile_justified": this.props.justified,
}); });
@ -99,26 +110,26 @@ module.exports = React.createClass({
info = ( info = (
<div className="mx_AddressTile_mx"> <div className="mx_AddressTile_mx">
<div className={nameClasses}>{ name }</div> <div className={nameClasses}>{ name }</div>
<div className={idClasses}>{ userId }</div> <div className={idClasses}>{ address.address }</div>
</div> </div>
); );
} else if (addrType === "mx") { } else if (address.addressType === "mx") {
var unknownMxClasses = classNames({ const unknownMxClasses = classNames({
"mx_AddressTile_unknownMx": true, "mx_AddressTile_unknownMx": true,
"mx_AddressTile_justified": this.props.justified, "mx_AddressTile_justified": this.props.justified,
}); });
info = ( info = (
<div className={unknownMxClasses}>{ this.props.address }</div> <div className={unknownMxClasses}>{ this.props.address.address }</div>
); );
} else if (email) { } else if (address.addressType === "email") {
var emailClasses = classNames({ var emailClasses = classNames({
"mx_AddressTile_email": true, "mx_AddressTile_email": true,
"mx_AddressTile_justified": this.props.justified, "mx_AddressTile_justified": this.props.justified,
}); });
info = ( info = (
<div className={emailClasses}>{ email }</div> <div className={emailClasses}>{ address.address }</div>
); );
} else { } else {
error = true; error = true;
@ -132,12 +143,12 @@ module.exports = React.createClass({
); );
} }
var classes = classNames({ const classes = classNames({
"mx_AddressTile": true, "mx_AddressTile": true,
"mx_AddressTile_error": error, "mx_AddressTile_error": error,
}); });
var dismiss; let dismiss;
if (this.props.canDismiss) { if (this.props.canDismiss) {
dismiss = ( dismiss = (
<div className="mx_AddressTile_dismiss" onClick={this.props.onDismissed} > <div className="mx_AddressTile_dismiss" onClick={this.props.onDismissed} >