Eduard Kuzmenko
5 years ago
5 changed files with 492 additions and 503 deletions
@ -0,0 +1,446 @@ |
|||||||
|
import Scrollable from "./scrollable"; |
||||||
|
import LazyLoadQueue from "./lazyLoadQueue"; |
||||||
|
import { RichTextProcessor } from "../lib/richtextprocessor"; |
||||||
|
import apiManager from "../lib/mtproto/apiManager"; |
||||||
|
import appWebPagesManager from "../lib/appManagers/appWebPagesManager"; |
||||||
|
import appImManager from "../lib/appManagers/appImManager"; |
||||||
|
import { calcImageInBox, getRichValue } from "../lib/utils"; |
||||||
|
import { wrapDocument, wrapReply } from "./wrappers"; |
||||||
|
import appMessagesManager from "../lib/appManagers/appMessagesManager"; |
||||||
|
import initEmoticonsDropdown, { EMOTICONSSTICKERGROUP } from "./emoticonsDropdown"; |
||||||
|
import lottieLoader from "../lib/lottieLoader"; |
||||||
|
|
||||||
|
export class ChatInput { |
||||||
|
public pageEl = document.querySelector('.page-chats') as HTMLDivElement; |
||||||
|
public messageInput = document.getElementById('input-message') as HTMLDivElement/* HTMLInputElement */; |
||||||
|
public fileInput = document.getElementById('input-file') as HTMLInputElement; |
||||||
|
public inputMessageContainer = document.getElementsByClassName('input-message-container')[0] as HTMLDivElement; |
||||||
|
public inputScroll = new Scrollable(this.inputMessageContainer); |
||||||
|
public btnSend = document.getElementById('btn-send') as HTMLButtonElement; |
||||||
|
public emoticonsDropdown: HTMLDivElement = null; |
||||||
|
public emoticonsTimeout: number = 0; |
||||||
|
public toggleEmoticons: HTMLButtonElement; |
||||||
|
public emoticonsLazyLoadQueue: LazyLoadQueue = null; |
||||||
|
public lastUrl = ''; |
||||||
|
public lastTimeType = 0; |
||||||
|
|
||||||
|
public attachMenu: { |
||||||
|
container?: HTMLButtonElement, |
||||||
|
media?: HTMLDivElement, |
||||||
|
document?: HTMLDivElement, |
||||||
|
poll?: HTMLDivElement |
||||||
|
} = {}; |
||||||
|
|
||||||
|
public attachMediaPopUp: { |
||||||
|
container?: HTMLDivElement, |
||||||
|
titleEl?: HTMLDivElement, |
||||||
|
sendBtn?: HTMLButtonElement, |
||||||
|
mediaContainer?: HTMLDivElement, |
||||||
|
captionInput?: HTMLInputElement |
||||||
|
} = {}; |
||||||
|
|
||||||
|
public replyElements: { |
||||||
|
container?: HTMLDivElement, |
||||||
|
cancelBtn?: HTMLButtonElement, |
||||||
|
titleEl?: HTMLDivElement, |
||||||
|
subtitleEl?: HTMLDivElement |
||||||
|
} = {}; |
||||||
|
|
||||||
|
public willSendWebPage: any = null; |
||||||
|
public replyToMsgID = 0; |
||||||
|
public editMsgID = 0; |
||||||
|
public noWebPage = false; |
||||||
|
|
||||||
|
constructor() { |
||||||
|
this.toggleEmoticons = this.pageEl.querySelector('.toggle-emoticons') as HTMLButtonElement; |
||||||
|
|
||||||
|
this.attachMenu.container = document.getElementById('attach-file') as HTMLButtonElement; |
||||||
|
this.attachMenu.media = this.attachMenu.container.querySelector('.menu-media') as HTMLDivElement; |
||||||
|
this.attachMenu.document = this.attachMenu.container.querySelector('.menu-document') as HTMLDivElement; |
||||||
|
this.attachMenu.poll = this.attachMenu.container.querySelector('.menu-poll') as HTMLDivElement; |
||||||
|
|
||||||
|
this.attachMediaPopUp.container = this.pageEl.querySelector('.popup-send-photo') as HTMLDivElement; |
||||||
|
this.attachMediaPopUp.titleEl = this.attachMediaPopUp.container.querySelector('.popup-title') as HTMLDivElement; |
||||||
|
this.attachMediaPopUp.sendBtn = this.attachMediaPopUp.container.querySelector('.btn-primary') as HTMLButtonElement; |
||||||
|
this.attachMediaPopUp.mediaContainer = this.attachMediaPopUp.container.querySelector('.popup-photo') as HTMLDivElement; |
||||||
|
this.attachMediaPopUp.captionInput = this.attachMediaPopUp.container.querySelector('input') as HTMLInputElement; |
||||||
|
|
||||||
|
this.replyElements.container = this.pageEl.querySelector('.reply-wrapper') as HTMLDivElement; |
||||||
|
this.replyElements.cancelBtn = this.replyElements.container.querySelector('.reply-cancel') as HTMLButtonElement; |
||||||
|
this.replyElements.titleEl = this.replyElements.container.querySelector('.reply-title') as HTMLDivElement; |
||||||
|
this.replyElements.subtitleEl = this.replyElements.container.querySelector('.reply-subtitle') as HTMLDivElement; |
||||||
|
|
||||||
|
this.messageInput.addEventListener('keydown', (e: KeyboardEvent) => { |
||||||
|
if(e.key == 'Enter') { |
||||||
|
/* if(e.ctrlKey || e.metaKey) { |
||||||
|
this.messageInput.innerHTML += '<br>'; |
||||||
|
placeCaretAtEnd(this.message) |
||||||
|
return; |
||||||
|
} */ |
||||||
|
|
||||||
|
if(e.shiftKey || e.ctrlKey || e.metaKey) { |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
this.sendMessage(); |
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
this.messageInput.addEventListener('input', (e) => { |
||||||
|
//console.log('messageInput input', this.messageInput.innerText, this.serializeNodes(Array.from(this.messageInput.childNodes)));
|
||||||
|
|
||||||
|
let value = this.messageInput.innerText; |
||||||
|
|
||||||
|
let entities = RichTextProcessor.parseEntities(value); |
||||||
|
//console.log('messageInput entities', entities);
|
||||||
|
|
||||||
|
let entityUrl = entities.find(e => e._ == 'messageEntityUrl'); |
||||||
|
if(entityUrl) { // need to get webpage
|
||||||
|
let url = value.slice(entityUrl.offset, entityUrl.offset + entityUrl.length); |
||||||
|
|
||||||
|
//console.log('messageInput url:', url);
|
||||||
|
|
||||||
|
if(this.lastUrl != url) { |
||||||
|
this.lastUrl = url; |
||||||
|
this.willSendWebPage = null; |
||||||
|
apiManager.invokeApi('messages.getWebPage', { |
||||||
|
url: url, |
||||||
|
hash: 0 |
||||||
|
}).then((webpage: any) => { |
||||||
|
appWebPagesManager.saveWebPage(webpage); |
||||||
|
if(this.lastUrl != url) return; |
||||||
|
console.log('got webpage: ', webpage); |
||||||
|
|
||||||
|
this.setTopInfo(webpage.site_name || webpage.title, webpage.description || webpage.url); |
||||||
|
|
||||||
|
this.replyToMsgID = 0; |
||||||
|
this.noWebPage = false; |
||||||
|
this.willSendWebPage = webpage; |
||||||
|
}); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if(!value.trim() && !this.serializeNodes(Array.from(this.messageInput.childNodes)).trim()) { |
||||||
|
this.messageInput.innerHTML = ''; |
||||||
|
this.btnSend.classList.remove('tgico-send'); |
||||||
|
this.btnSend.classList.add('tgico-microphone2'); |
||||||
|
|
||||||
|
appImManager.setTyping('sendMessageCancelAction'); |
||||||
|
} else if(!this.btnSend.classList.contains('tgico-send')) { |
||||||
|
this.btnSend.classList.add('tgico-send'); |
||||||
|
this.btnSend.classList.remove('tgico-microphone2'); |
||||||
|
|
||||||
|
let time = Date.now(); |
||||||
|
if(time - this.lastTimeType >= 6000) { |
||||||
|
this.lastTimeType = time; |
||||||
|
appImManager.setTyping('sendMessageTypingAction'); |
||||||
|
} |
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
if(!RichTextProcessor.emojiSupported) { |
||||||
|
this.messageInput.addEventListener('copy', (e) => { |
||||||
|
const selection = document.getSelection(); |
||||||
|
|
||||||
|
let range = selection.getRangeAt(0); |
||||||
|
let ancestorContainer = range.commonAncestorContainer; |
||||||
|
|
||||||
|
let str = ''; |
||||||
|
|
||||||
|
let selectedNodes = Array.from(ancestorContainer.childNodes).slice(range.startOffset, range.endOffset); |
||||||
|
if(selectedNodes.length) { |
||||||
|
str = this.serializeNodes(selectedNodes); |
||||||
|
} else { |
||||||
|
str = selection.toString(); |
||||||
|
} |
||||||
|
|
||||||
|
//console.log('messageInput copy', str, ancestorContainer.childNodes, range);
|
||||||
|
|
||||||
|
//let str = getRichValueWithCaret(this.messageInput);
|
||||||
|
//console.log('messageInput childNode copy:', str);
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
event.clipboardData.setData('text/plain', str); |
||||||
|
event.preventDefault(); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
this.messageInput.addEventListener('paste', (e) => { |
||||||
|
//console.log('messageInput paste');
|
||||||
|
|
||||||
|
e.preventDefault(); |
||||||
|
// @ts-ignore
|
||||||
|
let text = (e.originalEvent || e).clipboardData.getData('text/plain'); |
||||||
|
|
||||||
|
// console.log('messageInput paste', text);
|
||||||
|
text = RichTextProcessor.wrapEmojiText(text); |
||||||
|
|
||||||
|
// console.log('messageInput paste after', text);
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
//let html = (e.originalEvent || e).clipboardData.getData('text/html');
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
//console.log('paste text', text, );
|
||||||
|
window.document.execCommand('insertHTML', false, text); |
||||||
|
}); |
||||||
|
|
||||||
|
let attachFile = (file: File) => { |
||||||
|
console.log('selected file:', file, typeof(file)); |
||||||
|
|
||||||
|
willAttachFile = file; |
||||||
|
|
||||||
|
this.fileInput.value = ''; |
||||||
|
|
||||||
|
this.attachMediaPopUp.captionInput.value = ''; |
||||||
|
this.attachMediaPopUp.mediaContainer.innerHTML = ''; |
||||||
|
this.attachMediaPopUp.mediaContainer.style.width = ''; |
||||||
|
this.attachMediaPopUp.mediaContainer.style.height = ''; |
||||||
|
this.attachMediaPopUp.mediaContainer.classList.remove('is-document'); |
||||||
|
|
||||||
|
if(file.type.indexOf('video/') === 0) { |
||||||
|
willAttach = 'document'; |
||||||
|
} else if(file.type.indexOf('image/') === -1 && willAttach == 'media') { |
||||||
|
willAttach = 'document'; |
||||||
|
} |
||||||
|
|
||||||
|
switch(willAttach) { |
||||||
|
case 'media': { |
||||||
|
let img = new Image(); |
||||||
|
img.src = URL.createObjectURL(file); |
||||||
|
img.onload = () => { |
||||||
|
willAttachWidth = img.naturalWidth; |
||||||
|
willAttachHeight = img.naturalHeight; |
||||||
|
|
||||||
|
let {w, h} = calcImageInBox(willAttachWidth, willAttachHeight, 378, 256); |
||||||
|
this.attachMediaPopUp.mediaContainer.style.width = w + 'px'; |
||||||
|
this.attachMediaPopUp.mediaContainer.style.height = h + 'px'; |
||||||
|
this.attachMediaPopUp.mediaContainer.append(img); |
||||||
|
}; |
||||||
|
|
||||||
|
this.attachMediaPopUp.titleEl.innerText = 'Send Photo'; |
||||||
|
this.attachMediaPopUp.container.classList.add('active'); |
||||||
|
|
||||||
|
break; |
||||||
|
} |
||||||
|
|
||||||
|
case 'document': { |
||||||
|
let docDiv = wrapDocument({ |
||||||
|
file: file, |
||||||
|
file_name: file.name || '', |
||||||
|
size: file.size, |
||||||
|
type: ['image/jpeg', |
||||||
|
'image/png', |
||||||
|
'image/gif', |
||||||
|
'image/webp', |
||||||
|
'image/bmp'].indexOf(file.type) !== -1 ? 'photo' : 'doc' |
||||||
|
} as any, false, true); |
||||||
|
|
||||||
|
this.attachMediaPopUp.titleEl.innerText = 'Send File'; |
||||||
|
|
||||||
|
this.attachMediaPopUp.mediaContainer.append(docDiv); |
||||||
|
this.attachMediaPopUp.mediaContainer.classList.add('is-document'); |
||||||
|
this.attachMediaPopUp.container.classList.add('active'); |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
let willAttach = ''; |
||||||
|
let willAttachFile: File = null; |
||||||
|
let willAttachWidth = 0, willAttachHeight = 0; |
||||||
|
this.fileInput.addEventListener('change', (e) => { |
||||||
|
var file = (e.target as HTMLInputElement & EventTarget).files[0]; |
||||||
|
if(!file) { |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
attachFile(file); |
||||||
|
}, false); |
||||||
|
|
||||||
|
this.attachMenu.media.addEventListener('click', () => { |
||||||
|
willAttach = 'media'; |
||||||
|
this.fileInput.click(); |
||||||
|
}); |
||||||
|
|
||||||
|
this.attachMenu.document.addEventListener('click', () => { |
||||||
|
willAttach = 'document'; |
||||||
|
this.fileInput.click(); |
||||||
|
}); |
||||||
|
|
||||||
|
document.addEventListener('paste', (event) => { |
||||||
|
if(!appImManager.peerID || this.attachMediaPopUp.container.classList.contains('active')) { |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
//console.log('document paste');
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
var items = (event.clipboardData || event.originalEvent.clipboardData).items; |
||||||
|
//console.log('item', event.clipboardData.getData());
|
||||||
|
for(let i = 0; i < items.length; ++i) { |
||||||
|
if(items[i].kind == 'file') { |
||||||
|
event.preventDefault() |
||||||
|
event.cancelBubble = true; |
||||||
|
event.stopPropagation(); |
||||||
|
|
||||||
|
let file = items[i].getAsFile(); |
||||||
|
//console.log(items[i], file);
|
||||||
|
if(!file) continue; |
||||||
|
|
||||||
|
willAttach = file.type.indexOf('image/') === 0 ? 'media' : "document"; |
||||||
|
attachFile(file); |
||||||
|
} |
||||||
|
} |
||||||
|
}, true); |
||||||
|
|
||||||
|
this.attachMediaPopUp.sendBtn.addEventListener('click', () => { |
||||||
|
this.attachMediaPopUp.container.classList.remove('active'); |
||||||
|
let caption = this.attachMediaPopUp.captionInput.value; |
||||||
|
|
||||||
|
appMessagesManager.sendFile(appImManager.peerID, willAttachFile, { |
||||||
|
isMedia: true, |
||||||
|
caption, |
||||||
|
width: willAttachWidth, |
||||||
|
height: willAttachHeight |
||||||
|
}); |
||||||
|
appImManager.scroll.scrollTop = appImManager.scroll.scrollHeight; |
||||||
|
|
||||||
|
let dialog = appMessagesManager.getDialogByPeerID(appImManager.peerID)[0]; |
||||||
|
if(dialog && dialog.top_message) { |
||||||
|
appMessagesManager.readHistory(appImManager.peerID, dialog.top_message); // lol
|
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
this.btnSend.addEventListener('click', () => { |
||||||
|
if(this.btnSend.classList.contains('tgico-send')) { |
||||||
|
this.sendMessage(); |
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
this.toggleEmoticons.onmouseover = (e) => { |
||||||
|
clearTimeout(this.emoticonsTimeout); |
||||||
|
this.emoticonsTimeout = setTimeout(() => { |
||||||
|
if(!this.emoticonsDropdown) { |
||||||
|
let res = initEmoticonsDropdown(this.pageEl, appImManager, |
||||||
|
appMessagesManager, this.messageInput, this.toggleEmoticons, this.btnSend); |
||||||
|
|
||||||
|
this.emoticonsDropdown = res.dropdown; |
||||||
|
this.emoticonsLazyLoadQueue = res.lazyLoadQueue; |
||||||
|
|
||||||
|
this.toggleEmoticons.onmouseout = this.emoticonsDropdown.onmouseout = (e) => { |
||||||
|
clearTimeout(this.emoticonsTimeout); |
||||||
|
this.emoticonsTimeout = setTimeout(() => { |
||||||
|
this.emoticonsDropdown.classList.remove('active'); |
||||||
|
this.toggleEmoticons.classList.remove('active'); |
||||||
|
lottieLoader.checkAnimations(true, EMOTICONSSTICKERGROUP); |
||||||
|
}, 200); |
||||||
|
}; |
||||||
|
|
||||||
|
this.emoticonsDropdown.onmouseover = (e) => { |
||||||
|
clearTimeout(this.emoticonsTimeout); |
||||||
|
}; |
||||||
|
} else { |
||||||
|
this.emoticonsDropdown.classList.add('active'); |
||||||
|
this.emoticonsLazyLoadQueue.check(); |
||||||
|
} |
||||||
|
|
||||||
|
this.toggleEmoticons.classList.add('active'); |
||||||
|
|
||||||
|
lottieLoader.checkAnimations(false, EMOTICONSSTICKERGROUP); |
||||||
|
}, 0/* 200 */); |
||||||
|
}; |
||||||
|
|
||||||
|
this.replyElements.cancelBtn.addEventListener('click', () => { |
||||||
|
this.replyElements.container.classList.remove('active'); |
||||||
|
this.replyToMsgID = 0; |
||||||
|
|
||||||
|
if(this.editMsgID) { |
||||||
|
if(this.willSendWebPage) { |
||||||
|
let message = appMessagesManager.getMessage(this.editMsgID); |
||||||
|
this.setTopInfo('Editing', message.message); |
||||||
|
} else { |
||||||
|
this.editMsgID = 0; |
||||||
|
this.messageInput.innerHTML = ''; |
||||||
|
this.btnSend.classList.remove('tgico-send'); |
||||||
|
this.btnSend.classList.add('tgico-microphone2'); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
this.noWebPage = true; |
||||||
|
this.willSendWebPage = null; |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
public serializeNodes(nodes: Node[]): string { |
||||||
|
return nodes.reduce((str, child: any) => { |
||||||
|
//console.log('childNode', str, child, typeof(child), typeof(child) === 'string', child.innerText);
|
||||||
|
|
||||||
|
if(typeof(child) === 'object' && child.textContent) return str += child.textContent; |
||||||
|
if(child.innerText) return str += child.innerText; |
||||||
|
if(child.tagName == 'IMG' && child.classList && child.classList.contains('emoji')) return str += child.getAttribute('alt'); |
||||||
|
|
||||||
|
return str; |
||||||
|
}, ''); |
||||||
|
}; |
||||||
|
|
||||||
|
public sendMessage() { |
||||||
|
//let str = this.serializeNodes(Array.from(this.messageInput.childNodes));
|
||||||
|
let str = getRichValue(this.messageInput); |
||||||
|
|
||||||
|
//console.log('childnode str after:', str/* , getRichValue(this.messageInput) */);
|
||||||
|
|
||||||
|
//return;
|
||||||
|
this.lastUrl = ''; |
||||||
|
|
||||||
|
if(this.editMsgID) { |
||||||
|
appMessagesManager.editMessage(this.editMsgID, str, { |
||||||
|
noWebPage: this.noWebPage |
||||||
|
}); |
||||||
|
} else { |
||||||
|
appMessagesManager.sendText(appImManager.peerID, str, { |
||||||
|
replyToMsgID: this.replyToMsgID == 0 ? undefined : this.replyToMsgID, |
||||||
|
noWebPage: this.noWebPage, |
||||||
|
webPage: this.willSendWebPage |
||||||
|
}); |
||||||
|
|
||||||
|
appImManager.scroll.scrollTop = appImManager.scroll.scrollHeight; |
||||||
|
} |
||||||
|
|
||||||
|
let dialog = appMessagesManager.getDialogByPeerID(appImManager.peerID)[0]; |
||||||
|
if(dialog && dialog.top_message) { |
||||||
|
appMessagesManager.readHistory(appImManager.peerID, dialog.top_message); // lol
|
||||||
|
} |
||||||
|
|
||||||
|
this.editMsgID = 0; |
||||||
|
this.replyToMsgID = 0; |
||||||
|
this.noWebPage = false; |
||||||
|
this.replyElements.container.classList.remove('active'); |
||||||
|
this.willSendWebPage = null; |
||||||
|
this.messageInput.innerText = ''; |
||||||
|
|
||||||
|
this.btnSend.classList.remove('tgico-send'); |
||||||
|
this.btnSend.classList.add('tgico-microphone2'); |
||||||
|
}; |
||||||
|
|
||||||
|
public setTopInfo(title: string, subtitle: string, input?: string, media?: any) { |
||||||
|
//appImManager.scrollPosition.prepareFor('down');
|
||||||
|
|
||||||
|
if(this.replyElements.container.lastElementChild.tagName == 'DIV') { |
||||||
|
this.replyElements.container.lastElementChild.remove(); |
||||||
|
this.replyElements.container.append(wrapReply(title, subtitle, media)); |
||||||
|
} |
||||||
|
//this.replyElements.titleEl.innerHTML = title ? RichTextProcessor.wrapEmojiText(title) : '';
|
||||||
|
//this.replyElements.subtitleEl.innerHTML = subtitle ? RichTextProcessor.wrapEmojiText(subtitle) : '';
|
||||||
|
this.replyElements.container.classList.add('active'); |
||||||
|
|
||||||
|
if(input !== undefined) { |
||||||
|
this.messageInput.innerHTML = input ? RichTextProcessor.wrapRichText(input) : ''; |
||||||
|
|
||||||
|
this.btnSend.classList.remove('tgico-microphone2'); |
||||||
|
this.btnSend.classList.add('tgico-send'); |
||||||
|
} |
||||||
|
|
||||||
|
//appImManager.scrollPosition.restore();
|
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue