Telegram Web K with changes to work inside I2P
https://web.telegram.i2p/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
491 lines
14 KiB
491 lines
14 KiB
/* |
|
* https://github.com/morethanwords/tweb |
|
* Copyright (C) 2019-2021 Eduard Kuzmenko |
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE |
|
*/ |
|
|
|
import type { DialogFilter, Update } from "../../layer"; |
|
import { Modify } from "../../types"; |
|
import type { Dialog } from '../appManagers/appMessagesManager'; |
|
import forEachReverse from "../../helpers/array/forEachReverse"; |
|
import copy from "../../helpers/object/copy"; |
|
import safeReplaceObject from "../../helpers/object/safeReplaceObject"; |
|
import getPeerId from "../appManagers/utils/peers/getPeerId"; |
|
import { AppManager } from "../appManagers/manager"; |
|
import assumeType from "../../helpers/assumeType"; |
|
|
|
export type MyDialogFilter = Modify<DialogFilter.dialogFilter, { |
|
/* pinned_peers: PeerId[], |
|
include_peers: PeerId[], |
|
exclude_peers: PeerId[], */ |
|
pinnedPeerIds: PeerId[], |
|
includePeerIds: PeerId[], |
|
excludePeerIds: PeerId[] |
|
}>; |
|
|
|
const convertment = [ |
|
['pinned_peers', 'pinnedPeerIds'], |
|
['exclude_peers', 'excludePeerIds'], |
|
['include_peers', 'includePeerIds'] |
|
] as ['pinned_peers' | 'exclude_peers' | 'include_peers', 'pinnedPeerIds' | 'excludePeerIds' | 'includePeerIds'][]; |
|
|
|
const START_ORDER_INDEX = 2; |
|
|
|
// const LOCAL_FILTER: MyDialogFilter = { |
|
// _: 'dialogFilter', |
|
// id: 0, |
|
// title: '', |
|
// exclude_peers: [], |
|
// include_peers: [], |
|
// pinned_peers: [], |
|
// excludePeerIds: [], |
|
// includePeerIds: [], |
|
// pinnedPeerIds: [], |
|
// pFlags: {} |
|
// }; |
|
|
|
export default class FiltersStorage extends AppManager { |
|
private filters: {[filterId: string]: MyDialogFilter}; |
|
private orderIndex: number; |
|
private reloadedPeerIds: Set<PeerId>; |
|
|
|
protected after() { |
|
this.clear(true); |
|
this.filters = {}; |
|
|
|
this.apiUpdatesManager.addMultipleEventsListeners({ |
|
updateDialogFilter: this.onUpdateDialogFilter, |
|
|
|
updateDialogFilters: (update) => { |
|
//console.warn('updateDialogFilters', update); |
|
|
|
const oldFilters = copy(this.filters); |
|
|
|
this.getDialogFilters(true).then((filters) => { |
|
for(const _filterId in oldFilters) { |
|
const filterId = +_filterId; |
|
if(!filters.find((filter) => filter.id === filterId)) { // * deleted |
|
this.onUpdateDialogFilter({_: 'updateDialogFilter', id: filterId}); |
|
} |
|
} |
|
|
|
this.onUpdateDialogFilterOrder({_: 'updateDialogFilterOrder', order: filters.map((filter) => filter.id)}); |
|
}); |
|
}, |
|
|
|
updateDialogFilterOrder: this.onUpdateDialogFilterOrder |
|
}); |
|
|
|
// delete peers when dialog is being dropped |
|
/* rootScope.addEventListener('peer_deleted', (peerId) => { |
|
for(const filterId in this.filters) { |
|
const filter = this.filters[filterId]; |
|
let modified = false; |
|
[filter.pinned_peers, filter.include_peers, filter.exclude_peers].forEach((arr) => { |
|
forEachReverse(arr, (inputPeer, idx) => { |
|
if(getPeerId(inputPeer) === peerId) { |
|
arr.splice(idx, 1); |
|
modified = true; |
|
} |
|
}); |
|
}); |
|
|
|
if(modified) { |
|
this.saveDialogFilter(filter, true); |
|
} |
|
} |
|
}); */ |
|
|
|
return this.appStateManager.getState().then((state) => { |
|
safeReplaceObject(this.filters, state.filters); |
|
|
|
for(const filterId in this.filters) { |
|
const filter = this.filters[filterId]; |
|
if(filter.hasOwnProperty('orderIndex') && filter.orderIndex >= this.orderIndex) { |
|
this.orderIndex = filter.orderIndex + 1; |
|
} |
|
|
|
/* this.appMessagesManager.dialogsStorage.folders[+filterId] = { |
|
dialogs: [] |
|
}; */ |
|
} |
|
|
|
// delete this.filters[0]; |
|
// delete this.filters[1]; |
|
// this.getLocalFilter(0); |
|
// this.getLocalFilter(1); |
|
}); |
|
} |
|
|
|
// private getLocalFilter(id: number) { |
|
// return this.filters[id] ??= {...copy(LOCAL_FILTER), id}; |
|
// } |
|
|
|
public clear = (init?: boolean) => { |
|
if(!init) { |
|
// safeReplaceObject(this.filters, {}); |
|
this.reloadedPeerIds.clear(); |
|
this.clearFilters(); |
|
} else { |
|
this.filters = {}; |
|
this.reloadedPeerIds = new Set(); |
|
} |
|
|
|
this.orderIndex = START_ORDER_INDEX; |
|
}; |
|
|
|
private onUpdateDialogFilter = (update: Update.updateDialogFilter) => { |
|
if(update.filter) { |
|
this.saveDialogFilter(update.filter as any); |
|
} else if(this.filters[update.id]) { // Папка удалена |
|
//this.getDialogFilters(true); |
|
this.rootScope.dispatchEvent('filter_delete', this.filters[update.id]); |
|
delete this.filters[update.id]; |
|
} |
|
|
|
this.appStateManager.pushToState('filters', this.filters); |
|
}; |
|
|
|
private onUpdateDialogFilterOrder = (update: Update.updateDialogFilterOrder) => { |
|
//console.log('updateDialogFilterOrder', update); |
|
|
|
this.orderIndex = START_ORDER_INDEX; |
|
update.order.forEach((filterId, idx) => { |
|
const filter = this.filters[filterId]; |
|
delete filter.orderIndex; |
|
this.setOrderIndex(filter); |
|
}); |
|
|
|
this.rootScope.dispatchEvent('filter_order', update.order); |
|
|
|
this.appStateManager.pushToState('filters', this.filters); |
|
}; |
|
|
|
public testDialogForFilter(dialog: Dialog, filter: MyDialogFilter) { |
|
if(filter.id <= 1) { |
|
return dialog.folder_id === filter.id; |
|
} |
|
|
|
const peerId = dialog.peerId; |
|
|
|
// * check whether dialog exists |
|
if(!this.appMessagesManager.getDialogOnly(peerId)) { |
|
return false; |
|
} |
|
|
|
// exclude_peers |
|
if(filter.excludePeerIds.includes(peerId)) { |
|
return false; |
|
} |
|
|
|
// include_peers |
|
if(filter.includePeerIds.includes(peerId)) { |
|
return true; |
|
} |
|
|
|
const pFlags = filter.pFlags; |
|
|
|
// exclude_archived |
|
if(pFlags.exclude_archived && dialog.folder_id === 1) { |
|
return false; |
|
} |
|
|
|
// exclude_read |
|
if(pFlags.exclude_read && !this.appMessagesManager.isDialogUnread(dialog)) { |
|
return false; |
|
} |
|
|
|
// exclude_muted |
|
if(pFlags.exclude_muted && this.appNotificationsManager.isPeerLocalMuted(peerId) && !(dialog.unread_mentions_count && dialog.unread_count)) { |
|
return false; |
|
} |
|
|
|
if(this.appPeersManager.isAnyChat(peerId)) { |
|
// broadcasts |
|
if(pFlags.broadcasts && this.appPeersManager.isBroadcast(peerId)) { |
|
return true; |
|
} |
|
|
|
// groups |
|
if(pFlags.groups && this.appPeersManager.isAnyGroup(peerId)) { |
|
return true; |
|
} |
|
} else { |
|
const userId = peerId.toUserId(); |
|
|
|
// bots |
|
if(this.appUsersManager.isBot(userId)) { |
|
return !!pFlags.bots; |
|
} |
|
|
|
// non_contacts |
|
if(pFlags.non_contacts && !this.appUsersManager.isContact(userId)) { |
|
return true; |
|
} |
|
|
|
// contacts |
|
if(pFlags.contacts && this.appUsersManager.isContact(userId)) { |
|
return true; |
|
} |
|
} |
|
|
|
return false; |
|
} |
|
|
|
public testDialogForFilterId(dialog: Dialog, filterId: number) { |
|
return this.testDialogForFilter(dialog, this.filters[filterId]); |
|
} |
|
|
|
public getFilter(filterId: number) { |
|
return this.filters[filterId]; |
|
} |
|
|
|
public getFilters() { |
|
return this.filters; |
|
} |
|
|
|
public clearFilters() { |
|
const filters = this.getFilters(); |
|
for(const filterId in filters) { // delete filters |
|
this.onUpdateDialogFilter({ |
|
_: 'updateDialogFilter', |
|
id: +filterId, |
|
}); |
|
} |
|
} |
|
|
|
public async toggleDialogPin(peerId: PeerId, filterId: number) { |
|
const filter = this.filters[filterId]; |
|
|
|
const index = filter.pinnedPeerIds.indexOf(peerId); |
|
const wasPinned = index !== -1; |
|
|
|
if(wasPinned) { |
|
filter.pinned_peers.splice(index, 1); |
|
filter.pinnedPeerIds.splice(index, 1); |
|
} |
|
|
|
if(!wasPinned) { |
|
if(filter.pinned_peers.length >= (await this.apiManager.getConfig()).pinned_infolder_count_max) { |
|
return Promise.reject({type: 'PINNED_DIALOGS_TOO_MUCH'}); |
|
} |
|
|
|
filter.pinned_peers.unshift(this.appPeersManager.getInputPeerById(peerId)); |
|
filter.pinnedPeerIds.unshift(peerId); |
|
} |
|
|
|
return this.updateDialogFilter(filter); |
|
} |
|
|
|
public createDialogFilter(filter: MyDialogFilter, prepend?: boolean) { |
|
const maxId = Math.max(1, ...Object.keys(this.filters).map((i) => +i)); |
|
filter = copy(filter); |
|
filter.id = maxId + 1; |
|
return this.updateDialogFilter(filter, undefined, prepend); |
|
} |
|
|
|
public updateDialogFilter(filter: MyDialogFilter, remove = false, prepend = false) { |
|
const flags = remove ? 0 : 1; |
|
|
|
return this.apiManager.invokeApi('messages.updateDialogFilter', { |
|
flags, |
|
id: filter.id, |
|
filter: remove ? undefined : this.getOutputDialogFilter(filter) |
|
}).then((bool: boolean) => { // возможно нужна проверка и откат, если результат не ТРУ |
|
//console.log('updateDialogFilter bool:', bool); |
|
|
|
if(bool) { |
|
/* if(!this.filters[filter.id]) { |
|
this.saveDialogFilter(filter); |
|
} |
|
|
|
rootScope.$broadcast('filter_update', filter); */ |
|
|
|
this.onUpdateDialogFilter({ |
|
_: 'updateDialogFilter', |
|
id: filter.id, |
|
filter: remove ? undefined : filter as any |
|
}); |
|
|
|
if(prepend) { |
|
const f: MyDialogFilter[] = []; |
|
for(const filterId in this.filters) { |
|
const filter = this.filters[filterId]; |
|
++filter.orderIndex; |
|
f.push(filter); |
|
} |
|
|
|
filter.orderIndex = START_ORDER_INDEX; |
|
|
|
const order = f.sort((a, b) => a.orderIndex - b.orderIndex).map((filter) => filter.id); |
|
this.onUpdateDialogFilterOrder({ |
|
_: 'updateDialogFilterOrder', |
|
order |
|
}); |
|
} |
|
} |
|
|
|
return bool; |
|
}); |
|
} |
|
|
|
public getOutputDialogFilter(filter: MyDialogFilter) { |
|
const c = copy(filter); |
|
/* convertment.forEach(([from, to]) => { |
|
c[from] = c[to].map((peerId) => this.appPeersManager.getInputPeerById(peerId)); |
|
}); */ |
|
|
|
this.filterIncludedPinnedPeers(filter); |
|
|
|
return c; |
|
} |
|
|
|
private filterIncludedPinnedPeers(filter: MyDialogFilter) { |
|
forEachReverse(filter.includePeerIds, (peerId, idx) => { |
|
if(filter.pinnedPeerIds.includes(peerId)) { |
|
filter.include_peers.splice(idx, 1); |
|
filter.includePeerIds.splice(idx, 1); |
|
} |
|
}); |
|
} |
|
|
|
// private spliceMissingPeerIds(filterId: number, type: ArgumentTypes<FiltersStorage['reloadMissingPeerIds']>[1], missingPeerIds: PeerId[]) { |
|
// const filter = this.getFilter(filterId); |
|
// const peers = filter && filter[type]; |
|
// if(!peers?.length) { |
|
// return; |
|
// } |
|
|
|
// let spliced = false; |
|
// missingPeerIds.forEach((peerId) => { |
|
// const inputPeer = findAndSplice(peers, (inputPeer) => getPeerId(inputPeer) === peerId); |
|
// if(inputPeer) { |
|
// spliced = true; |
|
// } |
|
// }); |
|
|
|
// if(spliced) { |
|
// this.onUpdateDialogFilter({ |
|
// _: 'updateDialogFilter', |
|
// id: filterId, |
|
// filter |
|
// }); |
|
// } |
|
// } |
|
|
|
public reloadMissingPeerIds(filterId: number, type: 'pinned_peers' | 'include_peers' | 'exclude_peers' = 'pinned_peers') { |
|
const filter = this.getFilter(filterId); |
|
const peers = filter && filter[type]; |
|
if(!peers?.length) { |
|
return; |
|
} |
|
|
|
// const missingPeerIds: PeerId[] = []; |
|
const reloadDialogs = peers.filter((inputPeer) => { |
|
const peerId = getPeerId(inputPeer); |
|
const isAlreadyReloaded = this.reloadedPeerIds.has(peerId); |
|
const dialog = this.appMessagesManager.getDialogOnly(peerId); |
|
// if(isAlreadyReloaded && !dialog) { |
|
// missingPeerIds.push(peerId); |
|
// } |
|
|
|
const reload = !isAlreadyReloaded && !dialog; |
|
return reload; |
|
}); |
|
|
|
if(!reloadDialogs.length) { |
|
// if(missingPeerIds.length) { |
|
// this.spliceMissingPeerIds(filterId, type, missingPeerIds); |
|
// } |
|
|
|
return; |
|
} |
|
|
|
const reloadPromises = reloadDialogs.map((inputPeer) => { |
|
const peerId = getPeerId(inputPeer); |
|
const promise = this.appMessagesManager.reloadConversation(inputPeer) |
|
.then((dialog) => { |
|
this.reloadedPeerIds.add(peerId); |
|
|
|
return dialog ? undefined : peerId; |
|
}); |
|
|
|
return promise; |
|
}); |
|
|
|
const reloadPromise = Promise.all(reloadPromises).then((missingPeerIds) => { |
|
missingPeerIds = missingPeerIds.filter(Boolean); |
|
if(!missingPeerIds.length) { |
|
return; |
|
} |
|
|
|
// this.spliceMissingPeerIds(filterId, type, missingPeerIds); |
|
}); |
|
|
|
return reloadPromise; |
|
} |
|
|
|
public async getDialogFilters(overwrite = false): Promise<MyDialogFilter[]> { |
|
const keys = Object.keys(this.filters); |
|
if(keys.length && !overwrite) { |
|
return keys.map((filterId) => this.filters[filterId]).sort((a, b) => a.orderIndex - b.orderIndex); |
|
} |
|
|
|
const filters = await this.apiManager.invokeApiSingle('messages.getDialogFilters'); |
|
return filters.map((filter) => this.saveDialogFilter(filter, overwrite)).filter(Boolean); |
|
} |
|
|
|
public getSuggestedDialogsFilters() { |
|
return this.apiManager.invokeApi('messages.getSuggestedDialogFilters'); |
|
} |
|
|
|
public saveDialogFilter(filter: DialogFilter, update = true) { |
|
// defineNotNumerableProperties(filter, ['includePeerIds', 'excludePeerIds', 'pinnedPeerIds']); |
|
|
|
// if(filter._ === 'dialogFilterDefault') { |
|
// return; |
|
// // filter = this.getLocalFilter(0); |
|
// // delete filter.orderIndex; |
|
// } |
|
|
|
assumeType<MyDialogFilter>(filter); |
|
convertment.forEach(([from, to]) => { |
|
assumeType<MyDialogFilter>(filter); |
|
filter[to] = filter[from].map((peer) => getPeerId(peer)); |
|
}); |
|
|
|
this.filterIncludedPinnedPeers(filter); |
|
|
|
filter.include_peers = filter.pinned_peers.concat(filter.include_peers); |
|
filter.includePeerIds = filter.pinnedPeerIds.concat(filter.includePeerIds); |
|
|
|
const oldFilter = this.filters[filter.id]; |
|
if(oldFilter) { |
|
Object.assign(oldFilter, filter); |
|
} else { |
|
this.filters[filter.id] = filter; |
|
} |
|
|
|
this.setOrderIndex(filter); |
|
|
|
if(update) { |
|
this.rootScope.dispatchEvent('filter_update', filter); |
|
} else if(!oldFilter) { |
|
this.rootScope.dispatchEvent('filter_new', filter); |
|
} |
|
|
|
return filter; |
|
} |
|
|
|
public setOrderIndex(filter: MyDialogFilter) { |
|
if(filter.hasOwnProperty('orderIndex')) { |
|
if(filter.orderIndex >= this.orderIndex) { |
|
this.orderIndex = filter.orderIndex + 1; |
|
} |
|
} else { |
|
filter.orderIndex = this.orderIndex++ as MyDialogFilter['orderIndex']; |
|
} |
|
|
|
this.appStateManager.pushToState('filters', this.filters); |
|
} |
|
}
|
|
|