mirror of https://github.com/vector-im/riot-web
Merge pull request #2072 from matrix-org/bwindels/lazy_load_members
Lazy load room members - Part Ipull/21833/head
commit
0dd860242a
|
@ -34,6 +34,14 @@ interface MatrixClientCreds {
|
||||||
guest: boolean,
|
guest: boolean,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const FILTER_CONTENT = {
|
||||||
|
room: {
|
||||||
|
state: {
|
||||||
|
lazy_load_members: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Wrapper object for handling the js-sdk Matrix Client object in the react-sdk
|
* Wrapper object for handling the js-sdk Matrix Client object in the react-sdk
|
||||||
* Handles the creation/initialisation of client objects.
|
* Handles the creation/initialisation of client objects.
|
||||||
|
@ -99,6 +107,10 @@ class MatrixClientPeg {
|
||||||
// the react sdk doesn't work without this, so don't allow
|
// the react sdk doesn't work without this, so don't allow
|
||||||
opts.pendingEventOrdering = "detached";
|
opts.pendingEventOrdering = "detached";
|
||||||
|
|
||||||
|
if (SettingsStore.isFeatureEnabled('feature_lazyloading')) {
|
||||||
|
opts.filter = await this.matrixClient.createFilter(FILTER_CONTENT);
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const promise = this.matrixClient.store.startup();
|
const promise = this.matrixClient.store.startup();
|
||||||
console.log(`MatrixClientPeg: waiting for MatrixClient store to initialise`);
|
console.log(`MatrixClientPeg: waiting for MatrixClient store to initialise`);
|
||||||
|
|
51
src/Rooms.js
51
src/Rooms.js
|
@ -31,26 +31,26 @@ export function getDisplayAliasForRoom(room) {
|
||||||
* If the room contains only two members including the logged-in user,
|
* If the room contains only two members including the logged-in user,
|
||||||
* return the other one. Otherwise, return null.
|
* return the other one. Otherwise, return null.
|
||||||
*/
|
*/
|
||||||
export function getOnlyOtherMember(room, me) {
|
export function getOnlyOtherMember(room, myUserId) {
|
||||||
const joinedMembers = room.getJoinedMembers();
|
|
||||||
|
|
||||||
if (joinedMembers.length === 2) {
|
if (room.currentState.getJoinedMemberCount() === 2) {
|
||||||
return joinedMembers.filter(function(m) {
|
return room.getJoinedMembers().filter(function(m) {
|
||||||
return m.userId !== me.userId;
|
return m.userId !== myUserId;
|
||||||
})[0];
|
})[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function _isConfCallRoom(room, me, conferenceHandler) {
|
function _isConfCallRoom(room, myUserId, conferenceHandler) {
|
||||||
if (!conferenceHandler) return false;
|
if (!conferenceHandler) return false;
|
||||||
|
|
||||||
if (me.membership != "join") {
|
const myMembership = room.getMyMembership(myUserId);
|
||||||
|
if (myMembership != "join") {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const otherMember = getOnlyOtherMember(room, me);
|
const otherMember = getOnlyOtherMember(room, myUserId);
|
||||||
if (otherMember === null) {
|
if (otherMember === null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -68,29 +68,30 @@ const isConfCallRoomCache = {
|
||||||
// $roomId: bool
|
// $roomId: bool
|
||||||
};
|
};
|
||||||
|
|
||||||
export function isConfCallRoom(room, me, conferenceHandler) {
|
export function isConfCallRoom(room, myUserId, conferenceHandler) {
|
||||||
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
||||||
return isConfCallRoomCache[room.roomId];
|
return isConfCallRoomCache[room.roomId];
|
||||||
}
|
}
|
||||||
|
|
||||||
const result = _isConfCallRoom(room, me, conferenceHandler);
|
const result = _isConfCallRoom(room, myUserId, conferenceHandler);
|
||||||
|
|
||||||
isConfCallRoomCache[room.roomId] = result;
|
isConfCallRoomCache[room.roomId] = result;
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function looksLikeDirectMessageRoom(room, me) {
|
export function looksLikeDirectMessageRoom(room, myUserId) {
|
||||||
if (me.membership == "join" || me.membership === "ban" ||
|
const myMembership = room.getMyMembership(myUserId);
|
||||||
(me.membership === "leave" && me.events.member.getSender() !== me.events.member.getStateKey())) {
|
const me = room.getMember(myUserId);
|
||||||
|
|
||||||
|
if (myMembership == "join" || myMembership === "ban" || (me && me.isKicked())) {
|
||||||
// Used to split rooms via tags
|
// Used to split rooms via tags
|
||||||
const tagNames = Object.keys(room.tags);
|
const tagNames = Object.keys(room.tags);
|
||||||
// Used for 1:1 direct chats
|
// Used for 1:1 direct chats
|
||||||
const members = room.currentState.getMembers();
|
|
||||||
|
|
||||||
// Show 1:1 chats in seperate "Direct Messages" section as long as they haven't
|
// Show 1:1 chats in seperate "Direct Messages" section as long as they haven't
|
||||||
// been moved to a different tag section
|
// been moved to a different tag section
|
||||||
if (members.length === 2 && !tagNames.length) {
|
// TODO: Use SUMMARYAPI to take invited users into account
|
||||||
|
if (room.currentState.getJoinedMemberCount() === 2 && !tagNames.length) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -100,10 +101,10 @@ export function looksLikeDirectMessageRoom(room, me) {
|
||||||
export function guessAndSetDMRoom(room, isDirect) {
|
export function guessAndSetDMRoom(room, isDirect) {
|
||||||
let newTarget;
|
let newTarget;
|
||||||
if (isDirect) {
|
if (isDirect) {
|
||||||
const guessedTarget = guessDMRoomTarget(
|
const guessedUserId = guessDMRoomTargetId(
|
||||||
room, room.getMember(MatrixClientPeg.get().credentials.userId),
|
room, MatrixClientPeg.get().getUserId()
|
||||||
);
|
);
|
||||||
newTarget = guessedTarget.userId;
|
newTarget = guessedUserId;
|
||||||
} else {
|
} else {
|
||||||
newTarget = null;
|
newTarget = null;
|
||||||
}
|
}
|
||||||
|
@ -159,15 +160,15 @@ export function setDMRoom(roomId, userId) {
|
||||||
* Given a room, estimate which of its members is likely to
|
* Given a room, estimate which of its members is likely to
|
||||||
* be the target if the room were a DM room and return that user.
|
* be the target if the room were a DM room and return that user.
|
||||||
*/
|
*/
|
||||||
export function guessDMRoomTarget(room, me) {
|
function guessDMRoomTargetId(room, myUserId) {
|
||||||
let oldestTs;
|
let oldestTs;
|
||||||
let oldestUser;
|
let oldestUser;
|
||||||
|
|
||||||
// Pick the joined user who's been here longest (and isn't us),
|
// Pick the joined user who's been here longest (and isn't us),
|
||||||
for (const user of room.getJoinedMembers()) {
|
for (const user of room.getJoinedMembers()) {
|
||||||
if (user.userId == me.userId) continue;
|
if (user.userId == myUserId) continue;
|
||||||
|
|
||||||
if (oldestTs === undefined || user.events.member.getTs() < oldestTs) {
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
||||||
oldestUser = user;
|
oldestUser = user;
|
||||||
oldestTs = user.events.member.getTs();
|
oldestTs = user.events.member.getTs();
|
||||||
}
|
}
|
||||||
|
@ -176,14 +177,14 @@ export function guessDMRoomTarget(room, me) {
|
||||||
|
|
||||||
// if there are no joined members other than us, use the oldest member
|
// if there are no joined members other than us, use the oldest member
|
||||||
for (const user of room.currentState.getMembers()) {
|
for (const user of room.currentState.getMembers()) {
|
||||||
if (user.userId == me.userId) continue;
|
if (user.userId == myUserId) continue;
|
||||||
|
|
||||||
if (oldestTs === undefined || user.events.member.getTs() < oldestTs) {
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
||||||
oldestUser = user;
|
oldestUser = user;
|
||||||
oldestTs = user.events.member.getTs();
|
oldestTs = user.events.member.getTs();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldestUser === undefined) return me;
|
if (oldestUser === undefined) return myUserId;
|
||||||
return oldestUser;
|
return oldestUser;
|
||||||
}
|
}
|
||||||
|
|
|
@ -309,9 +309,21 @@ module.exports = React.createClass({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else if (room) {
|
} else if (room) {
|
||||||
|
//viewing a previously joined room, try to lazy load members
|
||||||
|
|
||||||
// Stop peeking because we have joined this room previously
|
// Stop peeking because we have joined this room previously
|
||||||
MatrixClientPeg.get().stopPeeking();
|
MatrixClientPeg.get().stopPeeking();
|
||||||
this.setState({isPeeking: false});
|
this.setState({isPeeking: false});
|
||||||
|
|
||||||
|
// lazy load members if enabled
|
||||||
|
if (SettingsStore.isFeatureEnabled('feature_lazyloading')) {
|
||||||
|
MatrixClientPeg.get().loadRoomMembersIfNeeded(room.roomId).catch((err) => {
|
||||||
|
const errorMessage = `Fetching room members for ${this.roomId} failed.` +
|
||||||
|
" Room members will appear incomplete.";
|
||||||
|
console.error(errorMessage);
|
||||||
|
console.error(err);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -746,40 +758,15 @@ module.exports = React.createClass({
|
||||||
},
|
},
|
||||||
|
|
||||||
_updateDMState() {
|
_updateDMState() {
|
||||||
const me = this.state.room.getMember(MatrixClientPeg.get().credentials.userId);
|
const me = this.state.room.getMember(MatrixClientPeg.get().getUserId());
|
||||||
if (!me || me.membership !== "join") {
|
if (!me || me.membership !== "join") {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const roomId = this.state.room.roomId;
|
||||||
|
const dmInviter = me.getDMInviter();
|
||||||
|
|
||||||
// The user may have accepted an invite with is_direct set
|
if (dmInviter) {
|
||||||
if (me.events.member.getPrevContent().membership === "invite" &&
|
Rooms.setDMRoom(roomId, dmInviter);
|
||||||
me.events.member.getPrevContent().is_direct
|
|
||||||
) {
|
|
||||||
// This is a DM with the sender of the invite event (which we assume
|
|
||||||
// preceded the join event)
|
|
||||||
Rooms.setDMRoom(
|
|
||||||
this.state.room.roomId,
|
|
||||||
me.events.member.getUnsigned().prev_sender,
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const invitedMembers = this.state.room.getMembersWithMembership("invite");
|
|
||||||
const joinedMembers = this.state.room.getMembersWithMembership("join");
|
|
||||||
|
|
||||||
// There must be one invited member and one joined member
|
|
||||||
if (invitedMembers.length !== 1 || joinedMembers.length !== 1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The user may have sent an invite with is_direct sent
|
|
||||||
const other = invitedMembers[0];
|
|
||||||
if (other &&
|
|
||||||
other.membership === "invite" &&
|
|
||||||
other.events.member.getContent().is_direct
|
|
||||||
) {
|
|
||||||
Rooms.setDMRoom(this.state.room.roomId, other.userId);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -598,7 +598,7 @@ module.exports = withMatrixClient(React.createClass({
|
||||||
|
|
||||||
onMemberAvatarClick: function() {
|
onMemberAvatarClick: function() {
|
||||||
const member = this.props.member;
|
const member = this.props.member;
|
||||||
const avatarUrl = member.user ? member.user.avatarUrl : member.events.member.getContent().avatar_url;
|
const avatarUrl = member.getMxcAvatarUrl();
|
||||||
if (!avatarUrl) return;
|
if (!avatarUrl) return;
|
||||||
|
|
||||||
const httpUrl = this.props.matrixClient.mxcUrlToHttp(avatarUrl);
|
const httpUrl = this.props.matrixClient.mxcUrlToHttp(avatarUrl);
|
||||||
|
|
|
@ -342,8 +342,8 @@ module.exports = React.createClass({
|
||||||
if (!taggedRoom) {
|
if (!taggedRoom) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const me = taggedRoom.getMember(MatrixClientPeg.get().credentials.userId);
|
const myUserId = MatrixClientPeg.get().getUserId();
|
||||||
if (HIDE_CONFERENCE_CHANS && Rooms.isConfCallRoom(taggedRoom, me, this.props.ConferenceHandler)) {
|
if (HIDE_CONFERENCE_CHANS && Rooms.isConfCallRoom(taggedRoom, myUserId, this.props.ConferenceHandler)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -98,15 +98,11 @@ module.exports = React.createClass({
|
||||||
</div>);
|
</div>);
|
||||||
}
|
}
|
||||||
|
|
||||||
const myMember = this.props.room ? this.props.room.currentState.members[
|
const myMember = this.props.room ?
|
||||||
MatrixClientPeg.get().credentials.userId
|
this.props.room.getMember(MatrixClientPeg.get().getUserId()) :
|
||||||
] : null;
|
null;
|
||||||
const kicked = (
|
const kicked = myMember && myMember.isKicked();
|
||||||
myMember &&
|
const banned = myMember && myMember && myMember.membership == 'ban';
|
||||||
myMember.membership == 'leave' &&
|
|
||||||
myMember.events.member.getSender() != MatrixClientPeg.get().credentials.userId
|
|
||||||
);
|
|
||||||
const banned = myMember && myMember.membership == 'ban';
|
|
||||||
|
|
||||||
if (this.props.inviterName) {
|
if (this.props.inviterName) {
|
||||||
let emailMatchBlock;
|
let emailMatchBlock;
|
||||||
|
|
|
@ -83,6 +83,11 @@ export const SETTINGS = {
|
||||||
supportedLevels: LEVELS_FEATURE,
|
supportedLevels: LEVELS_FEATURE,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
"feature_lazyloading": {
|
||||||
|
isFeature: true,
|
||||||
|
displayName: _td("Increase performance by loading room members on first view"),
|
||||||
|
supportedLevels: LEVELS_FEATURE,
|
||||||
|
},
|
||||||
"MessageComposerInput.dontSuggestEmoji": {
|
"MessageComposerInput.dontSuggestEmoji": {
|
||||||
supportedLevels: LEVELS_ACCOUNT_SETTINGS,
|
supportedLevels: LEVELS_ACCOUNT_SETTINGS,
|
||||||
displayName: _td('Disable Emoji suggestions while typing'),
|
displayName: _td('Disable Emoji suggestions while typing'),
|
||||||
|
|
|
@ -173,13 +173,13 @@ class RoomListStore extends Store {
|
||||||
if (!this._matrixClient) return;
|
if (!this._matrixClient) return;
|
||||||
|
|
||||||
this._matrixClient.getRooms().forEach((room, index) => {
|
this._matrixClient.getRooms().forEach((room, index) => {
|
||||||
const me = room.getMember(this._matrixClient.credentials.userId);
|
const myUserId = this._matrixClient.getUserId();
|
||||||
if (!me) return;
|
const membership = room.getMyMembership(myUserId);
|
||||||
|
const me = room.getMember(myUserId);
|
||||||
|
|
||||||
if (me.membership == "invite") {
|
if (membership == "invite") {
|
||||||
lists["im.vector.fake.invite"].push(room);
|
lists["im.vector.fake.invite"].push(room);
|
||||||
} else if (me.membership == "join" || me.membership === "ban" ||
|
} else if (membership == "join" || membership === "ban" || (me && me.isKicked())) {
|
||||||
(me.membership === "leave" && me.events.member.getSender() !== me.events.member.getStateKey())) {
|
|
||||||
// Used to split rooms via tags
|
// Used to split rooms via tags
|
||||||
let tagNames = Object.keys(room.tags);
|
let tagNames = Object.keys(room.tags);
|
||||||
|
|
||||||
|
@ -206,10 +206,10 @@ class RoomListStore extends Store {
|
||||||
} else {
|
} else {
|
||||||
lists["im.vector.fake.recent"].push(room);
|
lists["im.vector.fake.recent"].push(room);
|
||||||
}
|
}
|
||||||
} else if (me.membership === "leave") {
|
} else if (membership === "leave") {
|
||||||
lists["im.vector.fake.archived"].push(room);
|
lists["im.vector.fake.archived"].push(room);
|
||||||
} else {
|
} else {
|
||||||
console.error("unrecognised membership: " + me.membership + " - this should never happen");
|
console.error("unrecognised membership: " + membership + " - this should never happen");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -96,15 +96,10 @@ export default class DMRoomMap {
|
||||||
if (this.roomToUser[roomId] === undefined) {
|
if (this.roomToUser[roomId] === undefined) {
|
||||||
// no entry? if the room is an invite, look for the is_direct hint.
|
// no entry? if the room is an invite, look for the is_direct hint.
|
||||||
const room = this.matrixClient.getRoom(roomId);
|
const room = this.matrixClient.getRoom(roomId);
|
||||||
|
// TODO Use SUMMARYAPI to fix DM detection?
|
||||||
if (room) {
|
if (room) {
|
||||||
const me = room.getMember(this.matrixClient.credentials.userId);
|
const me = room.getMember(this.matrixClient.getUserId());
|
||||||
if (me.membership == 'invite') {
|
return me && me.getDMInviter();
|
||||||
// The 'direct' hihnt is there, so declare that this is a DM room for
|
|
||||||
// whoever invited us.
|
|
||||||
if (me.events.member.getContent().is_direct) {
|
|
||||||
return me.events.member.getSender();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return this.roomToUser[roomId];
|
return this.roomToUser[roomId];
|
||||||
|
|
|
@ -14,7 +14,7 @@ import dis from '../../../../src/dispatcher';
|
||||||
import DMRoomMap from '../../../../src/utils/DMRoomMap.js';
|
import DMRoomMap from '../../../../src/utils/DMRoomMap.js';
|
||||||
import GroupStore from '../../../../src/stores/GroupStore.js';
|
import GroupStore from '../../../../src/stores/GroupStore.js';
|
||||||
|
|
||||||
import { Room, RoomMember } from 'matrix-js-sdk';
|
import { MatrixClient, Room, RoomMember } from 'matrix-js-sdk';
|
||||||
|
|
||||||
function generateRoomId() {
|
function generateRoomId() {
|
||||||
return '!' + Math.random().toString().slice(2, 10) + ':domain';
|
return '!' + Math.random().toString().slice(2, 10) + ':domain';
|
||||||
|
@ -48,6 +48,8 @@ describe('RoomList', () => {
|
||||||
sandbox = TestUtils.stubClient(sandbox);
|
sandbox = TestUtils.stubClient(sandbox);
|
||||||
client = MatrixClientPeg.get();
|
client = MatrixClientPeg.get();
|
||||||
client.credentials = {userId: myUserId};
|
client.credentials = {userId: myUserId};
|
||||||
|
//revert this to prototype method as the test-utils monkey-patches this to return a hardcoded value
|
||||||
|
client.getUserId = MatrixClient.prototype.getUserId;
|
||||||
|
|
||||||
clock = lolex.install();
|
clock = lolex.install();
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue