Emoji animations
Fix performance of extra saving of state
This commit is contained in:
parent
3006054912
commit
f9fce9116b
@ -49,6 +49,13 @@ import { MiddleEllipsisElement } from './middleEllipsis';
|
|||||||
import { joinElementsWith } from '../lib/langPack';
|
import { joinElementsWith } from '../lib/langPack';
|
||||||
import throttleWithRaf from '../helpers/schedulers/throttleWithRaf';
|
import throttleWithRaf from '../helpers/schedulers/throttleWithRaf';
|
||||||
import { NULL_PEER_ID } from '../lib/mtproto/mtproto_config';
|
import { NULL_PEER_ID } from '../lib/mtproto/mtproto_config';
|
||||||
|
import findUpClassName from '../helpers/dom/findUpClassName';
|
||||||
|
import RLottiePlayer from '../lib/rlottie/rlottiePlayer';
|
||||||
|
import assumeType from '../helpers/assumeType';
|
||||||
|
import appMessagesIdsManager from '../lib/appManagers/appMessagesIdsManager';
|
||||||
|
import throttle from '../helpers/schedulers/throttle';
|
||||||
|
import { SendMessageEmojiInteractionData } from '../types';
|
||||||
|
import IS_VIBRATE_SUPPORTED from '../environment/vibrateSupport';
|
||||||
|
|
||||||
const MAX_VIDEO_AUTOPLAY_SIZE = 50 * 1024 * 1024; // 50 MB
|
const MAX_VIDEO_AUTOPLAY_SIZE = 50 * 1024 * 1024; // 50 MB
|
||||||
|
|
||||||
@ -1127,7 +1134,7 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
loadPromises?: Promise<any>[],
|
loadPromises?: Promise<any>[],
|
||||||
needFadeIn?: boolean,
|
needFadeIn?: boolean,
|
||||||
needUpscale?: boolean
|
needUpscale?: boolean
|
||||||
}) {
|
}): Promise<RLottiePlayer | void> {
|
||||||
const stickerType = doc.sticker;
|
const stickerType = doc.sticker;
|
||||||
|
|
||||||
if(!width) {
|
if(!width) {
|
||||||
@ -1160,7 +1167,7 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
|
|
||||||
let loadThumbPromise = deferredPromise<void>();
|
let loadThumbPromise = deferredPromise<void>();
|
||||||
let haveThumbCached = false;
|
let haveThumbCached = false;
|
||||||
if((doc.thumbs?.length || doc.stickerCachedThumbs) && !div.firstElementChild && (!downloaded || stickerType === 2 || onlyThumb)/* && doc.thumbs[0]._ !== 'photoSizeEmpty' */) {
|
if((doc.thumbs?.length || doc.stickerCachedThumbs) && !div.firstElementChild && (!downloaded || stickerType === 2 || onlyThumb) && withThumb !== false/* && doc.thumbs[0]._ !== 'photoSizeEmpty' */) {
|
||||||
let thumb = doc.stickerCachedThumbs && doc.stickerCachedThumbs[toneIndex] || doc.thumbs[0];
|
let thumb = doc.stickerCachedThumbs && doc.stickerCachedThumbs[toneIndex] || doc.thumbs[0];
|
||||||
|
|
||||||
//console.log('wrap sticker', thumb, div);
|
//console.log('wrap sticker', thumb, div);
|
||||||
@ -1267,7 +1274,7 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
|
|
||||||
//appDocsManager.downloadDocNew(doc.id).promise.then(res => res.json()).then(async(json) => {
|
//appDocsManager.downloadDocNew(doc.id).promise.then(res => res.json()).then(async(json) => {
|
||||||
//fetch(doc.url).then(res => res.json()).then(async(json) => {
|
//fetch(doc.url).then(res => res.json()).then(async(json) => {
|
||||||
/* return */ await appDocsManager.downloadDoc(doc, /* undefined, */lazyLoadQueue?.queueId)
|
return await appDocsManager.downloadDoc(doc, /* undefined, */lazyLoadQueue?.queueId)
|
||||||
.then(readBlobAsText)
|
.then(readBlobAsText)
|
||||||
//.then(JSON.parse)
|
//.then(JSON.parse)
|
||||||
.then(async(json) => {
|
.then(async(json) => {
|
||||||
@ -1324,6 +1331,13 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
}, {once: true});
|
}, {once: true});
|
||||||
|
|
||||||
if(emoji) {
|
if(emoji) {
|
||||||
|
const data: SendMessageEmojiInteractionData = {
|
||||||
|
a: [],
|
||||||
|
v: 1
|
||||||
|
};
|
||||||
|
|
||||||
|
let sendInteractionThrottled: () => void;
|
||||||
|
|
||||||
attachClickEvent(div, (e) => {
|
attachClickEvent(div, (e) => {
|
||||||
cancelEvent(e);
|
cancelEvent(e);
|
||||||
let animation = LottieLoader.getAnimation(div);
|
let animation = LottieLoader.getAnimation(div);
|
||||||
@ -1332,9 +1346,134 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
animation.autoplay = true;
|
animation.autoplay = true;
|
||||||
animation.restart();
|
animation.restart();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const doc = appStickersManager.getAnimatedEmojiSticker(emoji, true);
|
||||||
|
if(!doc) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const animationDiv = document.createElement('div');
|
||||||
|
animationDiv.classList.add('emoji-animation');
|
||||||
|
|
||||||
|
const size = 280;
|
||||||
|
animationDiv.style.width = size + 'px';
|
||||||
|
animationDiv.style.height = size + 'px';
|
||||||
|
|
||||||
|
wrapSticker({
|
||||||
|
div: animationDiv,
|
||||||
|
doc,
|
||||||
|
middleware,
|
||||||
|
withThumb: false,
|
||||||
|
needFadeIn: false,
|
||||||
|
loop: false,
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
play: true,
|
||||||
|
group: 'none'
|
||||||
|
}).then(animation => {
|
||||||
|
assumeType<RLottiePlayer>(animation);
|
||||||
|
animation.addEventListener('enterFrame', (frameNo) => {
|
||||||
|
if(frameNo === animation.maxFrame) {
|
||||||
|
animation.remove();
|
||||||
|
// animationDiv.remove();
|
||||||
|
appImManager.chat.bubbles.scrollable.container.removeEventListener('scroll', onScroll);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if(IS_VIBRATE_SUPPORTED) {
|
||||||
|
animation.addEventListener('firstFrame', () => {
|
||||||
|
navigator.vibrate(100);
|
||||||
|
}, {once: true});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const generateRandomSigned = (max: number) => {
|
||||||
|
const r = Math.random() * max * 2;
|
||||||
|
return r > max ? -r % max : r;
|
||||||
|
};
|
||||||
|
|
||||||
|
const bubble = findUpClassName(div, 'bubble');
|
||||||
|
const isOut = bubble.classList.contains('is-out');
|
||||||
|
|
||||||
|
const randomOffsetX = generateRandomSigned(16);
|
||||||
|
const randomOffsetY = generateRandomSigned(4);
|
||||||
|
const stableOffsetX = size / 8 * (isOut ? 1 : -1);
|
||||||
|
const setPosition = () => {
|
||||||
|
const rect = div.getBoundingClientRect();
|
||||||
|
/* const boxWidth = Math.max(rect.width, rect.height);
|
||||||
|
const boxHeight = Math.max(rect.width, rect.height);
|
||||||
|
const x = rect.left + ((boxWidth - size) / 2);
|
||||||
|
const y = rect.top + ((boxHeight - size) / 2); */
|
||||||
|
|
||||||
|
const rectX = isOut ? rect.right : rect.left;
|
||||||
|
|
||||||
|
const addOffsetX = (isOut ? -size : 0) + stableOffsetX + randomOffsetX;
|
||||||
|
const x = rectX + addOffsetX;
|
||||||
|
// const y = rect.bottom - size + size / 4;
|
||||||
|
const y = rect.top + ((rect.height - size) / 2) + randomOffsetY;
|
||||||
|
// animationDiv.style.transform = `translate(${x}px, ${y}px)`;
|
||||||
|
animationDiv.style.top = y + 'px';
|
||||||
|
animationDiv.style.left = x + 'px';
|
||||||
|
};
|
||||||
|
|
||||||
|
const onScroll = throttleWithRaf(setPosition);
|
||||||
|
|
||||||
|
appImManager.chat.bubbles.scrollable.container.addEventListener('scroll', onScroll);
|
||||||
|
|
||||||
|
setPosition();
|
||||||
|
|
||||||
|
if(bubble) {
|
||||||
|
if(isOut) {
|
||||||
|
animationDiv.classList.add('is-out');
|
||||||
|
} else {
|
||||||
|
animationDiv.classList.add('is-in');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
appImManager.emojiAnimationContainer.append(animationDiv);
|
||||||
|
|
||||||
|
if(!sendInteractionThrottled) {
|
||||||
|
sendInteractionThrottled = throttle(() => {
|
||||||
|
const length = data.a.length;
|
||||||
|
if(!length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const firstTime = data.a[0].t;
|
||||||
|
|
||||||
|
data.a.forEach((a) => {
|
||||||
|
a.t = (a.t - firstTime) / 1000;
|
||||||
|
});
|
||||||
|
|
||||||
|
const bubble = findUpClassName(div, 'bubble');
|
||||||
|
appMessagesManager.setTyping(appImManager.chat.peerId, {
|
||||||
|
_: 'sendMessageEmojiInteraction',
|
||||||
|
msg_id: appMessagesIdsManager.getServerMessageId(+bubble.dataset.mid),
|
||||||
|
emoticon: emoji,
|
||||||
|
interaction: {
|
||||||
|
_: 'dataJSON',
|
||||||
|
data: JSON.stringify(data)
|
||||||
|
}
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
data.a.length = 0;
|
||||||
|
}, 1000, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// using a trick here: simulated event from interlocutor's interaction won't fire ours
|
||||||
|
if(e.isTrusted) {
|
||||||
|
data.a.push({
|
||||||
|
i: 1,
|
||||||
|
t: Date.now()
|
||||||
|
});
|
||||||
|
|
||||||
|
sendInteractionThrottled();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return animation;
|
||||||
|
|
||||||
//return deferred;
|
//return deferred;
|
||||||
//await new Promise((resolve) => setTimeout(resolve, 5e3));
|
//await new Promise((resolve) => setTimeout(resolve, 5e3));
|
||||||
});
|
});
|
||||||
@ -1384,12 +1523,12 @@ export function wrapSticker({doc, div, middleware, lazyLoadQueue, group, play, o
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const loadPromise: Promise<any> = lazyLoadQueue && (!downloaded || stickerType === 2) ?
|
const loadPromise: Promise<RLottiePlayer | void> = lazyLoadQueue && (!downloaded || stickerType === 2) ?
|
||||||
(lazyLoadQueue.push({div, load}), Promise.resolve()) :
|
(lazyLoadQueue.push({div, load}), Promise.resolve()) :
|
||||||
load();
|
load();
|
||||||
|
|
||||||
if(downloaded && stickerType === 1) {
|
if(downloaded && stickerType === 1) {
|
||||||
loadThumbPromise = loadPromise;
|
loadThumbPromise = loadPromise as any;
|
||||||
if(loadPromises) {
|
if(loadPromises) {
|
||||||
loadPromises.push(loadThumbPromise);
|
loadPromises.push(loadThumbPromise);
|
||||||
}
|
}
|
||||||
|
3
src/environment/vibrateSupport.ts
Normal file
3
src/environment/vibrateSupport.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
const IS_VIBRATE_SUPPORTED = !!navigator?.vibrate;
|
||||||
|
|
||||||
|
export default IS_VIBRATE_SUPPORTED;
|
@ -854,6 +854,7 @@ const lang = {
|
|||||||
"Peer.Activity.User.RecordingAudio": "recording voice",
|
"Peer.Activity.User.RecordingAudio": "recording voice",
|
||||||
"Peer.Activity.User.SendingFile": "sending file",
|
"Peer.Activity.User.SendingFile": "sending file",
|
||||||
"Peer.Activity.User.ChoosingSticker": "choosing a sticker",
|
"Peer.Activity.User.ChoosingSticker": "choosing a sticker",
|
||||||
|
"Peer.Activity.User.EnjoyingAnimations": "watching %@",
|
||||||
"Peer.Activity.Chat.PlayingGame": "%@ is playing a game",
|
"Peer.Activity.Chat.PlayingGame": "%@ is playing a game",
|
||||||
"Peer.Activity.Chat.TypingText": "%@ is typing",
|
"Peer.Activity.Chat.TypingText": "%@ is typing",
|
||||||
"Peer.Activity.Chat.SendingPhoto": "%@ is sending a photo",
|
"Peer.Activity.Chat.SendingPhoto": "%@ is sending a photo",
|
||||||
@ -862,6 +863,7 @@ const lang = {
|
|||||||
"Peer.Activity.Chat.RecordingAudio": "%@ is recording voice",
|
"Peer.Activity.Chat.RecordingAudio": "%@ is recording voice",
|
||||||
"Peer.Activity.Chat.SendingFile": "%@ is sending a file",
|
"Peer.Activity.Chat.SendingFile": "%@ is sending a file",
|
||||||
"Peer.Activity.Chat.ChoosingSticker": "%@ is choosing a sticker",
|
"Peer.Activity.Chat.ChoosingSticker": "%@ is choosing a sticker",
|
||||||
|
"Peer.Activity.Chat.EnjoyingAnimations": "%@ is watching %@",
|
||||||
"Peer.Activity.Chat.Multi.PlayingGame1": "%@ and %d others are playing a game",
|
"Peer.Activity.Chat.Multi.PlayingGame1": "%@ and %d others are playing a game",
|
||||||
"Peer.Activity.Chat.Multi.TypingText1": "%@ and %d others are typing",
|
"Peer.Activity.Chat.Multi.TypingText1": "%@ and %d others are typing",
|
||||||
"Peer.Activity.Chat.Multi.SendingPhoto1": "%@ and %d others are sending photos",
|
"Peer.Activity.Chat.Multi.SendingPhoto1": "%@ and %d others are sending photos",
|
||||||
|
@ -1899,12 +1899,10 @@ export class AppDialogsManager {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let typingElement = dom.lastMessageSpan.querySelector('.peer-typing-container') as HTMLElement;
|
const oldTypingElement = dom.lastMessageSpan.querySelector('.peer-typing-container') as HTMLElement;
|
||||||
if(typingElement) {
|
const newTypingElement = appImManager.getPeerTyping(dialog.peerId, oldTypingElement);
|
||||||
appImManager.getPeerTyping(dialog.peerId, typingElement);
|
if(!oldTypingElement && newTypingElement) {
|
||||||
} else {
|
replaceContent(dom.lastMessageSpan, newTypingElement);
|
||||||
typingElement = appImManager.getPeerTyping(dialog.peerId);
|
|
||||||
replaceContent(dom.lastMessageSpan, typingElement);
|
|
||||||
dom.lastMessageSpan.classList.add('user-typing');
|
dom.lastMessageSpan.classList.add('user-typing');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -79,7 +79,10 @@ import IS_GROUP_CALL_SUPPORTED from '../../environment/groupCallSupport';
|
|||||||
import appAvatarsManager from './appAvatarsManager';
|
import appAvatarsManager from './appAvatarsManager';
|
||||||
import IS_CALL_SUPPORTED from '../../environment/callSupport';
|
import IS_CALL_SUPPORTED from '../../environment/callSupport';
|
||||||
import { CallType } from '../calls/types';
|
import { CallType } from '../calls/types';
|
||||||
import { Modify } from '../../types';
|
import { Modify, SendMessageEmojiInteractionData } from '../../types';
|
||||||
|
import htmlToSpan from '../../helpers/dom/htmlToSpan';
|
||||||
|
import getVisibleRect from '../../helpers/dom/getVisibleRect';
|
||||||
|
import { simulateClickEvent } from '../../helpers/dom/clickEvent';
|
||||||
|
|
||||||
//console.log('appImManager included33!');
|
//console.log('appImManager included33!');
|
||||||
|
|
||||||
@ -127,6 +130,7 @@ export class AppImManager {
|
|||||||
private backgroundPromises: {[slug: string]: Promise<string>} = {};
|
private backgroundPromises: {[slug: string]: Promise<string>} = {};
|
||||||
|
|
||||||
private topbarCall: TopbarCall;
|
private topbarCall: TopbarCall;
|
||||||
|
emojiAnimationContainer: HTMLDivElement;
|
||||||
|
|
||||||
get myId() {
|
get myId() {
|
||||||
return rootScope.myId;
|
return rootScope.myId;
|
||||||
@ -177,6 +181,10 @@ export class AppImManager {
|
|||||||
this.chatsContainer.classList.add('chats-container', 'tabs-container');
|
this.chatsContainer.classList.add('chats-container', 'tabs-container');
|
||||||
this.chatsContainer.dataset.animation = 'navigation';
|
this.chatsContainer.dataset.animation = 'navigation';
|
||||||
|
|
||||||
|
this.emojiAnimationContainer = document.createElement('div');
|
||||||
|
this.emojiAnimationContainer.classList.add('emoji-animation-container');
|
||||||
|
document.body.append(this.emojiAnimationContainer);
|
||||||
|
|
||||||
this.columnEl.append(this.chatsContainer);
|
this.columnEl.append(this.chatsContainer);
|
||||||
|
|
||||||
this.createNewChat();
|
this.createNewChat();
|
||||||
@ -204,6 +212,12 @@ export class AppImManager {
|
|||||||
&& document.body.classList.contains(RIGHT_COLUMN_ACTIVE_CLASSNAME)) {
|
&& document.body.classList.contains(RIGHT_COLUMN_ACTIVE_CLASSNAME)) {
|
||||||
appSidebarRight.toggleSidebar(false);
|
appSidebarRight.toggleSidebar(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(from === ScreenSize.mobile) {
|
||||||
|
document.body.append(this.emojiAnimationContainer);
|
||||||
|
} else if(to === ScreenSize.mobile) {
|
||||||
|
this.columnEl.append(this.emojiAnimationContainer);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
rootScope.addEventListener('history_focus', (e) => {
|
rootScope.addEventListener('history_focus', (e) => {
|
||||||
@ -232,6 +246,41 @@ export class AppImManager {
|
|||||||
this.setChoosingStickerTyping(!choosing);
|
this.setChoosingStickerTyping(!choosing);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
rootScope.addEventListener('peer_typings', ({peerId, typings}) => {
|
||||||
|
const chat = this.chat;
|
||||||
|
if(
|
||||||
|
!chat ||
|
||||||
|
chat.peerId !== peerId ||
|
||||||
|
rootScope.overlaysActive || (
|
||||||
|
mediaSizes.activeScreen === ScreenSize.mobile &&
|
||||||
|
this.tabId !== 1
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const typing = typings.find(typing => typing.action._ === 'sendMessageEmojiInteraction');
|
||||||
|
if(typing?.action?._ === 'sendMessageEmojiInteraction') {
|
||||||
|
const action = typing.action;
|
||||||
|
const bubble = chat.bubbles.bubbles[appMessagesIdsManager.generateMessageId(typing.action.msg_id)];
|
||||||
|
if(bubble && getVisibleRect(bubble, chat.bubbles.scrollable.container)) {
|
||||||
|
const stickerWrapper: HTMLElement = bubble.querySelector('.media-sticker-wrapper');
|
||||||
|
|
||||||
|
const data: SendMessageEmojiInteractionData = JSON.parse(action.interaction.data);
|
||||||
|
data.a.forEach(a => {
|
||||||
|
setTimeout(() => {
|
||||||
|
simulateClickEvent(stickerWrapper);
|
||||||
|
}, a.t * 1000);
|
||||||
|
});
|
||||||
|
|
||||||
|
appMessagesManager.setTyping(peerId, {
|
||||||
|
_: 'sendMessageEmojiInteractionSeen',
|
||||||
|
emoticon: action.emoticon
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
rootScope.addEventListener('instance_deactivated', () => {
|
rootScope.addEventListener('instance_deactivated', () => {
|
||||||
const popup = new PopupElement('popup-instance-deactivated', undefined, {overlayClosable: true});
|
const popup = new PopupElement('popup-instance-deactivated', undefined, {overlayClosable: true});
|
||||||
const c = document.createElement('div');
|
const c = document.createElement('div');
|
||||||
@ -1599,6 +1648,7 @@ export class AppImManager {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 'sendMessageEmojiInteractionSeen':
|
||||||
case 'sendMessageChooseStickerAction': {
|
case 'sendMessageChooseStickerAction': {
|
||||||
c += '-choosing-sticker';
|
c += '-choosing-sticker';
|
||||||
for(let i = 0; i < 2; ++i) {
|
for(let i = 0; i < 2; ++i) {
|
||||||
@ -1638,7 +1688,8 @@ export class AppImManager {
|
|||||||
'sendMessageRecordAudioAction': 'Peer.Activity.User.RecordingAudio',
|
'sendMessageRecordAudioAction': 'Peer.Activity.User.RecordingAudio',
|
||||||
'sendMessageRecordRoundAction': 'Peer.Activity.User.RecordingVideo',
|
'sendMessageRecordRoundAction': 'Peer.Activity.User.RecordingVideo',
|
||||||
'sendMessageGamePlayAction': 'Peer.Activity.User.PlayingGame',
|
'sendMessageGamePlayAction': 'Peer.Activity.User.PlayingGame',
|
||||||
'sendMessageChooseStickerAction': 'Peer.Activity.User.ChoosingSticker'
|
'sendMessageChooseStickerAction': 'Peer.Activity.User.ChoosingSticker',
|
||||||
|
'sendMessageEmojiInteractionSeen': 'Peer.Activity.User.EnjoyingAnimations'
|
||||||
},
|
},
|
||||||
chat: {
|
chat: {
|
||||||
'sendMessageTypingAction': 'Peer.Activity.Chat.TypingText',
|
'sendMessageTypingAction': 'Peer.Activity.Chat.TypingText',
|
||||||
@ -1651,7 +1702,8 @@ export class AppImManager {
|
|||||||
'sendMessageRecordAudioAction': 'Peer.Activity.Chat.RecordingAudio',
|
'sendMessageRecordAudioAction': 'Peer.Activity.Chat.RecordingAudio',
|
||||||
'sendMessageRecordRoundAction': 'Peer.Activity.Chat.RecordingVideo',
|
'sendMessageRecordRoundAction': 'Peer.Activity.Chat.RecordingVideo',
|
||||||
'sendMessageGamePlayAction': 'Peer.Activity.Chat.PlayingGame',
|
'sendMessageGamePlayAction': 'Peer.Activity.Chat.PlayingGame',
|
||||||
'sendMessageChooseStickerAction': 'Peer.Activity.Chat.ChoosingSticker'
|
'sendMessageChooseStickerAction': 'Peer.Activity.Chat.ChoosingSticker',
|
||||||
|
'sendMessageEmojiInteractionSeen': 'Peer.Activity.Chat.EnjoyingAnimations'
|
||||||
},
|
},
|
||||||
multi: {
|
multi: {
|
||||||
'sendMessageTypingAction': 'Peer.Activity.Chat.Multi.TypingText1',
|
'sendMessageTypingAction': 'Peer.Activity.Chat.Multi.TypingText1',
|
||||||
@ -1696,9 +1748,7 @@ export class AppImManager {
|
|||||||
container.classList.add('online', 'peer-typing-container');
|
container.classList.add('online', 'peer-typing-container');
|
||||||
}
|
}
|
||||||
|
|
||||||
if(action._ === 'sendMessageChooseStickerAction') {
|
container.classList.toggle('peer-typing-flex', action._ === 'sendMessageChooseStickerAction' || action._ === 'sendMessageEmojiInteractionSeen');
|
||||||
container.classList.add('peer-typing-flex');
|
|
||||||
}
|
|
||||||
|
|
||||||
let typingElement = container.firstElementChild as HTMLElement;
|
let typingElement = container.firstElementChild as HTMLElement;
|
||||||
if(!typingElement) {
|
if(!typingElement) {
|
||||||
@ -1717,6 +1767,18 @@ export class AppImManager {
|
|||||||
typings.length - 1
|
typings.length - 1
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(action._ === 'sendMessageEmojiInteractionSeen') {
|
||||||
|
if(args) {
|
||||||
|
args.pop();
|
||||||
|
} else {
|
||||||
|
args = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const span = htmlToSpan(RichTextProcessor.wrapEmojiText(action.emoticon));
|
||||||
|
args.push(span);
|
||||||
|
}
|
||||||
|
|
||||||
const descriptionElement = i18n(langPackKey, args);
|
const descriptionElement = i18n(langPackKey, args);
|
||||||
descriptionElement.classList.add('peer-typing-description');
|
descriptionElement.classList.add('peer-typing-description');
|
||||||
|
|
||||||
|
@ -201,7 +201,7 @@ export class AppMessagesManager {
|
|||||||
|
|
||||||
private groupedTempId = 0;
|
private groupedTempId = 0;
|
||||||
|
|
||||||
private typings: {[peerId: PeerId]: {type: SendMessageAction['_'], timeout?: number}} = {};
|
private typings: {[peerId: PeerId]: {action: SendMessageAction, timeout?: number}} = {};
|
||||||
|
|
||||||
private middleware: ReturnType<typeof getMiddleware>;
|
private middleware: ReturnType<typeof getMiddleware>;
|
||||||
|
|
||||||
@ -5806,13 +5806,14 @@ export class AppMessagesManager {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public setTyping(peerId: PeerId, action: SendMessageAction): Promise<boolean> {
|
public setTyping(peerId: PeerId, action: SendMessageAction, force?: boolean): Promise<boolean> {
|
||||||
let typing = this.typings[peerId];
|
let typing = this.typings[peerId];
|
||||||
if(!rootScope.myId ||
|
if(!rootScope.myId ||
|
||||||
!peerId ||
|
!peerId ||
|
||||||
!this.canSendToPeer(peerId) ||
|
!this.canSendToPeer(peerId) ||
|
||||||
peerId === rootScope.myId ||
|
peerId === rootScope.myId ||
|
||||||
typing?.type === action._
|
// (!force && deepEqual(typing?.action, action))
|
||||||
|
(!force && typing?.action?._ === action._)
|
||||||
) {
|
) {
|
||||||
return Promise.resolve(false);
|
return Promise.resolve(false);
|
||||||
}
|
}
|
||||||
@ -5822,7 +5823,7 @@ export class AppMessagesManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
typing = this.typings[peerId] = {
|
typing = this.typings[peerId] = {
|
||||||
type: action._
|
action
|
||||||
};
|
};
|
||||||
|
|
||||||
return apiManager.invokeApi('messages.setTyping', {
|
return apiManager.invokeApi('messages.setTyping', {
|
||||||
|
@ -28,6 +28,7 @@ import appPeersManager from "./appPeersManager";
|
|||||||
import appRuntimeManager from "./appRuntimeManager";
|
import appRuntimeManager from "./appRuntimeManager";
|
||||||
import appStateManager from "./appStateManager";
|
import appStateManager from "./appStateManager";
|
||||||
import appUsersManager from "./appUsersManager";
|
import appUsersManager from "./appUsersManager";
|
||||||
|
import IS_VIBRATE_SUPPORTED from "../../environment/vibrateSupport";
|
||||||
|
|
||||||
type MyNotification = Notification & {
|
type MyNotification = Notification & {
|
||||||
hidden?: boolean,
|
hidden?: boolean,
|
||||||
@ -60,7 +61,7 @@ export class AppNotificationsManager {
|
|||||||
private notificationIndex = 0;
|
private notificationIndex = 0;
|
||||||
private notificationsCount = 0;
|
private notificationsCount = 0;
|
||||||
private soundsPlayed: {[tag: string]: number} = {};
|
private soundsPlayed: {[tag: string]: number} = {};
|
||||||
private vibrateSupport = !!navigator.vibrate;
|
private vibrateSupport = IS_VIBRATE_SUPPORTED;
|
||||||
private nextSoundAt: number;
|
private nextSoundAt: number;
|
||||||
private prevSoundVolume: number;
|
private prevSoundVolume: number;
|
||||||
private peerSettings = {
|
private peerSettings = {
|
||||||
|
@ -22,6 +22,13 @@ import assumeType from '../../helpers/assumeType';
|
|||||||
|
|
||||||
const CACHE_TIME = 3600e3;
|
const CACHE_TIME = 3600e3;
|
||||||
|
|
||||||
|
const EMOJI_SET_LOCAL_ID = 'emoji';
|
||||||
|
const EMOJI_ANIMATIONS_SET_LOCAL_ID = 'emojiAnimations';
|
||||||
|
const LOCAL_IDS_SET = new Set([
|
||||||
|
EMOJI_SET_LOCAL_ID,
|
||||||
|
EMOJI_ANIMATIONS_SET_LOCAL_ID
|
||||||
|
]);
|
||||||
|
|
||||||
export type MyStickerSetInput = {
|
export type MyStickerSetInput = {
|
||||||
id: StickerSet.stickerSet['id'],
|
id: StickerSet.stickerSet['id'],
|
||||||
access_hash?: StickerSet.stickerSet['access_hash']
|
access_hash?: StickerSet.stickerSet['access_hash']
|
||||||
@ -104,11 +111,13 @@ export class AppStickersManager {
|
|||||||
|
|
||||||
return this.getStickerSetPromises[id] = new Promise(async(resolve) => {
|
return this.getStickerSetPromises[id] = new Promise(async(resolve) => {
|
||||||
if(!params.overwrite) {
|
if(!params.overwrite) {
|
||||||
|
// const perf = performance.now();
|
||||||
const cachedSet = await this.storage.get(id);
|
const cachedSet = await this.storage.get(id);
|
||||||
if(cachedSet && cachedSet.documents?.length && ((Date.now() - cachedSet.refreshTime) < CACHE_TIME || params.useCache)) {
|
if(cachedSet && cachedSet.documents?.length && ((Date.now() - cachedSet.refreshTime) < CACHE_TIME || params.useCache)) {
|
||||||
this.saveStickers(cachedSet.documents);
|
this.saveStickers(cachedSet.documents);
|
||||||
resolve(cachedSet);
|
resolve(cachedSet);
|
||||||
delete this.getStickerSetPromises[id];
|
delete this.getStickerSetPromises[id];
|
||||||
|
// console.log('get sticker set from cache time', id, performance.now() - perf);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -132,7 +141,12 @@ export class AppStickersManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public getAnimatedEmojiStickerSet() {
|
public getAnimatedEmojiStickerSet() {
|
||||||
return this.getStickerSet({id: 'emoji'}, {saveById: true});
|
return Promise.all([
|
||||||
|
this.getStickerSet({id: EMOJI_SET_LOCAL_ID}, {saveById: true}),
|
||||||
|
this.getStickerSet({id: EMOJI_ANIMATIONS_SET_LOCAL_ID}, {saveById: true})
|
||||||
|
]).then(([emoji, animations]) => {
|
||||||
|
return {emoji, animations};
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getRecentStickers(): Promise<Modify<MessagesRecentStickers.messagesRecentStickers, {
|
public async getRecentStickers(): Promise<Modify<MessagesRecentStickers.messagesRecentStickers, {
|
||||||
@ -150,8 +164,8 @@ export class AppStickersManager {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
public getAnimatedEmojiSticker(emoji: string) {
|
public getAnimatedEmojiSticker(emoji: string, isAnimation?: boolean) {
|
||||||
const stickerSet = this.storage.getFromCache('emoji');
|
const stickerSet = this.storage.getFromCache(isAnimation ? EMOJI_ANIMATIONS_SET_LOCAL_ID : EMOJI_SET_LOCAL_ID);
|
||||||
if(!stickerSet || !stickerSet.documents) return undefined;
|
if(!stickerSet || !stickerSet.documents) return undefined;
|
||||||
|
|
||||||
emoji = emoji.replace(/\ufe0f/g, '').replace(/🏻|🏼|🏽|🏾|🏿/g, '');
|
emoji = emoji.replace(/\ufe0f/g, '').replace(/🏻|🏼|🏽|🏾|🏿/g, '');
|
||||||
@ -207,7 +221,7 @@ export class AppStickersManager {
|
|||||||
this.saveStickers(res.documents);
|
this.saveStickers(res.documents);
|
||||||
|
|
||||||
//console.log('stickers wrote', this.stickerSets);
|
//console.log('stickers wrote', this.stickerSets);
|
||||||
const needSave = stickerSet.set.installed_date || id === 'emoji';
|
const needSave = stickerSet.set.installed_date || LOCAL_IDS_SET.has(id as any);
|
||||||
stickerSet.refreshTime = Date.now();
|
stickerSet.refreshTime = Date.now();
|
||||||
this.storage.set({[id]: stickerSet}, !needSave);
|
this.storage.set({[id]: stickerSet}, !needSave);
|
||||||
}
|
}
|
||||||
@ -247,10 +261,14 @@ export class AppStickersManager {
|
|||||||
} */
|
} */
|
||||||
|
|
||||||
public getStickerSetInput(set: MyStickerSetInput): InputStickerSet {
|
public getStickerSetInput(set: MyStickerSetInput): InputStickerSet {
|
||||||
if(set.id === 'emoji') {
|
if(set.id === EMOJI_SET_LOCAL_ID) {
|
||||||
return {
|
return {
|
||||||
_: 'inputStickerSetAnimatedEmoji'
|
_: 'inputStickerSetAnimatedEmoji'
|
||||||
};
|
};
|
||||||
|
} else if(set.id === EMOJI_ANIMATIONS_SET_LOCAL_ID) {
|
||||||
|
return {
|
||||||
|
_: 'inputStickerSetAnimatedEmojiAnimations'
|
||||||
|
};
|
||||||
} else if(!set.access_hash) {
|
} else if(!set.access_hash) {
|
||||||
return {
|
return {
|
||||||
_: 'inputStickerSetShortName',
|
_: 'inputStickerSetShortName',
|
||||||
|
@ -132,8 +132,8 @@ export default class RLottiePlayer extends EventListenerBase<{
|
|||||||
private color: RLottieColor;
|
private color: RLottieColor;
|
||||||
private inverseColor: RLottieColor;
|
private inverseColor: RLottieColor;
|
||||||
|
|
||||||
private minFrame: number;
|
public minFrame: number;
|
||||||
private maxFrame: number;
|
public maxFrame: number;
|
||||||
|
|
||||||
//private playedTimes = 0;
|
//private playedTimes = 0;
|
||||||
|
|
||||||
@ -294,7 +294,7 @@ export default class RLottiePlayer extends EventListenerBase<{
|
|||||||
}
|
}
|
||||||
|
|
||||||
private resetCurrentFrame() {
|
private resetCurrentFrame() {
|
||||||
return this.curFrame = this.initFrame || (this.direction === 1 ? this.minFrame : this.maxFrame);
|
return this.curFrame = this.initFrame ?? (this.direction === 1 ? this.minFrame : this.maxFrame);
|
||||||
}
|
}
|
||||||
|
|
||||||
public stop(renderFirstFrame = true) {
|
public stop(renderFirstFrame = true) {
|
||||||
@ -655,6 +655,11 @@ export default class RLottiePlayer extends EventListenerBase<{
|
|||||||
};
|
};
|
||||||
|
|
||||||
this.addEventListener('enterFrame', this.frameListener);
|
this.addEventListener('enterFrame', this.frameListener);
|
||||||
|
|
||||||
|
// ! fix autoplaying since there will be no animationIntersector for it,
|
||||||
|
if(this.group === 'none' && this.autoplay) {
|
||||||
|
this.play();
|
||||||
|
}
|
||||||
}, {once: true});
|
}, {once: true});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -41,7 +41,8 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
|
|
||||||
//private cache: Partial<{[key: string]: Storage[typeof key]}> = {};
|
//private cache: Partial<{[key: string]: Storage[typeof key]}> = {};
|
||||||
private cache: Partial<Storage> = {};
|
private cache: Partial<Storage> = {};
|
||||||
private useStorage = true;
|
private useStorage: boolean;
|
||||||
|
private savingFreezed: boolean;
|
||||||
|
|
||||||
private getPromises: Map<keyof Storage, CancellablePromise<Storage[keyof Storage]>> = new Map();
|
private getPromises: Map<keyof Storage, CancellablePromise<Storage[keyof Storage]>> = new Map();
|
||||||
private getThrottled: () => void;
|
private getThrottled: () => void;
|
||||||
@ -59,8 +60,12 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
|
|
||||||
if(AppStorage.STORAGES.length) {
|
if(AppStorage.STORAGES.length) {
|
||||||
this.useStorage = AppStorage.STORAGES[0].useStorage;
|
this.useStorage = AppStorage.STORAGES[0].useStorage;
|
||||||
|
} else {
|
||||||
|
this.useStorage = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.savingFreezed = false;
|
||||||
|
|
||||||
AppStorage.STORAGES.push(this);
|
AppStorage.STORAGES.push(this);
|
||||||
|
|
||||||
this.saveThrottled = throttle(async() => {
|
this.saveThrottled = throttle(async() => {
|
||||||
@ -140,6 +145,7 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
this.getThrottled = throttle(async() => {
|
this.getThrottled = throttle(async() => {
|
||||||
const keys = Array.from(this.getPromises.keys());
|
const keys = Array.from(this.getPromises.keys());
|
||||||
|
|
||||||
|
// const perf = performance.now();
|
||||||
this.storage.get(keys as string[]).then(values => {
|
this.storage.get(keys as string[]).then(values => {
|
||||||
for(let i = 0, length = keys.length; i < length; ++i) {
|
for(let i = 0, length = keys.length; i < length; ++i) {
|
||||||
const key = keys[i];
|
const key = keys[i];
|
||||||
@ -150,6 +156,8 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
this.getPromises.delete(key);
|
this.getPromises.delete(key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// console.log('[AS]: get time', keys, performance.now() - perf);
|
||||||
}, (error) => {
|
}, (error) => {
|
||||||
if(!['NO_ENTRY_FOUND', 'STORAGE_OFFLINE'].includes(error)) {
|
if(!['NO_ENTRY_FOUND', 'STORAGE_OFFLINE'].includes(error)) {
|
||||||
this.useStorage = false;
|
this.useStorage = false;
|
||||||
@ -214,6 +222,7 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
public set(obj: Partial<Storage>, onlyLocal = false) {
|
public set(obj: Partial<Storage>, onlyLocal = false) {
|
||||||
//console.log('storageSetValue', obj, callback, arguments);
|
//console.log('storageSetValue', obj, callback, arguments);
|
||||||
|
|
||||||
|
const canUseStorage = this.useStorage && !onlyLocal && !this.savingFreezed;
|
||||||
for(const key in obj) {
|
for(const key in obj) {
|
||||||
if(obj.hasOwnProperty(key)) {
|
if(obj.hasOwnProperty(key)) {
|
||||||
const value = obj[key];
|
const value = obj[key];
|
||||||
@ -233,7 +242,7 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
value = stringify(value);
|
value = stringify(value);
|
||||||
console.log('LocalStorage set: stringify time by own stringify:', performance.now() - perf); */
|
console.log('LocalStorage set: stringify time by own stringify:', performance.now() - perf); */
|
||||||
|
|
||||||
if(this.useStorage && !onlyLocal) {
|
if(canUseStorage) {
|
||||||
this.keysToSet.add(key);
|
this.keysToSet.add(key);
|
||||||
this.keysToDelete.delete(key);
|
this.keysToDelete.delete(key);
|
||||||
this.saveThrottled();
|
this.saveThrottled();
|
||||||
@ -241,7 +250,7 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.useStorage ? this.saveDeferred : Promise.resolve();
|
return canUseStorage ? this.saveDeferred : Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
public delete(key: keyof Storage, saveLocal = false) {
|
public delete(key: keyof Storage, saveLocal = false) {
|
||||||
@ -291,6 +300,14 @@ export default class AppStorage<Storage extends Record<string, any>, T extends D
|
|||||||
})).catch(noop);
|
})).catch(noop);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static freezeSaving<T extends Database<any>>(callback: () => any, names: T['stores'][number]['name'][]) {
|
||||||
|
this.STORAGES.forEach(storage => storage.savingFreezed = true);
|
||||||
|
try {
|
||||||
|
callback();
|
||||||
|
} catch(err) {}
|
||||||
|
this.STORAGES.forEach(storage => storage.savingFreezed = false);
|
||||||
|
}
|
||||||
|
|
||||||
/* public deleteDatabase() {
|
/* public deleteDatabase() {
|
||||||
return IDBStorage.deleteDatabase().catch(noop);
|
return IDBStorage.deleteDatabase().catch(noop);
|
||||||
} */
|
} */
|
||||||
|
@ -31,6 +31,8 @@ import { MyDialogFilter } from "./filters";
|
|||||||
import { NULL_PEER_ID } from "../mtproto/mtproto_config";
|
import { NULL_PEER_ID } from "../mtproto/mtproto_config";
|
||||||
import { NoneToVoidFunction } from "../../types";
|
import { NoneToVoidFunction } from "../../types";
|
||||||
import ctx from "../../environment/ctx";
|
import ctx from "../../environment/ctx";
|
||||||
|
import AppStorage from "../storage";
|
||||||
|
import type DATABASE_STATE from "../../config/databases/state";
|
||||||
|
|
||||||
export type FolderDialog = {
|
export type FolderDialog = {
|
||||||
dialog: Dialog,
|
dialog: Dialog,
|
||||||
@ -159,37 +161,41 @@ export default class DialogsStorage {
|
|||||||
|
|
||||||
const dialogs = appStateManager.storagesResults.dialogs;
|
const dialogs = appStateManager.storagesResults.dialogs;
|
||||||
if(dialogs.length) {
|
if(dialogs.length) {
|
||||||
for(let i = 0, length = dialogs.length; i < length; ++i) {
|
AppStorage.freezeSaving<typeof DATABASE_STATE>(this.setDialogsFromState.bind(this, dialogs), ['chats', 'dialogs', 'messages', 'users']);
|
||||||
const dialog = dialogs[i];
|
|
||||||
if(dialog) {
|
|
||||||
// if(dialog.peerId !== SERVICE_PEER_ID) {
|
|
||||||
dialog.top_message = this.appMessagesIdsManager.getServerMessageId(dialog.top_message); // * fix outgoing message to avoid copying dialog
|
|
||||||
// }
|
|
||||||
|
|
||||||
if(dialog.topMessage) {
|
|
||||||
this.appMessagesManager.saveMessages([dialog.topMessage]);
|
|
||||||
}
|
|
||||||
|
|
||||||
for(let i = 0; i <= 10; ++i) {
|
|
||||||
// @ts-ignore
|
|
||||||
delete dialog[`index_${i}`];
|
|
||||||
}
|
|
||||||
|
|
||||||
this.saveDialog(dialog, undefined, true);
|
|
||||||
|
|
||||||
// ! WARNING, убрать это когда нужно будет делать чтобы pending сообщения сохранялись
|
|
||||||
const message = this.appMessagesManager.getMessageByPeer(dialog.peerId, dialog.top_message);
|
|
||||||
if(message.deleted) {
|
|
||||||
this.appMessagesManager.reloadConversation(dialog.peerId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
this.allDialogsLoaded = state.allDialogsLoaded || {};
|
this.allDialogsLoaded = state.allDialogsLoaded || {};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private setDialogsFromState(dialogs: Dialog[]) {
|
||||||
|
for(let i = 0, length = dialogs.length; i < length; ++i) {
|
||||||
|
const dialog = dialogs[i];
|
||||||
|
if(dialog) {
|
||||||
|
// if(dialog.peerId !== SERVICE_PEER_ID) {
|
||||||
|
dialog.top_message = this.appMessagesIdsManager.getServerMessageId(dialog.top_message); // * fix outgoing message to avoid copying dialog
|
||||||
|
// }
|
||||||
|
|
||||||
|
if(dialog.topMessage) {
|
||||||
|
this.appMessagesManager.saveMessages([dialog.topMessage]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for(let i = 0; i <= 10; ++i) {
|
||||||
|
// @ts-ignore
|
||||||
|
delete dialog[`index_${i}`];
|
||||||
|
}
|
||||||
|
|
||||||
|
this.saveDialog(dialog, undefined, true);
|
||||||
|
|
||||||
|
// ! WARNING, убрать это когда нужно будет делать чтобы pending сообщения сохранялись
|
||||||
|
const message = this.appMessagesManager.getMessageByPeer(dialog.peerId, dialog.top_message);
|
||||||
|
if(message.deleted) {
|
||||||
|
this.appMessagesManager.reloadConversation(dialog.peerId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public isDialogsLoaded(folderId: number) {
|
public isDialogsLoaded(folderId: number) {
|
||||||
return !!this.allDialogsLoaded[folderId];
|
return !!this.allDialogsLoaded[folderId];
|
||||||
}
|
}
|
||||||
|
29
src/scss/partials/_emojiAnimation.scss
Normal file
29
src/scss/partials/_emojiAnimation.scss
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
/*
|
||||||
|
* https://github.com/morethanwords/tweb
|
||||||
|
* Copyright (C) 2019-2021 Eduard Kuzmenko
|
||||||
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
.emoji-animation {
|
||||||
|
position: absolute;
|
||||||
|
pointer-events: none;
|
||||||
|
|
||||||
|
// @include sidebar-transform(true);
|
||||||
|
|
||||||
|
&.is-in {
|
||||||
|
.rlottie {
|
||||||
|
transform: scaleX(-1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&-container {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
overflow: hidden;
|
||||||
|
z-index: 3;
|
||||||
|
}
|
||||||
|
}
|
@ -305,6 +305,7 @@ $chat-input-inner-padding-handhelds: .25rem;
|
|||||||
@import "partials/replyKeyboard";
|
@import "partials/replyKeyboard";
|
||||||
@import "partials/peopleNearby";
|
@import "partials/peopleNearby";
|
||||||
@import "partials/spoiler";
|
@import "partials/spoiler";
|
||||||
|
@import "partials/emojiAnimation";
|
||||||
|
|
||||||
@import "partials/popups/popup";
|
@import "partials/popups/popup";
|
||||||
@import "partials/popups/editAvatar";
|
@import "partials/popups/editAvatar";
|
||||||
|
5
src/types.d.ts
vendored
5
src/types.d.ts
vendored
@ -111,3 +111,8 @@ export namespace AuthState {
|
|||||||
_: 'authStateSignedIn'
|
_: 'authStateSignedIn'
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type SendMessageEmojiInteractionData = {
|
||||||
|
a: {t: number, i: 1}[],
|
||||||
|
v: 1
|
||||||
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user