2018-01-26 00:16:03 +03:00
|
|
|
/*
|
|
|
|
Copyright 2018 New Vector 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 {Store} from 'flux/utils';
|
|
|
|
import dis from '../dispatcher';
|
|
|
|
import DMRoomMap from '../utils/DMRoomMap';
|
2018-02-06 18:15:47 +03:00
|
|
|
import Unread from '../Unread';
|
2018-10-12 23:35:54 +03:00
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
2018-01-26 00:16:03 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
const CATEGORY_RED = "red";
|
|
|
|
const CATEGORY_GREY = "grey";
|
|
|
|
const CATEGORY_BOLD = "bold";
|
|
|
|
const CATEGORY_IDLE = "idle";
|
|
|
|
|
|
|
|
const CATEGORY_ORDER = [CATEGORY_RED, CATEGORY_GREY, CATEGORY_BOLD, CATEGORY_IDLE];
|
|
|
|
const LIST_ORDERS = {
|
|
|
|
"m.favourite": "manual",
|
|
|
|
"im.vector.fake.invite": "recent",
|
|
|
|
"im.vector.fake.recent": "recent",
|
|
|
|
"im.vector.fake.direct": "recent",
|
|
|
|
"m.lowpriority": "recent",
|
|
|
|
"im.vector.fake.archived": "recent",
|
|
|
|
};
|
|
|
|
|
2018-01-26 00:16:03 +03:00
|
|
|
/**
|
|
|
|
* A class for storing application state for categorising rooms in
|
|
|
|
* the RoomList.
|
|
|
|
*/
|
|
|
|
class RoomListStore extends Store {
|
|
|
|
constructor() {
|
|
|
|
super(dis);
|
|
|
|
|
|
|
|
this._init();
|
2018-02-06 18:15:47 +03:00
|
|
|
this._getManualComparator = this._getManualComparator.bind(this);
|
|
|
|
this._recentsComparator = this._recentsComparator.bind(this);
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_init() {
|
|
|
|
// Initialise state
|
2019-02-13 23:00:35 +03:00
|
|
|
const defaultLists = {
|
|
|
|
"m.server_notice": [/* { room: js-sdk room, category: string } */],
|
|
|
|
"im.vector.fake.invite": [],
|
|
|
|
"m.favourite": [],
|
|
|
|
"im.vector.fake.recent": [],
|
|
|
|
"im.vector.fake.direct": [],
|
|
|
|
"m.lowpriority": [],
|
|
|
|
"im.vector.fake.archived": [],
|
|
|
|
};
|
2018-01-26 00:16:03 +03:00
|
|
|
this._state = {
|
2019-02-13 23:00:35 +03:00
|
|
|
// The rooms in these arrays are ordered according to either the
|
|
|
|
// 'recents' behaviour or 'manual' behaviour.
|
|
|
|
lists: defaultLists,
|
|
|
|
presentationLists: defaultLists, // like `lists`, but with arrays of rooms instead
|
2018-01-26 00:16:03 +03:00
|
|
|
ready: false,
|
2019-02-13 23:00:35 +03:00
|
|
|
stickyRoomId: null,
|
2018-01-26 00:16:03 +03:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
_setState(newState) {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (newState['lists']) {
|
|
|
|
const presentationLists = {};
|
|
|
|
for (const key of Object.keys(newState['lists'])) {
|
|
|
|
presentationLists[key] = newState['lists'][key].map((e) => e.room);
|
|
|
|
}
|
|
|
|
newState['presentationLists'] = presentationLists;
|
|
|
|
}
|
2018-01-26 00:16:03 +03:00
|
|
|
this._state = Object.assign(this._state, newState);
|
|
|
|
this.__emitChange();
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
__onDispatch = (payload) => {
|
|
|
|
const logicallyReady = this._matrixClient && this._state.ready;
|
2018-01-26 00:16:03 +03:00
|
|
|
switch (payload.action) {
|
|
|
|
// Initialise state after initial sync
|
|
|
|
case 'MatrixActions.sync': {
|
|
|
|
if (!(payload.prevState !== 'PREPARED' && payload.state === 'PREPARED')) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-02-05 20:29:22 +03:00
|
|
|
this._matrixClient = payload.matrixClient;
|
2019-02-13 23:00:35 +03:00
|
|
|
this._generateInitialRoomLists();
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
break;
|
2019-02-14 03:03:27 +03:00
|
|
|
case 'MatrixActions.Room.receipt': {
|
|
|
|
if (!logicallyReady) break;
|
|
|
|
|
|
|
|
// First see if the receipt event is for our own user
|
|
|
|
const myUserId = this._matrixClient.getUserId();
|
|
|
|
for (const eventId of Object.keys(payload.event.getContent())) {
|
|
|
|
const receiptUsers = Object.keys(payload.event.getContent()[eventId]['m.read'] || {});
|
|
|
|
if (receiptUsers.includes(myUserId)) {
|
|
|
|
this._roomUpdateTriggered(payload.room.roomId);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2018-01-26 00:16:03 +03:00
|
|
|
case 'MatrixActions.Room.tags': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady) break;
|
2019-02-14 00:22:00 +03:00
|
|
|
// TODO: Figure out which rooms changed in the tag and only change those.
|
|
|
|
// This is very blunt and wipes out the sticky room stuff
|
|
|
|
this._generateInitialRoomLists();
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-02-16 17:16:50 +03:00
|
|
|
case 'MatrixActions.Room.timeline': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady ||
|
2018-02-16 17:16:50 +03:00
|
|
|
!payload.isLiveEvent ||
|
2018-02-19 12:56:03 +03:00
|
|
|
!payload.isLiveUnfilteredRoomTimelineEvent ||
|
2018-02-16 17:16:50 +03:00
|
|
|
!this._eventTriggersRecentReorder(payload.event)
|
2019-02-13 23:00:35 +03:00
|
|
|
) {
|
|
|
|
break;
|
|
|
|
}
|
2018-11-02 01:28:13 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
this._roomUpdateTriggered(payload.event.getRoomId());
|
2018-02-16 17:16:50 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-02-23 13:32:33 +03:00
|
|
|
// When an event is decrypted, it could mean we need to reorder the room
|
|
|
|
// list because we now know the type of the event.
|
|
|
|
case 'MatrixActions.Event.decrypted': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady) break;
|
2018-02-23 17:54:00 +03:00
|
|
|
|
|
|
|
const roomId = payload.event.getRoomId();
|
|
|
|
|
|
|
|
// We may have decrypted an event without a roomId (e.g to_device)
|
|
|
|
if (!roomId) break;
|
|
|
|
|
|
|
|
const room = this._matrixClient.getRoom(roomId);
|
|
|
|
|
|
|
|
// We somehow decrypted an event for a room our client is unaware of
|
|
|
|
if (!room) break;
|
|
|
|
|
2018-02-23 13:32:33 +03:00
|
|
|
const liveTimeline = room.getLiveTimeline();
|
|
|
|
const eventTimeline = room.getTimelineForEvent(payload.event.getId());
|
|
|
|
|
2018-02-23 17:54:00 +03:00
|
|
|
// Either this event was not added to the live timeline (e.g. pagination)
|
|
|
|
// or it doesn't affect the ordering of the room list.
|
2019-02-13 23:00:35 +03:00
|
|
|
if (liveTimeline !== eventTimeline || !this._eventTriggersRecentReorder(payload.event)) {
|
|
|
|
break;
|
|
|
|
}
|
2018-11-02 01:28:13 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
this._roomUpdateTriggered(roomId);
|
2018-02-23 13:32:33 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-02-05 21:06:29 +03:00
|
|
|
case 'MatrixActions.accountData': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady) break;
|
2018-02-05 21:06:29 +03:00
|
|
|
if (payload.event_type !== 'm.direct') break;
|
2019-02-14 00:22:00 +03:00
|
|
|
// TODO: Figure out which rooms changed in the direct chat and only change those.
|
|
|
|
// This is very blunt and wipes out the sticky room stuff
|
|
|
|
this._generateInitialRoomLists();
|
2018-11-05 05:47:24 +03:00
|
|
|
}
|
|
|
|
break;
|
2019-02-13 23:00:35 +03:00
|
|
|
// TODO: Remove if not actually needed
|
|
|
|
// case 'MatrixActions.Room.accountData': {
|
|
|
|
// if (!logicallyReady) break;
|
|
|
|
// if (payload.event_type === 'm.fully_read') {
|
|
|
|
// console.log("!! Fully read: ", payload);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// break;
|
2018-09-17 20:14:52 +03:00
|
|
|
case 'MatrixActions.Room.myMembership': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady) break;
|
|
|
|
this._roomUpdateTriggered(payload.room.roomId);
|
2018-02-05 21:27:50 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-02-20 20:57:46 +03:00
|
|
|
// This could be a new room that we've been invited to, joined or created
|
|
|
|
// we won't get a RoomMember.membership for these cases if we're not already
|
|
|
|
// a member.
|
|
|
|
case 'MatrixActions.Room': {
|
2019-02-13 23:00:35 +03:00
|
|
|
if (!logicallyReady) break;
|
|
|
|
this._roomUpdateTriggered(payload.room.roomId);
|
2018-02-20 20:57:46 +03:00
|
|
|
}
|
|
|
|
break;
|
2019-02-14 00:08:19 +03:00
|
|
|
// TODO: Re-enable optimistic updates when we support dragging again
|
|
|
|
// case 'RoomListActions.tagRoom.pending': {
|
|
|
|
// if (!logicallyReady) break;
|
|
|
|
// // XXX: we only show one optimistic update at any one time.
|
|
|
|
// // Ideally we should be making a list of in-flight requests
|
|
|
|
// // that are backed by transaction IDs. Until the js-sdk
|
|
|
|
// // supports this, we're stuck with only being able to use
|
|
|
|
// // the most recent optimistic update.
|
|
|
|
// console.log("!! Optimistic tag: ", payload);
|
|
|
|
// }
|
|
|
|
// break;
|
|
|
|
// case 'RoomListActions.tagRoom.failure': {
|
|
|
|
// if (!logicallyReady) break;
|
|
|
|
// // Reset state according to js-sdk
|
|
|
|
// console.log("!! Optimistic tag failure: ", payload);
|
|
|
|
// }
|
2019-02-14 00:22:00 +03:00
|
|
|
// break;
|
2018-01-26 00:16:03 +03:00
|
|
|
case 'on_logged_out': {
|
|
|
|
// Reset state without pushing an update to the view, which generally assumes that
|
|
|
|
// the matrix client isn't `null` and so causing a re-render will cause NPEs.
|
|
|
|
this._init();
|
2018-02-06 18:15:47 +03:00
|
|
|
this._matrixClient = null;
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
break;
|
2019-02-13 23:00:35 +03:00
|
|
|
case 'view_room': {
|
|
|
|
if (!logicallyReady) break;
|
|
|
|
|
|
|
|
// Note: it is important that we set a new stickyRoomId before setting the old room
|
|
|
|
// to IDLE. If we don't, the wrong room gets counted as sticky.
|
|
|
|
const currentSticky = this._state.stickyRoomId;
|
|
|
|
this._setState({stickyRoomId: payload.room_id});
|
|
|
|
if (currentSticky) {
|
|
|
|
this._setRoomCategory(this._matrixClient.getRoom(currentSticky), CATEGORY_IDLE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
_roomUpdateTriggered(roomId) {
|
|
|
|
const room = this._matrixClient.getRoom(roomId);
|
|
|
|
if (!room) return;
|
|
|
|
|
|
|
|
if (this._state.stickyRoomId !== room.roomId) {
|
2019-02-14 03:03:27 +03:00
|
|
|
const category = this._calculateCategory(room);
|
|
|
|
this._setRoomCategory(room, category);
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-14 04:19:18 +03:00
|
|
|
_setRoomCategory(room, category) {
|
2019-02-13 23:00:35 +03:00
|
|
|
const listsClone = {};
|
|
|
|
const targetCatIndex = CATEGORY_ORDER.indexOf(category);
|
2019-02-14 04:19:18 +03:00
|
|
|
const targetTs = this._tsOfNewestEvent(room);
|
2019-02-13 23:00:35 +03:00
|
|
|
|
2019-02-14 00:08:19 +03:00
|
|
|
const myMembership = room.getMyMembership();
|
|
|
|
let doInsert = true;
|
2019-02-14 04:19:18 +03:00
|
|
|
let targetTags = [];
|
2019-02-14 00:08:19 +03:00
|
|
|
if (myMembership !== "join" && myMembership !== "invite") {
|
|
|
|
doInsert = false;
|
2019-02-14 00:22:00 +03:00
|
|
|
} else {
|
|
|
|
const dmRoomMap = DMRoomMap.shared();
|
|
|
|
if (dmRoomMap.getUserIdForRoomId(room.roomId)) {
|
2019-02-14 04:19:18 +03:00
|
|
|
targetTags.push('im.vector.fake.direct');
|
2019-02-14 00:22:00 +03:00
|
|
|
} else {
|
2019-02-14 04:19:18 +03:00
|
|
|
targetTags.push('im.vector.fake.recent');
|
2019-02-14 00:22:00 +03:00
|
|
|
}
|
2019-02-14 00:08:19 +03:00
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
// We need to update all instances of a room to ensure that they are correctly organized
|
|
|
|
// in the list. We do this by shallow-cloning the entire `lists` object using a single
|
|
|
|
// iterator. Within the loop, we also rebuild the list of rooms per tag (key) so that the
|
|
|
|
// updated room gets slotted into the right spot.
|
|
|
|
|
2019-02-14 00:08:19 +03:00
|
|
|
let inserted = false;
|
2019-02-13 23:00:35 +03:00
|
|
|
for (const key of Object.keys(this._state.lists)) {
|
|
|
|
listsClone[key] = [];
|
|
|
|
let pushedEntry = false;
|
|
|
|
const hasRoom = !!this._state.lists[key].find((e) => e.room.roomId === room.roomId);
|
2019-02-14 04:19:18 +03:00
|
|
|
let lastCategoryBoundary = 0;
|
|
|
|
let lastCategoryIndex = 0;
|
2019-02-13 23:00:35 +03:00
|
|
|
for (const entry of this._state.lists[key]) {
|
2019-02-14 04:19:18 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
// if the list is a recent list, and the room appears in this list, and we're not looking at a sticky
|
|
|
|
// room (sticky rooms have unreliable categories), try to slot the new room in
|
|
|
|
if (LIST_ORDERS[key] === 'recent' && hasRoom && entry.room.roomId !== this._state.stickyRoomId) {
|
2019-02-14 04:19:18 +03:00
|
|
|
const inTag = targetTags.length === 0 || targetTags.includes(key);
|
2019-02-14 00:22:00 +03:00
|
|
|
if (!pushedEntry && doInsert && inTag) {
|
2019-02-14 04:19:18 +03:00
|
|
|
const entryTs = this._tsOfNewestEvent(entry.room);
|
|
|
|
const entryCategory = CATEGORY_ORDER.indexOf(entry.category);
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
// If we've hit the top of a boundary (either because there's no rooms in the target or
|
|
|
|
// we've reached the grouping of rooms), insert our room ahead of the others in the category.
|
|
|
|
// This ensures that our room is on top (more recent) than the others.
|
2019-02-14 04:19:18 +03:00
|
|
|
const changedBoundary = entryCategory > targetCatIndex;
|
|
|
|
const currentCategory = entryCategory === targetCatIndex;
|
|
|
|
if (changedBoundary || (currentCategory && targetTs >= entryTs)) {
|
|
|
|
if (changedBoundary) {
|
|
|
|
// If we changed a boundary, then we've gone too far - go to the top of the last
|
|
|
|
// section instead.
|
|
|
|
listsClone[key].splice(lastCategoryBoundary, 0, {room, category});
|
|
|
|
} else {
|
|
|
|
// If we're ordering by timestamp, just insert normally
|
|
|
|
listsClone[key].push({room, category});
|
|
|
|
}
|
2019-02-13 23:00:35 +03:00
|
|
|
pushedEntry = true;
|
2019-02-14 00:08:19 +03:00
|
|
|
inserted = true;
|
2019-02-13 23:00:35 +03:00
|
|
|
}
|
2019-02-14 04:19:18 +03:00
|
|
|
|
|
|
|
if (entryCategory !== lastCategoryIndex) {
|
|
|
|
lastCategoryBoundary = listsClone[key].length - 1;
|
|
|
|
}
|
|
|
|
lastCategoryIndex = entryCategory;
|
2019-02-13 23:00:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// We insert our own record as needed, so don't let the old one through.
|
|
|
|
if (entry.room.roomId === room.roomId) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall through and clone the list.
|
|
|
|
listsClone[key].push(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-14 00:08:19 +03:00
|
|
|
if (!inserted) {
|
|
|
|
// There's a good chance that we just joined the room, so we need to organize it
|
|
|
|
// We also could have left it...
|
|
|
|
let tags = [];
|
|
|
|
if (doInsert) {
|
|
|
|
tags = Object.keys(room.tags);
|
2019-02-14 00:22:00 +03:00
|
|
|
if (tags.length === 0) {
|
|
|
|
tags = targetTags;
|
|
|
|
}
|
2019-02-14 00:08:19 +03:00
|
|
|
if (tags.length === 0) {
|
|
|
|
tags.push(myMembership === 'join' ? 'im.vector.fake.recent' : 'im.vector.fake.invite');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tags = ['im.vector.fake.archived'];
|
|
|
|
}
|
|
|
|
for (const tag of tags) {
|
|
|
|
for (let i = 0; i < listsClone[tag].length; i++) {
|
|
|
|
const catIdxAtPosition = CATEGORY_ORDER.indexOf(listsClone[tag][i].category);
|
|
|
|
if (catIdxAtPosition >= targetCatIndex) {
|
|
|
|
listsClone[tag].splice(i, 0, {room: room, category: category});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
this._setState({lists: listsClone});
|
|
|
|
}
|
|
|
|
|
|
|
|
_generateInitialRoomLists() {
|
2018-01-26 00:16:03 +03:00
|
|
|
const lists = {
|
2018-08-06 20:00:40 +03:00
|
|
|
"m.server_notice": [],
|
2018-01-26 00:16:03 +03:00
|
|
|
"im.vector.fake.invite": [],
|
|
|
|
"m.favourite": [],
|
|
|
|
"im.vector.fake.recent": [],
|
|
|
|
"im.vector.fake.direct": [],
|
|
|
|
"m.lowpriority": [],
|
|
|
|
"im.vector.fake.archived": [],
|
|
|
|
};
|
|
|
|
|
|
|
|
const dmRoomMap = DMRoomMap.shared();
|
2019-02-07 21:04:30 +03:00
|
|
|
const isCustomTagsEnabled = SettingsStore.isFeatureEnabled("feature_custom_tags");
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
this._matrixClient.getRooms().forEach((room) => {
|
2018-07-25 17:08:44 +03:00
|
|
|
const myUserId = this._matrixClient.getUserId();
|
2018-08-14 12:47:05 +03:00
|
|
|
const membership = room.getMyMembership();
|
2018-07-25 17:08:44 +03:00
|
|
|
const me = room.getMember(myUserId);
|
2018-01-26 00:16:03 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
if (membership === "invite") {
|
|
|
|
lists["im.vector.fake.invite"].push({room, category: CATEGORY_RED});
|
|
|
|
} else if (membership === "join" || membership === "ban" || (me && me.isKicked())) {
|
2018-01-26 00:16:03 +03:00
|
|
|
// Used to split rooms via tags
|
2018-02-06 18:15:47 +03:00
|
|
|
let tagNames = Object.keys(room.tags);
|
|
|
|
|
2019-02-05 20:35:05 +03:00
|
|
|
// ignore any m. tag names we don't know about
|
2018-08-06 18:58:54 +03:00
|
|
|
tagNames = tagNames.filter((t) => {
|
2019-02-07 21:04:30 +03:00
|
|
|
return (isCustomTagsEnabled && !t.startsWith('m.')) || lists[t] !== undefined;
|
2018-08-06 18:58:54 +03:00
|
|
|
});
|
|
|
|
|
2018-01-26 00:16:03 +03:00
|
|
|
if (tagNames.length) {
|
|
|
|
for (let i = 0; i < tagNames.length; i++) {
|
|
|
|
const tagName = tagNames[i];
|
|
|
|
lists[tagName] = lists[tagName] || [];
|
2019-02-13 23:00:35 +03:00
|
|
|
|
|
|
|
// We categorize all the tagged rooms the same because we don't actually
|
|
|
|
// care about the order (it's defined elsewhere)
|
|
|
|
lists[tagName].push({room, category: CATEGORY_RED});
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
} else if (dmRoomMap.getUserIdForRoomId(room.roomId)) {
|
|
|
|
// "Direct Message" rooms (that we're still in and that aren't otherwise tagged)
|
2019-02-13 23:00:35 +03:00
|
|
|
lists["im.vector.fake.direct"].push({room, category: this._calculateCategory(room)});
|
2018-01-26 00:16:03 +03:00
|
|
|
} else {
|
2019-02-13 23:00:35 +03:00
|
|
|
lists["im.vector.fake.recent"].push({room, category: this._calculateCategory(room)});
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
2018-07-25 15:14:36 +03:00
|
|
|
} else if (membership === "leave") {
|
2019-02-13 23:00:35 +03:00
|
|
|
lists["im.vector.fake.archived"].push({room, category: this._calculateCategory(room)});
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-02-14 14:23:29 +03:00
|
|
|
Object.keys(lists).forEach((listKey) => {
|
|
|
|
let comparator;
|
2019-02-13 23:00:35 +03:00
|
|
|
switch (LIST_ORDERS[listKey]) {
|
2018-02-14 14:23:29 +03:00
|
|
|
case "recent":
|
2019-02-13 23:00:35 +03:00
|
|
|
comparator = this._recentsComparator;
|
2018-02-14 14:23:29 +03:00
|
|
|
break;
|
|
|
|
case "manual":
|
|
|
|
default:
|
2019-02-13 23:00:35 +03:00
|
|
|
comparator = this._getManualComparator(listKey);
|
2018-02-14 14:23:29 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
lists[listKey].sort(comparator);
|
2018-02-06 18:15:47 +03:00
|
|
|
});
|
|
|
|
|
2018-01-26 00:16:03 +03:00
|
|
|
this._setState({
|
|
|
|
lists,
|
2019-02-13 23:00:35 +03:00
|
|
|
ready: true, // Ready to receive updates to ordering
|
2018-01-26 00:16:03 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-02-16 17:16:50 +03:00
|
|
|
_eventTriggersRecentReorder(ev) {
|
|
|
|
return ev.getTs() && (
|
|
|
|
Unread.eventTriggersUnreadCount(ev) ||
|
|
|
|
ev.getSender() === this._matrixClient.credentials.userId
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-02-06 18:15:47 +03:00
|
|
|
_tsOfNewestEvent(room) {
|
|
|
|
for (let i = room.timeline.length - 1; i >= 0; --i) {
|
|
|
|
const ev = room.timeline[i];
|
2018-02-16 17:16:50 +03:00
|
|
|
if (this._eventTriggersRecentReorder(ev)) {
|
2018-02-06 18:15:47 +03:00
|
|
|
return ev.getTs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we might only have events that don't trigger the unread indicator,
|
|
|
|
// in which case use the oldest event even if normally it wouldn't count.
|
|
|
|
// This is better than just assuming the last event was forever ago.
|
|
|
|
if (room.timeline.length && room.timeline[0].getTs()) {
|
|
|
|
return room.timeline[0].getTs();
|
|
|
|
} else {
|
|
|
|
return Number.MAX_SAFE_INTEGER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
_calculateCategory(room) {
|
|
|
|
const mentions = room.getUnreadNotificationCount("highlight") > 0;
|
|
|
|
if (mentions) return CATEGORY_RED;
|
|
|
|
|
|
|
|
let unread = room.getUnreadNotificationCount() > 0;
|
|
|
|
if (unread) return CATEGORY_GREY;
|
|
|
|
|
|
|
|
unread = Unread.doesRoomHaveUnreadMessages(room);
|
|
|
|
if (unread) return CATEGORY_BOLD;
|
2018-10-12 23:35:54 +03:00
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
return CATEGORY_IDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_recentsComparator(entryA, entryB) {
|
|
|
|
const roomA = entryA.room;
|
|
|
|
const roomB = entryB.room;
|
|
|
|
const categoryA = entryA.category;
|
|
|
|
const categoryB = entryB.category;
|
|
|
|
|
|
|
|
if (categoryA !== categoryB) {
|
|
|
|
const idxA = CATEGORY_ORDER.indexOf(categoryA);
|
|
|
|
const idxB = CATEGORY_ORDER.indexOf(categoryB);
|
|
|
|
if (idxA > idxB) return 1;
|
|
|
|
if (idxA < idxB) return -1;
|
|
|
|
return 0;
|
2018-10-12 23:35:54 +03:00
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
const timestampA = this._tsOfNewestEvent(roomA);
|
|
|
|
const timestampB = this._tsOfNewestEvent(roomB);
|
|
|
|
return timestampB - timestampA;
|
2018-02-06 18:15:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
_lexicographicalComparator(roomA, roomB) {
|
|
|
|
return roomA.name > roomB.name ? 1 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
_getManualComparator(tagName, optimisticRequest) {
|
2019-02-13 23:00:35 +03:00
|
|
|
return (entryA, entryB) => {
|
|
|
|
const roomA = entryA.room;
|
|
|
|
const roomB = entryB.room;
|
|
|
|
|
2018-02-06 18:15:47 +03:00
|
|
|
let metaA = roomA.tags[tagName];
|
|
|
|
let metaB = roomB.tags[tagName];
|
|
|
|
|
|
|
|
if (optimisticRequest && roomA === optimisticRequest.room) metaA = optimisticRequest.metaData;
|
|
|
|
if (optimisticRequest && roomB === optimisticRequest.room) metaB = optimisticRequest.metaData;
|
|
|
|
|
|
|
|
// Make sure the room tag has an order element, if not set it to be the bottom
|
2019-02-13 23:00:35 +03:00
|
|
|
const a = metaA ? Number(metaA.order) : undefined;
|
|
|
|
const b = metaB ? Number(metaB.order) : undefined;
|
2018-02-06 18:15:47 +03:00
|
|
|
|
|
|
|
// Order undefined room tag orders to the bottom
|
|
|
|
if (a === undefined && b !== undefined) {
|
|
|
|
return 1;
|
|
|
|
} else if (a !== undefined && b === undefined) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-02-13 23:00:35 +03:00
|
|
|
return a === b ? this._lexicographicalComparator(roomA, roomB) : ( a > b ? 1 : -1);
|
2018-02-06 18:15:47 +03:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-01-26 00:16:03 +03:00
|
|
|
getRoomLists() {
|
2019-02-13 23:00:35 +03:00
|
|
|
return this._state.presentationLists;
|
2018-01-26 00:16:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (global.singletonRoomListStore === undefined) {
|
|
|
|
global.singletonRoomListStore = new RoomListStore();
|
|
|
|
}
|
|
|
|
export default global.singletonRoomListStore;
|