Select peers fixes:

Search contacts in forward
Display archived dialogs in select
This commit is contained in:
morethanwords 2020-08-31 00:50:25 +03:00
parent 773d97ee42
commit 6cddae1bf9
13 changed files with 109 additions and 48 deletions

View File

@ -73,7 +73,7 @@ class AppForward {
} else { } else {
this.sendBtn.classList.remove('is-visible'); this.sendBtn.classList.remove('is-visible');
} }
}, 'dialogs', () => { }, 'both', () => {
//console.log('forward rendered:', this.container.querySelector('.selector ul').childElementCount); //console.log('forward rendered:', this.container.querySelector('.selector ul').childElementCount);
this.sidebarWasActive = appSidebarRight.sidebarEl.classList.contains('active'); this.sidebarWasActive = appSidebarRight.sidebarEl.classList.contains('active');
appSidebarRight.toggleSidebar(true); appSidebarRight.toggleSidebar(true);

View File

@ -24,13 +24,16 @@ export class AppSelectPeers {
private myID = $rootScope.myID; private myID = $rootScope.myID;
private folderID = 0;
private offsetIndex = 0; private offsetIndex = 0;
private promise: Promise<any>; private promise: Promise<any>;
private query = ''; private query = '';
private cachedContacts: number[]; private cachedContacts: number[];
private loadedWhat: Partial<{[k in 'dialogs' | 'archived' | 'contacts']: true}> = {};
constructor(private appendTo: HTMLElement, private onChange?: (length: number) => void, private peerType: 'contacts' | 'dialogs' = 'dialogs', onFirstRender?: () => void, private renderResultsFunc?: (peerIDs: number[]) => void) { constructor(private appendTo: HTMLElement, private onChange?: (length: number) => void, private peerType: 'contacts' | 'dialogs' | 'both' = 'dialogs', onFirstRender?: () => void, private renderResultsFunc?: (peerIDs: number[]) => void) {
this.container.classList.add('selector'); this.container.classList.add('selector');
if(!this.renderResultsFunc) { if(!this.renderResultsFunc) {
@ -93,15 +96,25 @@ export class AppSelectPeers {
this.input.addEventListener('input', () => { this.input.addEventListener('input', () => {
let value = this.input.value; let value = this.input.value;
if(this.query != value) { if(this.query != value) {
if(this.peerType == 'contacts') { if(this.peerType == 'contacts' || this.peerType == 'both') {
delete this.loadedWhat.contacts;
this.cachedContacts = null; this.cachedContacts = null;
} else { }
if(this.peerType == 'dialogs' || this.peerType == 'both') {
delete this.loadedWhat.dialogs;
delete this.loadedWhat.archived;
this.folderID = 0;
this.offsetIndex = 0; this.offsetIndex = 0;
} }
this.promise = null; this.promise = null;
this.list.innerHTML = ''; this.list.innerHTML = '';
this.query = value; this.query = value;
if(this.query && 'saved messages'.includes(this.query.toLowerCase())) {
this.renderResultsFunc([$rootScope.myID]);
}
//console.log('selectPeers input:', this.query); //console.log('selectPeers input:', this.query);
this.getMoreResults(); this.getMoreResults();
@ -126,33 +139,51 @@ export class AppSelectPeers {
}, 0); }, 0);
} }
private async getMoreDialogs() { private async getMoreDialogs(): Promise<any> {
if(this.promise) return this.promise; if(this.promise) return this.promise;
if(this.loadedWhat.dialogs && this.loadedWhat.archived) {
return;
}
// в десктопе - сначала без группы, потом архивные, потом контакты без сообщений // в десктопе - сначала без группы, потом архивные, потом контакты без сообщений
const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0; const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0;
this.promise = appMessagesManager.getConversations(this.query, this.offsetIndex, pageCount, 0); this.promise = appMessagesManager.getConversations(this.query, this.offsetIndex, pageCount, this.folderID);
const value = await this.promise; const value = await this.promise;
let dialogs = value.dialogs as Dialog[]; let dialogs = value.dialogs as Dialog[];
if(!dialogs.length) { if(dialogs.length) {
return; const newOffsetIndex = dialogs[dialogs.length - 1].index || 0;
dialogs = dialogs.filter(d => d.peerID != this.myID);
if(!this.offsetIndex && this.folderID == 0 && !this.query) {
dialogs.unshift({
peerID: this.myID,
pFlags: {}
} as any);
}
this.offsetIndex = newOffsetIndex;
this.renderResultsFunc(dialogs.map(dialog => dialog.peerID));
} else {
if(!this.loadedWhat.dialogs) {
this.loadedWhat.dialogs = true;
this.offsetIndex = 0;
this.folderID = 1;
this.promise = null;
return this.getMoreDialogs();
} else {
this.loadedWhat.archived = true;
if(!this.loadedWhat.contacts && this.peerType == 'both') {
this.promise = null;
return this.getMoreContacts();
}
}
} }
const newOffsetIndex = dialogs[dialogs.length - 1].index || 0;
dialogs = dialogs.filter(d => d.peerID != this.myID);
if(!this.offsetIndex && !this.query) {
dialogs.unshift({
peerID: this.myID,
pFlags: {}
} as any);
}
this.offsetIndex = newOffsetIndex;
this.renderResultsFunc(dialogs.map(dialog => dialog.peerID));
this.promise = null; this.promise = null;
} }
@ -160,6 +191,10 @@ export class AppSelectPeers {
private async getMoreContacts() { private async getMoreContacts() {
if(this.promise) return this.promise; if(this.promise) return this.promise;
if(this.loadedWhat.contacts) {
return;
}
if(!this.cachedContacts) { if(!this.cachedContacts) {
this.promise = appUsersManager.getContacts(this.query); this.promise = appUsersManager.getContacts(this.query);
this.cachedContacts = (await this.promise).slice(); this.cachedContacts = (await this.promise).slice();
@ -171,15 +206,26 @@ export class AppSelectPeers {
const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0; const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0;
const arr = this.cachedContacts.splice(0, pageCount); const arr = this.cachedContacts.splice(0, pageCount);
this.renderResultsFunc(arr); this.renderResultsFunc(arr);
} else {
this.loadedWhat.contacts = true;
} }
} }
private getMoreResults() { private getMoreResults() {
if(this.peerType == 'dialogs') { const promises: Promise<any>[] = [];
return this.getMoreDialogs(); if(this.peerType == 'dialogs' || this.peerType == 'both') {
} else { promises.push(this.getMoreDialogs());
return this.getMoreContacts();
if(!this.loadedWhat.archived) {
return Promise.all(promises);
}
} }
if(this.peerType == 'contacts' || this.peerType == 'both') {
promises.push(this.getMoreContacts());
}
return Promise.all(promises);
} }
private renderResults(peerIDs: number[]) { private renderResults(peerIDs: number[]) {

View File

@ -7,6 +7,7 @@ import appPeersManager from "../../lib/appManagers/appPeersManager";
import { $rootScope, cancelEvent } from "../../lib/utils"; import { $rootScope, cancelEvent } from "../../lib/utils";
import appSidebarLeft from "../../lib/appManagers/appSidebarLeft"; import appSidebarLeft from "../../lib/appManagers/appSidebarLeft";
import { ripple } from "../ripple"; import { ripple } from "../ripple";
import { toast } from "../toast";
type DialogFilterSuggested = { type DialogFilterSuggested = {
_: 'dialogFilterSuggested', _: 'dialogFilterSuggested',
@ -98,7 +99,11 @@ export default class AppChatFoldersTab implements SliderTab {
this.createFolderBtn = this.container.querySelector('.btn-create-folder'); this.createFolderBtn = this.container.querySelector('.btn-create-folder');
this.createFolderBtn.addEventListener('click', () => { this.createFolderBtn.addEventListener('click', () => {
appSidebarLeft.editFolderTab.open(); if(Object.keys(this.filtersRendered).length >= 10) {
toast('Sorry, you can\'t create more folders.');
} else {
appSidebarLeft.editFolderTab.open();
}
}); });
lottieLoader.loadAnimationFromURL({ lottieLoader.loadAnimationFromURL({
@ -162,6 +167,12 @@ export default class AppChatFoldersTab implements SliderTab {
button.addEventListener('click', (e) => { button.addEventListener('click', (e) => {
cancelEvent(e); cancelEvent(e);
if(Object.keys(this.filtersRendered).length >= 10) {
toast('Sorry, you can\'t create more folders.');
return;
}
button.setAttribute('disabled', 'true'); button.setAttribute('disabled', 'true');
appMessagesManager.filtersStorage.createDialogFilter(filter.filter).then(bool => { appMessagesManager.filtersStorage.createDialogFilter(filter.filter).then(bool => {

View File

@ -108,6 +108,12 @@ export default class AppEditFolderTab implements SliderTab {
if(bool) { if(bool) {
this.closeBtn.click(); this.closeBtn.click();
} }
}).catch(err => {
if(err.type == 'DIALOG_FILTERS_TOO_MUCH') {
toast('Sorry, you can\'t create more folders.');
} else {
console.error('updateDialogFilter error:', err);
}
}).finally(() => { }).finally(() => {
this.confirmBtn.removeAttribute('disabled'); this.confirmBtn.removeAttribute('disabled');
}); });

View File

@ -7,6 +7,8 @@ import Scrollable from "../scrollable_new";
import appUsersManager from "../../lib/appManagers/appUsersManager"; import appUsersManager from "../../lib/appManagers/appUsersManager";
import { $rootScope } from "../../lib/utils"; import { $rootScope } from "../../lib/utils";
// TODO: аватарка не поменяется в этой вкладке после изменения почему-то (если поставить в другом клиенте, и потом тут проверить, для этого ещё вышел в чатлист)
export default class AppEditProfileTab implements SliderTab { export default class AppEditProfileTab implements SliderTab {
private container = document.querySelector('.edit-profile-container') as HTMLDivElement; private container = document.querySelector('.edit-profile-container') as HTMLDivElement;
private scrollWrapper = this.container.querySelector('.scroll-wrapper') as HTMLDivElement; private scrollWrapper = this.container.querySelector('.scroll-wrapper') as HTMLDivElement;

View File

@ -414,7 +414,7 @@ function wrapMediaWithTail(photo: MTPhoto | MTDocument, message: {mid: number, m
return img; return img;
} }
export function wrapPhoto(photo: MTPhoto | MTDocument, message: any, container: HTMLDivElement, boxWidth = mediaSizes.active.regular.width, boxHeight = mediaSizes.active.regular.height, withTail = true, isOut = false, lazyLoadQueue: LazyLoadQueue, middleware: () => boolean, size: MTPhotoSize = null) { export function wrapPhoto(photo: MTPhoto | MTDocument, message: any, container: HTMLDivElement, boxWidth = mediaSizes.active.regular.width, boxHeight = mediaSizes.active.regular.height, withTail: boolean, isOut: boolean, lazyLoadQueue: LazyLoadQueue, middleware: () => boolean, size: MTPhotoSize = null) {
let image: HTMLImageElement; let image: HTMLImageElement;
if(withTail) { if(withTail) {
image = wrapMediaWithTail(photo, message, container, boxWidth, boxHeight, isOut); image = wrapMediaWithTail(photo, message, container, boxWidth, boxHeight, isOut);

View File

@ -1,7 +1,7 @@
export const readBlobAsText = (blob: Blob) => { export const readBlobAsText = (blob: Blob) => {
return new Promise<string>(resolve => { return new Promise<string>(resolve => {
const reader = new FileReader(); const reader = new FileReader();
reader.addEventListener('loadend', async(e) => { reader.addEventListener('loadend', (e) => {
// @ts-ignore // @ts-ignore
resolve(e.srcElement.result); resolve(e.srcElement.result);
}); });

View File

@ -1,7 +1,7 @@
import { putPreloader, renderImageFromUrl } from "../../components/misc"; import { putPreloader, renderImageFromUrl } from "../../components/misc";
//import Scrollable from '../../components/scrollable'; //import Scrollable from '../../components/scrollable';
import Scrollable from '../../components/scrollable_new'; import Scrollable from '../../components/scrollable_new';
import { $rootScope, findUpClassName } from "../utils"; import { $rootScope } from "../utils";
import appMessagesManager from "./appMessagesManager"; import appMessagesManager from "./appMessagesManager";
import appPhotosManager from "./appPhotosManager"; import appPhotosManager from "./appPhotosManager";
import appPeersManager from "./appPeersManager"; import appPeersManager from "./appPeersManager";
@ -12,12 +12,12 @@ import { logger, LogLevels } from "../logger";
import appImManager from "./appImManager"; import appImManager from "./appImManager";
import appMediaViewer from "./appMediaViewer"; import appMediaViewer from "./appMediaViewer";
import LazyLoadQueue from "../../components/lazyLoadQueue"; import LazyLoadQueue from "../../components/lazyLoadQueue";
import { wrapDocument, wrapAudio, wrapSticker } from "../../components/wrappers"; import { wrapDocument, wrapAudio } from "../../components/wrappers";
import AppSearch, { SearchGroup } from "../../components/appSearch"; import AppSearch, { SearchGroup } from "../../components/appSearch";
import AvatarElement from "../../components/avatar"; import AvatarElement from "../../components/avatar";
import appForward from "../../components/appForward"; import appForward from "../../components/appForward";
import { mediaSizes } from "../config"; import { mediaSizes } from "../config";
import SidebarSlider, { SliderTab } from "../../components/slider"; import SidebarSlider from "../../components/slider";
import SearchInput from "../../components/searchInput"; import SearchInput from "../../components/searchInput";
import { horizontalMenu } from "../../components/horizontalMenu"; import { horizontalMenu } from "../../components/horizontalMenu";
import AppStickersTab from "../../components/sidebarRight/stickers"; import AppStickersTab from "../../components/sidebarRight/stickers";
@ -55,6 +55,9 @@ const contentToSharedMap: {[contentType in ContentType]: SharedMediaType} = {
contentAudio: 'inputMessagesFilterMusic' contentAudio: 'inputMessagesFilterMusic'
}; };
// TODO: отправленное сообщение с картинкой, или же новое полученное апдейтом сообщение не отобразится в медии
// TODO: по-хорошему, нужно просто сделать апдейты для всего сайдбара
export class AppSidebarRight extends SidebarSlider { export class AppSidebarRight extends SidebarSlider {
public static SLIDERITEMSIDS = { public static SLIDERITEMSIDS = {
search: 1, search: 1,
@ -633,7 +636,7 @@ export class AppSidebarRight extends SidebarSlider {
//this.contentContainer.classList.add('loaded'); //this.contentContainer.classList.add('loaded');
if(!messages.length) { if(!messages.length && !sharedMediaDiv.childElementCount) {
const div = document.createElement('div'); const div = document.createElement('div');
div.innerText = 'Nothing interesting here yet...'; div.innerText = 'Nothing interesting here yet...';
div.classList.add('position-center', 'text-center', 'content-empty', 'no-select'); div.classList.add('position-center', 'text-center', 'content-empty', 'no-select');

View File

@ -151,17 +151,6 @@ export function getRichElementValue(node: any, lines: string[], line: string[],
} }
} */ } */
type BroadcastKeys = 'download_progress' | 'user_update' | 'user_auth' | 'peer_changed' |
'filter_delete' | 'filter_update' | 'message_edit' | 'dialog_draft' | 'messages_pending' |
'history_append' | 'history_update' | 'dialogs_multiupdate' | 'dialog_unread' | 'dialog_flush' |
'dialog_drop' | 'dialog_migrate' | 'dialog_top' | 'history_reply_markup' | 'history_multiappend' |
'messages_read' | 'history_delete' | 'history_forbidden' | 'history_reload' | 'message_views' |
'message_sent' | 'history_request' | 'messages_downloaded' | 'contacts_update' | 'avatar_update' |
'stickers_installed' | 'stickers_deleted' | 'chat_full_update' | 'peer_pinned_message' |
'poll_update' | 'dialogs_archived_unread' | 'audio_play' | 'audio_pause' | 'chat_update' |
'apiUpdate' | 'stateSynchronized' | 'channel_settings' | 'webpage_updated' | 'draft_updated' |
'dialog_notify_settings';
type BroadcastEvents = { type BroadcastEvents = {
'download_progress': any, 'download_progress': any,
'user_update': any, 'user_update': any,
@ -222,7 +211,7 @@ export const $rootScope = {
// @ts-ignore // @ts-ignore
document.addEventListener(name, callback); document.addEventListener(name, callback);
}, },
$off: (name: BroadcastKeys, callback: (e: CustomEvent) => any) => { $off: <T extends keyof BroadcastEvents>(name: T, callback: (e: Omit<CustomEvent, 'detail'> & {detail: BroadcastEvents[T]}) => any) => {
// @ts-ignore // @ts-ignore
document.removeEventListener(name, callback); document.removeEventListener(name, callback);
}, },

View File

@ -247,7 +247,7 @@ let onFirstMount = () => {
} }
//lang_code: navigator.language || 'en' //lang_code: navigator.language || 'en'
}).then((code: any) => { }).then((code: any) => {
console.log('got code', code); //console.log('got code', code);
pageAuthCode.mount(Object.assign(code, {phone_number: phone_number})); pageAuthCode.mount(Object.assign(code, {phone_number: phone_number}));
}).catch(err => { }).catch(err => {

View File

@ -105,7 +105,7 @@
.user-title { .user-title {
&:after { &:after {
content: $tgico-nosound; content: $tgico-nosound;
color: #707579; color: #a3a3a3;
font-size: 1.125rem; font-size: 1.125rem;
line-height: 27px; line-height: 27px;
vertical-align: middle; vertical-align: middle;

View File

@ -238,6 +238,10 @@
> div { > div {
//height: 100%; //height: 100%;
position: relative; position: relative;
/* > div:not(:empty) + .content-empty {
display: none;
} */
} }
/* > div > div:not(.scroll-padding) { /* > div > div:not(.scroll-padding) {

View File

@ -6,7 +6,7 @@ const postcssPresetEnv = require('postcss-preset-env');
const ServiceWorkerWebpackPlugin = require('serviceworker-webpack-plugin'); const ServiceWorkerWebpackPlugin = require('serviceworker-webpack-plugin');
const fs = require('fs'); const fs = require('fs');
const allowedIPs = ['195.66.140.39', '192.168.31.144', '127.0.0.1', '192.168.31.1', '192.168.31.192', '176.100.18.181', '46.219.250.22', '193.42.119.184', '46.133.168.67']; const allowedIPs = ['195.66.140.39', '192.168.31.144', '127.0.0.1', '192.168.31.1', '192.168.31.192', '176.100.18.181', '46.219.250.22', '193.42.119.184', '46.133.168.67', '78.26.144.197', '46.133.225.88'];
const devMode = process.env.NODE_ENV !== 'production'; const devMode = process.env.NODE_ENV !== 'production';
const useLocal = false; const useLocal = false;