Telegram Web K with changes to work inside I2P
https://web.telegram.i2p/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2772 lines
96 KiB
2772 lines
96 KiB
/* |
|
* https://github.com/morethanwords/tweb |
|
* Copyright (C) 2019-2021 Eduard Kuzmenko |
|
* https://github.com/morethanwords/tweb/blob/master/LICENSE |
|
*/ |
|
|
|
import type { MyDocument } from "../../lib/appManagers/appDocsManager"; |
|
import type { AppImManager } from '../../lib/appManagers/appImManager'; |
|
import type { MyDraftMessage } from '../../lib/appManagers/appDraftsManager'; |
|
import type Chat from './chat'; |
|
import Recorder from '../../../public/recorder.min'; |
|
import IS_TOUCH_SUPPORTED from "../../environment/touchSupport"; |
|
//import Recorder from '../opus-recorder/dist/recorder.min'; |
|
import opusDecodeController from "../../lib/opusDecodeController"; |
|
import ButtonMenu, { ButtonMenuItemOptions } from '../buttonMenu'; |
|
import emoticonsDropdown from "../emoticonsDropdown"; |
|
import PopupCreatePoll from "../popups/createPoll"; |
|
import PopupForward from '../popups/forward'; |
|
import PopupNewMedia from '../popups/newMedia'; |
|
import { toast } from "../toast"; |
|
import { wrapReply } from "../wrappers"; |
|
import InputField from '../inputField'; |
|
import { MessageEntity, DraftMessage, WebPage, Message, ChatFull, UserFull } from '../../layer'; |
|
import StickersHelper from './stickersHelper'; |
|
import ButtonIcon from '../buttonIcon'; |
|
import ButtonMenuToggle from '../buttonMenuToggle'; |
|
import ListenerSetter, { Listener } from '../../helpers/listenerSetter'; |
|
import Button from '../button'; |
|
import PopupSchedule from '../popups/schedule'; |
|
import SendMenu from './sendContextMenu'; |
|
import rootScope from '../../lib/rootScope'; |
|
import PopupPinMessage from '../popups/unpinMessage'; |
|
import tsNow from '../../helpers/tsNow'; |
|
import appNavigationController, { NavigationItem } from '../appNavigationController'; |
|
import { IS_MOBILE, IS_MOBILE_SAFARI } from '../../environment/userAgent'; |
|
import I18n, { i18n, join, LangPackKey } from '../../lib/langPack'; |
|
import { generateTail } from './bubbles'; |
|
import findUpClassName from '../../helpers/dom/findUpClassName'; |
|
import ButtonCorner from '../buttonCorner'; |
|
import blurActiveElement from '../../helpers/dom/blurActiveElement'; |
|
import cancelEvent from '../../helpers/dom/cancelEvent'; |
|
import cancelSelection from '../../helpers/dom/cancelSelection'; |
|
import { attachClickEvent, simulateClickEvent } from '../../helpers/dom/clickEvent'; |
|
import getRichValue from '../../helpers/dom/getRichValue'; |
|
import isInputEmpty from '../../helpers/dom/isInputEmpty'; |
|
import isSendShortcutPressed from '../../helpers/dom/isSendShortcutPressed'; |
|
import placeCaretAtEnd from '../../helpers/dom/placeCaretAtEnd'; |
|
import { MarkdownType, markdownTags } from '../../helpers/dom/getRichElementValue'; |
|
import getRichValueWithCaret from '../../helpers/dom/getRichValueWithCaret'; |
|
import EmojiHelper from './emojiHelper'; |
|
import CommandsHelper from './commandsHelper'; |
|
import AutocompleteHelperController from './autocompleteHelperController'; |
|
import AutocompleteHelper from './autocompleteHelper'; |
|
import MentionsHelper from './mentionsHelper'; |
|
import fixSafariStickyInput from '../../helpers/dom/fixSafariStickyInput'; |
|
import ReplyKeyboard from './replyKeyboard'; |
|
import InlineHelper from './inlineHelper'; |
|
import debounce from '../../helpers/schedulers/debounce'; |
|
import noop from '../../helpers/noop'; |
|
import { putPreloader } from '../putPreloader'; |
|
import SetTransition from '../singleTransition'; |
|
import PeerTitle from '../peerTitle'; |
|
import { fastRaf } from '../../helpers/schedulers'; |
|
import PopupDeleteMessages from '../popups/deleteMessages'; |
|
import fixSafariStickyInputFocusing, { IS_STICKY_INPUT_BUGGED } from '../../helpers/dom/fixSafariStickyInputFocusing'; |
|
import PopupPeer from '../popups/peer'; |
|
import MEDIA_MIME_TYPES_SUPPORTED from '../../environment/mediaMimeTypesSupport'; |
|
import appMediaPlaybackController from '../appMediaPlaybackController'; |
|
import { BOT_START_PARAM, NULL_PEER_ID } from '../../lib/mtproto/mtproto_config'; |
|
import setCaretAt from '../../helpers/dom/setCaretAt'; |
|
import CheckboxField from '../checkboxField'; |
|
import DropdownHover from '../../helpers/dropdownHover'; |
|
import RadioForm from '../radioForm'; |
|
import findUpTag from '../../helpers/dom/findUpTag'; |
|
import toggleDisability from '../../helpers/dom/toggleDisability'; |
|
import callbackify from '../../helpers/callbackify'; |
|
import ChatBotCommands from './botCommands'; |
|
import copy from '../../helpers/object/copy'; |
|
import toHHMMSS from '../../helpers/string/toHHMMSS'; |
|
import documentFragmentToHTML from '../../helpers/dom/documentFragmentToHTML'; |
|
import PopupElement from '../popups'; |
|
import getEmojiEntityFromEmoji from '../../lib/richTextProcessor/getEmojiEntityFromEmoji'; |
|
import mergeEntities from '../../lib/richTextProcessor/mergeEntities'; |
|
import parseEntities from '../../lib/richTextProcessor/parseEntities'; |
|
import parseMarkdown from '../../lib/richTextProcessor/parseMarkdown'; |
|
import wrapDraftText from '../../lib/richTextProcessor/wrapDraftText'; |
|
import wrapDraft from '../wrappers/draft'; |
|
import wrapMessageForReply from '../wrappers/messageForReply'; |
|
import getServerMessageId from '../../lib/appManagers/utils/messageId/getServerMessageId'; |
|
import { AppManagers } from '../../lib/appManagers/managers'; |
|
import contextMenuController from "../../helpers/contextMenuController"; |
|
import { emojiFromCodePoints } from "../../vendor/emoji"; |
|
import { modifyAckedPromise } from "../../helpers/modifyAckedResult"; |
|
import ChatSendAs from "./sendAs"; |
|
import filterAsync from "../../helpers/array/filterAsync"; |
|
|
|
const RECORD_MIN_TIME = 500; |
|
const POSTING_MEDIA_NOT_ALLOWED = 'Posting media content isn\'t allowed in this group.'; |
|
|
|
type ChatInputHelperType = 'edit' | 'webpage' | 'forward' | 'reply'; |
|
|
|
export default class ChatInput { |
|
// private static AUTO_COMPLETE_REG_EXP = /(\s|^)((?::|.)(?!.*[:@]).*|(?:[@\/]\S*))$/; |
|
private static AUTO_COMPLETE_REG_EXP = /(\s|^)((?:(?:@|^\/)\S*)|(?::|^[^:@\/])(?!.*[:@\/]).*)$/; |
|
public messageInput: HTMLElement; |
|
public messageInputField: InputField; |
|
private fileInput: HTMLInputElement; |
|
private inputMessageContainer: HTMLDivElement; |
|
private btnSend: HTMLButtonElement; |
|
private btnCancelRecord: HTMLButtonElement; |
|
private lastUrl = ''; |
|
private lastTimeType = 0; |
|
|
|
public chatInput: HTMLElement; |
|
public inputContainer: HTMLElement; |
|
public rowsWrapper: HTMLDivElement; |
|
private newMessageWrapper: HTMLDivElement; |
|
private btnToggleEmoticons: HTMLButtonElement; |
|
private btnToggleReplyMarkup: HTMLButtonElement; |
|
private btnSendContainer: HTMLDivElement; |
|
|
|
private replyKeyboard: ReplyKeyboard; |
|
|
|
private attachMenu: HTMLElement; |
|
private attachMenuButtons: (ButtonMenuItemOptions & {verify: (peerId: PeerId, threadId: number) => boolean | Promise<boolean>})[]; |
|
|
|
private sendMenu: SendMenu; |
|
|
|
private replyElements: { |
|
container: HTMLElement, |
|
cancelBtn: HTMLButtonElement, |
|
iconBtn: HTMLButtonElement |
|
} = {} as any; |
|
|
|
private forwardElements: { |
|
changePeer: ButtonMenuItemOptions, |
|
showSender: ButtonMenuItemOptions, |
|
hideSender: ButtonMenuItemOptions, |
|
showCaption: ButtonMenuItemOptions, |
|
hideCaption: ButtonMenuItemOptions, |
|
container: HTMLElement, |
|
modifyArgs?: ButtonMenuItemOptions[] |
|
}; |
|
private forwardHover: DropdownHover; |
|
private forwardWasDroppingAuthor: boolean; |
|
|
|
private getWebPagePromise: Promise<void>; |
|
private willSendWebPage: WebPage = null; |
|
private forwarding: {[fromPeerId: PeerId]: number[]}; |
|
public replyToMsgId: number; |
|
public editMsgId: number; |
|
public editMessage: Message.message; |
|
private noWebPage: true; |
|
public scheduleDate: number; |
|
public sendSilent: true; |
|
public startParam: string; |
|
|
|
private recorder: any; |
|
public recording = false; |
|
private recordCanceled = false; |
|
private recordTimeEl: HTMLElement; |
|
private recordRippleEl: HTMLElement; |
|
private recordStartTime = 0; |
|
private recordingOverlayListener: Listener; |
|
private recordingNavigationItem: NavigationItem; |
|
|
|
// private scrollTop = 0; |
|
// private scrollOffsetTop = 0; |
|
// private scrollDiff = 0; |
|
|
|
public helperType: Exclude<ChatInputHelperType, 'webpage'>; |
|
private helperFunc: () => void; |
|
private helperWaitingForward: boolean; |
|
|
|
public willAttachType: 'document' | 'media'; |
|
|
|
private lockRedo = false; |
|
private canRedoFromHTML = ''; |
|
private readonly undoHistory: string[] = []; |
|
private readonly executedHistory: string[] = []; |
|
private canUndoFromHTML = ''; |
|
|
|
private autocompleteHelperController: AutocompleteHelperController; |
|
private stickersHelper: StickersHelper; |
|
private emojiHelper: EmojiHelper; |
|
private commandsHelper: CommandsHelper; |
|
private mentionsHelper: MentionsHelper; |
|
private inlineHelper: InlineHelper; |
|
private listenerSetter: ListenerSetter; |
|
|
|
private pinnedControlBtn: HTMLButtonElement; |
|
|
|
private goDownBtn: HTMLButtonElement; |
|
private goDownUnreadBadge: HTMLElement; |
|
private goMentionBtn: HTMLButtonElement; |
|
private goMentionUnreadBadge: HTMLSpanElement; |
|
private btnScheduled: HTMLButtonElement; |
|
|
|
private btnPreloader: HTMLButtonElement; |
|
|
|
private saveDraftDebounced: () => void; |
|
|
|
private fakeRowsWrapper: HTMLDivElement; |
|
|
|
private previousQuery: string; |
|
|
|
private releaseMediaPlayback: () => void; |
|
|
|
private botStartBtn: HTMLButtonElement; |
|
private rowsWrapperWrapper: HTMLDivElement; |
|
private controlContainer: HTMLElement; |
|
private fakeSelectionWrapper: HTMLDivElement; |
|
|
|
private fakeWrapperTo: HTMLElement; |
|
private toggleBotStartBtnDisability: () => void; |
|
|
|
private botCommandsToggle: HTMLElement; |
|
private botCommands: ChatBotCommands; |
|
private botCommandsIcon: HTMLDivElement; |
|
private hasBotCommands: boolean; |
|
|
|
// private activeContainer: HTMLElement; |
|
|
|
private sendAs: ChatSendAs; |
|
public sendAsPeerId: PeerId; |
|
|
|
constructor( |
|
private chat: Chat, |
|
private appImManager: AppImManager, |
|
private managers: AppManagers |
|
) { |
|
this.listenerSetter = new ListenerSetter(); |
|
} |
|
|
|
public construct() { |
|
this.chatInput = document.createElement('div'); |
|
this.chatInput.classList.add('chat-input', 'hide'); |
|
|
|
this.inputContainer = document.createElement('div'); |
|
this.inputContainer.classList.add('chat-input-container'); |
|
|
|
this.rowsWrapperWrapper = document.createElement('div'); |
|
this.rowsWrapperWrapper.classList.add('rows-wrapper-wrapper'); |
|
|
|
this.rowsWrapper = document.createElement('div'); |
|
this.rowsWrapper.classList.add('rows-wrapper', 'chat-input-wrapper'); |
|
|
|
this.rowsWrapperWrapper.append(this.rowsWrapper); |
|
|
|
const tail = generateTail(); |
|
this.rowsWrapper.append(tail); |
|
|
|
const fakeRowsWrapper = this.fakeRowsWrapper = document.createElement('div'); |
|
fakeRowsWrapper.classList.add('fake-wrapper', 'fake-rows-wrapper'); |
|
|
|
const fakeSelectionWrapper = this.fakeSelectionWrapper = document.createElement('div'); |
|
fakeSelectionWrapper.classList.add('fake-wrapper', 'fake-selection-wrapper'); |
|
|
|
this.inputContainer.append(this.rowsWrapperWrapper, fakeRowsWrapper, fakeSelectionWrapper); |
|
this.chatInput.append(this.inputContainer); |
|
|
|
this.goDownBtn = ButtonCorner({icon: 'arrow_down', className: 'bubbles-corner-button bubbles-go-down hide'}); |
|
this.inputContainer.append(this.goDownBtn); |
|
|
|
attachClickEvent(this.goDownBtn, (e) => { |
|
cancelEvent(e); |
|
this.chat.bubbles.onGoDownClick(); |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
// * constructor end |
|
|
|
/* let setScrollTopTimeout: number; |
|
// @ts-ignore |
|
let height = window.visualViewport.height; */ |
|
// @ts-ignore |
|
// this.listenerSetter.add(window.visualViewport)('resize', () => { |
|
// const scrollable = this.chat.bubbles.scrollable; |
|
// const wasScrolledDown = scrollable.isScrolledDown; |
|
|
|
// /* if(wasScrolledDown) { |
|
// this.saveScroll(); |
|
// } */ |
|
|
|
// // @ts-ignore |
|
// let newHeight = window.visualViewport.height; |
|
// const diff = height - newHeight; |
|
// const scrollTop = scrollable.scrollTop; |
|
// const needScrollTop = wasScrolledDown ? scrollable.scrollHeight : scrollTop + diff; // * wasScrolledDown это проверка для десктоп хрома, когда пропадает панель загрузок снизу |
|
|
|
// console.log('resize before', scrollable.scrollTop, scrollable.container.clientHeight, scrollable.scrollHeight, wasScrolledDown, scrollable.lastScrollTop, diff, needScrollTop); |
|
|
|
// scrollable.scrollTop = needScrollTop; |
|
|
|
// if(setScrollTopTimeout) clearTimeout(setScrollTopTimeout); |
|
// setScrollTopTimeout = window.setTimeout(() => { |
|
// const diff = height - newHeight; |
|
// const isScrolledDown = scrollable.scrollHeight - Math.round(scrollable.scrollTop + scrollable.container.offsetHeight + diff) <= 1; |
|
// height = newHeight; |
|
|
|
// scrollable.scrollTop = needScrollTop; |
|
|
|
// console.log('resize after', scrollable.scrollTop, scrollable.container.clientHeight, scrollable.scrollHeight, scrollable.isScrolledDown, scrollable.lastScrollTop, isScrolledDown); |
|
|
|
// /* if(isScrolledDown) { |
|
// scrollable.scrollTop = scrollable.scrollHeight; |
|
// } */ |
|
|
|
// //scrollable.scrollTop += diff; |
|
// setScrollTopTimeout = 0; |
|
// }, 0); |
|
// }); |
|
|
|
// ! Can't use it with resizeObserver |
|
/* this.listenerSetter.add(window.visualViewport)('resize', () => { |
|
const scrollable = this.chat.bubbles.scrollable; |
|
const wasScrolledDown = scrollable.isScrolledDown; |
|
|
|
// @ts-ignore |
|
let newHeight = window.visualViewport.height; |
|
const diff = height - newHeight; |
|
const needScrollTop = wasScrolledDown ? scrollable.scrollHeight : scrollable.scrollTop + diff; // * wasScrolledDown это проверка для десктоп хрома, когда пропадает панель загрузок снизу |
|
|
|
//console.log('resize before', scrollable.scrollTop, scrollable.container.clientHeight, scrollable.scrollHeight, wasScrolledDown, scrollable.lastScrollTop, diff, needScrollTop); |
|
|
|
scrollable.scrollTop = needScrollTop; |
|
height = newHeight; |
|
|
|
if(setScrollTopTimeout) clearTimeout(setScrollTopTimeout); |
|
setScrollTopTimeout = window.setTimeout(() => { // * try again for scrolled down Android Chrome |
|
scrollable.scrollTop = needScrollTop; |
|
|
|
//console.log('resize after', scrollable.scrollTop, scrollable.container.clientHeight, scrollable.scrollHeight, scrollable.isScrolledDown, scrollable.lastScrollTop, isScrolledDown); |
|
setScrollTopTimeout = 0; |
|
}, 0); |
|
}); */ |
|
|
|
const c = this.controlContainer = document.createElement('div'); |
|
c.classList.add('chat-input-control', 'chat-input-wrapper'); |
|
this.inputContainer.append(c); |
|
} |
|
|
|
public constructPeerHelpers() { |
|
this.replyElements.container = document.createElement('div'); |
|
this.replyElements.container.classList.add('reply-wrapper'); |
|
|
|
this.replyElements.iconBtn = ButtonIcon(''); |
|
this.replyElements.cancelBtn = ButtonIcon('close reply-cancel', {noRipple: true}); |
|
|
|
this.replyElements.container.append(this.replyElements.iconBtn, this.replyElements.cancelBtn); |
|
|
|
// |
|
|
|
const onHideAuthorClick = () => { |
|
isChangingAuthor = true; |
|
return this.canToggleHideAuthor(); |
|
}; |
|
|
|
const onHideCaptionClick = () => { |
|
isChangingAuthor = false; |
|
}; |
|
|
|
const forwardElements: ChatInput['forwardElements'] = this.forwardElements = {} as any; |
|
let isChangingAuthor = false; |
|
const forwardButtons: ButtonMenuItemOptions[] = [ |
|
forwardElements.showSender = { |
|
text: 'Chat.Alert.Forward.Action.Show1', |
|
onClick: onHideAuthorClick, |
|
checkboxField: new CheckboxField({checked: true}) |
|
}, |
|
forwardElements.hideSender = { |
|
text: 'Chat.Alert.Forward.Action.Hide1', |
|
onClick: onHideAuthorClick, |
|
checkboxField: new CheckboxField({checked: false}) |
|
}, |
|
forwardElements.showCaption = { |
|
text: 'Chat.Alert.Forward.Action.ShowCaption', |
|
onClick: onHideCaptionClick, |
|
checkboxField: new CheckboxField({checked: true}) |
|
}, |
|
forwardElements.hideCaption = { |
|
text: 'Chat.Alert.Forward.Action.HideCaption', |
|
onClick: onHideCaptionClick, |
|
checkboxField: new CheckboxField({checked: false}) |
|
}, |
|
forwardElements.changePeer = { |
|
text: 'Chat.Alert.Forward.Action.Another', |
|
onClick: () => { |
|
this.changeForwardRecipient(); |
|
}, |
|
icon: 'replace' |
|
} |
|
]; |
|
const forwardBtnMenu = forwardElements.container = ButtonMenu(forwardButtons, this.listenerSetter); |
|
// forwardBtnMenu.classList.add('top-center'); |
|
|
|
const children = Array.from(forwardBtnMenu.children) as HTMLElement[]; |
|
const groups: { |
|
elements: HTMLElement[], |
|
onChange: (value: string, event: Event) => void |
|
}[] = [{ |
|
elements: children.slice(0, 2), |
|
onChange: (value, e) => { |
|
const checked = !!+value; |
|
if(isChangingAuthor) { |
|
this.forwardWasDroppingAuthor = !checked; |
|
} |
|
|
|
const replyTitle = this.replyElements.container.querySelector('.reply-title'); |
|
if(replyTitle) { |
|
const el = replyTitle.firstElementChild as HTMLElement; |
|
const i = I18n.weakMap.get(el) as I18n.IntlElement; |
|
const langPackKey: LangPackKey = forwardElements.showSender.checkboxField.checked ? 'Chat.Accessory.Forward' : 'Chat.Accessory.Hidden'; |
|
i.key = langPackKey; |
|
i.update(); |
|
} |
|
} |
|
}, { |
|
elements: children.slice(2, 4), |
|
onChange: (value) => { |
|
const checked = !!+value; |
|
let b: ButtonMenuItemOptions; |
|
if(checked && this.forwardWasDroppingAuthor !== undefined) { |
|
b = this.forwardWasDroppingAuthor ? forwardElements.hideSender : forwardElements.showSender; |
|
} else { |
|
b = checked ? forwardElements.showSender : forwardElements.hideSender; |
|
} |
|
|
|
b.checkboxField.checked = true; |
|
} |
|
}]; |
|
groups.forEach((group) => { |
|
const container = RadioForm(group.elements.map((e) => { |
|
return { |
|
container: e, |
|
input: e.querySelector('input') |
|
}; |
|
}), group.onChange); |
|
|
|
const hr = document.createElement('hr'); |
|
container.append(hr); |
|
forwardBtnMenu.append(container); |
|
}); |
|
|
|
forwardBtnMenu.append(forwardElements.changePeer.element); |
|
|
|
if(!IS_TOUCH_SUPPORTED) { |
|
const forwardHover = this.forwardHover = new DropdownHover({ |
|
element: forwardBtnMenu |
|
}); |
|
} |
|
|
|
forwardElements.modifyArgs = forwardButtons.slice(0, -1); |
|
this.replyElements.container.append(forwardBtnMenu); |
|
|
|
forwardElements.modifyArgs.forEach((b, idx) => { |
|
const {input} = b.checkboxField; |
|
input.type = 'radio'; |
|
input.name = idx < 2 ? 'author' : 'caption'; |
|
input.value = '' + +!(idx % 2); |
|
}); |
|
|
|
// |
|
|
|
this.newMessageWrapper = document.createElement('div'); |
|
this.newMessageWrapper.classList.add('new-message-wrapper'); |
|
|
|
this.btnToggleEmoticons = ButtonIcon('none toggle-emoticons', {noRipple: true}); |
|
|
|
this.inputMessageContainer = document.createElement('div'); |
|
this.inputMessageContainer.classList.add('input-message-container'); |
|
|
|
if(this.chat.type === 'chat') { |
|
this.goDownUnreadBadge = document.createElement('span'); |
|
this.goDownUnreadBadge.classList.add('badge', 'badge-24', 'badge-primary'); |
|
this.goDownBtn.append(this.goDownUnreadBadge); |
|
|
|
this.goMentionBtn = ButtonCorner({icon: 'mention', className: 'bubbles-corner-button bubbles-go-mention'}); |
|
this.goMentionUnreadBadge = document.createElement('span'); |
|
this.goMentionUnreadBadge.classList.add('badge', 'badge-24', 'badge-primary'); |
|
this.goMentionBtn.append(this.goMentionUnreadBadge); |
|
this.inputContainer.append(this.goMentionBtn); |
|
|
|
attachClickEvent(this.goMentionBtn, (e) => { |
|
cancelEvent(e); |
|
const middleware = this.chat.bubbles.getMiddleware(); |
|
this.managers.appMessagesManager.goToNextMention(this.chat.peerId).then((mid) => { |
|
if(!middleware()) { |
|
return; |
|
} |
|
|
|
if(mid) { |
|
this.chat.setMessageId(mid); |
|
} |
|
}); |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
this.btnScheduled = ButtonIcon('scheduled btn-scheduled float hide', {noRipple: true}); |
|
|
|
attachClickEvent(this.btnScheduled, (e) => { |
|
this.appImManager.openScheduled(this.chat.peerId); |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
this.listenerSetter.add(rootScope)('scheduled_new', ({peerId}) => { |
|
if(this.chat.peerId !== peerId) { |
|
return; |
|
} |
|
|
|
this.btnScheduled.classList.remove('hide'); |
|
}); |
|
|
|
this.listenerSetter.add(rootScope)('scheduled_delete', ({peerId}) => { |
|
if(this.chat.peerId !== peerId) { |
|
return; |
|
} |
|
|
|
this.managers.appMessagesManager.getScheduledMessages(this.chat.peerId).then((value) => { |
|
this.btnScheduled.classList.toggle('hide', !value.length); |
|
}); |
|
}); |
|
|
|
this.btnToggleReplyMarkup = ButtonIcon('botcom toggle-reply-markup float hide', {noRipple: true}); |
|
this.replyKeyboard = new ReplyKeyboard({ |
|
appendTo: this.rowsWrapper, |
|
listenerSetter: this.listenerSetter, |
|
managers: this.managers, |
|
btnHover: this.btnToggleReplyMarkup, |
|
chatInput: this |
|
}); |
|
this.listenerSetter.add(this.replyKeyboard)('open', () => this.btnToggleReplyMarkup.classList.add('active')); |
|
this.listenerSetter.add(this.replyKeyboard)('close', () => this.btnToggleReplyMarkup.classList.remove('active')); |
|
|
|
this.botCommands = new ChatBotCommands(this.rowsWrapper, this, this.managers); |
|
this.botCommandsToggle = document.createElement('div'); |
|
this.botCommandsToggle.classList.add('new-message-bot-commands'); |
|
|
|
const scaler = document.createElement('div'); |
|
scaler.classList.add('new-message-bot-commands-icon-scale'); |
|
|
|
const icon = this.botCommandsIcon = document.createElement('div'); |
|
icon.classList.add('animated-menu-icon', 'animated-menu-close-icon'); |
|
scaler.append(icon); |
|
this.botCommandsToggle.append(scaler); |
|
|
|
attachClickEvent(this.botCommandsToggle, (e) => { |
|
cancelEvent(e); |
|
const isShown = icon.classList.contains('state-back'); |
|
if(isShown) { |
|
this.botCommands.toggle(true); |
|
icon.classList.remove('state-back'); |
|
} else { |
|
this.botCommands.setUserId(this.chat.peerId.toUserId(), this.chat.bubbles.getMiddleware()); |
|
icon.classList.add('state-back'); |
|
} |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
this.botCommands.addEventListener('visible', () => { |
|
icon.classList.add('state-back'); |
|
}); |
|
|
|
this.botCommands.addEventListener('hiding', () => { |
|
icon.classList.remove('state-back'); |
|
}); |
|
} |
|
|
|
this.attachMenuButtons = [{ |
|
icon: 'image', |
|
text: 'Chat.Input.Attach.PhotoOrVideo', |
|
onClick: () => { |
|
this.fileInput.value = ''; |
|
const accept = [...MEDIA_MIME_TYPES_SUPPORTED].join(', '); |
|
this.fileInput.setAttribute('accept', accept); |
|
this.willAttachType = 'media'; |
|
this.fileInput.click(); |
|
}, |
|
verify: () => this.chat.canSend('send_media') |
|
}, { |
|
icon: 'document', |
|
text: 'Chat.Input.Attach.Document', |
|
onClick: () => { |
|
this.fileInput.value = ''; |
|
this.fileInput.removeAttribute('accept'); |
|
this.willAttachType = 'document'; |
|
this.fileInput.click(); |
|
}, |
|
verify: () => this.chat.canSend('send_media') |
|
}, { |
|
icon: 'poll', |
|
text: 'Poll', |
|
onClick: () => { |
|
PopupElement.createPopup(PopupCreatePoll, this.chat).show(); |
|
}, |
|
verify: (peerId) => peerId.isAnyChat() && this.chat.canSend('send_polls') |
|
}]; |
|
|
|
this.attachMenu = ButtonMenuToggle({noRipple: true, listenerSetter: this.listenerSetter}, 'top-left', this.attachMenuButtons); |
|
this.attachMenu.classList.add('attach-file', 'tgico-attach'); |
|
this.attachMenu.classList.remove('tgico-more'); |
|
|
|
//this.inputContainer.append(this.sendMenu); |
|
|
|
this.recordTimeEl = document.createElement('div'); |
|
this.recordTimeEl.classList.add('record-time'); |
|
|
|
this.fileInput = document.createElement('input'); |
|
this.fileInput.type = 'file'; |
|
this.fileInput.multiple = true; |
|
this.fileInput.style.display = 'none'; |
|
|
|
this.newMessageWrapper.append(...[this.botCommandsToggle, this.btnToggleEmoticons, this.inputMessageContainer, this.btnScheduled, this.btnToggleReplyMarkup, this.attachMenu, this.recordTimeEl, this.fileInput].filter(Boolean)); |
|
|
|
this.rowsWrapper.append(this.replyElements.container); |
|
this.autocompleteHelperController = new AutocompleteHelperController(); |
|
this.stickersHelper = new StickersHelper(this.rowsWrapper, this.autocompleteHelperController, this.managers); |
|
this.emojiHelper = new EmojiHelper(this.rowsWrapper, this.autocompleteHelperController, this, this.managers); |
|
this.commandsHelper = new CommandsHelper(this.rowsWrapper, this.autocompleteHelperController, this, this.managers); |
|
this.mentionsHelper = new MentionsHelper(this.rowsWrapper, this.autocompleteHelperController, this, this.managers); |
|
this.inlineHelper = new InlineHelper(this.rowsWrapper, this.autocompleteHelperController, this.chat, this.managers); |
|
this.rowsWrapper.append(this.newMessageWrapper); |
|
|
|
this.btnCancelRecord = ButtonIcon('delete btn-circle z-depth-1 btn-record-cancel'); |
|
|
|
this.btnSendContainer = document.createElement('div'); |
|
this.btnSendContainer.classList.add('btn-send-container'); |
|
|
|
this.recordRippleEl = document.createElement('div'); |
|
this.recordRippleEl.classList.add('record-ripple'); |
|
|
|
this.btnSend = ButtonIcon('none btn-circle z-depth-1 btn-send animated-button-icon'); |
|
this.btnSend.insertAdjacentHTML('afterbegin', ` |
|
<span class="tgico tgico-send"></span> |
|
<span class="tgico tgico-schedule"></span> |
|
<span class="tgico tgico-check"></span> |
|
<span class="tgico tgico-microphone_filled"></span> |
|
`); |
|
|
|
this.btnSendContainer.append(this.recordRippleEl, this.btnSend); |
|
|
|
if(this.chat.type !== 'scheduled') { |
|
this.sendMenu = new SendMenu({ |
|
onSilentClick: () => { |
|
this.sendSilent = true; |
|
this.sendMessage(); |
|
}, |
|
onScheduleClick: () => { |
|
this.scheduleSending(undefined); |
|
}, |
|
listenerSetter: this.listenerSetter, |
|
openSide: 'top-left', |
|
onContextElement: this.btnSend, |
|
onOpen: () => { |
|
return !this.isInputEmpty() || !!Object.keys(this.forwarding).length; |
|
} |
|
}); |
|
|
|
this.btnSendContainer.append(this.sendMenu.sendMenu); |
|
} |
|
|
|
this.inputContainer.append(this.btnCancelRecord, this.btnSendContainer); |
|
|
|
emoticonsDropdown.attachButtonListener(this.btnToggleEmoticons, this.listenerSetter); |
|
this.listenerSetter.add(emoticonsDropdown)('open', this.onEmoticonsOpen); |
|
this.listenerSetter.add(emoticonsDropdown)('close', this.onEmoticonsClose); |
|
|
|
this.attachMessageInputField(); |
|
|
|
/* this.attachMenu.addEventListener('mousedown', (e) => { |
|
const hidden = this.attachMenu.querySelectorAll('.hide'); |
|
if(hidden.length === this.attachMenuButtons.length) { |
|
toast(POSTING_MEDIA_NOT_ALLOWED); |
|
cancelEvent(e); |
|
return false; |
|
} |
|
}, {passive: false, capture: true}); */ |
|
|
|
this.listenerSetter.add(rootScope)('settings_updated', () => { |
|
if(this.stickersHelper || this.emojiHelper) { |
|
// this.previousQuery = undefined; |
|
this.previousQuery = ''; |
|
this.checkAutocomplete(); |
|
/* if(!rootScope.settings.stickers.suggest) { |
|
this.stickersHelper.checkEmoticon(''); |
|
} else { |
|
this.onMessageInput(); |
|
} */ |
|
} |
|
|
|
if(this.messageInputField) { |
|
this.messageInputField.onFakeInput(); |
|
} |
|
}); |
|
|
|
this.listenerSetter.add(rootScope)('draft_updated', ({peerId, threadId, draft, force}) => { |
|
if(this.chat.threadId !== threadId || this.chat.peerId !== peerId) return; |
|
this.setDraft(draft, true, force); |
|
}); |
|
|
|
this.listenerSetter.add(this.appImManager)('peer_changing', (chat) => { |
|
if(this.chat === chat) { |
|
this.saveDraft(); |
|
} |
|
}); |
|
|
|
this.listenerSetter.add(this.appImManager)('chat_changing', ({from, to}) => { |
|
if(this.chat === from) { |
|
this.autocompleteHelperController.toggleListNavigation(false); |
|
} else if(this.chat === to) { |
|
this.autocompleteHelperController.toggleListNavigation(true); |
|
} |
|
}); |
|
|
|
if(this.chat.type === 'scheduled') { |
|
this.listenerSetter.add(rootScope)('scheduled_delete', ({peerId, mids}) => { |
|
if(this.chat.peerId === peerId && mids.includes(this.editMsgId)) { |
|
this.onMessageSent(); |
|
} |
|
}); |
|
} else { |
|
this.listenerSetter.add(rootScope)('history_delete', ({peerId, msgs}) => { |
|
if(this.chat.peerId === peerId) { |
|
if(msgs.has(this.editMsgId)) { |
|
this.onMessageSent(); |
|
} |
|
|
|
if(this.replyToMsgId && msgs.has(this.replyToMsgId)) { |
|
this.clearHelper('reply'); |
|
} |
|
|
|
/* if(this.chat.isStartButtonNeeded()) { |
|
this.setStartParam(BOT_START_PARAM); |
|
} */ |
|
} |
|
}); |
|
|
|
this.listenerSetter.add(rootScope)('dialogs_multiupdate', (dialogs) => { |
|
if(dialogs[this.chat.peerId]) { |
|
if(this.startParam === BOT_START_PARAM) { |
|
this.setStartParam(); |
|
} else { // updateNewMessage comes earlier than dialog appers |
|
this.center(true); |
|
} |
|
} |
|
}); |
|
} |
|
|
|
try { |
|
this.recorder = new Recorder({ |
|
//encoderBitRate: 32, |
|
//encoderPath: "../dist/encoderWorker.min.js", |
|
encoderSampleRate: 48000, |
|
monitorGain: 0, |
|
numberOfChannels: 1, |
|
recordingGain: 1, |
|
reuseWorker: true |
|
}); |
|
} catch(err) { |
|
console.error('Recorder constructor error:', err); |
|
} |
|
|
|
this.updateSendBtn(); |
|
|
|
this.listenerSetter.add(this.fileInput)('change', (e) => { |
|
let files = (e.target as HTMLInputElement & EventTarget).files; |
|
if(!files.length) { |
|
return; |
|
} |
|
|
|
PopupElement.createPopup(PopupNewMedia, this.chat, Array.from(files).slice(), this.willAttachType); |
|
this.fileInput.value = ''; |
|
}, false); |
|
|
|
/* let time = Date.now(); |
|
this.btnSend.addEventListener('touchstart', (e) => { |
|
time = Date.now(); |
|
}); |
|
|
|
let eventName1 = 'touchend'; |
|
this.btnSend.addEventListener(eventName1, (e: Event) => { |
|
//cancelEvent(e); |
|
console.log(eventName1 + ', time: ' + (Date.now() - time)); |
|
}); |
|
|
|
let eventName = 'mousedown'; |
|
this.btnSend.addEventListener(eventName, (e: Event) => { |
|
cancelEvent(e); |
|
console.log(eventName + ', time: ' + (Date.now() - time)); |
|
}); */ |
|
attachClickEvent(this.btnSend, this.onBtnSendClick, {listenerSetter: this.listenerSetter, touchMouseDown: true}); |
|
|
|
if(this.recorder) { |
|
attachClickEvent(this.btnCancelRecord, this.onCancelRecordClick, {listenerSetter: this.listenerSetter}); |
|
|
|
this.recorder.onstop = () => { |
|
this.setRecording(false); |
|
this.chatInput.classList.remove('is-locked'); |
|
this.recordRippleEl.style.transform = ''; |
|
}; |
|
|
|
this.recorder.ondataavailable = (typedArray: Uint8Array) => { |
|
if(this.releaseMediaPlayback) { |
|
this.releaseMediaPlayback(); |
|
this.releaseMediaPlayback = undefined; |
|
} |
|
|
|
if(this.recordingOverlayListener) { |
|
this.listenerSetter.remove(this.recordingOverlayListener); |
|
this.recordingOverlayListener = undefined; |
|
} |
|
|
|
if(this.recordingNavigationItem) { |
|
appNavigationController.removeItem(this.recordingNavigationItem); |
|
this.recordingNavigationItem = undefined; |
|
} |
|
|
|
if(this.recordCanceled) { |
|
return; |
|
} |
|
|
|
const {peerId, threadId} = this.chat; |
|
const replyToMsgId = this.replyToMsgId; |
|
|
|
const duration = (Date.now() - this.recordStartTime) / 1000 | 0; |
|
const dataBlob = new Blob([typedArray], {type: 'audio/ogg'}); |
|
/* const fileName = new Date().toISOString() + ".opus"; |
|
console.log('Recorder data received', typedArray, dataBlob); */ |
|
|
|
//let perf = performance.now(); |
|
opusDecodeController.decode(typedArray, true).then((result) => { |
|
//console.log('WAVEFORM!:', /* waveform, */performance.now() - perf); |
|
|
|
opusDecodeController.setKeepAlive(false); |
|
|
|
// тут objectURL ставится уже с audio/wav |
|
this.managers.appMessagesManager.sendFile(peerId, dataBlob, { |
|
isVoiceMessage: true, |
|
isMedia: true, |
|
duration, |
|
waveform: result.waveform, |
|
objectURL: result.url, |
|
replyToMsgId, |
|
threadId, |
|
clearDraft: true |
|
}); |
|
|
|
this.onMessageSent(false, true); |
|
}); |
|
}; |
|
} |
|
|
|
attachClickEvent(this.replyElements.cancelBtn, this.onHelperCancel, {listenerSetter: this.listenerSetter}); |
|
attachClickEvent(this.replyElements.container, this.onHelperClick, {listenerSetter: this.listenerSetter}); |
|
|
|
this.saveDraftDebounced = debounce(() => this.saveDraft(), 2500, false, true); |
|
|
|
this.botStartBtn = Button('btn-primary btn-transparent text-bold chat-input-control-button'); |
|
this.botStartBtn.append(i18n('BotStart')); |
|
|
|
attachClickEvent(this.botStartBtn, () => { |
|
const {startParam} = this; |
|
if(startParam === undefined) { |
|
return; |
|
} |
|
|
|
const toggle = this.toggleBotStartBtnDisability = toggleDisability([this.botStartBtn], true); |
|
const peerId = this.chat.peerId; |
|
const middleware = this.chat.bubbles.getMiddleware(() => { |
|
return this.chat.peerId === peerId && this.startParam === startParam && this.toggleBotStartBtnDisability === toggle; |
|
}); |
|
|
|
this.managers.appMessagesManager.startBot(peerId.toUserId(), undefined, startParam).then(() => { |
|
if(middleware()) { |
|
toggle(); |
|
this.toggleBotStartBtnDisability = undefined; |
|
this.setStartParam(); |
|
} |
|
}); |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
this.controlContainer.append(this.botStartBtn); |
|
} |
|
|
|
public constructPinnedHelpers() { |
|
this.pinnedControlBtn = Button('btn-primary btn-transparent text-bold chat-input-control-button', {icon: 'unpin'}); |
|
this.controlContainer.append(this.pinnedControlBtn); |
|
|
|
this.listenerSetter.add(this.pinnedControlBtn)('click', () => { |
|
const peerId = this.chat.peerId; |
|
|
|
new PopupPinMessage(peerId, 0, true, () => { |
|
this.chat.appImManager.setPeer(); // * close tab |
|
|
|
// ! костыль, это скроет закреплённые сообщения сразу, вместо того, чтобы ждать пока анимация перехода закончится |
|
const originalChat = this.chat.appImManager.chat; |
|
if(originalChat.topbar.pinnedMessage) { |
|
originalChat.topbar.pinnedMessage.pinnedMessageContainer.toggle(true); |
|
} |
|
}); |
|
}); |
|
|
|
this.chatInput.classList.add('type-pinned'); |
|
} |
|
|
|
public _center(neededFakeContainer: HTMLElement, animate?: boolean) { |
|
if(!neededFakeContainer && !this.inputContainer.classList.contains('is-centering')) { |
|
return; |
|
} |
|
|
|
if(neededFakeContainer === this.fakeWrapperTo) { |
|
return; |
|
} |
|
|
|
/* if(neededFakeContainer === this.botStartContainer && this.fakeWrapperTo === this.fakeSelectionWrapper) { |
|
this.inputContainer.classList.remove('is-centering'); |
|
void this.rowsWrapper.offsetLeft; // reflow |
|
// this.inputContainer.classList.add('is-centering'); |
|
// void this.rowsWrapper.offsetLeft; // reflow |
|
} */ |
|
|
|
const fakeSelectionWrapper = neededFakeContainer || this.fakeWrapperTo; |
|
const forwards = !!neededFakeContainer; |
|
const oldFakeWrapperTo = this.fakeWrapperTo; |
|
let transform = '', borderRadius = '', needTranslateX: number; |
|
// if(forwards) {] |
|
const fakeSelectionRect = fakeSelectionWrapper.getBoundingClientRect(); |
|
const fakeRowsRect = this.fakeRowsWrapper.getBoundingClientRect(); |
|
const widthFrom = fakeRowsRect.width; |
|
const widthTo = fakeSelectionRect.width; |
|
|
|
if(widthFrom !== widthTo) { |
|
const scale = (widthTo/* - 8 */) / widthFrom; |
|
const initTranslateX = (widthFrom - widthTo) / 2; |
|
needTranslateX = fakeSelectionRect.left - fakeRowsRect.left - initTranslateX; |
|
|
|
if(forwards) { |
|
transform = `translateX(${needTranslateX}px) scaleX(${scale})`; |
|
// transform = `translateX(0px) scaleX(${scale})`; |
|
|
|
if(scale < 1) { |
|
const br = 12; |
|
borderRadius = '' + (br + br * (1 - scale)) + 'px'; |
|
} |
|
} |
|
//scale = widthTo / widthFrom; |
|
} |
|
// } |
|
|
|
this.fakeWrapperTo = neededFakeContainer; |
|
|
|
const duration = animate ? 200 : 0; |
|
SetTransition(this.inputContainer, 'is-centering', forwards, duration); |
|
SetTransition(this.rowsWrapperWrapper, 'is-centering-to-control', !!(forwards && neededFakeContainer && neededFakeContainer.classList.contains('chat-input-control')), duration); |
|
this.rowsWrapper.style.transform = transform; |
|
this.rowsWrapper.style.borderRadius = borderRadius; |
|
|
|
return { |
|
transform, |
|
borderRadius, |
|
needTranslateX: oldFakeWrapperTo && ( |
|
( |
|
neededFakeContainer && |
|
neededFakeContainer.classList.contains('chat-input-control') && |
|
oldFakeWrapperTo === this.fakeSelectionWrapper |
|
) || oldFakeWrapperTo.classList.contains('chat-input-control') |
|
) ? needTranslateX * -.5 : needTranslateX, |
|
widthFrom, |
|
widthTo |
|
}; |
|
} |
|
|
|
public async center(animate = false) { |
|
return this._center(await this.getNeededFakeContainer(), animate); |
|
} |
|
|
|
public setStartParam(startParam?: string) { |
|
if(this.startParam === startParam) { |
|
return; |
|
} |
|
|
|
this.startParam = startParam; |
|
this.center(true); |
|
} |
|
|
|
public async getNeededFakeContainer() { |
|
if(this.chat.selection.isSelecting) { |
|
return this.fakeSelectionWrapper; |
|
} else if( |
|
this.startParam !== undefined || |
|
!(await this.chat.canSend()) || |
|
this.chat.type === 'pinned' || |
|
await this.chat.isStartButtonNeeded() |
|
) { |
|
return this.controlContainer; |
|
} |
|
} |
|
|
|
// public getActiveContainer() { |
|
// if(this.chat.selection.isSelecting) { |
|
// return this.chat |
|
// } |
|
// return this.startParam !== undefined ? this.botStartContainer : this.rowsWrapper; |
|
// } |
|
|
|
// public setActiveContainer() { |
|
// const container = this.activeContainer; |
|
// const newContainer = this.getActiveContainer(); |
|
// if(newContainer === container) { |
|
// return; |
|
// } |
|
|
|
|
|
// } |
|
|
|
private onCancelRecordClick = (e?: Event) => { |
|
if(e) { |
|
cancelEvent(e); |
|
} |
|
|
|
this.recordCanceled = true; |
|
this.recorder.stop(); |
|
opusDecodeController.setKeepAlive(false); |
|
}; |
|
|
|
private onEmoticonsOpen = () => { |
|
const toggleClass = IS_TOUCH_SUPPORTED ? 'flip-icon' : 'active'; |
|
this.btnToggleEmoticons.classList.toggle(toggleClass, true); |
|
}; |
|
|
|
private onEmoticonsClose = () => { |
|
const toggleClass = IS_TOUCH_SUPPORTED ? 'flip-icon' : 'active'; |
|
this.btnToggleEmoticons.classList.toggle(toggleClass, false); |
|
}; |
|
|
|
public getReadyToSend(callback: () => void) { |
|
return this.chat.type === 'scheduled' ? (this.scheduleSending(callback), true) : (callback(), false); |
|
} |
|
|
|
public scheduleSending = async(callback: () => void = this.sendMessage.bind(this, true), initDate = new Date()) => { |
|
const {peerId} = this.chat; |
|
const middleware = this.chat.bubbles.getMiddleware(); |
|
const canSendWhenOnline = rootScope.myId !== peerId && peerId.isUser() && await this.managers.appUsersManager.isUserOnlineVisible(peerId); |
|
|
|
new PopupSchedule(initDate, (timestamp) => { |
|
if(!middleware()) { |
|
return; |
|
} |
|
|
|
const minTimestamp = (Date.now() / 1000 | 0) + 10; |
|
if(timestamp <= minTimestamp) { |
|
timestamp = undefined; |
|
} |
|
|
|
this.scheduleDate = timestamp; |
|
callback(); |
|
|
|
if(this.chat.type !== 'scheduled' && timestamp) { |
|
setTimeout(() => { // ! need timeout here because .forwardMessages will be called after timeout |
|
if(!middleware()) { |
|
return; |
|
} |
|
|
|
this.appImManager.openScheduled(peerId); |
|
}, 0); |
|
} |
|
}, canSendWhenOnline).show(); |
|
}; |
|
|
|
public async setUnreadCount() { |
|
if(!this.goDownUnreadBadge) { |
|
return; |
|
} |
|
|
|
const dialog = await this.managers.appMessagesManager.getDialogOnly(this.chat.peerId); |
|
const count = dialog?.unread_count; |
|
this.goDownUnreadBadge.innerText = '' + (count || ''); |
|
this.goDownUnreadBadge.classList.toggle('badge-gray', await this.managers.appNotificationsManager.isPeerLocalMuted(this.chat.peerId, true)); |
|
|
|
if(this.goMentionUnreadBadge && this.chat.type === 'chat') { |
|
const hasMentions = !!(dialog?.unread_mentions_count && dialog.unread_count); |
|
this.goMentionUnreadBadge.innerText = hasMentions ? '' + (dialog.unread_mentions_count) : ''; |
|
this.goMentionBtn.classList.toggle('is-visible', hasMentions); |
|
} |
|
} |
|
|
|
public saveDraft() { |
|
if(!this.chat.peerId || this.editMsgId || this.chat.type === 'scheduled') return; |
|
|
|
const {value, entities} = getRichValue(this.messageInputField.input); |
|
|
|
let draft: DraftMessage.draftMessage; |
|
if(value.length || this.replyToMsgId) { |
|
draft = { |
|
_: 'draftMessage', |
|
date: tsNow(true), |
|
message: value, |
|
entities: entities.length ? entities : undefined, |
|
pFlags: { |
|
no_webpage: this.noWebPage |
|
}, |
|
reply_to_msg_id: this.replyToMsgId |
|
}; |
|
} |
|
|
|
this.managers.appDraftsManager.syncDraft(this.chat.peerId, this.chat.threadId, draft); |
|
} |
|
|
|
public destroy() { |
|
//this.chat.log.error('Input destroying'); |
|
|
|
this.listenerSetter.removeAll(); |
|
} |
|
|
|
public cleanup(helperToo = true) { |
|
if(!this.chat.peerId) { |
|
this.chatInput.classList.add('hide'); |
|
this.goDownBtn.classList.add('hide'); |
|
} |
|
|
|
cancelSelection(); |
|
|
|
this.lastTimeType = 0; |
|
this.startParam = undefined; |
|
|
|
if(this.toggleBotStartBtnDisability) { |
|
this.toggleBotStartBtnDisability(); |
|
this.toggleBotStartBtnDisability = undefined; |
|
} |
|
|
|
if(this.messageInput) { |
|
this.clearInput(); |
|
helperToo && this.clearHelper(); |
|
} |
|
} |
|
|
|
public async setDraft(draft?: MyDraftMessage, fromUpdate = true, force = false) { |
|
if((!force && !isInputEmpty(this.messageInput)) || this.chat.type === 'scheduled') return false; |
|
|
|
if(!draft) { |
|
draft = await this.managers.appDraftsManager.getDraft(this.chat.peerId, this.chat.threadId); |
|
|
|
if(!draft) { |
|
if(force) { // this situation can only happen when sending message with clearDraft |
|
/* const height = this.chatInput.getBoundingClientRect().height; |
|
const willChangeHeight = 78 - height; |
|
this.willChangeHeight = willChangeHeight; */ |
|
if(this.chat.container.classList.contains('is-helper-active')) { |
|
this.t(); |
|
} |
|
|
|
this.messageInputField.inputFake.textContent = ''; |
|
this.messageInputField.onFakeInput(false); |
|
|
|
((this.chat.bubbles.messagesQueuePromise || Promise.resolve()) as Promise<any>).then(() => { |
|
fastRaf(() => { |
|
this.onMessageSent(); |
|
}); |
|
}); |
|
} |
|
|
|
return false; |
|
} |
|
} |
|
|
|
const wrappedDraft = wrapDraft(draft); |
|
|
|
if(this.messageInputField.value === wrappedDraft && this.replyToMsgId === draft.reply_to_msg_id) return false; |
|
|
|
if(fromUpdate) { |
|
this.clearHelper(); |
|
} |
|
|
|
this.noWebPage = draft.pFlags.no_webpage; |
|
if(draft.reply_to_msg_id) { |
|
this.initMessageReply(draft.reply_to_msg_id); |
|
} |
|
|
|
this.setInputValue(wrappedDraft, fromUpdate, fromUpdate); |
|
return true; |
|
} |
|
|
|
private createSendAs() { |
|
this.sendAsPeerId = undefined; |
|
|
|
if(this.chat.type === 'chat' || this.chat.type === 'discussion') { |
|
let firstChange = true; |
|
this.sendAs = new ChatSendAs( |
|
this.managers, |
|
(container, skipAnimation) => { |
|
let useRafs = 0; |
|
if(!container.parentElement) { |
|
this.newMessageWrapper.prepend(container); |
|
useRafs = 2; |
|
} |
|
|
|
this.updateOffset('as', true, skipAnimation, useRafs); |
|
}, |
|
(sendAsPeerId) => { |
|
this.sendAsPeerId = sendAsPeerId; |
|
|
|
// do not change placeholder earlier than finishPeerChange does |
|
if(firstChange) { |
|
firstChange = false; |
|
return; |
|
} |
|
|
|
this.getPlaceholderKey().then((key) => { |
|
this.updateMessageInputPlaceholder(key); |
|
}); |
|
} |
|
); |
|
} else { |
|
this.sendAs = undefined; |
|
} |
|
|
|
return this.sendAs; |
|
} |
|
|
|
public async finishPeerChange(startParam?: string) { |
|
const peerId = this.chat.peerId; |
|
|
|
const {forwardElements, btnScheduled, replyKeyboard, sendMenu, goDownBtn, chatInput, botCommandsToggle} = this; |
|
|
|
const previousSendAs = this.sendAs; |
|
const sendAs = this.createSendAs(); |
|
|
|
const [ |
|
isBroadcast, |
|
canPinMessage, |
|
isBot, |
|
canSend, |
|
neededFakeContainer, |
|
ackedPeerFull, |
|
ackedScheduledMids, |
|
setSendAsCallback, |
|
filteredAttachMenuButtons |
|
] = await Promise.all([ |
|
this.managers.appPeersManager.isBroadcast(peerId), |
|
this.managers.appPeersManager.canPinMessage(peerId), |
|
this.managers.appPeersManager.isBot(peerId), |
|
this.chat.canSend(), |
|
this.getNeededFakeContainer(), |
|
modifyAckedPromise(this.managers.acknowledged.appProfileManager.getProfileByPeerId(peerId)), |
|
btnScheduled ? modifyAckedPromise(this.managers.acknowledged.appMessagesManager.getScheduledMessages(peerId)) : undefined, |
|
sendAs ? (sendAs.setPeerId(this.chat.peerId), sendAs.updateManual(true)) : undefined, |
|
this.filterAttachMenuButtons() |
|
]); |
|
|
|
const placeholderKey = this.messageInput ? await this.getPlaceholderKey() : undefined; |
|
|
|
return () => { |
|
// console.warn('[input] finishpeerchange start'); |
|
|
|
chatInput.classList.remove('hide'); |
|
goDownBtn.classList.toggle('is-broadcast', isBroadcast); |
|
goDownBtn.classList.remove('hide'); |
|
|
|
if(this.goDownUnreadBadge) { |
|
this.setUnreadCount(); |
|
} |
|
|
|
if(this.chat.type === 'pinned') { |
|
chatInput.classList.toggle('can-pin', canPinMessage); |
|
}/* else if(this.chat.type === 'chat') { |
|
} */ |
|
|
|
if(forwardElements) { |
|
this.forwardWasDroppingAuthor = false; |
|
forwardElements.showCaption.checkboxField.setValueSilently(true); |
|
forwardElements.showSender.checkboxField.setValueSilently(true); |
|
} |
|
|
|
if(btnScheduled && ackedScheduledMids) { |
|
btnScheduled.classList.add('hide'); |
|
const middleware = this.chat.bubbles.getMiddleware(); |
|
callbackify(ackedScheduledMids.result, (mids) => { |
|
if(!middleware() || !mids) return; |
|
btnScheduled.classList.toggle('hide', !mids.length); |
|
}); |
|
} |
|
|
|
if(this.newMessageWrapper) { |
|
this.updateOffset(null, false, true); |
|
} |
|
|
|
if(botCommandsToggle) { |
|
this.hasBotCommands = undefined; |
|
this.botCommands.toggle(true, undefined, true); |
|
this.updateBotCommandsToggle(true); |
|
botCommandsToggle.remove(); |
|
if(isBot) { |
|
const middleware = this.chat.bubbles.getMiddleware(); |
|
const result = ackedPeerFull.result; |
|
callbackify(result, (userFull) => { |
|
if(!middleware()) return; |
|
this.updateBotCommands(userFull as UserFull.userFull, !(result instanceof Promise)); |
|
}); |
|
} |
|
} |
|
|
|
if(previousSendAs) { |
|
previousSendAs.destroy(); |
|
} |
|
|
|
if(setSendAsCallback) { |
|
setSendAsCallback(); |
|
} |
|
|
|
if(replyKeyboard) { |
|
replyKeyboard.setPeer(peerId); |
|
} |
|
|
|
if(sendMenu) { |
|
sendMenu.setPeerId(peerId); |
|
} |
|
|
|
if(this.messageInput) { |
|
this.updateMessageInput(canSend, placeholderKey, filteredAttachMenuButtons); |
|
} else if(this.pinnedControlBtn) { |
|
this.pinnedControlBtn.append(i18n(canPinMessage ? 'Chat.Input.UnpinAll' : 'Chat.Pinned.DontShow')); |
|
} |
|
|
|
// * testing |
|
// this.startParam = this.appPeersManager.isBot(peerId) ? '123' : undefined; |
|
|
|
this.startParam = startParam; |
|
|
|
this._center(neededFakeContainer, false); |
|
|
|
// console.warn('[input] finishpeerchange ends'); |
|
}; |
|
} |
|
|
|
private updateOffset(type: 'commands' | 'as', forwards: boolean, skipAnimation?: boolean, useRafs?: number) { |
|
if(type) { |
|
this.newMessageWrapper.dataset.offset = type; |
|
} else { |
|
delete this.newMessageWrapper.dataset.offset; |
|
} |
|
|
|
SetTransition(this.newMessageWrapper, 'has-offset', forwards, skipAnimation ? 0 : 300, undefined, useRafs); |
|
} |
|
|
|
private updateBotCommands(userFull: UserFull.userFull, skipAnimation?: boolean) { |
|
this.hasBotCommands = !!userFull.bot_info?.commands?.length; |
|
this.updateBotCommandsToggle(skipAnimation); |
|
} |
|
|
|
private updateBotCommandsToggle(skipAnimation?: boolean) { |
|
const {botCommandsToggle, hasBotCommands} = this; |
|
|
|
const show = !!hasBotCommands && this.isInputEmpty(); |
|
if(!hasBotCommands) { |
|
if(!botCommandsToggle.parentElement) { |
|
return; |
|
} |
|
|
|
botCommandsToggle.remove(); |
|
} |
|
|
|
const forwards = show; |
|
const useRafs = botCommandsToggle.parentElement ? 0 : 2; |
|
|
|
if(!botCommandsToggle.parentElement) { |
|
this.newMessageWrapper.prepend(botCommandsToggle); |
|
} |
|
|
|
this.updateOffset('commands', forwards, skipAnimation, useRafs); |
|
} |
|
|
|
private async getPlaceholderKey() { |
|
const {peerId, threadId} = this.chat; |
|
let key: LangPackKey; |
|
if(threadId) { |
|
key = 'Comment'; |
|
} else if(await this.managers.appPeersManager.isBroadcast(peerId)) { |
|
key = 'ChannelBroadcast'; |
|
} else if( |
|
(this.sendAsPeerId !== undefined && this.sendAsPeerId !== rootScope.myId) || |
|
await this.managers.appMessagesManager.isAnonymousSending(peerId) |
|
) { |
|
key = 'SendAnonymously'; |
|
} else { |
|
key = 'Message'; |
|
} |
|
|
|
return key; |
|
} |
|
|
|
private updateMessageInputPlaceholder(key: LangPackKey) { |
|
// console.warn('[input] update placeholder'); |
|
const i = I18n.weakMap.get(this.messageInput) as I18n.IntlElement; |
|
if(!i) { |
|
return; |
|
} |
|
|
|
i.compareAndUpdate({key}); |
|
} |
|
|
|
private filterAttachMenuButtons() { |
|
if(!this.attachMenuButtons) return; |
|
const {peerId, threadId} = this.chat; |
|
return filterAsync(this.attachMenuButtons, (button) => { |
|
return button.verify(peerId, threadId); |
|
}); |
|
} |
|
|
|
public updateMessageInput(canSend: boolean, placeholderKey: LangPackKey, visible: ChatInput['attachMenuButtons']) { |
|
const {chatInput, attachMenu, messageInput} = this; |
|
const {peerId, threadId} = this.chat; |
|
const isHidden = chatInput.classList.contains('is-hidden'); |
|
const willBeHidden = !canSend; |
|
if(isHidden !== willBeHidden) { |
|
chatInput.classList.add('no-transition'); |
|
chatInput.classList.toggle('is-hidden', !canSend); |
|
void chatInput.offsetLeft; // reflow |
|
chatInput.classList.remove('no-transition'); |
|
} |
|
|
|
this.updateMessageInputPlaceholder(placeholderKey); |
|
|
|
this.attachMenuButtons && this.attachMenuButtons.forEach((button) => { |
|
button.element.classList.toggle('hide', !visible.includes(button)); |
|
}); |
|
|
|
if(!canSend) { |
|
messageInput.removeAttribute('contenteditable'); |
|
} else { |
|
messageInput.setAttribute('contenteditable', 'true'); |
|
this.setDraft(undefined, false); |
|
|
|
if(!messageInput.innerHTML) { |
|
this.messageInputField.onFakeInput(); |
|
} |
|
} |
|
|
|
if(attachMenu) { |
|
attachMenu.toggleAttribute('disabled', !visible.length); |
|
attachMenu.classList.toggle('btn-disabled', !visible.length); |
|
} |
|
|
|
this.updateSendBtn(); |
|
} |
|
|
|
private attachMessageInputField() { |
|
const oldInputField = this.messageInputField; |
|
this.messageInputField = new InputField({ |
|
placeholder: 'Message', |
|
name: 'message', |
|
animate: true |
|
}); |
|
|
|
this.messageInputField.input.classList.replace('input-field-input', 'input-message-input'); |
|
this.messageInputField.inputFake.classList.replace('input-field-input', 'input-message-input'); |
|
this.messageInput = this.messageInputField.input; |
|
this.messageInput.classList.add('no-scrollbar'); |
|
this.attachMessageInputListeners(); |
|
|
|
if(IS_STICKY_INPUT_BUGGED) { |
|
fixSafariStickyInputFocusing(this.messageInput); |
|
} |
|
|
|
if(oldInputField) { |
|
oldInputField.input.replaceWith(this.messageInputField.input); |
|
oldInputField.inputFake.replaceWith(this.messageInputField.inputFake); |
|
} else { |
|
this.inputMessageContainer.append(this.messageInputField.input, this.messageInputField.inputFake); |
|
} |
|
} |
|
|
|
private attachMessageInputListeners() { |
|
this.listenerSetter.add(this.messageInput)('keydown', (e: KeyboardEvent) => { |
|
const key = e.key; |
|
if(isSendShortcutPressed(e)) { |
|
cancelEvent(e); |
|
this.sendMessage(); |
|
} else if(e.ctrlKey || e.metaKey) { |
|
this.handleMarkdownShortcut(e); |
|
} else if((key === 'PageUp' || key === 'PageDown') && !e.shiftKey) { // * fix pushing page to left (Chrome Windows) |
|
e.preventDefault(); |
|
|
|
if(key === 'PageUp') { |
|
const range = document.createRange(); |
|
const sel = window.getSelection(); |
|
|
|
range.setStart(this.messageInput.childNodes[0] || this.messageInput, 0); |
|
range.collapse(true); |
|
|
|
sel.removeAllRanges(); |
|
sel.addRange(range); |
|
} else { |
|
placeCaretAtEnd(this.messageInput); |
|
} |
|
} |
|
}); |
|
|
|
if(IS_TOUCH_SUPPORTED) { |
|
attachClickEvent(this.messageInput, (e) => { |
|
this.appImManager.selectTab(1); // * set chat tab for album orientation |
|
//this.saveScroll(); |
|
emoticonsDropdown.toggle(false); |
|
}, {listenerSetter: this.listenerSetter}); |
|
|
|
/* this.listenerSetter.add(window)('resize', () => { |
|
this.restoreScroll(); |
|
}); */ |
|
|
|
/* if(isSafari) { |
|
this.listenerSetter.add(this.messageInput)('mousedown', () => { |
|
window.requestAnimationFrame(() => { |
|
window.requestAnimationFrame(() => { |
|
emoticonsDropdown.toggle(false); |
|
}); |
|
}); |
|
}); |
|
} */ |
|
} |
|
|
|
/* this.listenerSetter.add(this.messageInput)('beforeinput', (e: Event) => { |
|
// * validate due to manual formatting through browser's context menu |
|
const inputType = (e as InputEvent).inputType; |
|
//console.log('message beforeinput event', e); |
|
|
|
if(inputType.indexOf('format') === 0) { |
|
//console.log('message beforeinput format', e, inputType, this.messageInput.innerHTML); |
|
const markdownType = inputType.split('format')[1].toLowerCase() as MarkdownType; |
|
if(this.applyMarkdown(markdownType)) { |
|
cancelEvent(e); // * cancel legacy markdown event |
|
} |
|
} |
|
}); */ |
|
this.listenerSetter.add(this.messageInput)('input', this.onMessageInput); |
|
this.listenerSetter.add(this.messageInput)('keyup', () => { |
|
this.checkAutocomplete(); |
|
}); |
|
|
|
if(this.chat.type === 'chat' || this.chat.type === 'discussion') { |
|
this.listenerSetter.add(this.messageInput)('focusin', () => { |
|
if(this.chat.bubbles.scrollable.loadedAll.bottom) { |
|
this.managers.appMessagesManager.readAllHistory(this.chat.peerId, this.chat.threadId); |
|
} |
|
}); |
|
} |
|
} |
|
|
|
private prepareDocumentExecute = () => { |
|
this.executedHistory.push(this.messageInput.innerHTML); |
|
return () => this.canUndoFromHTML = this.messageInput.innerHTML; |
|
}; |
|
|
|
private undoRedo = (e: Event, type: 'undo' | 'redo', needHTML: string) => { |
|
cancelEvent(e); // cancel legacy event |
|
|
|
let html = this.messageInput.innerHTML; |
|
if(html && html !== needHTML) { |
|
this.lockRedo = true; |
|
|
|
let sameHTMLTimes = 0; |
|
do { |
|
document.execCommand(type, false, null); |
|
const currentHTML = this.messageInput.innerHTML; |
|
if(html === currentHTML) { |
|
if(++sameHTMLTimes > 2) { // * unlink, removeFormat (а может и нет, случай: заболдить подчёркнутый текст (выделить ровно его), попробовать отменить) |
|
break; |
|
} |
|
} else { |
|
sameHTMLTimes = 0; |
|
} |
|
|
|
html = currentHTML; |
|
} while(html !== needHTML); |
|
|
|
this.lockRedo = false; |
|
} |
|
}; |
|
|
|
public applyMarkdown(type: MarkdownType, href?: string) { |
|
const MONOSPACE_FONT = 'var(--font-monospace)'; |
|
const SPOILER_FONT = 'spoiler'; |
|
const commandsMap: Partial<{[key in typeof type]: string | (() => void)}> = { |
|
bold: 'Bold', |
|
italic: 'Italic', |
|
underline: 'Underline', |
|
strikethrough: 'Strikethrough', |
|
monospace: () => document.execCommand('fontName', false, MONOSPACE_FONT), |
|
link: href ? () => document.execCommand('createLink', false, href) : () => document.execCommand('unlink', false, null), |
|
spoiler: () => document.execCommand('fontName', false, SPOILER_FONT) |
|
}; |
|
|
|
if(!commandsMap[type]) { |
|
return false; |
|
} |
|
|
|
const command = commandsMap[type]; |
|
|
|
//type = 'monospace'; |
|
|
|
const saveExecuted = this.prepareDocumentExecute(); |
|
const executed: any[] = []; |
|
/** |
|
* * clear previous formatting, due to Telegram's inability to handle several entities |
|
*/ |
|
/* const checkForSingle = () => { |
|
const nodes = getSelectedNodes(); |
|
//console.log('Using formatting:', commandsMap[type], nodes, this.executedHistory); |
|
|
|
const parents = [...new Set(nodes.map((node) => node.parentNode))]; |
|
//const differentParents = !!nodes.find((node) => node.parentNode !== firstParent); |
|
const differentParents = parents.length > 1; |
|
|
|
let notSingle = false; |
|
if(differentParents) { |
|
notSingle = true; |
|
} else { |
|
const node = nodes[0]; |
|
if(node && (node.parentNode as HTMLElement) !== this.messageInput && (node.parentNode.parentNode as HTMLElement) !== this.messageInput) { |
|
notSingle = true; |
|
} |
|
} |
|
|
|
if(notSingle) { |
|
//if(type === 'monospace') { |
|
executed.push(document.execCommand('styleWithCSS', false, 'true')); |
|
//} |
|
|
|
executed.push(document.execCommand('unlink', false, null)); |
|
executed.push(document.execCommand('removeFormat', false, null)); |
|
executed.push(typeof(command) === 'function' ? command() : document.execCommand(command, false, null)); |
|
|
|
//if(type === 'monospace') { |
|
executed.push(document.execCommand('styleWithCSS', false, 'false')); |
|
//} |
|
} |
|
}; */ |
|
|
|
executed.push(document.execCommand('styleWithCSS', false, 'true')); |
|
|
|
if(type === 'monospace' || type === 'spoiler') { |
|
let haveThisType = false; |
|
//executed.push(document.execCommand('styleWithCSS', false, 'true')); |
|
|
|
const selection = window.getSelection(); |
|
if(!selection.isCollapsed) { |
|
const range = selection.getRangeAt(0); |
|
const tag = markdownTags[type]; |
|
|
|
const node = range.commonAncestorContainer; |
|
if((node.parentNode as HTMLElement).matches(tag.match) || (node instanceof HTMLElement && node.matches(tag.match))) { |
|
haveThisType = true; |
|
} |
|
} |
|
|
|
//executed.push(document.execCommand('removeFormat', false, null)); |
|
|
|
if(haveThisType) { |
|
executed.push(this.resetCurrentFormatting()); |
|
} else { |
|
executed.push(typeof(command) === 'function' ? command() : document.execCommand(command, false, null)); |
|
} |
|
} else { |
|
executed.push(typeof(command) === 'function' ? command() : document.execCommand(command, false, null)); |
|
} |
|
|
|
executed.push(document.execCommand('styleWithCSS', false, 'false')); |
|
|
|
//checkForSingle(); |
|
saveExecuted(); |
|
if(this.appImManager.markupTooltip) { |
|
this.appImManager.markupTooltip.setActiveMarkupButton(); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
private resetCurrentFormatting() { |
|
return document.execCommand('fontName', false, 'Roboto'); |
|
} |
|
|
|
private handleMarkdownShortcut = (e: KeyboardEvent) => { |
|
// console.log('handleMarkdownShortcut', e); |
|
const formatKeys: {[key: string]: MarkdownType} = { |
|
'KeyB': 'bold', |
|
'KeyI': 'italic', |
|
'KeyU': 'underline', |
|
'KeyS': 'strikethrough', |
|
'KeyM': 'monospace', |
|
'KeyP': 'spoiler' |
|
}; |
|
|
|
if(this.appImManager.markupTooltip) { |
|
formatKeys['KeyK'] = 'link'; |
|
} |
|
|
|
const code = e.code; |
|
const applyMarkdown = formatKeys[code]; |
|
|
|
const selection = document.getSelection(); |
|
if(selection.toString().trim().length && applyMarkdown) { |
|
// * костыльчик |
|
if(code === 'KeyK') { |
|
this.appImManager.markupTooltip.showLinkEditor(); |
|
} else { |
|
this.applyMarkdown(applyMarkdown); |
|
} |
|
|
|
cancelEvent(e); // cancel legacy event |
|
} |
|
|
|
//return; |
|
if(code === 'KeyZ') { |
|
let html = this.messageInput.innerHTML; |
|
|
|
if(e.shiftKey) { |
|
if(this.undoHistory.length) { |
|
this.executedHistory.push(html); |
|
html = this.undoHistory.pop(); |
|
this.undoRedo(e, 'redo', html); |
|
html = this.messageInput.innerHTML; |
|
this.canRedoFromHTML = this.undoHistory.length ? html : ''; |
|
this.canUndoFromHTML = html; |
|
} |
|
} else { |
|
// * подождём, когда пользователь сам восстановит поле до нужного состояния, которое стало сразу после saveExecuted |
|
if(this.executedHistory.length && (!this.canUndoFromHTML || html === this.canUndoFromHTML)) { |
|
this.undoHistory.push(html); |
|
html = this.executedHistory.pop(); |
|
this.undoRedo(e, 'undo', html); |
|
|
|
// * поставим новое состояние чтобы снова подождать, если пользователь изменит что-то, и потом попробует откатить до предыдущего состояния |
|
this.canUndoFromHTML = this.canRedoFromHTML = this.messageInput.innerHTML; |
|
} |
|
} |
|
} |
|
}; |
|
|
|
private onMessageInput = (e?: Event) => { |
|
// * validate due to manual formatting through browser's context menu |
|
/* const inputType = (e as InputEvent).inputType; |
|
console.log('message input event', e); |
|
if(inputType === 'formatBold') { |
|
console.log('message input format', this.messageInput.innerHTML); |
|
cancelEvent(e); |
|
} |
|
|
|
if(!isSelectionSingle()) { |
|
alert('not single'); |
|
} */ |
|
|
|
//console.log('messageInput input', this.messageInput.innerText); |
|
//const value = this.messageInput.innerText; |
|
const {value: richValue, entities: markdownEntities, caretPos} = getRichValueWithCaret(this.messageInputField.input); |
|
|
|
//const entities = parseEntities(value); |
|
const value = parseMarkdown(richValue, markdownEntities, true); |
|
const entities = mergeEntities(markdownEntities, parseEntities(value)); |
|
|
|
//this.chat.log('messageInput entities', richValue, value, markdownEntities, caretPos); |
|
|
|
if(this.canRedoFromHTML && !this.lockRedo && this.messageInput.innerHTML !== this.canRedoFromHTML) { |
|
this.canRedoFromHTML = ''; |
|
this.undoHistory.length = 0; |
|
} |
|
|
|
const urlEntities: Array<MessageEntity.messageEntityUrl | MessageEntity.messageEntityTextUrl> = (!this.editMessage?.media || this.editMessage.media._ === 'messageMediaWebPage') && entities.filter((e) => e._ === 'messageEntityUrl' || e._ === 'messageEntityTextUrl') as any; |
|
if(urlEntities.length) { |
|
for(const entity of urlEntities) { |
|
let url: string; |
|
if(entity._ === 'messageEntityTextUrl') { |
|
url = entity.url; |
|
} else { |
|
url = richValue.slice(entity.offset, entity.offset + entity.length); |
|
|
|
if(!(url.includes('http://') || url.includes('https://'))) { |
|
continue; |
|
} |
|
} |
|
|
|
//console.log('messageInput url:', url); |
|
|
|
if(this.lastUrl !== url) { |
|
this.lastUrl = url; |
|
// this.willSendWebPage = null; |
|
const promise = this.getWebPagePromise = this.managers.appWebPagesManager.getWebPage(url).then((webpage) => { |
|
if(this.getWebPagePromise === promise) this.getWebPagePromise = undefined; |
|
if(this.lastUrl !== url) return; |
|
if(webpage._ === 'webPage') { |
|
//console.log('got webpage: ', webpage); |
|
|
|
this.setTopInfo('webpage', () => {}, webpage.site_name || webpage.title || 'Webpage', webpage.description || webpage.url || ''); |
|
delete this.noWebPage; |
|
this.willSendWebPage = webpage; |
|
} else if(this.willSendWebPage) { |
|
this.onHelperCancel(); |
|
} |
|
}); |
|
} |
|
|
|
break; |
|
} |
|
} else if(this.lastUrl) { |
|
this.lastUrl = ''; |
|
delete this.noWebPage; |
|
this.willSendWebPage = null; |
|
|
|
if(this.helperType) { |
|
this.helperFunc(); |
|
} else { |
|
this.clearHelper(); |
|
} |
|
} |
|
|
|
const isEmpty = !richValue.trim(); |
|
if(isEmpty) { |
|
if(this.lastTimeType) { |
|
this.managers.appMessagesManager.setTyping(this.chat.peerId, {_: 'sendMessageCancelAction'}); |
|
} |
|
|
|
if(this.appImManager.markupTooltip) { |
|
this.appImManager.markupTooltip.hide(); |
|
} |
|
|
|
// * Chrome has a bug - it will preserve the formatting if the input with monospace text is cleared |
|
// * so have to reset formatting |
|
if(document.activeElement === this.messageInput) { |
|
// document.execCommand('styleWithCSS', false, 'true'); |
|
setTimeout(() => { |
|
if(document.activeElement === this.messageInput) { |
|
this.resetCurrentFormatting(); |
|
} |
|
}, 0); |
|
// document.execCommand('styleWithCSS', false, 'false'); |
|
} |
|
} else { |
|
const time = Date.now(); |
|
if(time - this.lastTimeType >= 6000) { |
|
this.lastTimeType = time; |
|
this.managers.appMessagesManager.setTyping(this.chat.peerId, {_: 'sendMessageTypingAction'}); |
|
} |
|
|
|
if(this.botCommands) { |
|
this.botCommands.toggle(true); |
|
} |
|
} |
|
|
|
if(this.botCommands) { |
|
this.updateBotCommandsToggle(); |
|
} |
|
|
|
if(!this.editMsgId) { |
|
this.saveDraftDebounced(); |
|
} |
|
|
|
this.checkAutocomplete(richValue, caretPos, entities); |
|
|
|
this.updateSendBtn(); |
|
}; |
|
|
|
public insertAtCaret(insertText: string, insertEntity?: MessageEntity, isHelper = true) { |
|
const {value: fullValue, caretPos, entities} = getRichValueWithCaret(this.messageInput); |
|
const pos = caretPos >= 0 ? caretPos : fullValue.length; |
|
const prefix = fullValue.substr(0, pos); |
|
const suffix = fullValue.substr(pos); |
|
|
|
const matches = isHelper ? prefix.match(ChatInput.AUTO_COMPLETE_REG_EXP) : null; |
|
|
|
const matchIndex = matches ? matches.index + (matches[0].length - matches[2].length) : prefix.length; |
|
const newPrefix = prefix.slice(0, matchIndex); |
|
const newValue = newPrefix + insertText + suffix; |
|
|
|
// merge emojis |
|
const hadEntities = parseEntities(fullValue); |
|
mergeEntities(entities, hadEntities); |
|
|
|
// max for additional whitespace |
|
const insertLength = insertEntity ? Math.max(insertEntity.length, insertText.length) : insertText.length; |
|
const addEntities: MessageEntity[] = []; |
|
if(insertEntity) { |
|
addEntities.push(insertEntity); |
|
insertEntity.offset = matchIndex; |
|
} |
|
|
|
// add offset to entities next to emoji |
|
const diff = matches ? insertLength - matches[2].length : insertLength; |
|
entities.forEach((entity) => { |
|
if(entity.offset >= matchIndex) { |
|
entity.offset += diff; |
|
} |
|
}); |
|
|
|
mergeEntities(entities, addEntities); |
|
|
|
if(/* caretPos !== -1 && caretPos !== fullValue.length */true) { |
|
const caretEntity: MessageEntity.messageEntityCaret = { |
|
_: 'messageEntityCaret', |
|
offset: matchIndex + insertLength, |
|
length: 0 |
|
}; |
|
|
|
let insertCaretAtIndex = 0; |
|
for(let length = entities.length; insertCaretAtIndex < length; ++insertCaretAtIndex) { |
|
const entity = entities[insertCaretAtIndex]; |
|
if(entity.offset > caretEntity.offset) { |
|
break; |
|
} |
|
} |
|
|
|
entities.splice(insertCaretAtIndex, 0, caretEntity); |
|
} |
|
|
|
//const saveExecuted = this.prepareDocumentExecute(); |
|
// can't exec .value here because it will instantly check for autocomplete |
|
const value = documentFragmentToHTML(wrapDraftText(newValue, {entities})); |
|
this.messageInputField.setValueSilently(value, true); |
|
|
|
const caret = this.messageInput.querySelector('.composer-sel'); |
|
if(caret) { |
|
setCaretAt(caret); |
|
caret.remove(); |
|
} |
|
|
|
// but it's needed to be checked only here |
|
this.onMessageInput(); |
|
|
|
//saveExecuted(); |
|
|
|
//document.execCommand('insertHTML', true, wrapEmojiText(emoji)); |
|
} |
|
|
|
public onEmojiSelected = (emoji: string, autocomplete: boolean) => { |
|
this.insertAtCaret(emoji, getEmojiEntityFromEmoji(emoji), autocomplete); |
|
}; |
|
|
|
private async checkAutocomplete(value?: string, caretPos?: number, entities?: MessageEntity[]) { |
|
//return; |
|
|
|
if(value === undefined) { |
|
const r = getRichValueWithCaret(this.messageInputField.input, true); |
|
value = r.value; |
|
caretPos = r.caretPos; |
|
entities = r.entities; |
|
} |
|
|
|
if(caretPos === -1) { |
|
caretPos = value.length; |
|
} |
|
|
|
if(entities === undefined) { |
|
const _value = parseMarkdown(value, entities, true); |
|
entities = mergeEntities(entities, parseEntities(_value)); |
|
} |
|
|
|
value = value.slice(0, caretPos); |
|
|
|
if(this.previousQuery === value) { |
|
return; |
|
} |
|
|
|
this.previousQuery = value; |
|
|
|
const matches = value.match(ChatInput.AUTO_COMPLETE_REG_EXP); |
|
let foundHelper: AutocompleteHelper; |
|
if(matches) { |
|
const entity = entities[0]; |
|
|
|
let query = matches[2]; |
|
const firstChar = query[0]; |
|
|
|
if(this.stickersHelper && |
|
rootScope.settings.stickers.suggest && |
|
await this.chat.canSend('send_stickers') && |
|
entity?._ === 'messageEntityEmoji' && entity.length === value.length && !entity.offset) { |
|
foundHelper = this.stickersHelper; |
|
this.stickersHelper.checkEmoticon(value); |
|
} else if(firstChar === '@') { // mentions |
|
const topMsgId = this.chat.threadId ? getServerMessageId(this.chat.threadId) : undefined; |
|
if(await this.mentionsHelper.checkQuery(query, this.chat.peerId.isUser() ? NULL_PEER_ID : this.chat.peerId, topMsgId)) { |
|
foundHelper = this.mentionsHelper; |
|
} |
|
} else if(!matches[1] && firstChar === '/') { // commands |
|
if(await this.commandsHelper.checkQuery(query, this.chat.peerId)) { |
|
foundHelper = this.commandsHelper; |
|
} |
|
} else if(rootScope.settings.emoji.suggest) { // emoji |
|
query = query.replace(/^\s*/, ''); |
|
if(!value.match(/^\s*:(.+):\s*$/) && !value.match(/:[;!@#$%^&*()-=|]/) && query) { |
|
foundHelper = this.emojiHelper; |
|
this.emojiHelper.checkQuery(query, firstChar); |
|
} |
|
} |
|
} |
|
|
|
foundHelper = this.checkInlineAutocomplete(value, foundHelper); |
|
|
|
this.autocompleteHelperController.hideOtherHelpers(foundHelper); |
|
} |
|
|
|
private checkInlineAutocomplete(value: string, foundHelper?: AutocompleteHelper): AutocompleteHelper { |
|
let needPlaceholder = false; |
|
|
|
if(!foundHelper) { |
|
const inlineMatch = value.match(/^@([a-zA-Z\\d_]{3,32})\s/); |
|
if(inlineMatch) { |
|
const username = inlineMatch[1]; |
|
const query = value.slice(inlineMatch[0].length); |
|
needPlaceholder = inlineMatch[0].length === value.length; |
|
|
|
foundHelper = this.inlineHelper; |
|
|
|
if(!this.btnPreloader) { |
|
this.btnPreloader = ButtonIcon('none btn-preloader float show disable-hover', {noRipple: true}); |
|
putPreloader(this.btnPreloader, true); |
|
this.inputMessageContainer.parentElement.insertBefore(this.btnPreloader, this.inputMessageContainer.nextSibling); |
|
} else { |
|
SetTransition(this.btnPreloader, 'show', true, 400); |
|
} |
|
|
|
this.inlineHelper.checkQuery(this.chat.peerId, username, query).then(({user, renderPromise}) => { |
|
if(needPlaceholder && user.bot_inline_placeholder) { |
|
this.messageInput.dataset.inlinePlaceholder = user.bot_inline_placeholder; |
|
} |
|
|
|
renderPromise.then(() => { |
|
SetTransition(this.btnPreloader, 'show', false, 400); |
|
}); |
|
}).catch(noop); |
|
} |
|
} |
|
|
|
if(!needPlaceholder) { |
|
delete this.messageInput.dataset.inlinePlaceholder; |
|
} |
|
|
|
if(foundHelper !== this.inlineHelper) { |
|
if(this.btnPreloader) { |
|
SetTransition(this.btnPreloader, 'show', false, 400); |
|
} |
|
} |
|
|
|
return foundHelper; |
|
} |
|
|
|
private setRecording(value: boolean) { |
|
if(this.recording === value) { |
|
return; |
|
} |
|
|
|
SetTransition(this.chatInput, 'is-recording', value, 200); |
|
this.recording = value; |
|
this.updateSendBtn(); |
|
} |
|
|
|
private onBtnSendClick = async(e: Event) => { |
|
cancelEvent(e); |
|
|
|
if(!this.recorder || this.recording || !this.isInputEmpty() || this.forwarding || this.editMsgId) { |
|
if(this.recording) { |
|
if((Date.now() - this.recordStartTime) < RECORD_MIN_TIME) { |
|
this.onCancelRecordClick(); |
|
} else { |
|
this.recorder.stop(); |
|
} |
|
} else { |
|
this.sendMessage(); |
|
} |
|
} else { |
|
if(this.chat.peerId.isAnyChat() && !(await this.chat.canSend('send_media'))) { |
|
toast(POSTING_MEDIA_NOT_ALLOWED); |
|
return; |
|
} |
|
|
|
this.chatInput.classList.add('is-locked'); |
|
blurActiveElement(); |
|
|
|
this.recorder.start().then(() => { |
|
this.releaseMediaPlayback = appMediaPlaybackController.setSingleMedia(); |
|
this.recordCanceled = false; |
|
|
|
this.setRecording(true); |
|
opusDecodeController.setKeepAlive(true); |
|
|
|
const showDiscardPopup = () => { |
|
new PopupPeer('popup-cancel-record', { |
|
titleLangKey: 'DiscardVoiceMessageTitle', |
|
descriptionLangKey: 'DiscardVoiceMessageDescription', |
|
buttons: [{ |
|
langKey: 'DiscardVoiceMessageAction', |
|
callback: () => { |
|
simulateClickEvent(this.btnCancelRecord); |
|
} |
|
}, { |
|
langKey: 'Continue', |
|
isCancel: true |
|
}] |
|
}).show(); |
|
}; |
|
|
|
this.recordingOverlayListener = this.listenerSetter.add(document.body)('mousedown', (e) => { |
|
if(!findUpClassName(e.target, 'chat-input') && !findUpClassName(e.target, 'popup-cancel-record')) { |
|
cancelEvent(e); |
|
showDiscardPopup(); |
|
} |
|
}, {capture: true, passive: false}) as any; |
|
|
|
appNavigationController.pushItem(this.recordingNavigationItem = { |
|
type: 'voice', |
|
onPop: () => { |
|
setTimeout(() => { |
|
showDiscardPopup(); |
|
}, 0); |
|
|
|
return false; |
|
} |
|
}); |
|
|
|
this.recordStartTime = Date.now(); |
|
|
|
const sourceNode: MediaStreamAudioSourceNode = this.recorder.sourceNode; |
|
const context = sourceNode.context; |
|
|
|
const analyser = context.createAnalyser(); |
|
sourceNode.connect(analyser); |
|
//analyser.connect(context.destination); |
|
analyser.fftSize = 32; |
|
|
|
const frequencyData = new Uint8Array(analyser.frequencyBinCount); |
|
const max = frequencyData.length * 255; |
|
const min = 54 / 150; |
|
let r = () => { |
|
if(!this.recording) return; |
|
|
|
analyser.getByteFrequencyData(frequencyData); |
|
|
|
let sum = 0; |
|
frequencyData.forEach((value) => { |
|
sum += value; |
|
}); |
|
|
|
let percents = Math.min(1, (sum / max) + min); |
|
//console.log('frequencyData', frequencyData, percents); |
|
|
|
this.recordRippleEl.style.transform = `scale(${percents})`; |
|
|
|
let diff = Date.now() - this.recordStartTime; |
|
let ms = diff % 1000; |
|
|
|
let formatted = toHHMMSS(diff / 1000) + ',' + ('00' + Math.round(ms / 10)).slice(-2); |
|
|
|
this.recordTimeEl.innerText = formatted; |
|
|
|
fastRaf(r); |
|
}; |
|
|
|
r(); |
|
}).catch((e: Error) => { |
|
switch(e.name as string) { |
|
case 'NotAllowedError': { |
|
toast('Please allow access to your microphone'); |
|
break; |
|
} |
|
|
|
case 'NotReadableError': { |
|
toast(e.message); |
|
break; |
|
} |
|
|
|
default: |
|
console.error('Recorder start error:', e, e.name, e.message); |
|
toast(e.message); |
|
break; |
|
} |
|
|
|
this.setRecording(false); |
|
this.chatInput.classList.remove('is-locked'); |
|
}); |
|
} |
|
}; |
|
|
|
private onHelperCancel = (e?: Event, force?: boolean) => { |
|
if(e) { |
|
cancelEvent(e); |
|
} |
|
|
|
if(this.willSendWebPage) { |
|
const lastUrl = this.lastUrl; |
|
let needReturn = false; |
|
if(this.helperType) { |
|
//if(this.helperFunc) { |
|
this.helperFunc(); |
|
//} |
|
|
|
needReturn = true; |
|
} |
|
|
|
// * restore values |
|
this.lastUrl = lastUrl; |
|
this.noWebPage = true; |
|
this.willSendWebPage = null; |
|
|
|
if(needReturn) return; |
|
} |
|
|
|
if(this.helperType === 'edit' && !force) { |
|
const message = this.editMessage |
|
const value = parseMarkdown(this.messageInputField.value, []); |
|
if(message.message !== value) { |
|
new PopupPeer('discard-editing', { |
|
buttons: [{ |
|
langKey: 'Alert.Confirm.Discard', |
|
callback: () => { |
|
this.onHelperCancel(undefined, true); |
|
} |
|
}], |
|
descriptionLangKey: 'Chat.Edit.Cancel.Text' |
|
}).show(); |
|
|
|
return; |
|
} |
|
} |
|
|
|
this.clearHelper(); |
|
this.updateSendBtn(); |
|
}; |
|
|
|
private onHelperClick = (e: Event) => { |
|
cancelEvent(e); |
|
|
|
if(!findUpClassName(e.target, 'reply')) return; |
|
if(this.helperType === 'forward') { |
|
const {forwardElements} = this; |
|
if(forwardElements && IS_TOUCH_SUPPORTED && !forwardElements.container.classList.contains('active')) { |
|
contextMenuController.openBtnMenu(forwardElements.container); |
|
} |
|
} else if(this.helperType === 'reply') { |
|
this.chat.setMessageId(this.replyToMsgId); |
|
} else if(this.helperType === 'edit') { |
|
this.chat.setMessageId(this.editMsgId); |
|
} |
|
}; |
|
|
|
private changeForwardRecipient() { |
|
if(this.helperWaitingForward) return; |
|
this.helperWaitingForward = true; |
|
|
|
const forwarding = copy(this.forwarding); |
|
const helperFunc = this.helperFunc; |
|
this.clearHelper(); |
|
this.updateSendBtn(); |
|
let selected = false; |
|
const popup = new PopupForward(forwarding, () => { |
|
selected = true; |
|
}); |
|
|
|
popup.addEventListener('close', () => { |
|
this.helperWaitingForward = false; |
|
|
|
if(!selected) { |
|
helperFunc(); |
|
} |
|
}); |
|
} |
|
|
|
public async clearInput(canSetDraft = true, fireEvent = true, clearValue = '') { |
|
if(document.activeElement === this.messageInput && IS_MOBILE_SAFARI) { // fix first char uppercase |
|
const i = document.createElement('input'); |
|
document.body.append(i); |
|
fixSafariStickyInput(i); |
|
this.messageInputField.setValueSilently(clearValue); |
|
fixSafariStickyInput(this.messageInput); |
|
i.remove(); |
|
} else { |
|
this.messageInputField.setValueSilently(clearValue); |
|
} |
|
|
|
if(IS_TOUCH_SUPPORTED) { |
|
//this.messageInput.innerText = ''; |
|
} else { |
|
//this.attachMessageInputField(); |
|
//this.messageInput.innerText = ''; |
|
|
|
// clear executions |
|
this.canRedoFromHTML = ''; |
|
this.undoHistory.length = 0; |
|
this.executedHistory.length = 0; |
|
this.canUndoFromHTML = ''; |
|
} |
|
|
|
let set = false; |
|
if(canSetDraft) { |
|
set = await this.setDraft(undefined, false); |
|
} |
|
|
|
if(!set && fireEvent) { |
|
this.onMessageInput(); |
|
} |
|
} |
|
|
|
public isInputEmpty() { |
|
return isInputEmpty(this.messageInput); |
|
} |
|
|
|
public updateSendBtn() { |
|
let icon: 'send' | 'record' | 'edit' | 'schedule'; |
|
|
|
const isInputEmpty = this.isInputEmpty(); |
|
|
|
if(this.editMsgId) icon = 'edit'; |
|
else if(!this.recorder || this.recording || !isInputEmpty || this.forwarding) icon = this.chat.type === 'scheduled' ? 'schedule' : 'send'; |
|
else icon = 'record'; |
|
|
|
['send', 'record', 'edit', 'schedule'].forEach((i) => { |
|
this.btnSend.classList.toggle(i, icon === i); |
|
}); |
|
|
|
if(this.btnScheduled) { |
|
this.btnScheduled.classList.toggle('show', isInputEmpty); |
|
} |
|
|
|
if(this.btnToggleReplyMarkup) { |
|
this.btnToggleReplyMarkup.classList.toggle('show', isInputEmpty); |
|
} |
|
} |
|
|
|
public onMessageSent(clearInput = true, clearReply?: boolean) { |
|
if(this.chat.type !== 'scheduled') { |
|
this.managers.appMessagesManager.readAllHistory(this.chat.peerId, this.chat.threadId, true); |
|
} |
|
|
|
this.scheduleDate = undefined; |
|
this.sendSilent = undefined; |
|
|
|
const value = this.messageInputField.value; |
|
const entities = parseEntities(value); |
|
const emojiEntities: MessageEntity.messageEntityEmoji[] = entities.filter((entity) => entity._ === 'messageEntityEmoji') as any; |
|
emojiEntities.forEach((entity) => { |
|
const emoji = emojiFromCodePoints(entity.unicode); |
|
this.managers.appEmojiManager.pushRecentEmoji(emoji); |
|
}); |
|
|
|
if(clearInput) { |
|
this.lastUrl = ''; |
|
delete this.noWebPage; |
|
this.willSendWebPage = null; |
|
this.clearInput(); |
|
} |
|
|
|
if(clearReply || clearInput) { |
|
this.clearHelper(); |
|
} |
|
|
|
this.updateSendBtn(); |
|
} |
|
|
|
public sendMessage(force = false) { |
|
const {editMsgId, chat} = this; |
|
if(chat.type === 'scheduled' && !force && !editMsgId) { |
|
this.scheduleSending(); |
|
return; |
|
} |
|
|
|
const {peerId} = chat; |
|
const {noWebPage} = this; |
|
const sendingParams = this.chat.getMessageSendingParams(); |
|
|
|
const {value, entities} = getRichValue(this.messageInputField.input); |
|
|
|
//return; |
|
if(editMsgId) { |
|
const message = this.editMessage; |
|
if(value.trim() || message.media) { |
|
this.managers.appMessagesManager.editMessage(message, value, { |
|
entities, |
|
noWebPage: noWebPage |
|
}); |
|
|
|
this.onMessageSent(); |
|
} else { |
|
new PopupDeleteMessages(peerId, [editMsgId], chat.type); |
|
|
|
return; |
|
} |
|
} else if(value.trim()) { |
|
this.managers.appMessagesManager.sendText(peerId, value, { |
|
entities, |
|
...sendingParams, |
|
noWebPage: noWebPage, |
|
webPage: this.getWebPagePromise ? undefined : this.willSendWebPage, |
|
clearDraft: true |
|
}); |
|
|
|
if(this.chat.type === 'scheduled') { |
|
this.onMessageSent(true); |
|
} else { |
|
this.onMessageSent(false, false); |
|
} |
|
// this.onMessageSent(); |
|
} |
|
|
|
// * wait for sendText set messageId for invokeAfterMsg |
|
if(this.forwarding) { |
|
const forwarding = copy(this.forwarding); |
|
setTimeout(() => { |
|
for(const fromPeerId in forwarding) { |
|
this.managers.appMessagesManager.forwardMessages(peerId, fromPeerId.toPeerId(), forwarding[fromPeerId], { |
|
...sendingParams, |
|
dropAuthor: this.forwardElements && this.forwardElements.hideSender.checkboxField.checked, |
|
dropCaptions: this.isDroppingCaptions() |
|
}); |
|
} |
|
|
|
if(!value) { |
|
this.onMessageSent(); |
|
} |
|
}, 0); |
|
} |
|
|
|
// this.onMessageSent(); |
|
} |
|
|
|
public async sendMessageWithDocument(document: MyDocument | string, force = false, clearDraft = false) { |
|
document = await this.managers.appDocsManager.getDoc(document); |
|
|
|
const flag = document.type === 'sticker' ? 'send_stickers' : (document.type === 'gif' ? 'send_gifs' : 'send_media'); |
|
if(this.chat.peerId.isAnyChat() && !(await this.chat.canSend(flag))) { |
|
toast(POSTING_MEDIA_NOT_ALLOWED); |
|
return false; |
|
} |
|
|
|
if(this.chat.type === 'scheduled' && !force) { |
|
this.scheduleSending(() => this.sendMessageWithDocument(document, true, clearDraft)); |
|
return false; |
|
} |
|
|
|
if(document) { |
|
this.managers.appMessagesManager.sendFile(this.chat.peerId, document, { |
|
...this.chat.getMessageSendingParams(), |
|
isMedia: true, |
|
clearDraft: clearDraft || undefined |
|
}); |
|
this.onMessageSent(clearDraft, true); |
|
|
|
if(document.type === 'sticker') { |
|
emoticonsDropdown.stickersTab?.pushRecentSticker(document); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
private canToggleHideAuthor() { |
|
const {forwardElements} = this; |
|
if(!forwardElements) return false; |
|
const hideCaptionCheckboxField = forwardElements.hideCaption.checkboxField; |
|
return !hideCaptionCheckboxField.checked || |
|
findUpTag(hideCaptionCheckboxField.label, 'FORM').classList.contains('hide'); |
|
} |
|
|
|
private isDroppingCaptions() { |
|
return !this.canToggleHideAuthor(); |
|
} |
|
|
|
/* public sendSomething(callback: () => void, force = false) { |
|
if(this.chat.type === 'scheduled' && !force) { |
|
this.scheduleSending(() => this.sendSomething(callback, true)); |
|
return false; |
|
} |
|
|
|
callback(); |
|
this.onMessageSent(false, true); |
|
|
|
return true; |
|
} */ |
|
|
|
public async initMessageEditing(mid: number) { |
|
const message = (await this.chat.getMessage(mid)) as Message.message; |
|
|
|
let input = documentFragmentToHTML(wrapDraftText(message.message, {entities: message.totalEntities})); |
|
const f = async() => { |
|
const replyFragment = await wrapMessageForReply(message, undefined, [message.mid]); |
|
this.setTopInfo('edit', f, i18n('AccDescrEditing'), replyFragment, input, message); |
|
|
|
this.editMsgId = mid; |
|
this.editMessage = message; |
|
input = undefined; |
|
}; |
|
f(); |
|
} |
|
|
|
public initMessagesForward(fromPeerIdsMids: {[fromPeerId: PeerId]: number[]}) { |
|
const f = async() => { |
|
//const peerTitles: string[] |
|
const fromPeerIds = Object.keys(fromPeerIdsMids).map((fromPeerId) => fromPeerId.toPeerId()); |
|
const smth: Set<string> = new Set(); |
|
let length = 0, messagesWithCaptionsLength = 0; |
|
|
|
const p = fromPeerIds.map(async(fromPeerId) => { |
|
const mids = fromPeerIdsMids[fromPeerId]; |
|
const promises = mids.map(async(mid) => { |
|
const message = (await this.managers.appMessagesManager.getMessageByPeer(fromPeerId, mid)) as Message.message; |
|
if(message.fwd_from?.from_name && !message.fromId && !message.fwdFromId) { |
|
smth.add('N' + message.fwd_from.from_name); |
|
} else { |
|
smth.add('P' + message.fromId); |
|
} |
|
|
|
if(message.media && message.message) { |
|
++messagesWithCaptionsLength; |
|
} |
|
}); |
|
|
|
await Promise.all(promises); |
|
|
|
length += mids.length; |
|
}); |
|
|
|
await Promise.all(p); |
|
|
|
const onlyFirstName = smth.size > 2; |
|
const peerTitles = [...smth].map((smth) => { |
|
const type = smth[0]; |
|
smth = smth.slice(1); |
|
if(type === 'P') { |
|
const peerId = smth.toPeerId(); |
|
return peerId === rootScope.myId ? i18n('Chat.Accessory.Forward.You') : new PeerTitle({peerId, dialog: false, onlyFirstName}).element; |
|
} else { |
|
return onlyFirstName ? smth.split(' ')[0] : smth; |
|
} |
|
}); |
|
|
|
const {forwardElements} = this; |
|
const form = findUpTag(forwardElements.showCaption.checkboxField.label, 'FORM'); |
|
form.classList.toggle('hide', !messagesWithCaptionsLength); |
|
const hideCaption = forwardElements.hideCaption.checkboxField.checked; |
|
if(messagesWithCaptionsLength && hideCaption) { |
|
forwardElements.hideSender.checkboxField.setValueSilently(true); |
|
} else if(this.forwardWasDroppingAuthor !== undefined) { |
|
(this.forwardWasDroppingAuthor ? forwardElements.hideSender : forwardElements.showSender).checkboxField.setValueSilently(true); |
|
} |
|
|
|
const titleKey: LangPackKey = forwardElements.showSender.checkboxField.checked ? 'Chat.Accessory.Forward' : 'Chat.Accessory.Hidden'; |
|
const title = i18n(titleKey, [length]); |
|
|
|
const senderTitles = document.createDocumentFragment(); |
|
if(peerTitles.length < 3) { |
|
senderTitles.append(...join(peerTitles, false)); |
|
} else { |
|
senderTitles.append(peerTitles[0], i18n('AndOther', [peerTitles.length - 1])); |
|
} |
|
|
|
let firstMessage: Message.message, usingFullAlbum: boolean; |
|
if(fromPeerIds.length === 1) { |
|
const fromPeerId = fromPeerIds[0]; |
|
const mids = fromPeerIdsMids[fromPeerId]; |
|
firstMessage = (await this.managers.appMessagesManager.getMessageByPeer(fromPeerId, mids[0])) as Message.message; |
|
|
|
usingFullAlbum = !!firstMessage.grouped_id; |
|
if(usingFullAlbum) { |
|
const albumMids = await this.managers.appMessagesManager.getMidsByMessage(firstMessage); |
|
if(albumMids.length !== length || albumMids.find((mid) => !mids.includes(mid))) { |
|
usingFullAlbum = false; |
|
} |
|
} |
|
} |
|
|
|
const subtitleFragment = document.createDocumentFragment(); |
|
const delimiter = ': '; |
|
if(usingFullAlbum || length === 1) { |
|
const mids = fromPeerIdsMids[fromPeerIds[0]]; |
|
const replyFragment = await wrapMessageForReply(firstMessage, undefined, mids); |
|
subtitleFragment.append( |
|
senderTitles, |
|
delimiter, |
|
replyFragment |
|
); |
|
} else { |
|
subtitleFragment.append( |
|
i18n('Chat.Accessory.Forward.From'), |
|
delimiter, |
|
senderTitles |
|
); |
|
} |
|
|
|
let newReply = this.setTopInfo('forward', f, title, subtitleFragment); |
|
|
|
forwardElements.modifyArgs.forEach((b, idx) => { |
|
const text = b.textElement; |
|
const intl: I18n.IntlElement = I18n.weakMap.get(text) as any; |
|
intl.args = [idx < 2 ? fromPeerIds.length : messagesWithCaptionsLength]; |
|
intl.update(); |
|
}); |
|
|
|
if(this.forwardHover) { |
|
this.forwardHover.attachButtonListener(newReply, this.listenerSetter); |
|
} |
|
|
|
this.forwarding = fromPeerIdsMids; |
|
}; |
|
|
|
f(); |
|
} |
|
|
|
public async initMessageReply(mid: number) { |
|
if(this.replyToMsgId === mid) { |
|
return; |
|
} |
|
|
|
let message = await this.chat.getMessage(mid); |
|
const f = () => { |
|
let peerTitleEl: HTMLElement; |
|
if(!message) { // load missing replying message |
|
peerTitleEl = i18n('Loading'); |
|
|
|
this.managers.appMessagesManager.wrapSingleMessage(this.chat.peerId, mid).then((_message) => { |
|
if(this.replyToMsgId !== mid) { |
|
return; |
|
} |
|
|
|
message = _message; |
|
if(!message) { |
|
this.clearHelper('reply'); |
|
} else { |
|
f(); |
|
} |
|
}); |
|
} else { |
|
peerTitleEl = new PeerTitle({ |
|
peerId: message.fromId, |
|
dialog: false |
|
}).element; |
|
} |
|
|
|
this.setTopInfo('reply', f, peerTitleEl, message && (message as Message.message).message, undefined, message); |
|
this.replyToMsgId = mid; |
|
}; |
|
f(); |
|
} |
|
|
|
public clearHelper(type?: ChatInputHelperType) { |
|
if(this.helperType === 'edit' && type !== 'edit') { |
|
this.clearInput(); |
|
} |
|
|
|
if(type) { |
|
this.lastUrl = ''; |
|
delete this.noWebPage; |
|
this.willSendWebPage = null; |
|
} |
|
|
|
if(type !== 'reply') { |
|
this.replyToMsgId = undefined; |
|
this.forwarding = undefined; |
|
} |
|
|
|
this.editMsgId = this.editMessage = undefined; |
|
this.helperType = this.helperFunc = undefined; |
|
|
|
if(this.chat.container.classList.contains('is-helper-active')) { |
|
appNavigationController.removeByType('input-helper'); |
|
this.chat.container.classList.remove('is-helper-active'); |
|
this.t(); |
|
} |
|
} |
|
|
|
private t() { |
|
const className = 'is-toggling-helper'; |
|
SetTransition(this.chat.container, className, true, 150, () => { |
|
this.chat.container.classList.remove(className); |
|
}); |
|
} |
|
|
|
public setInputValue(value: string, clear = true, focus = true) { |
|
if(!value) value = ''; |
|
|
|
if(clear) this.clearInput(false, false, value); |
|
else this.messageInputField.setValueSilently(value); |
|
|
|
fastRaf(() => { |
|
focus && placeCaretAtEnd(this.messageInput); |
|
this.onMessageInput(); |
|
this.messageInput.scrollTop = this.messageInput.scrollHeight; |
|
}); |
|
} |
|
|
|
public setTopInfo( |
|
type: ChatInputHelperType, |
|
callerFunc: () => void, |
|
title: Parameters<typeof wrapReply>[0] = '', |
|
subtitle: Parameters<typeof wrapReply>[1] = '', |
|
input?: string, |
|
message?: any |
|
) { |
|
if(this.willSendWebPage && type === 'reply') { |
|
return; |
|
} |
|
|
|
if(type !== 'webpage') { |
|
this.clearHelper(type); |
|
this.helperType = type; |
|
this.helperFunc = callerFunc; |
|
} |
|
|
|
const replyParent = this.replyElements.container; |
|
const oldReply = replyParent.lastElementChild.previousElementSibling; |
|
const haveReply = oldReply.classList.contains('reply'); |
|
|
|
this.replyElements.iconBtn.replaceWith(this.replyElements.iconBtn = ButtonIcon((type === 'webpage' ? 'link' : type) + ' active reply-icon', {noRipple: true})); |
|
const {container} = wrapReply(title, subtitle, message); |
|
if(haveReply) { |
|
oldReply.replaceWith(container); |
|
} else { |
|
replyParent.insertBefore(container, replyParent.lastElementChild); |
|
} |
|
|
|
if(type === 'webpage') { |
|
container.style.cursor = 'default'; |
|
} |
|
|
|
if(!this.chat.container.classList.contains('is-helper-active')) { |
|
this.chat.container.classList.add('is-helper-active'); |
|
this.t(); |
|
} |
|
|
|
/* const scroll = appImManager.scrollable; |
|
if(scroll.isScrolledDown && !scroll.scrollLocked && !appImManager.messagesQueuePromise && !appImManager.setPeerPromise) { |
|
scroll.scrollTo(scroll.scrollHeight, 'top', true, true, 200); |
|
} */ |
|
|
|
if(!IS_MOBILE) { |
|
appNavigationController.pushItem({ |
|
type: 'input-helper', |
|
onPop: () => { |
|
this.onHelperCancel(); |
|
} |
|
}); |
|
} |
|
|
|
if(input !== undefined) { |
|
this.setInputValue(input); |
|
} |
|
|
|
setTimeout(() => { |
|
this.updateSendBtn(); |
|
}, 0); |
|
|
|
return container; |
|
} |
|
|
|
// public saveScroll() { |
|
// this.scrollTop = this.chat.bubbles.scrollable.container.scrollTop; |
|
// this.scrollOffsetTop = this.chatInput.offsetTop; |
|
// } |
|
|
|
// public restoreScroll() { |
|
// if(this.chatInput.style.display) return; |
|
// //console.log('input resize', offsetTop, this.chatInput.offsetTop); |
|
// let newOffsetTop = this.chatInput.offsetTop; |
|
// let container = this.chat.bubbles.scrollable.container; |
|
// let scrollTop = container.scrollTop; |
|
// let clientHeight = container.clientHeight; |
|
// let maxScrollTop = container.scrollHeight; |
|
|
|
// if(newOffsetTop < this.scrollOffsetTop) { |
|
// this.scrollDiff = this.scrollOffsetTop - newOffsetTop; |
|
// container.scrollTop += this.scrollDiff; |
|
// } else if(scrollTop !== this.scrollTop) { |
|
// let endDiff = maxScrollTop - (scrollTop + clientHeight); |
|
// if(endDiff < this.scrollDiff/* && false */) { |
|
// //container.scrollTop -= endDiff; |
|
// } else { |
|
// container.scrollTop -= this.scrollDiff; |
|
// } |
|
// } |
|
// } |
|
}
|
|
|