Merge pull request #62 from vector-im/bwindels/heroes
Implement heroes logic
This commit is contained in:
commit
16f13a3b4f
7 changed files with 173 additions and 28 deletions
|
@ -84,7 +84,7 @@ export class RoomViewModel extends ViewModel {
|
|||
}
|
||||
|
||||
get name() {
|
||||
return this._room.name;
|
||||
return this._room.name || this.i18n`Empty Room`;
|
||||
}
|
||||
|
||||
get timelineViewModel() {
|
||||
|
@ -102,7 +102,7 @@ export class RoomViewModel extends ViewModel {
|
|||
}
|
||||
|
||||
get avatarLetter() {
|
||||
return avatarInitials(this._room.name);
|
||||
return avatarInitials(this.name);
|
||||
}
|
||||
|
||||
get avatarColorNumber() {
|
||||
|
|
|
@ -70,7 +70,7 @@ export class RoomTileViewModel extends ViewModel {
|
|||
const timeDiff = theirTimestamp - myTimestamp;
|
||||
if (timeDiff === 0) {
|
||||
// sort alphabetically
|
||||
const nameCmp = this._room.name.localeCompare(other._room.name);
|
||||
const nameCmp = this.name.localeCompare(other.name);
|
||||
if (nameCmp === 0) {
|
||||
return this._room.id.localeCompare(other._room.id);
|
||||
}
|
||||
|
@ -88,12 +88,12 @@ export class RoomTileViewModel extends ViewModel {
|
|||
}
|
||||
|
||||
get name() {
|
||||
return this._room.name;
|
||||
return this._room.name || this.i18n`Empty Room`;
|
||||
}
|
||||
|
||||
// Avatar view model contract
|
||||
get avatarLetter() {
|
||||
return avatarInitials(this._room.name);
|
||||
return avatarInitials(this.name);
|
||||
}
|
||||
|
||||
get avatarColorNumber() {
|
||||
|
|
|
@ -36,7 +36,7 @@ export class Session {
|
|||
const txn = await this._storage.readTxn([
|
||||
this._storage.storeNames.session,
|
||||
this._storage.storeNames.roomSummary,
|
||||
this._storage.storeNames.roomState,
|
||||
this._storage.storeNames.roomMembers,
|
||||
this._storage.storeNames.timelineEvents,
|
||||
this._storage.storeNames.timelineFragments,
|
||||
this._storage.storeNames.pendingEvents,
|
||||
|
|
|
@ -24,6 +24,7 @@ import {SendQueue} from "./sending/SendQueue.js";
|
|||
import {WrappedError} from "../error.js"
|
||||
import {fetchOrLoadMembers} from "./members/load.js";
|
||||
import {MemberList} from "./members/MemberList.js";
|
||||
import {Heroes} from "./members/Heroes.js";
|
||||
|
||||
export class Room extends EventEmitter {
|
||||
constructor({roomId, storage, hsApi, emitCollectionChange, sendScheduler, pendingEvents, user}) {
|
||||
|
@ -50,15 +51,32 @@ export class Room extends EventEmitter {
|
|||
isInitialSync, isTimelineOpen,
|
||||
txn);
|
||||
const {entries, newLiveKey, changedMembers} = await this._syncWriter.writeSync(roomResponse, txn);
|
||||
// room name disappeared, open heroes
|
||||
if (!summaryChanges.name && summaryChanges.heroes && !this._heroes) {
|
||||
this._heroes = new Heroes(this._roomId);
|
||||
}
|
||||
// fetch new members while we have txn open,
|
||||
// but don't make any in-memory changes yet
|
||||
let heroChanges;
|
||||
if (summaryChanges.heroes && this._heroes) {
|
||||
heroChanges = await this._heroes.calculateChanges(summaryChanges.heroes, changedMembers, txn);
|
||||
}
|
||||
let removedPendingEvents;
|
||||
if (roomResponse.timeline && roomResponse.timeline.events) {
|
||||
removedPendingEvents = this._sendQueue.removeRemoteEchos(roomResponse.timeline.events, txn);
|
||||
}
|
||||
return {summaryChanges, newTimelineEntries: entries, newLiveKey, removedPendingEvents, changedMembers};
|
||||
return {
|
||||
summaryChanges,
|
||||
newTimelineEntries: entries,
|
||||
newLiveKey,
|
||||
removedPendingEvents,
|
||||
changedMembers,
|
||||
heroChanges
|
||||
};
|
||||
}
|
||||
|
||||
/** @package */
|
||||
afterSync({summaryChanges, newTimelineEntries, newLiveKey, removedPendingEvents, changedMembers}) {
|
||||
afterSync({summaryChanges, newTimelineEntries, newLiveKey, removedPendingEvents, changedMembers, heroChanges}) {
|
||||
this._syncWriter.afterSync(newLiveKey);
|
||||
if (changedMembers.length) {
|
||||
if (this._changedMembersDuringSync) {
|
||||
|
@ -70,8 +88,22 @@ export class Room extends EventEmitter {
|
|||
this._memberList.afterSync(changedMembers);
|
||||
}
|
||||
}
|
||||
let emitChange = false;
|
||||
if (summaryChanges) {
|
||||
this._summary.applyChanges(summaryChanges);
|
||||
if (this._summary.name && this._heroes) {
|
||||
this._heroes = null;
|
||||
}
|
||||
emitChange = true;
|
||||
}
|
||||
if (this._heroes && heroChanges) {
|
||||
const oldName = this.name;
|
||||
this._heroes.applyChanges(heroChanges, this._summary);
|
||||
if (oldName !== this.name) {
|
||||
emitChange = true;
|
||||
}
|
||||
}
|
||||
if (emitChange) {
|
||||
this.emit("change");
|
||||
this._emitCollectionChange(this);
|
||||
}
|
||||
|
@ -89,9 +121,15 @@ export class Room extends EventEmitter {
|
|||
}
|
||||
|
||||
/** @package */
|
||||
load(summary, txn) {
|
||||
async load(summary, txn) {
|
||||
try {
|
||||
this._summary.load(summary);
|
||||
// need to load members for name?
|
||||
if (!this._summary.name && this._summary.heroes) {
|
||||
this._heroes = new Heroes(this._roomId);
|
||||
const changes = await this._heroes.calculateChanges(this._summary.heroes, [], txn);
|
||||
this._heroes.applyChanges(changes, this._summary);
|
||||
}
|
||||
return this._syncWriter.load(txn);
|
||||
} catch (err) {
|
||||
throw new WrappedError(`Could not load room ${this._roomId}`, err);
|
||||
|
@ -177,6 +215,9 @@ export class Room extends EventEmitter {
|
|||
|
||||
/** @public */
|
||||
get name() {
|
||||
if (this._heroes) {
|
||||
return this._heroes.roomName;
|
||||
}
|
||||
return this._summary.name;
|
||||
}
|
||||
|
||||
|
@ -186,7 +227,12 @@ export class Room extends EventEmitter {
|
|||
}
|
||||
|
||||
get avatarUrl() {
|
||||
if (this._summary.avatarUrl) {
|
||||
return this._summary.avatarUrl;
|
||||
} else if (this._heroes) {
|
||||
return this._heroes.roomAvatarUrl;
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
get lastMessageTimestamp() {
|
||||
|
|
|
@ -73,7 +73,6 @@ function processStateEvent(data, event) {
|
|||
const content = event.content;
|
||||
data = data.cloneIfNeeded();
|
||||
data.canonicalAlias = content.alias;
|
||||
data.altAliases = content.alt_aliases;
|
||||
}
|
||||
return data;
|
||||
}
|
||||
|
@ -97,10 +96,10 @@ function processTimelineEvent(data, event, isInitialSync, isTimelineOpen, ownUse
|
|||
|
||||
function updateSummary(data, summary) {
|
||||
const heroes = summary["m.heroes"];
|
||||
const inviteCount = summary["m.joined_member_count"];
|
||||
const joinCount = summary["m.invited_member_count"];
|
||||
const joinCount = summary["m.joined_member_count"];
|
||||
const inviteCount = summary["m.invited_member_count"];
|
||||
|
||||
if (heroes) {
|
||||
if (heroes && Array.isArray(heroes)) {
|
||||
data = data.cloneIfNeeded();
|
||||
data.heroes = heroes;
|
||||
}
|
||||
|
@ -129,7 +128,6 @@ class SummaryData {
|
|||
this.joinCount = copy ? copy.joinCount : 0;
|
||||
this.heroes = copy ? copy.heroes : null;
|
||||
this.canonicalAlias = copy ? copy.canonicalAlias : null;
|
||||
this.altAliases = copy ? copy.altAliases : null;
|
||||
this.hasFetchedMembers = copy ? copy.hasFetchedMembers : false;
|
||||
this.lastPaginationToken = copy ? copy.lastPaginationToken : null;
|
||||
this.avatarUrl = copy ? copy.avatarUrl : null;
|
||||
|
@ -165,13 +163,11 @@ export class RoomSummary {
|
|||
if (this._data.canonicalAlias) {
|
||||
return this._data.canonicalAlias;
|
||||
}
|
||||
if (Array.isArray(this._data.altAliases) && this._data.altAliases.length !== 0) {
|
||||
return this._data.altAliases[0];
|
||||
return null;
|
||||
}
|
||||
if (Array.isArray(this._data.heroes) && this._data.heroes.length !== 0) {
|
||||
return this._data.heroes.join(", ");
|
||||
}
|
||||
return this._data.roomId;
|
||||
|
||||
get heroes() {
|
||||
return this._data.heroes;
|
||||
}
|
||||
|
||||
get isUnread() {
|
||||
|
@ -240,12 +236,6 @@ export class RoomSummary {
|
|||
isInitialSync, isTimelineOpen,
|
||||
this._ownUserId);
|
||||
if (data !== this._data) {
|
||||
// need to think here how we want to persist
|
||||
// things like unread status (as read marker, or unread count)?
|
||||
// we could very well load additional things in the load method
|
||||
// ... the trade-off is between constantly writing the summary
|
||||
// on every sync, or doing a bit of extra reading on load
|
||||
// and have in-memory only variables for visualization
|
||||
txn.roomSummary.set(data.serialize());
|
||||
return data;
|
||||
}
|
||||
|
|
101
src/matrix/room/members/Heroes.js
Normal file
101
src/matrix/room/members/Heroes.js
Normal file
|
@ -0,0 +1,101 @@
|
|||
/*
|
||||
Copyright 2020 The Matrix.org Foundation C.I.C.
|
||||
|
||||
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 {RoomMember} from "./RoomMember.js";
|
||||
|
||||
function calculateRoomName(sortedMembers, summary) {
|
||||
const countWithoutMe = summary.joinCount + summary.inviteCount - 1;
|
||||
if (sortedMembers.length >= countWithoutMe) {
|
||||
if (sortedMembers.length > 1) {
|
||||
const lastMember = sortedMembers[sortedMembers.length - 1];
|
||||
const firstMembers = sortedMembers.slice(0, sortedMembers.length - 1);
|
||||
return firstMembers.map(m => m.displayName).join(", ") + " and " + lastMember.displayName;
|
||||
} else {
|
||||
return sortedMembers[0].displayName;
|
||||
}
|
||||
} else if (sortedMembers.length < countWithoutMe) {
|
||||
return sortedMembers.map(m => m.displayName).join(", ") + ` and ${countWithoutMe} others`;
|
||||
} else {
|
||||
// Empty Room
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export class Heroes {
|
||||
constructor(roomId) {
|
||||
this._roomId = roomId;
|
||||
this._members = new Map();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string[]} newHeroes array of user ids
|
||||
* @param {RoomMember[]} changedMembers array of changed members in this sync
|
||||
* @param {Transaction} txn
|
||||
* @return {Promise}
|
||||
*/
|
||||
async calculateChanges(newHeroes, changedMembers, txn) {
|
||||
const updatedHeroMembers = new Map();
|
||||
const removedUserIds = [];
|
||||
// remove non-present members
|
||||
for (const existingUserId of this._members.keys()) {
|
||||
if (newHeroes.indexOf(existingUserId) === -1) {
|
||||
removedUserIds.push(existingUserId);
|
||||
}
|
||||
}
|
||||
// update heroes with synced member changes
|
||||
for (const member of changedMembers) {
|
||||
if (this._members.has(member.userId) || newHeroes.indexOf(member.userId) !== -1) {
|
||||
updatedHeroMembers.set(member.userId, member);
|
||||
}
|
||||
}
|
||||
// load member for new heroes from storage
|
||||
for (const userId of newHeroes) {
|
||||
if (!this._members.has(userId) && !updatedHeroMembers.has(userId)) {
|
||||
const memberData = await txn.roomMembers.get(this._roomId, userId);
|
||||
if (memberData) {
|
||||
const member = new RoomMember(memberData);
|
||||
updatedHeroMembers.set(member.userId, member);
|
||||
}
|
||||
}
|
||||
}
|
||||
return {updatedHeroMembers: updatedHeroMembers.values(), removedUserIds};
|
||||
}
|
||||
|
||||
applyChanges({updatedHeroMembers, removedUserIds}, summary) {
|
||||
for (const userId of removedUserIds) {
|
||||
this._members.delete(userId);
|
||||
}
|
||||
for (const member of updatedHeroMembers) {
|
||||
this._members.set(member.userId, member);
|
||||
}
|
||||
const sortedMembers = Array.from(this._members.values()).sort((a, b) => a.displayName.localeCompare(b.displayName));
|
||||
this._roomName = calculateRoomName(sortedMembers, summary);
|
||||
}
|
||||
|
||||
get roomName() {
|
||||
return this._roomName;
|
||||
}
|
||||
|
||||
get roomAvatarUrl() {
|
||||
if (this._members.size === 1) {
|
||||
for (const member of this._members.values()) {
|
||||
console.log("roomAvatarUrl", member, member.avatarUrl);
|
||||
return member.avatarUrl;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
|
@ -54,6 +54,14 @@ export class RoomMember {
|
|||
});
|
||||
}
|
||||
|
||||
get displayName() {
|
||||
return this._data.displayName;
|
||||
}
|
||||
|
||||
get avatarUrl() {
|
||||
return this._data.avatarUrl;
|
||||
}
|
||||
|
||||
get roomId() {
|
||||
return this._data.roomId;
|
||||
}
|
||||
|
|
Reference in a new issue