Fold BaseAvatar state mainly into state.urls to avoid too many code paths.

pull/21833/head
Kegan Dougal 2016-01-15 17:05:05 +00:00
parent d8d79722ac
commit 855bef17fa
2 changed files with 78 additions and 59 deletions

View File

@ -26,12 +26,12 @@ module.exports = React.createClass({
name: React.PropTypes.string.isRequired, // The name (first initial used as default) name: React.PropTypes.string.isRequired, // The name (first initial used as default)
idName: React.PropTypes.string, // ID for generating hash colours idName: React.PropTypes.string, // ID for generating hash colours
title: React.PropTypes.string, // onHover title text title: React.PropTypes.string, // onHover title text
url: React.PropTypes.string, // highest priority of them all url: React.PropTypes.string, // highest priority of them all, shortcut to set in urls[0]
urls: React.PropTypes.array, // [highest_priority, ... , lowest_priority] urls: React.PropTypes.array, // [highest_priority, ... , lowest_priority]
width: React.PropTypes.number, width: React.PropTypes.number,
height: React.PropTypes.number, height: React.PropTypes.number,
resizeMethod: React.PropTypes.string, resizeMethod: React.PropTypes.string,
defaultToInitialLetter: React.PropTypes.bool defaultToInitialLetter: React.PropTypes.bool // true to add default url
}, },
getDefaultProps: function() { getDefaultProps: function() {
@ -44,51 +44,58 @@ module.exports = React.createClass({
}, },
getInitialState: function() { getInitialState: function() {
return this._getState(this.props);
},
componentWillReceiveProps: function(nextProps) {
// work out if we need to call setState (if the image URLs array has changed)
var newState = this._getState(nextProps);
var newImageUrls = newState.imageUrls;
var oldImageUrls = this.state.imageUrls;
if (newImageUrls.length !== oldImageUrls.length) {
this.setState(newState); // detected a new entry
}
else {
// check each one to see if they are the same
for (var i = 0; i < newImageUrls.length; i++) {
if (oldImageUrls[i] !== newImageUrls[i]) {
this.setState(newState); // detected a diff
break;
}
}
}
},
_getState: function(props) {
// work out the full set of urls to try to load. This is formed like so:
// imageUrls: [ props.url, props.urls, default image ]
var urls = props.urls || [];
if (props.url) {
urls.unshift(props.url); // put in urls[0]
}
var defaultImageUrl = null; var defaultImageUrl = null;
if (this.props.defaultToInitialLetter) { if (props.defaultToInitialLetter) {
defaultImageUrl = AvatarLogic.defaultAvatarUrlForString( defaultImageUrl = AvatarLogic.defaultAvatarUrlForString(
this.props.idName || this.props.name props.idName || props.name
); );
urls.push(defaultImageUrl); // lowest priority
} }
return { return {
imageUrl: this.props.url || (this.props.urls ? this.props.urls[0] : null), imageUrls: urls,
defaultImageUrl: defaultImageUrl, defaultImageUrl: defaultImageUrl,
urlsIndex: 0 urlsIndex: 0
}; };
}, },
componentWillReceiveProps: function(nextProps) {
// retry all the urls again, they may have changed.
if (this.props.urls && this.state.urlsIndex > 0) {
this.setState({
urlsIndex: 0,
imageUrl: this.props.urls[0]
});
}
},
onError: function(ev) { onError: function(ev) {
var failedUrl = ev.target.src;
if (this.props.urls) {
var nextIndex = this.state.urlsIndex + 1; var nextIndex = this.state.urlsIndex + 1;
if (nextIndex < this.props.urls.length) { if (nextIndex < this.state.imageUrls.length) {
// try another // try the next one
this.setState({ this.setState({
urlsIndex: nextIndex, urlsIndex: nextIndex
imageUrl: this.props.urls[nextIndex]
}); });
return;
}
}
// either no urls array or we've reached the end of it, we may have a default
// we can use...
if (this.props.defaultToInitialLetter) {
if (failedUrl === this.state.defaultImageUrl) {
return; // don't tightloop if the browser can't load the default URL
}
this.setState({ imageUrl: this.state.defaultImageUrl })
} }
}, },
@ -104,7 +111,9 @@ module.exports = React.createClass({
render: function() { render: function() {
var name = this.props.name; var name = this.props.name;
if (this.state.imageUrl === this.state.defaultImageUrl) { var imageUrl = this.state.imageUrls[this.state.urlsIndex];
if (imageUrl === this.state.defaultImageUrl) {
var initialLetter = this._getInitialLetter(); var initialLetter = this._getInitialLetter();
return ( return (
<span className="mx_BaseAvatar" {...this.props}> <span className="mx_BaseAvatar" {...this.props}>
@ -114,14 +123,14 @@ module.exports = React.createClass({
lineHeight: this.props.height + "px" }}> lineHeight: this.props.height + "px" }}>
{ initialLetter } { initialLetter }
</span> </span>
<img className="mx_BaseAvatar_image" src={this.state.imageUrl} <img className="mx_BaseAvatar_image" src={imageUrl}
title={this.props.title} onError={this.onError} title={this.props.title} onError={this.onError}
width={this.props.width} height={this.props.height} /> width={this.props.width} height={this.props.height} />
</span> </span>
); );
} }
return ( return (
<img className="mx_BaseAvatar mx_BaseAvatar_image" src={this.state.imageUrl} <img className="mx_BaseAvatar mx_BaseAvatar_image" src={imageUrl}
onError={this.onError} onError={this.onError}
width={this.props.width} height={this.props.height} width={this.props.width} height={this.props.height}
title={this.props.title} title={this.props.title}

View File

@ -38,43 +38,53 @@ module.exports = React.createClass({
getInitialState: function() { getInitialState: function() {
return { return {
urls: [ urls: this.getImageUrls(this.props)
this.getRoomAvatarUrl(), // highest priority
this.getOneToOneAvatar(),
this.getFallbackAvatar() // lowest priority
].filter(function(url) {
return url != null;
})
}; };
}, },
getRoomAvatarUrl: function() { componentWillReceiveProps: function(newProps) {
return this.props.room.getAvatarUrl( this.setState({
urls: this.getImageUrls(newProps)
})
},
getImageUrls: function(props) {
return [
this.getRoomAvatarUrl(props), // highest priority
this.getOneToOneAvatar(props),
this.getFallbackAvatar(props) // lowest priority
].filter(function(url) {
return url != null;
});
},
getRoomAvatarUrl: function(props) {
return props.room.getAvatarUrl(
MatrixClientPeg.get().getHomeserverUrl(), MatrixClientPeg.get().getHomeserverUrl(),
this.props.width, this.props.height, this.props.resizeMethod, props.width, props.height, props.resizeMethod,
false false
); );
}, },
getOneToOneAvatar: function() { getOneToOneAvatar: function(props) {
var userIds = Object.keys(this.props.room.currentState.members); var userIds = Object.keys(props.room.currentState.members);
if (userIds.length == 2) { if (userIds.length == 2) {
var theOtherGuy = null; var theOtherGuy = null;
if (this.props.room.currentState.members[userIds[0]].userId == MatrixClientPeg.get().credentials.userId) { if (props.room.currentState.members[userIds[0]].userId == MatrixClientPeg.get().credentials.userId) {
theOtherGuy = this.props.room.currentState.members[userIds[1]]; theOtherGuy = props.room.currentState.members[userIds[1]];
} else { } else {
theOtherGuy = this.props.room.currentState.members[userIds[0]]; theOtherGuy = props.room.currentState.members[userIds[0]];
} }
return theOtherGuy.getAvatarUrl( return theOtherGuy.getAvatarUrl(
MatrixClientPeg.get().getHomeserverUrl(), MatrixClientPeg.get().getHomeserverUrl(),
this.props.width, this.props.height, this.props.resizeMethod, props.width, props.height, props.resizeMethod,
false false
); );
} else if (userIds.length == 1) { } else if (userIds.length == 1) {
return this.props.room.currentState.members[userIds[0]].getAvatarUrl( return props.room.currentState.members[userIds[0]].getAvatarUrl(
MatrixClientPeg.get().getHomeserverUrl(), MatrixClientPeg.get().getHomeserverUrl(),
this.props.width, this.props.height, this.props.resizeMethod, props.width, props.height, props.resizeMethod,
false false
); );
} else { } else {
@ -82,8 +92,8 @@ module.exports = React.createClass({
} }
}, },
getFallbackAvatar: function() { getFallbackAvatar: function(props) {
return Avatar.defaultAvatarUrlForString(this.props.room.roomId); return Avatar.defaultAvatarUrlForString(props.room.roomId);
}, },
render: function() { render: function() {