2021-09-11 22:57:35 +09:00
|
|
|
import EventEmitter from 'events';
|
|
|
|
import cons from './cons';
|
|
|
|
|
2021-12-08 00:34:07 +09:00
|
|
|
function isNotifEvent(mEvent) {
|
|
|
|
const eType = mEvent.getType();
|
|
|
|
if (!cons.supportEventTypes.includes(eType)) return false;
|
|
|
|
if (eType === 'm.room.member') return false;
|
|
|
|
|
|
|
|
if (mEvent.isRedacted()) return false;
|
|
|
|
if (mEvent.getRelation()?.rel_type === 'm.replace') return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-11 22:57:35 +09:00
|
|
|
class Notifications extends EventEmitter {
|
|
|
|
constructor(roomList) {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.matrixClient = roomList.matrixClient;
|
|
|
|
this.roomList = roomList;
|
|
|
|
|
|
|
|
this.roomIdToNoti = new Map();
|
|
|
|
|
|
|
|
this._initNoti();
|
|
|
|
this._listenEvents();
|
|
|
|
|
|
|
|
// TODO:
|
|
|
|
window.notifications = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
_initNoti() {
|
|
|
|
const addNoti = (roomId) => {
|
|
|
|
const room = this.matrixClient.getRoom(roomId);
|
|
|
|
if (this.doesRoomHaveUnread(room) === false) return;
|
|
|
|
const total = room.getUnreadNotificationCount('total');
|
|
|
|
const highlight = room.getUnreadNotificationCount('highlight');
|
|
|
|
const noti = this.getNoti(room.roomId);
|
|
|
|
this._setNoti(room.roomId, total - noti.total, highlight - noti.highlight);
|
|
|
|
};
|
|
|
|
[...this.roomList.rooms].forEach(addNoti);
|
|
|
|
[...this.roomList.directs].forEach(addNoti);
|
|
|
|
}
|
|
|
|
|
|
|
|
doesRoomHaveUnread(room) {
|
|
|
|
const userId = this.matrixClient.getUserId();
|
|
|
|
const readUpToId = room.getEventReadUpTo(userId);
|
2021-12-03 22:02:10 +09:00
|
|
|
const liveEvents = room.getLiveTimeline().getEvents();
|
2021-09-11 22:57:35 +09:00
|
|
|
|
2021-12-08 00:34:07 +09:00
|
|
|
if (liveEvents[liveEvents.length - 1]?.getSender() === userId) {
|
2021-09-11 22:57:35 +09:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-12-03 22:02:10 +09:00
|
|
|
for (let i = liveEvents.length - 1; i >= 0; i -= 1) {
|
|
|
|
const event = liveEvents[i];
|
2021-09-11 22:57:35 +09:00
|
|
|
if (event.getId() === readUpToId) return false;
|
2021-12-08 00:34:07 +09:00
|
|
|
if (isNotifEvent(event)) return true;
|
2021-09-11 22:57:35 +09:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
getNoti(roomId) {
|
|
|
|
return this.roomIdToNoti.get(roomId) || { total: 0, highlight: 0, from: null };
|
|
|
|
}
|
|
|
|
|
2021-09-13 00:14:13 +09:00
|
|
|
getTotalNoti(roomId) {
|
|
|
|
const { total } = this.getNoti(roomId);
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
|
|
|
getHighlightNoti(roomId) {
|
|
|
|
const { highlight } = this.getNoti(roomId);
|
|
|
|
return highlight;
|
|
|
|
}
|
|
|
|
|
|
|
|
getFromNoti(roomId) {
|
|
|
|
const { from } = this.getNoti(roomId);
|
|
|
|
return from;
|
|
|
|
}
|
|
|
|
|
2021-09-11 22:57:35 +09:00
|
|
|
hasNoti(roomId) {
|
|
|
|
return this.roomIdToNoti.has(roomId);
|
|
|
|
}
|
|
|
|
|
2021-10-28 18:38:26 +09:00
|
|
|
_getAllParentIds(roomId) {
|
|
|
|
let allParentIds = this.roomList.roomIdToParents.get(roomId);
|
|
|
|
if (allParentIds === undefined) return new Set();
|
|
|
|
const parentIds = [...allParentIds];
|
|
|
|
|
|
|
|
parentIds.forEach((pId) => {
|
|
|
|
allParentIds = new Set(
|
|
|
|
[...allParentIds, ...this._getAllParentIds(pId)],
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
return allParentIds;
|
|
|
|
}
|
|
|
|
|
2021-09-11 22:57:35 +09:00
|
|
|
_setNoti(roomId, total, highlight, childId) {
|
2021-09-13 00:14:13 +09:00
|
|
|
const prevTotal = this.roomIdToNoti.get(roomId)?.total ?? null;
|
2021-09-11 22:57:35 +09:00
|
|
|
const noti = this.getNoti(roomId);
|
|
|
|
|
2021-10-28 18:38:26 +09:00
|
|
|
if (!childId || this._remainingParentIds?.has(roomId)) {
|
|
|
|
noti.total += total;
|
|
|
|
noti.highlight += highlight;
|
|
|
|
}
|
2021-09-11 22:57:35 +09:00
|
|
|
if (childId) {
|
|
|
|
if (noti.from === null) noti.from = new Set();
|
|
|
|
noti.from.add(childId);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.roomIdToNoti.set(roomId, noti);
|
2021-09-13 00:14:13 +09:00
|
|
|
this.emit(cons.events.notifications.NOTI_CHANGED, roomId, noti.total, prevTotal);
|
2021-09-11 22:57:35 +09:00
|
|
|
|
2021-10-28 18:38:26 +09:00
|
|
|
if (!childId) this._remainingParentIds = this._getAllParentIds(roomId);
|
|
|
|
else this._remainingParentIds.delete(roomId);
|
|
|
|
|
2021-09-11 22:57:35 +09:00
|
|
|
const parentIds = this.roomList.roomIdToParents.get(roomId);
|
2021-10-28 18:38:26 +09:00
|
|
|
if (typeof parentIds === 'undefined') {
|
|
|
|
if (!childId) this._remainingParentIds = undefined;
|
|
|
|
return;
|
|
|
|
}
|
2021-09-11 22:57:35 +09:00
|
|
|
[...parentIds].forEach((parentId) => this._setNoti(parentId, total, highlight, roomId));
|
2021-10-28 18:38:26 +09:00
|
|
|
if (!childId) this._remainingParentIds = undefined;
|
2021-09-11 22:57:35 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
_deleteNoti(roomId, total, highlight, childId) {
|
|
|
|
if (this.roomIdToNoti.has(roomId) === false) return;
|
|
|
|
|
|
|
|
const noti = this.getNoti(roomId);
|
2021-09-13 00:14:13 +09:00
|
|
|
const prevTotal = noti.total;
|
2021-09-11 22:57:35 +09:00
|
|
|
noti.total -= total;
|
|
|
|
noti.highlight -= highlight;
|
2021-10-27 19:33:41 +09:00
|
|
|
if (noti.total < 0) {
|
|
|
|
noti.total = 0;
|
|
|
|
noti.highlight = 0;
|
|
|
|
}
|
2021-09-11 22:57:35 +09:00
|
|
|
if (childId && noti.from !== null) {
|
2021-10-28 18:38:26 +09:00
|
|
|
if (!this.hasNoti(childId)) noti.from.delete(childId);
|
2021-09-11 22:57:35 +09:00
|
|
|
}
|
|
|
|
if (noti.from === null || noti.from.size === 0) {
|
|
|
|
this.roomIdToNoti.delete(roomId);
|
2021-09-13 00:14:13 +09:00
|
|
|
this.emit(cons.events.notifications.FULL_READ, roomId);
|
|
|
|
this.emit(cons.events.notifications.NOTI_CHANGED, roomId, null, prevTotal);
|
2021-09-11 22:57:35 +09:00
|
|
|
} else {
|
|
|
|
this.roomIdToNoti.set(roomId, noti);
|
2021-09-13 00:14:13 +09:00
|
|
|
this.emit(cons.events.notifications.NOTI_CHANGED, roomId, noti.total, prevTotal);
|
2021-09-11 22:57:35 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
const parentIds = this.roomList.roomIdToParents.get(roomId);
|
|
|
|
if (typeof parentIds === 'undefined') return;
|
|
|
|
[...parentIds].forEach((parentId) => this._deleteNoti(parentId, total, highlight, roomId));
|
|
|
|
}
|
|
|
|
|
|
|
|
_listenEvents() {
|
|
|
|
this.matrixClient.on('Room.timeline', (mEvent, room) => {
|
2021-12-08 00:34:07 +09:00
|
|
|
if (!isNotifEvent(mEvent)) return;
|
2021-12-03 22:02:10 +09:00
|
|
|
const liveEvents = room.getLiveTimeline().getEvents();
|
2021-09-11 22:57:35 +09:00
|
|
|
|
2021-12-03 22:02:10 +09:00
|
|
|
const lastTimelineEvent = liveEvents[liveEvents.length - 1];
|
2021-09-11 22:57:35 +09:00
|
|
|
if (lastTimelineEvent.getId() !== mEvent.getId()) return;
|
|
|
|
if (mEvent.getSender() === this.matrixClient.getUserId()) return;
|
|
|
|
|
|
|
|
const total = room.getUnreadNotificationCount('total');
|
|
|
|
const highlight = room.getUnreadNotificationCount('highlight');
|
|
|
|
|
|
|
|
const noti = this.getNoti(room.roomId);
|
|
|
|
this._setNoti(room.roomId, total - noti.total, highlight - noti.highlight);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.matrixClient.on('Room.receipt', (mEvent, room) => {
|
|
|
|
if (mEvent.getType() === 'm.receipt') {
|
|
|
|
const content = mEvent.getContent();
|
|
|
|
const readedEventId = Object.keys(content)[0];
|
|
|
|
const readerUserId = Object.keys(content[readedEventId]['m.read'])[0];
|
|
|
|
if (readerUserId !== this.matrixClient.getUserId()) return;
|
|
|
|
|
|
|
|
if (this.hasNoti(room.roomId)) {
|
|
|
|
const noti = this.getNoti(room.roomId);
|
|
|
|
this._deleteNoti(room.roomId, noti.total, noti.highlight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.matrixClient.on('Room.myMembership', (room, membership) => {
|
|
|
|
if (membership === 'leave' && this.hasNoti(room.roomId)) {
|
|
|
|
const noti = this.getNoti(room.roomId);
|
|
|
|
this._deleteNoti(room.roomId, noti.total, noti.highlight);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Notifications;
|