User permissions
Fix forward layout
This commit is contained in:
parent
2a6e7d1178
commit
5f00fb9a87
@ -91,6 +91,8 @@ export class AppNavigationController {
|
|||||||
}, options);
|
}, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
history.scrollRestoration = 'manual';
|
||||||
|
|
||||||
this.pushState(); // * push init state
|
this.pushState(); // * push init state
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ export class SearchGroup {
|
|||||||
list: HTMLUListElement;
|
list: HTMLUListElement;
|
||||||
|
|
||||||
constructor(public name: string, public type: string, private clearable = true, className?: string, clickable = true, public autonomous = true, public onFound?: () => void) {
|
constructor(public name: string, public type: string, private clearable = true, className?: string, clickable = true, public autonomous = true, public onFound?: () => void) {
|
||||||
this.list = document.createElement('ul');
|
this.list = appDialogsManager.createChatList();
|
||||||
this.container = document.createElement('div');
|
this.container = document.createElement('div');
|
||||||
if(className) this.container.className = className;
|
if(className) this.container.className = className;
|
||||||
|
|
||||||
|
@ -9,15 +9,19 @@ import { cancelEvent, findUpAttribute, findUpClassName } from "../helpers/dom";
|
|||||||
import Scrollable from "./scrollable";
|
import Scrollable from "./scrollable";
|
||||||
import { FocusDirection } from "../helpers/fastSmoothScroll";
|
import { FocusDirection } from "../helpers/fastSmoothScroll";
|
||||||
import CheckboxField from "./checkboxField";
|
import CheckboxField from "./checkboxField";
|
||||||
|
import appProfileManager from "../lib/appManagers/appProfileManager";
|
||||||
|
|
||||||
type PeerType = 'contacts' | 'dialogs';
|
type PeerType = 'contacts' | 'dialogs' | 'channelParticipants';
|
||||||
|
|
||||||
// TODO: правильная сортировка для addMembers, т.е. для peerType: 'contacts', потому что там идут сначала контакты - потом неконтакты, а должно всё сортироваться по имени
|
// TODO: правильная сортировка для addMembers, т.е. для peerType: 'contacts', потому что там идут сначала контакты - потом неконтакты, а должно всё сортироваться по имени
|
||||||
|
|
||||||
let loadedAllDialogs = false, loadAllDialogsPromise: Promise<any>;
|
let loadedAllDialogs = false, loadAllDialogsPromise: Promise<any>;
|
||||||
export default class AppSelectPeers {
|
export default class AppSelectPeers {
|
||||||
public container = document.createElement('div');
|
public container = document.createElement('div');
|
||||||
public list = document.createElement('ul');
|
public list = appDialogsManager.createChatList(/* {
|
||||||
|
handheldsSize: 66,
|
||||||
|
avatarSize: 48
|
||||||
|
} */);
|
||||||
public chatsContainer = document.createElement('div');
|
public chatsContainer = document.createElement('div');
|
||||||
public scrollable: Scrollable;
|
public scrollable: Scrollable;
|
||||||
public selectedScrollable: Scrollable;
|
public selectedScrollable: Scrollable;
|
||||||
@ -37,7 +41,7 @@ export default class AppSelectPeers {
|
|||||||
private query = '';
|
private query = '';
|
||||||
private cachedContacts: number[];
|
private cachedContacts: number[];
|
||||||
|
|
||||||
private loadedWhat: Partial<{[k in 'dialogs' | 'archived' | 'contacts']: true}> = {};
|
private loadedWhat: Partial<{[k in 'dialogs' | 'archived' | 'contacts' | 'channelParticipants']: true}> = {};
|
||||||
|
|
||||||
private renderedPeerIds: Set<number> = new Set();
|
private renderedPeerIds: Set<number> = new Set();
|
||||||
|
|
||||||
@ -48,16 +52,22 @@ export default class AppSelectPeers {
|
|||||||
private chatRightsAction: ChatRights;
|
private chatRightsAction: ChatRights;
|
||||||
private multiSelect = true;
|
private multiSelect = true;
|
||||||
private rippleEnabled = true;
|
private rippleEnabled = true;
|
||||||
|
private avatarSize = 48;
|
||||||
|
|
||||||
|
private tempIds: {[k in keyof AppSelectPeers['loadedWhat']]: number} = {};
|
||||||
|
private peerId = 0;
|
||||||
|
|
||||||
constructor(options: {
|
constructor(options: {
|
||||||
appendTo: AppSelectPeers['appendTo'],
|
appendTo: AppSelectPeers['appendTo'],
|
||||||
onChange?: AppSelectPeers['onChange'],
|
onChange?: AppSelectPeers['onChange'],
|
||||||
peerType?: AppSelectPeers['peerType'],
|
peerType?: AppSelectPeers['peerType'],
|
||||||
|
peerId?: number,
|
||||||
onFirstRender?: () => void,
|
onFirstRender?: () => void,
|
||||||
renderResultsFunc?: AppSelectPeers['renderResultsFunc'],
|
renderResultsFunc?: AppSelectPeers['renderResultsFunc'],
|
||||||
chatRightsAction?: AppSelectPeers['chatRightsAction'],
|
chatRightsAction?: AppSelectPeers['chatRightsAction'],
|
||||||
multiSelect?: AppSelectPeers['multiSelect'],
|
multiSelect?: AppSelectPeers['multiSelect'],
|
||||||
rippleEnabled?: boolean
|
rippleEnabled?: boolean,
|
||||||
|
avatarSize?: AppSelectPeers['avatarSize'],
|
||||||
}) {
|
}) {
|
||||||
for(let i in options) {
|
for(let i in options) {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
@ -66,8 +76,19 @@ export default class AppSelectPeers {
|
|||||||
|
|
||||||
this.container.classList.add('selector');
|
this.container.classList.add('selector');
|
||||||
|
|
||||||
|
this.peerType.forEach(type => {
|
||||||
|
this.tempIds[type] = 0;
|
||||||
|
});
|
||||||
|
|
||||||
|
let needSwitchList = false;
|
||||||
const f = (this.renderResultsFunc || this.renderResults).bind(this);
|
const f = (this.renderResultsFunc || this.renderResults).bind(this);
|
||||||
this.renderResultsFunc = (peerIds: number[]) => {
|
this.renderResultsFunc = (peerIds: number[]) => {
|
||||||
|
if(needSwitchList) {
|
||||||
|
this.scrollable.splitUp.replaceWith(this.list);
|
||||||
|
this.scrollable.setVirtualContainer(this.list);
|
||||||
|
needSwitchList = false;
|
||||||
|
}
|
||||||
|
|
||||||
peerIds = peerIds.filter(peerId => {
|
peerIds = peerIds.filter(peerId => {
|
||||||
const notRendered = !this.renderedPeerIds.has(peerId);
|
const notRendered = !this.renderedPeerIds.has(peerId);
|
||||||
if(notRendered) this.renderedPeerIds.add(peerId);
|
if(notRendered) this.renderedPeerIds.add(peerId);
|
||||||
@ -149,21 +170,26 @@ export default class AppSelectPeers {
|
|||||||
const value = this.input.value;
|
const value = this.input.value;
|
||||||
if(this.query !== value) {
|
if(this.query !== value) {
|
||||||
if(this.peerType.includes('contacts')) {
|
if(this.peerType.includes('contacts')) {
|
||||||
delete this.loadedWhat.contacts;
|
|
||||||
this.cachedContacts = null;
|
this.cachedContacts = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
//if(this.peerType.includes('dialogs')) {
|
//if(this.peerType.includes('dialogs')) {
|
||||||
delete this.loadedWhat.dialogs;
|
|
||||||
delete this.loadedWhat.archived;
|
|
||||||
this.folderId = 0;
|
this.folderId = 0;
|
||||||
this.offsetIndex = 0;
|
this.offsetIndex = 0;
|
||||||
//}
|
//}
|
||||||
|
|
||||||
|
for(let i in this.tempIds) {
|
||||||
|
// @ts-ignore
|
||||||
|
++this.tempIds[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
this.list = appDialogsManager.createChatList();
|
||||||
|
|
||||||
this.promise = null;
|
this.promise = null;
|
||||||
this.list.innerHTML = '';
|
this.loadedWhat = {};
|
||||||
this.query = value;
|
this.query = value;
|
||||||
this.renderedPeerIds.clear();
|
this.renderedPeerIds.clear();
|
||||||
|
needSwitchList = true;
|
||||||
|
|
||||||
//console.log('selectPeers input:', this.query);
|
//console.log('selectPeers input:', this.query);
|
||||||
this.getMoreResults();
|
this.getMoreResults();
|
||||||
@ -204,10 +230,16 @@ export default class AppSelectPeers {
|
|||||||
// в десктопе - сначала без группы, потом архивные, потом контакты без сообщений
|
// в десктопе - сначала без группы, потом архивные, потом контакты без сообщений
|
||||||
const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0;
|
const pageCount = appPhotosManager.windowH / 72 * 1.25 | 0;
|
||||||
|
|
||||||
|
const tempId = ++this.tempIds.dialogs;
|
||||||
|
|
||||||
this.promise = appMessagesManager.getConversations(this.query, this.offsetIndex, pageCount, this.folderId);
|
this.promise = appMessagesManager.getConversations(this.query, this.offsetIndex, pageCount, this.folderId);
|
||||||
const value = await this.promise;
|
const value = await this.promise;
|
||||||
this.promise = null;
|
this.promise = null;
|
||||||
|
|
||||||
|
if(this.tempIds.dialogs !== tempId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let dialogs = value.dialogs as Dialog[];
|
let dialogs = value.dialogs as Dialog[];
|
||||||
if(dialogs.length) {
|
if(dialogs.length) {
|
||||||
const newOffsetIndex = dialogs[dialogs.length - 1].index || 0;
|
const newOffsetIndex = dialogs[dialogs.length - 1].index || 0;
|
||||||
@ -260,8 +292,13 @@ export default class AppSelectPeers {
|
|||||||
|
|
||||||
this.promise = Promise.all(promises);
|
this.promise = Promise.all(promises);
|
||||||
this.cachedContacts = (await this.promise)[0].slice(); */
|
this.cachedContacts = (await this.promise)[0].slice(); */
|
||||||
|
const tempId = ++this.tempIds.contacts;
|
||||||
this.promise = appUsersManager.getContacts(this.query);
|
this.promise = appUsersManager.getContacts(this.query);
|
||||||
this.cachedContacts = (await this.promise).slice();
|
this.cachedContacts = (await this.promise).slice();
|
||||||
|
if(this.tempIds.contacts !== tempId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.cachedContacts.findAndSplice(userId => userId === rootScope.myId); // no my account
|
this.cachedContacts.findAndSplice(userId => userId === rootScope.myId); // no my account
|
||||||
this.promise = null;
|
this.promise = null;
|
||||||
}
|
}
|
||||||
@ -282,6 +319,31 @@ export default class AppSelectPeers {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async getMoreChannelParticipants() {
|
||||||
|
if(this.promise) return this.promise;
|
||||||
|
|
||||||
|
if(this.loadedWhat.channelParticipants) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const pageCount = 50; // same as in group permissions to use cache
|
||||||
|
|
||||||
|
const tempId = ++this.tempIds.channelParticipants;
|
||||||
|
const promise = appProfileManager.getChannelParticipants(-this.peerId, {_: 'channelParticipantsSearch', q: this.query}, pageCount, this.list.childElementCount);
|
||||||
|
const participants = await promise;
|
||||||
|
if(this.tempIds.channelParticipants !== tempId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const userIds = participants.participants.map(participant => participant.user_id);
|
||||||
|
userIds.findAndSplice(u => u === rootScope.myId);
|
||||||
|
this.renderResultsFunc(userIds);
|
||||||
|
|
||||||
|
if(this.list.childElementCount >= participants.count || participants.participants.length < pageCount) {
|
||||||
|
this.loadedWhat.channelParticipants = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
checkForTriggers = () => {
|
checkForTriggers = () => {
|
||||||
this.scrollable.checkForTriggers();
|
this.scrollable.checkForTriggers();
|
||||||
};
|
};
|
||||||
@ -290,13 +352,15 @@ export default class AppSelectPeers {
|
|||||||
const get = () => {
|
const get = () => {
|
||||||
const promises: Promise<any>[] = [];
|
const promises: Promise<any>[] = [];
|
||||||
|
|
||||||
if(!loadedAllDialogs && !loadAllDialogsPromise) {
|
if(!loadedAllDialogs && (this.peerType.includes('dialogs') || this.peerType.includes('contacts'))) {
|
||||||
loadAllDialogsPromise = appMessagesManager.getConversationsAll()
|
if(!loadAllDialogsPromise) {
|
||||||
.then(() => {
|
loadAllDialogsPromise = appMessagesManager.getConversationsAll()
|
||||||
loadedAllDialogs = true;
|
.then(() => {
|
||||||
}, () => {
|
loadedAllDialogs = true;
|
||||||
loadAllDialogsPromise = null;
|
}).finally(() => {
|
||||||
});
|
loadAllDialogsPromise = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
promises.push(loadAllDialogsPromise);
|
promises.push(loadAllDialogsPromise);
|
||||||
}
|
}
|
||||||
@ -312,6 +376,10 @@ export default class AppSelectPeers {
|
|||||||
if(this.peerType.includes('contacts') && !this.loadedWhat.contacts) {
|
if(this.peerType.includes('contacts') && !this.loadedWhat.contacts) {
|
||||||
promises.push(this.getMoreContacts());
|
promises.push(this.getMoreContacts());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(this.peerType.includes('channelParticipants') && !this.loadedWhat.channelParticipants) {
|
||||||
|
promises.push(this.getMoreChannelParticipants());
|
||||||
|
}
|
||||||
|
|
||||||
return promises;
|
return promises;
|
||||||
};
|
};
|
||||||
@ -340,8 +408,8 @@ export default class AppSelectPeers {
|
|||||||
dialog: peerId,
|
dialog: peerId,
|
||||||
container: this.scrollable,
|
container: this.scrollable,
|
||||||
drawStatus: false,
|
drawStatus: false,
|
||||||
rippleEnabled: true,
|
rippleEnabled: this.rippleEnabled,
|
||||||
avatarSize: 48
|
avatarSize: this.avatarSize
|
||||||
});
|
});
|
||||||
|
|
||||||
if(this.multiSelect) {
|
if(this.multiSelect) {
|
||||||
@ -449,4 +517,4 @@ export default class AppSelectPeers {
|
|||||||
this.selectedScrollable.scrollIntoViewNew(this.input, 'center', undefined, undefined, FocusDirection.Static);
|
this.selectedScrollable.scrollIntoViewNew(this.input, 'center', undefined, undefined, FocusDirection.Static);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ export default class PopupPeer extends PopupElement {
|
|||||||
description: string,
|
description: string,
|
||||||
buttons: Array<PopupButton>
|
buttons: Array<PopupButton>
|
||||||
}> = {}) {
|
}> = {}) {
|
||||||
super('popup-peer' + (className ? ' ' + className : ''), options.buttons);
|
super('popup-peer' + (className ? ' ' + className : ''), options.buttons, {overlayClosable: true});
|
||||||
|
|
||||||
let avatarEl = new AvatarElement();
|
let avatarEl = new AvatarElement();
|
||||||
avatarEl.setAttribute('dialog', '1');
|
avatarEl.setAttribute('dialog', '1');
|
||||||
|
@ -10,7 +10,8 @@ export default class PopupPickUser extends PopupElement {
|
|||||||
onSelect?: (peerId: number) => Promise<void> | void,
|
onSelect?: (peerId: number) => Promise<void> | void,
|
||||||
onClose?: () => void,
|
onClose?: () => void,
|
||||||
placeholder: string,
|
placeholder: string,
|
||||||
chatRightsAction?: AppSelectPeers['chatRightsAction']
|
chatRightsAction?: AppSelectPeers['chatRightsAction'],
|
||||||
|
peerId?: number,
|
||||||
}) {
|
}) {
|
||||||
super('popup-forward', null, {closable: true, overlayClosable: true, body: true});
|
super('popup-forward', null, {closable: true, overlayClosable: true, body: true});
|
||||||
|
|
||||||
@ -42,7 +43,9 @@ export default class PopupPickUser extends PopupElement {
|
|||||||
},
|
},
|
||||||
chatRightsAction: options.chatRightsAction,
|
chatRightsAction: options.chatRightsAction,
|
||||||
multiSelect: false,
|
multiSelect: false,
|
||||||
rippleEnabled: false
|
rippleEnabled: false,
|
||||||
|
avatarSize: 46,
|
||||||
|
peerId: options.peerId,
|
||||||
});
|
});
|
||||||
|
|
||||||
//this.scrollable = new Scrollable(this.body);
|
//this.scrollable = new Scrollable(this.body);
|
||||||
|
@ -22,7 +22,7 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
//let animationEndPromise: Promise<number>;
|
//let animationEndPromise: Promise<number>;
|
||||||
const drawRipple = (clientX: number, clientY: number) => {
|
const drawRipple = (clientX: number, clientY: number) => {
|
||||||
const startTime = Date.now();
|
const startTime = Date.now();
|
||||||
const span = document.createElement('span');
|
const elem = document.createElement('div');
|
||||||
|
|
||||||
const clickId = rippleClickId++;
|
const clickId = rippleClickId++;
|
||||||
|
|
||||||
@ -40,18 +40,18 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
let elapsedTime = Date.now() - startTime;
|
let elapsedTime = Date.now() - startTime;
|
||||||
if(elapsedTime < duration) {
|
if(elapsedTime < duration) {
|
||||||
let delay = Math.max(duration - elapsedTime, duration / 2);
|
let delay = Math.max(duration - elapsedTime, duration / 2);
|
||||||
setTimeout(() => span.classList.add('hiding'), Math.max(delay - duration / 2, 0));
|
setTimeout(() => elem.classList.add('hiding'), Math.max(delay - duration / 2, 0));
|
||||||
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
//console.log('ripple elapsedTime total pre-remove:', Date.now() - startTime);
|
//console.log('ripple elapsedTime total pre-remove:', Date.now() - startTime);
|
||||||
span.remove();
|
elem.remove();
|
||||||
if(onEnd) onEnd(clickId);
|
if(onEnd) onEnd(clickId);
|
||||||
}, delay);
|
}, delay);
|
||||||
} else {
|
} else {
|
||||||
span.classList.add('hiding');
|
elem.classList.add('hiding');
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
//console.log('ripple elapsedTime total pre-remove:', Date.now() - startTime);
|
//console.log('ripple elapsedTime total pre-remove:', Date.now() - startTime);
|
||||||
span.remove();
|
elem.remove();
|
||||||
if(onEnd) onEnd(clickId);
|
if(onEnd) onEnd(clickId);
|
||||||
}, duration / 2);
|
}, duration / 2);
|
||||||
}
|
}
|
||||||
@ -82,7 +82,7 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
|
|
||||||
window.requestAnimationFrame(() => {
|
window.requestAnimationFrame(() => {
|
||||||
let rect = r.getBoundingClientRect();
|
let rect = r.getBoundingClientRect();
|
||||||
span.classList.add('c-ripple__circle');
|
elem.classList.add('c-ripple__circle');
|
||||||
|
|
||||||
let clickX = clientX - rect.left;
|
let clickX = clientX - rect.left;
|
||||||
let clickY = clientY - rect.top;
|
let clickY = clientY - rect.top;
|
||||||
@ -106,9 +106,9 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
|
|
||||||
//console.log('ripple click', offsetFromCenter, size, clickX, clickY);
|
//console.log('ripple click', offsetFromCenter, size, clickX, clickY);
|
||||||
|
|
||||||
span.style.width = span.style.height = size + 'px';
|
elem.style.width = elem.style.height = size + 'px';
|
||||||
span.style.left = x + 'px';
|
elem.style.left = x + 'px';
|
||||||
span.style.top = y + 'px';
|
elem.style.top = y + 'px';
|
||||||
|
|
||||||
// нижний код выполняется с задержкой
|
// нижний код выполняется с задержкой
|
||||||
/* animationEndPromise = new Promise((resolve) => {
|
/* animationEndPromise = new Promise((resolve) => {
|
||||||
@ -124,7 +124,7 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
duration = +window.getComputedStyle(span).getPropertyValue('animation-duration').replace('s', '') * 1000;
|
duration = +window.getComputedStyle(span).getPropertyValue('animation-duration').replace('s', '') * 1000;
|
||||||
span.style.display = ''; */
|
span.style.display = ''; */
|
||||||
|
|
||||||
r.append(span);
|
r.append(elem);
|
||||||
|
|
||||||
//r.classList.add('active');
|
//r.classList.add('active');
|
||||||
//handler();
|
//handler();
|
||||||
@ -190,4 +190,4 @@ export function ripple(elem: HTMLElement, callback: (id: number) => Promise<bool
|
|||||||
window.addEventListener('contextmenu', handler, {once: true});
|
window.addEventListener('contextmenu', handler, {once: true});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,7 @@ export default class AppBlockedUsersTab extends SliderSuperTab {
|
|||||||
});
|
});
|
||||||
}, {listenerSetter: this.listenerSetter});
|
}, {listenerSetter: this.listenerSetter});
|
||||||
|
|
||||||
const list = document.createElement('ul');
|
const list = appDialogsManager.createChatList();
|
||||||
this.scrollable.container.classList.add('chatlist-container');
|
this.scrollable.container.classList.add('chatlist-container');
|
||||||
this.scrollable.append(list);
|
this.scrollable.append(list);
|
||||||
|
|
||||||
@ -122,7 +122,7 @@ export default class AppBlockedUsersTab extends SliderSuperTab {
|
|||||||
add(peerId, true);
|
add(peerId, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(res.peerIds.length < LOAD_COUNT) {
|
if(res.peerIds.length < LOAD_COUNT || list.childElementCount === res.count) {
|
||||||
this.scrollable.onScrolledBottom = null;
|
this.scrollable.onScrolledBottom = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ export default class AppContactsTab extends SliderSuperTab {
|
|||||||
init() {
|
init() {
|
||||||
this.container.id = 'contacts-container';
|
this.container.id = 'contacts-container';
|
||||||
|
|
||||||
this.list = document.createElement('ul');
|
this.list = appDialogsManager.createChatList(/* {avatarSize: 48, handheldsSize: 66} */);
|
||||||
this.list.id = 'contacts';
|
this.list.id = 'contacts';
|
||||||
this.list.classList.add('contacts-container');
|
this.list.classList.add('contacts-container');
|
||||||
|
|
||||||
|
@ -254,7 +254,7 @@ export default class AppEditFolderTab extends SliderSuperTab {
|
|||||||
|
|
||||||
(['include_peers', 'exclude_peers'] as ['include_peers', 'exclude_peers']).forEach(key => {
|
(['include_peers', 'exclude_peers'] as ['include_peers', 'exclude_peers']).forEach(key => {
|
||||||
const container = this[key];
|
const container = this[key];
|
||||||
const ul = document.createElement('ul');
|
const ul = appDialogsManager.createChatList();
|
||||||
|
|
||||||
const peers = filter[key].slice();
|
const peers = filter[key].slice();
|
||||||
|
|
||||||
|
@ -113,7 +113,7 @@ export default class AppEditGroupTab extends SliderSuperTab {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const setPermissionsLength = () => {
|
const setPermissionsLength = () => {
|
||||||
permissionsRow.subtitle.innerHTML = flags.reduce((acc, f) => acc + +appChatsManager.hasRights(this.chatId, f, 0), 0) + '/' + flags.length;
|
permissionsRow.subtitle.innerHTML = flags.reduce((acc, f) => acc + +appChatsManager.hasRights(this.chatId, f, chat.default_banned_rights), 0) + '/' + flags.length;
|
||||||
};
|
};
|
||||||
|
|
||||||
setPermissionsLength();
|
setPermissionsLength();
|
||||||
@ -180,7 +180,8 @@ export default class AppEditGroupTab extends SliderSuperTab {
|
|||||||
|
|
||||||
if(appChatsManager.hasRights(this.chatId, 'change_permissions')) {
|
if(appChatsManager.hasRights(this.chatId, 'change_permissions')) {
|
||||||
const showChatHistoryCheckboxField = new CheckboxField({
|
const showChatHistoryCheckboxField = new CheckboxField({
|
||||||
text: 'Show chat history for new members'
|
text: 'Show chat history for new members',
|
||||||
|
withRipple: true
|
||||||
});
|
});
|
||||||
|
|
||||||
if(appChatsManager.isChannel(this.chatId) && !(chatFull as ChatFull.channelFull).pFlags.hidden_prehistory) {
|
if(appChatsManager.isChannel(this.chatId) && !(chatFull as ChatFull.channelFull).pFlags.hidden_prehistory) {
|
||||||
|
@ -1,10 +1,116 @@
|
|||||||
|
import { attachClickEvent, cancelEvent, findUpTag } from "../../../helpers/dom";
|
||||||
import ListenerSetter from "../../../helpers/listenerSetter";
|
import ListenerSetter from "../../../helpers/listenerSetter";
|
||||||
import { Chat, ChatBannedRights } from "../../../layer";
|
import ScrollableLoader from "../../../helpers/listLoader";
|
||||||
|
import { ChannelParticipant, Chat, ChatBannedRights, Update } from "../../../layer";
|
||||||
import appChatsManager, { ChatRights } from "../../../lib/appManagers/appChatsManager";
|
import appChatsManager, { ChatRights } from "../../../lib/appManagers/appChatsManager";
|
||||||
|
import appDialogsManager from "../../../lib/appManagers/appDialogsManager";
|
||||||
|
import appProfileManager from "../../../lib/appManagers/appProfileManager";
|
||||||
|
import appUsersManager from "../../../lib/appManagers/appUsersManager";
|
||||||
|
import rootScope from "../../../lib/rootScope";
|
||||||
import CheckboxField from "../../checkboxField";
|
import CheckboxField from "../../checkboxField";
|
||||||
|
import PopupPickUser from "../../popups/pickUser";
|
||||||
import Row from "../../row";
|
import Row from "../../row";
|
||||||
import { SettingSection } from "../../sidebarLeft";
|
import { SettingSection } from "../../sidebarLeft";
|
||||||
import { SliderSuperTabEventable } from "../../sliderTab";
|
import { SliderSuperTabEventable } from "../../sliderTab";
|
||||||
|
import { toast } from "../../toast";
|
||||||
|
import AppUserPermissionsTab from "./userPermissions";
|
||||||
|
|
||||||
|
export class ChatPermissions {
|
||||||
|
public v: Array<{
|
||||||
|
flags: ChatRights[],
|
||||||
|
text: string,
|
||||||
|
checkboxField?: CheckboxField
|
||||||
|
}>;
|
||||||
|
private toggleWith: Partial<{[chatRight in ChatRights]: ChatRights[]}>;
|
||||||
|
|
||||||
|
constructor(options: {
|
||||||
|
chatId: number,
|
||||||
|
listenerSetter: ListenerSetter,
|
||||||
|
appendTo: HTMLElement,
|
||||||
|
participant?: ChannelParticipant.channelParticipantBanned
|
||||||
|
}) {
|
||||||
|
this.v = [
|
||||||
|
{flags: ['send_messages'], text: 'Send Messages'},
|
||||||
|
{flags: ['send_media'], text: 'Send Media'},
|
||||||
|
{flags: ['send_stickers', 'send_gifs'], text: 'Send Stickers & GIFs'},
|
||||||
|
{flags: ['send_polls'], text: 'Send Polls'},
|
||||||
|
{flags: ['embed_links'], text: 'Send Links'},
|
||||||
|
{flags: ['invite_users'], text: 'Add Users'},
|
||||||
|
{flags: ['pin_messages'], text: 'Pin Messages'},
|
||||||
|
{flags: ['change_info'], text: 'Change Chat Info'}
|
||||||
|
];
|
||||||
|
|
||||||
|
this.toggleWith = {
|
||||||
|
'send_messages': ['send_media', 'send_stickers', 'send_polls', 'embed_links']
|
||||||
|
};
|
||||||
|
|
||||||
|
const chat: Chat.chat = appChatsManager.getChat(options.chatId);
|
||||||
|
const defaultBannedRights = chat.default_banned_rights;
|
||||||
|
const rights = options.participant ? appChatsManager.combineParticipantBannedRights(options.chatId, options.participant.banned_rights) : defaultBannedRights;
|
||||||
|
|
||||||
|
for(const info of this.v) {
|
||||||
|
const mainFlag = info.flags[0];
|
||||||
|
info.checkboxField = new CheckboxField({
|
||||||
|
text: info.text,
|
||||||
|
checked: appChatsManager.hasRights(options.chatId, mainFlag, rights),
|
||||||
|
restriction: true,
|
||||||
|
withRipple: true
|
||||||
|
});
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
if(options.participant && defaultBannedRights.pFlags[mainFlag]) {
|
||||||
|
info.checkboxField.input.disabled = true;
|
||||||
|
|
||||||
|
/* options.listenerSetter.add(info.checkboxField.input, 'change', (e) => {
|
||||||
|
if(!e.isTrusted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
cancelEvent(e);
|
||||||
|
toast('This option is disabled for all members in Group Permissions.');
|
||||||
|
info.checkboxField.checked = false;
|
||||||
|
}); */
|
||||||
|
|
||||||
|
attachClickEvent(info.checkboxField.label, (e) => {
|
||||||
|
toast('This option is disabled for all members in Group Permissions.');
|
||||||
|
}, {listenerSetter: options.listenerSetter});
|
||||||
|
}
|
||||||
|
|
||||||
|
if(this.toggleWith[mainFlag]) {
|
||||||
|
options.listenerSetter.add(info.checkboxField.input, 'change', () => {
|
||||||
|
if(!info.checkboxField.checked) {
|
||||||
|
const other = this.v.filter(i => this.toggleWith[mainFlag].includes(i.flags[0]));
|
||||||
|
other.forEach(info => {
|
||||||
|
info.checkboxField.checked = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
options.appendTo.append(info.checkboxField.label);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public takeOut() {
|
||||||
|
const rights: ChatBannedRights = {
|
||||||
|
_: 'chatBannedRights',
|
||||||
|
until_date: 0x7FFFFFFF,
|
||||||
|
pFlags: {}
|
||||||
|
};
|
||||||
|
|
||||||
|
for(const info of this.v) {
|
||||||
|
const banned = !info.checkboxField.checked;
|
||||||
|
if(banned) {
|
||||||
|
info.flags.forEach(flag => {
|
||||||
|
// @ts-ignore
|
||||||
|
rights.pFlags[flag] = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rights;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default class AppGroupPermissionsTab extends SliderSuperTabEventable {
|
export default class AppGroupPermissionsTab extends SliderSuperTabEventable {
|
||||||
public chatId: number;
|
public chatId: number;
|
||||||
@ -13,93 +119,20 @@ export default class AppGroupPermissionsTab extends SliderSuperTabEventable {
|
|||||||
this.container.classList.add('edit-peer-container', 'group-permissions-container');
|
this.container.classList.add('edit-peer-container', 'group-permissions-container');
|
||||||
this.title.innerHTML = 'Permissions';
|
this.title.innerHTML = 'Permissions';
|
||||||
|
|
||||||
class ChatPermissions {
|
let chatPermissions: ChatPermissions;
|
||||||
private v: Array<{
|
|
||||||
flags: ChatRights[],
|
|
||||||
text: string,
|
|
||||||
checkboxField?: CheckboxField
|
|
||||||
}>;
|
|
||||||
private toggleWith: Partial<{[chatRight in ChatRights]: ChatRights[]}>;
|
|
||||||
|
|
||||||
constructor(options: {
|
|
||||||
chatId: number,
|
|
||||||
listenerSetter: ListenerSetter,
|
|
||||||
appendTo: HTMLElement,
|
|
||||||
userId: number
|
|
||||||
}) {
|
|
||||||
this.v = [
|
|
||||||
{flags: ['send_messages'], text: 'Send Messages'},
|
|
||||||
{flags: ['send_media'], text: 'Send Media'},
|
|
||||||
{flags: ['send_stickers', 'send_gifs'], text: 'Send Stickers & GIFs'},
|
|
||||||
{flags: ['send_polls'], text: 'Send Polls'},
|
|
||||||
{flags: ['embed_links'], text: 'Send Links'},
|
|
||||||
{flags: ['invite_users'], text: 'Add Users'},
|
|
||||||
{flags: ['pin_messages'], text: 'Pin Messages'},
|
|
||||||
{flags: ['change_info'], text: 'Change Chat Info'}
|
|
||||||
];
|
|
||||||
|
|
||||||
this.toggleWith = {
|
|
||||||
'send_messages': ['send_media', 'send_stickers', 'send_polls', 'embed_links']
|
|
||||||
};
|
|
||||||
|
|
||||||
for(const info of this.v) {
|
|
||||||
const mainFlag = info.flags[0];
|
|
||||||
info.checkboxField = new CheckboxField({
|
|
||||||
text: info.text,
|
|
||||||
checked: appChatsManager.hasRights(options.chatId, mainFlag, options.userId),
|
|
||||||
restriction: true
|
|
||||||
});
|
|
||||||
|
|
||||||
if(this.toggleWith[mainFlag]) {
|
|
||||||
options.listenerSetter.add(info.checkboxField.input, 'change', () => {
|
|
||||||
if(!info.checkboxField.checked) {
|
|
||||||
const other = this.v.filter(i => this.toggleWith[mainFlag].includes(i.flags[0]));
|
|
||||||
other.forEach(info => {
|
|
||||||
info.checkboxField.checked = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
options.appendTo.append(info.checkboxField.label);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public takeOut() {
|
|
||||||
const rights: ChatBannedRights = {
|
|
||||||
_: 'chatBannedRights',
|
|
||||||
until_date: 0x7FFFFFFF,
|
|
||||||
pFlags: {}
|
|
||||||
};
|
|
||||||
|
|
||||||
for(const info of this.v) {
|
|
||||||
const banned = !info.checkboxField.checked;
|
|
||||||
if(banned) {
|
|
||||||
info.flags.forEach(flag => {
|
|
||||||
// @ts-ignore
|
|
||||||
rights.pFlags[flag] = true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return rights;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
{
|
||||||
const section = new SettingSection({
|
const section = new SettingSection({
|
||||||
name: 'What can members of this group do?',
|
name: 'What can members of this group do?',
|
||||||
});
|
});
|
||||||
|
|
||||||
const p = new ChatPermissions({
|
chatPermissions = new ChatPermissions({
|
||||||
chatId: this.chatId,
|
chatId: this.chatId,
|
||||||
listenerSetter: this.listenerSetter,
|
listenerSetter: this.listenerSetter,
|
||||||
appendTo: section.content,
|
appendTo: section.content,
|
||||||
userId: 0
|
|
||||||
});
|
});
|
||||||
|
|
||||||
this.eventListener.addEventListener('destroy', () => {
|
this.eventListener.addEventListener('destroy', () => {
|
||||||
appChatsManager.editChatDefaultBannedRights(this.chatId, p.takeOut());
|
appChatsManager.editChatDefaultBannedRights(this.chatId, chatPermissions.takeOut());
|
||||||
});
|
});
|
||||||
|
|
||||||
this.scrollable.append(section.container);
|
this.scrollable.append(section.container);
|
||||||
@ -110,6 +143,44 @@ export default class AppGroupPermissionsTab extends SliderSuperTabEventable {
|
|||||||
name: 'Exceptions'
|
name: 'Exceptions'
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const addExceptionRow = new Row({
|
||||||
|
title: 'Add Exception',
|
||||||
|
subtitle: 'Loading...',
|
||||||
|
icon: 'adduser',
|
||||||
|
clickable: () => {
|
||||||
|
new PopupPickUser({
|
||||||
|
peerTypes: ['channelParticipants'],
|
||||||
|
onSelect: (peerId) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
openPermissions(peerId);
|
||||||
|
}, 0);
|
||||||
|
},
|
||||||
|
placeholder: 'Add Exception...',
|
||||||
|
peerId: -this.chatId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const openPermissions = async(peerId: number) => {
|
||||||
|
let participant: AppUserPermissionsTab['participant'];
|
||||||
|
try {
|
||||||
|
participant = await appProfileManager.getChannelParticipant(this.chatId, peerId) as any;
|
||||||
|
|
||||||
|
if(participant._ !== 'channelParticipantBanned') {
|
||||||
|
participant = undefined;
|
||||||
|
}
|
||||||
|
} catch(err) {
|
||||||
|
toast('User is no longer participant');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tab = new AppUserPermissionsTab(this.slider);
|
||||||
|
tab.participant = participant;
|
||||||
|
tab.chatId = this.chatId;
|
||||||
|
tab.userId = peerId;
|
||||||
|
tab.open();
|
||||||
|
};
|
||||||
|
|
||||||
const removedUsersRow = new Row({
|
const removedUsersRow = new Row({
|
||||||
title: 'Removed Users',
|
title: 'Removed Users',
|
||||||
subtitle: 'No removed users',
|
subtitle: 'No removed users',
|
||||||
@ -117,13 +188,126 @@ export default class AppGroupPermissionsTab extends SliderSuperTabEventable {
|
|||||||
clickable: true
|
clickable: true
|
||||||
});
|
});
|
||||||
|
|
||||||
section.content.append(removedUsersRow.container);
|
section.content.append(addExceptionRow.container, removedUsersRow.container);
|
||||||
|
|
||||||
const c = section.generateContentElement();
|
const c = section.generateContentElement();
|
||||||
c.classList.add('chatlist-container');
|
c.classList.add('chatlist-container');
|
||||||
|
|
||||||
|
const list = appDialogsManager.createChatList();
|
||||||
|
c.append(list);
|
||||||
|
|
||||||
|
attachClickEvent(list, (e) => {
|
||||||
|
const target = findUpTag(e.target, 'LI');
|
||||||
|
if(!target) return;
|
||||||
|
|
||||||
|
const peerId = +target.dataset.peerId;
|
||||||
|
openPermissions(peerId);
|
||||||
|
}, {listenerSetter: this.listenerSetter});
|
||||||
|
|
||||||
|
const setSubtitle = (li: Element, participant: ChannelParticipant.channelParticipantBanned) => {
|
||||||
|
const bannedRights = participant.banned_rights;//appChatsManager.combineParticipantBannedRights(this.chatId, participant.banned_rights);
|
||||||
|
const defaultBannedRights = (appChatsManager.getChat(this.chatId) as Chat.channel).default_banned_rights;
|
||||||
|
const combinedRights = appChatsManager.combineParticipantBannedRights(this.chatId, bannedRights);
|
||||||
|
|
||||||
|
const cantWhat: string[] = [], canWhat: string[] = [];
|
||||||
|
chatPermissions.v.forEach(info => {
|
||||||
|
const mainFlag = info.flags[0];
|
||||||
|
// @ts-ignore
|
||||||
|
if(bannedRights.pFlags[mainFlag] && !defaultBannedRights.pFlags[mainFlag]) {
|
||||||
|
cantWhat.push(info.text);
|
||||||
|
// @ts-ignore
|
||||||
|
} else if(!combinedRights.pFlags[mainFlag]) {
|
||||||
|
canWhat.push(info.text);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const el = li.querySelector('.user-last-message');
|
||||||
|
let str: string;
|
||||||
|
if(cantWhat.length) {
|
||||||
|
str = 'Can\'t ' + cantWhat.join(cantWhat.length === 2 ? ' and ' : ', ');
|
||||||
|
} else if(canWhat.length) {
|
||||||
|
str = 'Can ' + canWhat.join(canWhat.length === 2 ? ' and ' : ', ');
|
||||||
|
}
|
||||||
|
|
||||||
|
//const user = appUsersManager.getUser(participant.user_id);
|
||||||
|
if(str) {
|
||||||
|
el.innerHTML = str;
|
||||||
|
}
|
||||||
|
|
||||||
|
el.classList.toggle('hide', !str);
|
||||||
|
};
|
||||||
|
|
||||||
|
const add = (participant: ChannelParticipant.channelParticipantBanned, append: boolean) => {
|
||||||
|
const {dom} = appDialogsManager.addDialogNew({
|
||||||
|
dialog: participant.user_id,
|
||||||
|
container: list,
|
||||||
|
drawStatus: false,
|
||||||
|
rippleEnabled: true,
|
||||||
|
avatarSize: 48,
|
||||||
|
append
|
||||||
|
});
|
||||||
|
|
||||||
|
setSubtitle(dom.listEl, participant);
|
||||||
|
|
||||||
|
//dom.titleSpan.innerHTML = 'Chinaza Akachi';
|
||||||
|
//dom.lastMessageSpan.innerHTML = 'Can Add Users and Pin Messages';
|
||||||
|
};
|
||||||
|
|
||||||
|
this.listenerSetter.add(rootScope, 'apiUpdate', (update: Update) => {
|
||||||
|
if(update._ === 'updateChannelParticipant') {
|
||||||
|
const needAdd = update.new_participant?._ === 'channelParticipantBanned';
|
||||||
|
const li = list.querySelector(`[data-peer-id="${update.user_id}"]`);
|
||||||
|
if(needAdd) {
|
||||||
|
if(!li) {
|
||||||
|
add(update.new_participant as ChannelParticipant.channelParticipantBanned, false);
|
||||||
|
} else {
|
||||||
|
setSubtitle(li, update.new_participant as ChannelParticipant.channelParticipantBanned);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(update.prev_participant?._ !== 'channelParticipantBanned') {
|
||||||
|
++exceptionsCount;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if(li) {
|
||||||
|
li.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
if(update.prev_participant?._ === 'channelParticipantBanned') {
|
||||||
|
--exceptionsCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setLength();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const setLength = () => {
|
||||||
|
addExceptionRow.subtitle.innerHTML = exceptionsCount ? exceptionsCount + ' exceptions' : 'None';
|
||||||
|
};
|
||||||
|
|
||||||
|
let exceptionsCount = 0;
|
||||||
|
const LOAD_COUNT = 50;
|
||||||
|
const loader = new ScrollableLoader({
|
||||||
|
scrollable: this.scrollable,
|
||||||
|
getPromise: () => {
|
||||||
|
return appProfileManager.getChannelParticipants(this.chatId, {_: 'channelParticipantsBanned', q: ''}, LOAD_COUNT, list.childElementCount).then(res => {
|
||||||
|
for(const participant of res.participants) {
|
||||||
|
add(participant as ChannelParticipant.channelParticipantBanned, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
exceptionsCount = res.count;
|
||||||
|
setLength();
|
||||||
|
|
||||||
|
return res.participants.length < LOAD_COUNT || res.count === list.childElementCount;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
this.scrollable.append(section.container);
|
this.scrollable.append(section.container);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onOpenAfterTimeout() {
|
||||||
|
this.scrollable.onScroll();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -51,7 +51,7 @@ export default class AppPollResultsTab extends SliderSuperTab {
|
|||||||
answerEl.append(answerTitle, answerPercents);
|
answerEl.append(answerTitle, answerPercents);
|
||||||
|
|
||||||
// Humans
|
// Humans
|
||||||
const list = document.createElement('ul');
|
const list = appDialogsManager.createChatList();
|
||||||
list.classList.add('poll-results-voters');
|
list.classList.add('poll-results-voters');
|
||||||
|
|
||||||
appDialogsManager.setListClickListener(list, () => {
|
appDialogsManager.setListClickListener(list, () => {
|
||||||
|
95
src/components/sidebarRight/tabs/userPermissions.ts
Normal file
95
src/components/sidebarRight/tabs/userPermissions.ts
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
import { attachClickEvent } from "../../../helpers/dom";
|
||||||
|
import { deepEqual } from "../../../helpers/object";
|
||||||
|
import { ChannelParticipant } from "../../../layer";
|
||||||
|
import appChatsManager from "../../../lib/appManagers/appChatsManager";
|
||||||
|
import appDialogsManager from "../../../lib/appManagers/appDialogsManager";
|
||||||
|
import appProfileManager from "../../../lib/appManagers/appProfileManager";
|
||||||
|
import appUsersManager from "../../../lib/appManagers/appUsersManager";
|
||||||
|
import Button from "../../button";
|
||||||
|
import { SettingSection } from "../../sidebarLeft";
|
||||||
|
import { SliderSuperTabEventable } from "../../sliderTab";
|
||||||
|
import { ChatPermissions } from "./groupPermissions";
|
||||||
|
|
||||||
|
export default class AppUserPermissionsTab extends SliderSuperTabEventable {
|
||||||
|
public participant: ChannelParticipant.channelParticipantBanned;
|
||||||
|
public chatId: number;
|
||||||
|
public userId: number;
|
||||||
|
|
||||||
|
protected init() {
|
||||||
|
this.container.classList.add('edit-peer-container', 'user-permissions-container');
|
||||||
|
this.title.innerHTML = 'User Permissions';
|
||||||
|
|
||||||
|
{
|
||||||
|
const section = new SettingSection({
|
||||||
|
name: 'What can this user do?',
|
||||||
|
});
|
||||||
|
|
||||||
|
const div = document.createElement('div');
|
||||||
|
div.classList.add('chatlist-container');
|
||||||
|
section.content.insertBefore(div, section.title);
|
||||||
|
|
||||||
|
const list = appDialogsManager.createChatList();
|
||||||
|
div.append(list);
|
||||||
|
|
||||||
|
const {dom} = appDialogsManager.addDialogNew({
|
||||||
|
dialog: this.userId,
|
||||||
|
container: list,
|
||||||
|
drawStatus: false,
|
||||||
|
rippleEnabled: true,
|
||||||
|
avatarSize: 48
|
||||||
|
});
|
||||||
|
|
||||||
|
dom.lastMessageSpan.innerHTML = appUsersManager.getUserStatusString(this.userId);
|
||||||
|
|
||||||
|
const p = new ChatPermissions({
|
||||||
|
chatId: this.chatId,
|
||||||
|
listenerSetter: this.listenerSetter,
|
||||||
|
appendTo: section.content,
|
||||||
|
participant: this.participant
|
||||||
|
});
|
||||||
|
|
||||||
|
this.eventListener.addEventListener('destroy', () => {
|
||||||
|
//appChatsManager.editChatDefaultBannedRights(this.chatId, p.takeOut());
|
||||||
|
const rights = p.takeOut();
|
||||||
|
if(deepEqual(this.participant.banned_rights.pFlags, rights.pFlags)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
appChatsManager.editBanned(this.chatId, this.participant, rights);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.scrollable.append(section.container);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const section = new SettingSection({});
|
||||||
|
|
||||||
|
const btnDelete = Button('btn-primary btn-transparent danger', {icon: 'deleteuser', text: 'Ban and Remove From Group'});
|
||||||
|
|
||||||
|
attachClickEvent(btnDelete, () => {
|
||||||
|
/* new PopupPeer('popup-delete-group', {
|
||||||
|
peerId: -this.chatId,
|
||||||
|
title: 'Delete Group?',
|
||||||
|
description: `Are you sure you want to delete this group? All members will be removed, and all messages will be lost.`,
|
||||||
|
buttons: addCancelButton([{
|
||||||
|
text: 'DELETE',
|
||||||
|
callback: () => {
|
||||||
|
toggleDisability([btnDelete], true);
|
||||||
|
|
||||||
|
appChatsManager.deleteChannel(this.chatId).then(() => {
|
||||||
|
this.close();
|
||||||
|
}, () => {
|
||||||
|
toggleDisability([btnDelete], false);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
isDanger: true
|
||||||
|
}])
|
||||||
|
}).show(); */
|
||||||
|
}, {listenerSetter: this.listenerSetter});
|
||||||
|
|
||||||
|
section.content.append(btnDelete);
|
||||||
|
|
||||||
|
this.scrollable.append(section.container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
28
src/helpers/listLoader.ts
Normal file
28
src/helpers/listLoader.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import Scrollable from "../components/scrollable";
|
||||||
|
|
||||||
|
export default class ScrollableLoader {
|
||||||
|
constructor(options: {
|
||||||
|
scrollable: Scrollable,
|
||||||
|
getPromise: () => Promise<any>
|
||||||
|
}) {
|
||||||
|
let loading = false;
|
||||||
|
options.scrollable.onScrolledBottom = () => {
|
||||||
|
if(loading) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
loading = true;
|
||||||
|
options.getPromise().then(done => {
|
||||||
|
loading = false;
|
||||||
|
|
||||||
|
if(done) {
|
||||||
|
options.scrollable.onScrolledBottom = null;
|
||||||
|
} else {
|
||||||
|
options.scrollable.checkForTriggers();
|
||||||
|
}
|
||||||
|
}, () => {
|
||||||
|
loading = false;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -121,7 +121,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="tabs-container" id="folders-container">
|
<div class="tabs-container" id="folders-container">
|
||||||
<div>
|
<div>
|
||||||
<ul id="dialogs"></ul>
|
<ul id="dialogs" class="chatlist chatlist-72"></ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
import { MOUNT_CLASS_TO } from "../../config/debug";
|
import { MOUNT_CLASS_TO } from "../../config/debug";
|
||||||
import { numberThousandSplitter } from "../../helpers/number";
|
import { numberThousandSplitter } from "../../helpers/number";
|
||||||
import { isObject, safeReplaceObject, copy, deepEqual } from "../../helpers/object";
|
import { isObject, safeReplaceObject, copy, deepEqual } from "../../helpers/object";
|
||||||
import { Chat, ChatAdminRights, ChatBannedRights, ChatFull, ChatParticipants, InputChannel, InputChatPhoto, InputFile, InputPeer, SendMessageAction, Update, Updates } from "../../layer";
|
import { ChannelParticipant, Chat, ChatAdminRights, ChatBannedRights, ChatFull, ChatParticipant, ChatParticipants, InputChannel, InputChatPhoto, InputFile, InputPeer, SendMessageAction, Update, Updates } from "../../layer";
|
||||||
|
import apiManagerProxy from "../mtproto/mtprotoworker";
|
||||||
import apiManager from '../mtproto/mtprotoworker';
|
import apiManager from '../mtproto/mtprotoworker';
|
||||||
import { RichTextProcessor } from "../richtextprocessor";
|
import { RichTextProcessor } from "../richtextprocessor";
|
||||||
import rootScope from "../rootScope";
|
import rootScope from "../rootScope";
|
||||||
@ -40,6 +41,13 @@ export class AppChatsManager {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'updateChannelParticipant': {
|
||||||
|
apiManagerProxy.clearCache('channels.getParticipants', (params) => {
|
||||||
|
return (params.channel as InputChannel.inputChannel).channel_id === update.channel_id;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case 'updateChatDefaultBannedRights': {
|
case 'updateChatDefaultBannedRights': {
|
||||||
const chatId = -appPeersManager.getPeerId(update.peer);
|
const chatId = -appPeersManager.getPeerId(update.peer);
|
||||||
const chat: Chat = this.getChat(chatId);
|
const chat: Chat = this.getChat(chatId);
|
||||||
@ -182,7 +190,22 @@ export class AppChatsManager {
|
|||||||
return this.chats[id] || {_: 'chatEmpty', id, deleted: true, access_hash: '', pFlags: {}/* this.channelAccess[id] */};
|
return this.chats[id] || {_: 'chatEmpty', id, deleted: true, access_hash: '', pFlags: {}/* this.channelAccess[id] */};
|
||||||
}
|
}
|
||||||
|
|
||||||
public hasRights(id: number, action: ChatRights, userId?: number) {
|
public combineParticipantBannedRights(id: number, rights: ChatBannedRights) {
|
||||||
|
const chat: Chat.channel = this.getChat(id);
|
||||||
|
|
||||||
|
if(chat.default_banned_rights) {
|
||||||
|
rights = copy(rights);
|
||||||
|
const defaultRights = chat.default_banned_rights.pFlags;
|
||||||
|
for(let i in defaultRights) {
|
||||||
|
// @ts-ignore
|
||||||
|
rights.pFlags[i] = defaultRights[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rights;
|
||||||
|
}
|
||||||
|
|
||||||
|
public hasRights(id: number, action: ChatRights, rights?: ChatAdminRights | ChatBannedRights) {
|
||||||
const chat: Chat = this.getChat(id);
|
const chat: Chat = this.getChat(id);
|
||||||
if(chat._ === 'chatEmpty') return false;
|
if(chat._ === 'chatEmpty') return false;
|
||||||
|
|
||||||
@ -193,15 +216,14 @@ export class AppChatsManager {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(userId !== undefined && appUsersManager.getSelf().id === userId) {
|
if(chat.pFlags.creator && rights === undefined) {
|
||||||
userId = undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(chat.pFlags.creator && userId === undefined) {
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
const rights = (userId === undefined && (chat.admin_rights || (chat as Chat.channel).banned_rights)) || chat.default_banned_rights;
|
if(!rights) {
|
||||||
|
rights = chat.admin_rights || (chat as Chat.channel).banned_rights || chat.default_banned_rights;
|
||||||
|
}
|
||||||
|
|
||||||
if(!rights) {
|
if(!rights) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -641,6 +663,47 @@ export class AppChatsManager {
|
|||||||
rootScope.broadcast('peer_bio_edit', -id);
|
rootScope.broadcast('peer_bio_edit', -id);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public editBanned(id: number, participant: number | ChannelParticipant, banned_rights: ChatBannedRights) {
|
||||||
|
const userId = typeof(participant) === 'number' ? participant : participant.user_id;
|
||||||
|
return apiManager.invokeApi('channels.editBanned', {
|
||||||
|
channel: this.getChannelInput(id),
|
||||||
|
user_id: appUsersManager.getUserInput(userId),
|
||||||
|
banned_rights
|
||||||
|
}).then((updates) => {
|
||||||
|
this.onChatUpdated(id, updates);
|
||||||
|
|
||||||
|
if(typeof(participant) !== 'number') {
|
||||||
|
const timestamp = Date.now() / 1000 | 0;
|
||||||
|
apiUpdatesManager.processUpdateMessage({
|
||||||
|
_: 'updateShort',
|
||||||
|
update: {
|
||||||
|
_: 'updateChannelParticipant',
|
||||||
|
channel_id: id,
|
||||||
|
date: timestamp,
|
||||||
|
//qts: 0,
|
||||||
|
user_id: userId,
|
||||||
|
prev_participant: participant,
|
||||||
|
new_participant: Object.keys(banned_rights.pFlags).length ? {
|
||||||
|
_: 'channelParticipantBanned',
|
||||||
|
date: timestamp,
|
||||||
|
banned_rights,
|
||||||
|
kicked_by: appUsersManager.getSelf().id,
|
||||||
|
user_id: userId,
|
||||||
|
pFlags: {}
|
||||||
|
} : undefined
|
||||||
|
} as Update.updateChannelParticipant
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public kickFromChannel(id: number, userId: number) {
|
||||||
|
return this.editBanned(id, userId, {
|
||||||
|
_: 'chatBannedRights',
|
||||||
|
until_date: 0
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const appChatsManager = new AppChatsManager();
|
const appChatsManager = new AppChatsManager();
|
||||||
|
@ -39,7 +39,7 @@ type DialogDom = {
|
|||||||
lastTimeSpan: HTMLSpanElement,
|
lastTimeSpan: HTMLSpanElement,
|
||||||
unreadMessagesSpan: HTMLSpanElement,
|
unreadMessagesSpan: HTMLSpanElement,
|
||||||
lastMessageSpan: HTMLSpanElement,
|
lastMessageSpan: HTMLSpanElement,
|
||||||
containerEl: HTMLDivElement,
|
containerEl: HTMLElement,
|
||||||
listEl: HTMLLIElement,
|
listEl: HTMLLIElement,
|
||||||
muteAnimationTimeout?: number
|
muteAnimationTimeout?: number
|
||||||
};
|
};
|
||||||
@ -222,7 +222,7 @@ export class AppDialogsManager {
|
|||||||
private lastActiveElements: Set<HTMLElement> = new Set();
|
private lastActiveElements: Set<HTMLElement> = new Set();
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.chatListArchived = document.createElement('ul');
|
this.chatListArchived = this.createChatList();
|
||||||
this.chatListArchived.id = 'dialogs-archived';
|
this.chatListArchived.id = 'dialogs-archived';
|
||||||
|
|
||||||
this.chatLists = {
|
this.chatLists = {
|
||||||
@ -686,7 +686,7 @@ export class AppDialogsManager {
|
|||||||
positionElementByIndex(menuTab, containerToAppend, filter.orderIndex);
|
positionElementByIndex(menuTab, containerToAppend, filter.orderIndex);
|
||||||
//containerToAppend.append(li);
|
//containerToAppend.append(li);
|
||||||
|
|
||||||
const ul = document.createElement('ul');
|
const ul = this.createChatList();
|
||||||
const div = document.createElement('div');
|
const div = document.createElement('div');
|
||||||
div.append(ul);
|
div.append(ul);
|
||||||
div.dataset.filterId = '' + filter.id;
|
div.dataset.filterId = '' + filter.id;
|
||||||
@ -914,17 +914,15 @@ export class AppDialogsManager {
|
|||||||
//cancelEvent(e);
|
//cancelEvent(e);
|
||||||
|
|
||||||
this.log('dialogs click list');
|
this.log('dialogs click list');
|
||||||
let target = e.target as HTMLElement;
|
const target = e.target as HTMLElement;
|
||||||
let elem = target.classList.contains('rp') ? target : findUpClassName(target, 'rp');
|
const elem = findUpTag(target, 'LI');
|
||||||
|
|
||||||
if(!elem) {
|
if(!elem) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
elem = elem.parentElement;
|
|
||||||
|
|
||||||
if(autonomous) {
|
if(autonomous) {
|
||||||
let sameElement = lastActiveListElement === elem;
|
const sameElement = lastActiveListElement === elem;
|
||||||
if(lastActiveListElement && !sameElement) {
|
if(lastActiveListElement && !sameElement) {
|
||||||
lastActiveListElement.classList.remove('active');
|
lastActiveListElement.classList.remove('active');
|
||||||
}
|
}
|
||||||
@ -939,8 +937,8 @@ export class AppDialogsManager {
|
|||||||
if(elem) {
|
if(elem) {
|
||||||
if(onFound) onFound();
|
if(onFound) onFound();
|
||||||
|
|
||||||
let peerId = +elem.dataset.peerId;
|
const peerId = +elem.dataset.peerId;
|
||||||
let lastMsgId = +elem.dataset.mid || undefined;
|
const lastMsgId = +elem.dataset.mid || undefined;
|
||||||
|
|
||||||
appImManager.setPeer(peerId, lastMsgId);
|
appImManager.setPeer(peerId, lastMsgId);
|
||||||
} else {
|
} else {
|
||||||
@ -963,6 +961,22 @@ export class AppDialogsManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public createChatList(/* options: {
|
||||||
|
avatarSize?: number,
|
||||||
|
handheldsSize?: number,
|
||||||
|
//size?: number,
|
||||||
|
} = {} */) {
|
||||||
|
const list = document.createElement('ul');
|
||||||
|
list.classList.add('chatlist'/* ,
|
||||||
|
'chatlist-avatar-' + (options.avatarSize || 54) *//* , 'chatlist-' + (options.size || 72) */);
|
||||||
|
|
||||||
|
/* if(options.handheldsSize) {
|
||||||
|
list.classList.add('chatlist-handhelds-' + options.handheldsSize);
|
||||||
|
} */
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
private reorderDialogs() {
|
private reorderDialogs() {
|
||||||
//const perf = performance.now();
|
//const perf = performance.now();
|
||||||
if(this.reorderDialogsTimeout) {
|
if(this.reorderDialogsTimeout) {
|
||||||
@ -1291,16 +1305,12 @@ export class AppDialogsManager {
|
|||||||
//captionDiv.append(titleSpan);
|
//captionDiv.append(titleSpan);
|
||||||
//captionDiv.append(span);
|
//captionDiv.append(span);
|
||||||
|
|
||||||
const paddingDiv = document.createElement('div');
|
const li = document.createElement('li');
|
||||||
paddingDiv.classList.add('rp');
|
|
||||||
paddingDiv.append(avatarEl, captionDiv);
|
|
||||||
|
|
||||||
if(rippleEnabled) {
|
if(rippleEnabled) {
|
||||||
ripple(paddingDiv);
|
ripple(li);
|
||||||
}
|
}
|
||||||
|
|
||||||
const li = document.createElement('li');
|
li.append(avatarEl, captionDiv);
|
||||||
li.append(paddingDiv);
|
|
||||||
li.dataset.peerId = '' + peerId;
|
li.dataset.peerId = '' + peerId;
|
||||||
|
|
||||||
const statusSpan = document.createElement('span');
|
const statusSpan = document.createElement('span');
|
||||||
@ -1335,7 +1345,7 @@ export class AppDialogsManager {
|
|||||||
lastTimeSpan,
|
lastTimeSpan,
|
||||||
unreadMessagesSpan,
|
unreadMessagesSpan,
|
||||||
lastMessageSpan: span,
|
lastMessageSpan: span,
|
||||||
containerEl: paddingDiv,
|
containerEl: li,
|
||||||
listEl: li
|
listEl: li
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -53,6 +53,21 @@ export class ApiManagerProxy extends CryptoWorkerMethods {
|
|||||||
|
|
||||||
private hashes: {[method: string]: HashOptions} = {};
|
private hashes: {[method: string]: HashOptions} = {};
|
||||||
|
|
||||||
|
private apiPromisesSingle: {
|
||||||
|
[q: string]: Promise<any>
|
||||||
|
} = {};
|
||||||
|
private apiPromisesCacheable: {
|
||||||
|
[method: string]: {
|
||||||
|
[queryJSON: string]: {
|
||||||
|
timestamp: number,
|
||||||
|
promise: Promise<any>,
|
||||||
|
fulfilled: boolean,
|
||||||
|
timeout?: number,
|
||||||
|
params: any
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} = {};
|
||||||
|
|
||||||
private isSWRegistered = true;
|
private isSWRegistered = true;
|
||||||
|
|
||||||
private debug = DEBUG /* && false */;
|
private debug = DEBUG /* && false */;
|
||||||
@ -354,6 +369,71 @@ export class ApiManagerProxy extends CryptoWorkerMethods {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public invokeApiSingle<T extends keyof MethodDeclMap>(method: T, params: MethodDeclMap[T]['req'] = {} as any, options: InvokeApiOptions = {}): Promise<MethodDeclMap[T]['res']> {
|
||||||
|
const q = method + '-' + JSON.stringify(params);
|
||||||
|
if(this.apiPromisesSingle[q]) {
|
||||||
|
return this.apiPromisesSingle[q];
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.apiPromisesSingle[q] = this.invokeApi(method, params, options).finally(() => {
|
||||||
|
delete this.apiPromisesSingle[q];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public invokeApiCacheable<T extends keyof MethodDeclMap>(method: T, params: MethodDeclMap[T]['req'] = {} as any, options: InvokeApiOptions & Partial<{cacheSeconds: number, override: boolean}> = {}): Promise<MethodDeclMap[T]['res']> {
|
||||||
|
const cache = this.apiPromisesCacheable[method] ?? (this.apiPromisesCacheable[method] = {});
|
||||||
|
const queryJSON = JSON.stringify(params);
|
||||||
|
const item = cache[queryJSON];
|
||||||
|
if(item && (!options.override || !item.fulfilled)) {
|
||||||
|
return item.promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(options.override) {
|
||||||
|
if(item && item.timeout) {
|
||||||
|
clearTimeout(item.timeout);
|
||||||
|
delete item.timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete options.override;
|
||||||
|
}
|
||||||
|
|
||||||
|
let timeout: number;
|
||||||
|
if(options.cacheSeconds) {
|
||||||
|
timeout = window.setTimeout(() => {
|
||||||
|
delete cache[queryJSON];
|
||||||
|
}, options.cacheSeconds * 1000);
|
||||||
|
delete options.cacheSeconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
const promise = this.invokeApi(method, params, options);
|
||||||
|
|
||||||
|
cache[queryJSON] = {
|
||||||
|
timestamp: Date.now(),
|
||||||
|
fulfilled: false,
|
||||||
|
timeout,
|
||||||
|
promise,
|
||||||
|
params
|
||||||
|
};
|
||||||
|
|
||||||
|
return promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
public clearCache<T extends keyof MethodDeclMap>(method: T, verify: (params: MethodDeclMap[T]['req']) => boolean) {
|
||||||
|
const cache = this.apiPromisesCacheable[method];
|
||||||
|
if(cache) {
|
||||||
|
for(const queryJSON in cache) {
|
||||||
|
const item = cache[queryJSON];
|
||||||
|
if(verify(item.params)) {
|
||||||
|
if(item.timeout) {
|
||||||
|
clearTimeout(item.timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
delete cache[queryJSON];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* private computeHash(smth: any[]) {
|
/* private computeHash(smth: any[]) {
|
||||||
smth = smth.slice().sort((a, b) => a.id - b.id);
|
smth = smth.slice().sort((a, b) => a.id - b.id);
|
||||||
//return smth.reduce((hash, v) => (((hash * 0x4F25) & 0x7FFFFFFF) + v.id) & 0x7FFFFFFF, 0);
|
//return smth.reduce((hash, v) => (((hash * 0x4F25) & 0x7FFFFFFF) + v.id) & 0x7FFFFFFF, 0);
|
||||||
|
@ -7,25 +7,124 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ul {
|
.search-group {
|
||||||
margin: 0;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
position: relative;
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
//border-bottom: 1px solid #DADCE0;
|
||||||
|
padding: 1rem 0 .5rem;
|
||||||
|
margin-bottom: 17px;
|
||||||
|
|
||||||
user-select: none;
|
@include respond-to(handhelds) {
|
||||||
-webkit-user-select: none; /* disable selection/Copy of UIWebView */
|
margin-bottom: 0;
|
||||||
-webkit-touch-callout: none; /* disable the IOS popup when long-press on a link */
|
}
|
||||||
|
|
||||||
|
&__name {
|
||||||
|
color: $color-gray;
|
||||||
|
padding: 0 23px;
|
||||||
|
padding-bottom: 1rem;
|
||||||
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
@include respond-to(handhelds) {
|
||||||
|
padding: 5px 9px 0 16px;
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&-contacts {
|
||||||
|
border-bottom: 1px solid #dadce0;
|
||||||
|
|
||||||
|
@include respond-to(handhelds) {
|
||||||
|
padding: 0px 0 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
// .search-group__name {
|
||||||
|
// padding-bottom: 17px;
|
||||||
|
|
||||||
|
// @include respond-to(handhelds) {
|
||||||
|
// padding-bottom: 0;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
|
||||||
|
&-people.search-group-contacts {
|
||||||
|
padding: 5px 0 5px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.search-super {
|
||||||
|
.search-group {
|
||||||
|
margin-bottom: 0px;
|
||||||
|
padding: 4px 0 0;
|
||||||
|
|
||||||
|
&__name {
|
||||||
|
padding-top: 1rem;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.chatlist {
|
||||||
|
padding: 0 .5rem;
|
||||||
|
|
||||||
|
@include respond-to(handhelds) {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatlist {
|
||||||
|
//--avatarSize: 54px;
|
||||||
|
//--height: 72px;
|
||||||
|
margin: 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
user-select: none;
|
||||||
|
-webkit-user-select: none; /* disable selection/Copy of UIWebView */
|
||||||
|
-webkit-touch-callout: none; /* disable the IOS popup when long-press on a link */
|
||||||
|
|
||||||
|
/* &.chatlist-avatar-48 {
|
||||||
|
--avatarSize: 48px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include respond-to(handhelds) {
|
||||||
|
&.chatlist-handhelds-66 {
|
||||||
|
--height: 66px;
|
||||||
|
}
|
||||||
|
} */
|
||||||
|
|
||||||
li {
|
li {
|
||||||
background-color: #fff;
|
background-color: #fff;
|
||||||
|
//height: var(--height);
|
||||||
|
height: 72px;
|
||||||
|
//max-height: var(--height);
|
||||||
|
border-radius: $border-radius-medium;
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start; // TODO: проверить разницу в производительности с align-items: center;
|
||||||
|
flex-direction: row;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 9px 8.5px;
|
||||||
|
/* padding-top: calc((var(--height) - var(--avatarSize)) / 2);
|
||||||
|
padding-bottom: calc((var(--height) - var(--avatarSize)) / 2);
|
||||||
|
padding-right: 8.5px;
|
||||||
|
padding-left: 8.5px; */
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
padding-bottom: 0px;
|
border-radius: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@include hover-background-effect();
|
||||||
|
|
||||||
&.is-muted {
|
&.is-muted {
|
||||||
.user-title {
|
.user-title {
|
||||||
&:after {
|
&:after {
|
||||||
@ -87,44 +186,15 @@
|
|||||||
margin-left: 0;
|
margin-left: 0;
|
||||||
}
|
}
|
||||||
} */
|
} */
|
||||||
}
|
|
||||||
|
|
||||||
li > .rp {
|
&.menu-open {
|
||||||
height: 72px;
|
|
||||||
max-height: 72px;
|
|
||||||
border-radius: $border-radius-medium;
|
|
||||||
display: flex;
|
|
||||||
align-items: flex-start; // TODO: проверить разницу в производительности с align-items: center;
|
|
||||||
flex-direction: row;
|
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 9px 8.5px;
|
|
||||||
margin: 0 8px;
|
|
||||||
overflow: hidden;
|
|
||||||
|
|
||||||
/* html.is-safari & {
|
|
||||||
margin-right: 3px;
|
|
||||||
} */
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
padding: 9px 12px 9px 9px !important;
|
|
||||||
border-radius: 0;
|
|
||||||
margin: 0;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
@include hover-background-effect();
|
|
||||||
}
|
|
||||||
|
|
||||||
li.menu-open {
|
|
||||||
> .rp {
|
|
||||||
background: var(--color-gray-hover);
|
background: var(--color-gray-hover);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@include respond-to(not-handhelds) {
|
@include respond-to(not-handhelds) {
|
||||||
li.active > .rp {
|
&.active {
|
||||||
background: var(--color-gray-hover);
|
background: var(--color-gray-hover);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -261,96 +331,25 @@
|
|||||||
li.is-muted .unread {
|
li.is-muted .unread {
|
||||||
background: #c5c9cc;
|
background: #c5c9cc;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-group {
|
|
||||||
width: 100%;
|
|
||||||
//border-bottom: 1px solid #DADCE0;
|
|
||||||
padding: 1rem 0 .5rem;
|
|
||||||
margin-bottom: 17px;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
&__name {
|
|
||||||
color: $color-gray;
|
|
||||||
padding: 0 23px;
|
|
||||||
padding-bottom: 1rem;
|
|
||||||
font-weight: 500;
|
|
||||||
user-select: none;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
padding: 5px 9px 0 16px;
|
|
||||||
font-size: 15px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&-contacts {
|
|
||||||
border-bottom: 1px solid #dadce0;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
padding: 0px 0 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
// .search-group__name {
|
|
||||||
// padding-bottom: 17px;
|
|
||||||
|
|
||||||
// @include respond-to(handhelds) {
|
|
||||||
// padding-bottom: 0;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
&-people.search-group-contacts {
|
|
||||||
padding: 5px 0 5px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:last-child {
|
|
||||||
border-bottom: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-super {
|
|
||||||
.search-group {
|
|
||||||
margin-bottom: 0px;
|
|
||||||
padding: 4px 0 0;
|
|
||||||
|
|
||||||
&__name {
|
|
||||||
padding-top: 1rem;
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// use together like class="chatlist-container contacts-container"
|
// use together like class="chatlist-container contacts-container"
|
||||||
.contacts-container, .search-group-contacts {
|
.contacts-container, .search-group-contacts {
|
||||||
li {
|
li {
|
||||||
//margin-bottom: 2px;
|
padding: .75rem;
|
||||||
padding-bottom: 4px;
|
|
||||||
padding-top: 2px;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
padding: 0;
|
height: 66px;
|
||||||
|
padding-top: 9px;
|
||||||
|
padding-bottom: 9px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
li > .rp {
|
|
||||||
padding: 9px 11.5px !important;
|
|
||||||
height: 66px;
|
|
||||||
|
|
||||||
//@include respond-to(handhelds) {
|
|
||||||
//height: 62px;
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
padding: 1px 3.5px 1px 13px;
|
padding: 1px 3.5px 1px 13px;
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
padding: 0px 4px 0px 14px;
|
padding: 0 4px 0 14px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -95,6 +95,10 @@
|
|||||||
.checkbox-ripple {
|
.checkbox-ripple {
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
border-radius: $border-radius-medium;
|
border-radius: $border-radius-medium;
|
||||||
|
|
||||||
|
.checkbox-box, .checkbox-caption {
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.checkbox-field-round {
|
.checkbox-field-round {
|
||||||
@ -168,7 +172,6 @@
|
|||||||
&::before {
|
&::before {
|
||||||
border: 2px solid #707579;
|
border: 2px solid #707579;
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
background-color: white;
|
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
transition: border-color .1s ease, opacity .1s ease;
|
transition: border-color .1s ease, opacity .1s ease;
|
||||||
}
|
}
|
||||||
|
@ -263,7 +263,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.search-group-people {
|
.search-group-people {
|
||||||
ul {
|
.chatlist {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
@ -272,13 +272,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
margin-right: 5px;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.rp {
|
|
||||||
height: 98px;
|
height: 98px;
|
||||||
max-height: 98px;
|
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
max-width: 78px;
|
max-width: 78px;
|
||||||
width: 78px;
|
width: 78px;
|
||||||
@ -286,7 +280,8 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
padding: 12px 0 0 !important;
|
padding: 12px 0 0 !important;
|
||||||
margin: 0;
|
margin: 0 5px 0 0;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
width: 77px;
|
width: 77px;
|
||||||
@ -294,7 +289,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.dialog-title-details {
|
.dialog-title-details, .dialog-subtitle {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -628,16 +623,8 @@
|
|||||||
|
|
||||||
.folder-list {
|
.folder-list {
|
||||||
li {
|
li {
|
||||||
padding-bottom: 2px;
|
padding: 9px 11px;
|
||||||
|
height: 50px;
|
||||||
.rp {
|
|
||||||
padding: 8px 11px !important;
|
|
||||||
height: 48px !important;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
padding: 8px 12px !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
@ -680,16 +667,10 @@
|
|||||||
|
|
||||||
.popup-forward, .included-chatlist-container {
|
.popup-forward, .included-chatlist-container {
|
||||||
.selector {
|
.selector {
|
||||||
ul {
|
.chatlist {
|
||||||
li > .rp {
|
li {
|
||||||
margin: 0 .5rem;
|
|
||||||
padding: 7px .75rem !important;
|
padding: 7px .75rem !important;
|
||||||
height: 3.75rem;
|
height: 3.75rem;
|
||||||
max-height: 3.75rem;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
@ -705,13 +686,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.popup-forward {
|
|
||||||
li > .rp {
|
|
||||||
height: 3.875rem !important;
|
|
||||||
max-height: 3.875rem !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.included-chatlist-container {
|
.included-chatlist-container {
|
||||||
.sidebar-left-h2 {
|
.sidebar-left-h2 {
|
||||||
padding: 6px 24px 8px 24px;
|
padding: 6px 24px 8px 24px;
|
||||||
@ -776,7 +750,9 @@
|
|||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
li {
|
li {
|
||||||
padding-top: 0;
|
height: 62px;
|
||||||
|
padding-top: 7px;
|
||||||
|
padding-bottom: 7px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
@ -791,16 +767,12 @@
|
|||||||
--size: 46px;
|
--size: 46px;
|
||||||
--multiplier: 1.173913;
|
--multiplier: 1.173913;
|
||||||
}
|
}
|
||||||
|
|
||||||
li > .rp {
|
|
||||||
height: 62px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
@include respond-to(handhelds) {
|
||||||
.search-group-recent.search-group.search-group-contacts ul {
|
.search-group-recent.search-group.search-group-contacts ul {
|
||||||
margin-top: -2px;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-group.search-group-contacts ul, .search-group.search-group-messages ul {
|
.search-group.search-group-contacts ul, .search-group.search-group-messages ul {
|
||||||
@ -1045,9 +1017,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.blocked-users-container {
|
.blocked-users-container {
|
||||||
li > .rp {
|
li {
|
||||||
height: 66px;
|
height: 66px;
|
||||||
max-height: 66px;
|
padding-top: 9px;
|
||||||
|
padding-bottom: 9px;
|
||||||
|
|
||||||
|
border-radius: $border-radius-medium;
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
@ -1061,7 +1036,7 @@
|
|||||||
|
|
||||||
ul {
|
ul {
|
||||||
margin-top: .3125rem;
|
margin-top: .3125rem;
|
||||||
padding: 0 3px;
|
padding: 0 .6875rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -696,7 +696,6 @@
|
|||||||
color: #707579;
|
color: #707579;
|
||||||
padding: 0 16px 8px 16px;
|
padding: 0 16px 8px 16px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding-bottom: 8px;
|
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
display: flex;
|
display: flex;
|
||||||
@ -753,15 +752,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
padding-bottom: 2px;
|
height: 50px;
|
||||||
|
padding: 9px;
|
||||||
|
|
||||||
> .rp {
|
@include respond-to(not-handhelds) {
|
||||||
padding: 8px 5px;
|
padding: 9px 12px;
|
||||||
height: 48px;
|
|
||||||
|
|
||||||
@include respond-to(not-handhelds) {
|
|
||||||
padding: 8px 12px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -789,8 +784,35 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.checkbox-field {
|
// * supernew and correct layout
|
||||||
margin: 0 1.1875rem;
|
.chatlist {
|
||||||
|
padding: 0;
|
||||||
|
|
||||||
|
li {
|
||||||
|
height: 72px;
|
||||||
|
padding: 0 .75rem;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-caption {
|
||||||
|
padding-left: .75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
span {
|
||||||
|
line-height: 1.3125;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dialog-subtitle {
|
||||||
|
margin-top: .125rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-last-message {
|
||||||
|
font-size: .875rem;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -817,7 +839,6 @@
|
|||||||
.group-type-container {
|
.group-type-container {
|
||||||
.sidebar-left-section-caption {
|
.sidebar-left-section-caption {
|
||||||
font-size: .875rem;
|
font-size: .875rem;
|
||||||
line-height: 1rem;
|
|
||||||
margin-top: .8125rem;
|
margin-top: .8125rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,9 +125,21 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ul {
|
.chatlist {
|
||||||
|
li {
|
||||||
|
padding-top: .75rem;
|
||||||
|
padding-bottom: .75rem;
|
||||||
|
|
||||||
|
@include respond-to(handhelds) {
|
||||||
|
height: 66px;
|
||||||
|
padding-top: 9px;
|
||||||
|
padding-bottom: 9px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.user-caption {
|
.user-caption {
|
||||||
padding: 1px 3.5px 1px 12px;
|
padding-left: .75rem;
|
||||||
|
padding-right: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
p {
|
p {
|
||||||
@ -137,25 +149,6 @@
|
|||||||
span.user-last-message {
|
span.user-last-message {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
li {
|
|
||||||
padding-bottom: 0;
|
|
||||||
|
|
||||||
> .rp {
|
|
||||||
margin: 0px 9px 0px 8px;
|
|
||||||
padding: 12px 8.5px;
|
|
||||||
|
|
||||||
@include respond-to(handhelds) {
|
|
||||||
height: 66px;
|
|
||||||
max-height: 66px;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* html.is-safari & {
|
|
||||||
margin-right: 4px;
|
|
||||||
} */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
hr {
|
hr {
|
||||||
@ -194,4 +187,4 @@
|
|||||||
--offset: 6px;
|
--offset: 6px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,14 +6,14 @@
|
|||||||
width: 420px;
|
width: 420px;
|
||||||
max-width: 420px;
|
max-width: 420px;
|
||||||
//padding: 12px 20px 32.5px;
|
//padding: 12px 20px 32.5px;
|
||||||
padding: 9px 0 0 0;
|
padding: 7px 0 0 0;
|
||||||
max-height: unquote('min(40.625rem, 100%)');
|
max-height: unquote('min(40.625rem, 100%)');
|
||||||
height: 40.625rem;
|
height: 40.625rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
&-header {
|
&-header {
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
margin-bottom: 4px;
|
margin-bottom: 3px;
|
||||||
padding: 0 1rem;
|
padding: 0 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -35,10 +35,17 @@
|
|||||||
font-size: 1.25rem;
|
font-size: 1.25rem;
|
||||||
padding: .5rem 1.5rem;
|
padding: .5rem 1.5rem;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
line-height: 1.3125;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ul li > .rp {
|
.chatlist {
|
||||||
margin-left: 0;
|
margin-top: 0 !important;
|
||||||
} */
|
|
||||||
|
li {
|
||||||
|
height: 3.875rem !important;
|
||||||
|
padding-top: .5rem !important;
|
||||||
|
padding-bottom: .5rem !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user