import { MessageEntity } from "../layer"; import { MOUNT_CLASS_TO } from "../lib/mtproto/mtproto_config"; import RichTextProcessor from "../lib/richtextprocessor"; import ListenerSetter from "./listenerSetter"; import { isTouchSupported } from "./touchSupport"; import { isApple } from "./userAgent"; import rootScope from "../lib/rootScope"; /* export function isInDOM(element: Element, parentNode?: HTMLElement): boolean { if(!element) { return false; } parentNode = parentNode || document.body; if(element == parentNode) { return true; } return isInDOM(element.parentNode as HTMLElement, parentNode); } */ export function isInDOM(element: Element): boolean { return element?.isConnected; } /* export function checkDragEvent(e: any) { if(!e || e.target && (e.target.tagName == 'IMG' || e.target.tagName == 'A')) return false if(e.dataTransfer && e.dataTransfer.types) { for(var i = 0; i < e.dataTransfer.types.length; i++) { if(e.dataTransfer.types[i] == 'Files') { return true; } } } else { return true; } return false; } */ export function cancelEvent(event: Event) { event = event || window.event; if(event) { // @ts-ignore event = event.originalEvent || event; try { if(event.stopPropagation) event.stopPropagation(); if(event.preventDefault) event.preventDefault(); event.returnValue = false; event.cancelBubble = true; } catch(err) {} } return false; } export function placeCaretAtEnd(el: HTMLElement) { if(isTouchSupported) { return; } el.focus(); if(typeof window.getSelection != "undefined" && typeof document.createRange != "undefined") { var range = document.createRange(); range.selectNodeContents(el); range.collapse(false); var sel = window.getSelection(); sel.removeAllRanges(); sel.addRange(range); // @ts-ignore } else if(typeof document.body.createTextRange != "undefined") { // @ts-ignore var textRange = document.body.createTextRange(); textRange.moveToElementText(el); textRange.collapse(false); textRange.select(); } } /* export function getFieldSelection(field: any) { if(field.selectionStart) { return field.selectionStart; // @ts-ignore } else if(!document.selection) { return 0; } const c = '\x01'; // @ts-ignore const sel = document.selection.createRange(); const txt = sel.text; const dup = sel.duplicate(); let len = 0; try { dup.moveToElementText(field); } catch(e) { return 0; } sel.text = txt + c; len = dup.text.indexOf(c); sel.moveStart('character', -1); sel.text = ''; // if (browser.msie && len == -1) { // return field.value.length // } return len; } */ export function getRichValue(field: HTMLElement, entities?: MessageEntity[]) { if(!field) { return ''; } const lines: string[] = []; const line: string[] = []; getRichElementValue(field, lines, line, undefined, undefined, entities); if(line.length) { lines.push(line.join('')); } let value = lines.join('\n'); value = value.replace(/\u00A0/g, ' '); if(entities) { RichTextProcessor.combineSameEntities(entities); } //console.log('getRichValue:', value, entities); return value; } MOUNT_CLASS_TO && (MOUNT_CLASS_TO.getRichValue = getRichValue); export type MarkdownType = 'bold' | 'italic' | 'underline' | 'strikethrough' | 'monospace' | 'link'; export type MarkdownTag = { match: string, entityName: 'messageEntityBold' | 'messageEntityUnderline' | 'messageEntityItalic' | 'messageEntityPre' | 'messageEntityStrike' | 'messageEntityTextUrl'; }; export const markdownTags: {[type in MarkdownType]: MarkdownTag} = { bold: { match: '[style*="font-weight"], b', entityName: 'messageEntityBold' }, underline: { match: '[style*="underline"], u', entityName: 'messageEntityUnderline' }, italic: { match: '[style*="italic"], i', entityName: 'messageEntityItalic' }, monospace: { match: '[style*="monospace"], [face="monospace"]', entityName: 'messageEntityPre' }, strikethrough: { match: '[style*="line-through"], strike', entityName: 'messageEntityStrike' }, link: { match: 'A', entityName: 'messageEntityTextUrl' } }; export function getRichElementValue(node: HTMLElement, lines: string[], line: string[], selNode?: Node, selOffset?: number, entities?: MessageEntity[], offset = {offset: 0}) { if(node.nodeType == 3) { // TEXT if(selNode === node) { const value = node.nodeValue; line.push(value.substr(0, selOffset) + '\x01' + value.substr(selOffset)); } else { const nodeValue = node.nodeValue; line.push(nodeValue); if(entities && nodeValue.trim()) { if(node.parentNode) { const parentElement = node.parentElement; for(const type in markdownTags) { const tag = markdownTags[type as MarkdownType]; const closest = parentElement.closest(tag.match + ', [contenteditable]'); if(closest && closest.getAttribute('contenteditable') === null) { if(tag.entityName === 'messageEntityTextUrl') { entities.push({ _: tag.entityName as any, url: (parentElement as HTMLAnchorElement).href, offset: offset.offset, length: nodeValue.length }); } else { entities.push({ _: tag.entityName as any, offset: offset.offset, length: nodeValue.length }); } } } } } offset.offset += nodeValue.length; } return; } if(node.nodeType != 1) { // NON-ELEMENT return; } const isSelected = (selNode === node); const isBlock = node.tagName == 'DIV' || node.tagName == 'P'; if(isBlock && line.length || node.tagName == 'BR') { lines.push(line.join('')); line.splice(0, line.length); } else if(node.tagName == 'IMG') { const alt = (node as HTMLImageElement).alt; if(alt) { line.push(alt); offset.offset += alt.length; } } if(isSelected && !selOffset) { line.push('\x01'); } let curChild = node.firstChild as HTMLElement; while(curChild) { getRichElementValue(curChild, lines, line, selNode, selOffset, entities, offset); curChild = curChild.nextSibling as any; } if(isSelected && selOffset) { line.push('\x01'); } if(isBlock && line.length) { lines.push(line.join('')); line.splice(0, line.length); } } export function isInputEmpty(element: HTMLElement) { if(element.hasAttribute('contenteditable') || element.tagName != 'INPUT') { /* const value = element.innerText; return !value.trim() && !serializeNodes(Array.from(element.childNodes)).trim(); */ return !getRichValue(element).trim(); } else { return !(element as HTMLInputElement).value.trim(); } } /* export function 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; }, ''); } */ /* if (Config.Modes.animations && typeof window.requestAnimationFrame == 'function') { window.onAnimationFrameCallback = function (cb) { return (function () { window.requestAnimationFrame(cb) }) } } else { window.onAnimationFrameCallback = function (cb) { return cb } } */ // generate a path's arc data parameter // http://www.w3.org/TR/SVG/paths.html#PathDataEllipticalArcCommands var arcParameter = function(rx: number, ry: number, xAxisRotation: number, largeArcFlag: number, sweepFlag: number, x: number, y: number) { return [rx, ',', ry, ' ', xAxisRotation, ' ', largeArcFlag, ',', sweepFlag, ' ', x, ',', y ].join(''); }; export function generatePathData(x: number, y: number, width: number, height: number, tl: number, tr: number, br: number, bl: number) { const data: string[] = []; // start point in top-middle of the rectangle data.push('M' + (x + width / 2) + ',' + y); // next we go to the right data.push('H' + (x + width - tr)); if(tr > 0) { // now we draw the arc in the top-right corner data.push('A' + arcParameter(tr, tr, 0, 0, 1, (x + width), (y + tr))); } // next we go down data.push('V' + (y + height - br)); if(br > 0) { // now we draw the arc in the lower-right corner data.push('A' + arcParameter(br, br, 0, 0, 1, (x + width - br), (y + height))); } // now we go to the left data.push('H' + (x + bl)); if(bl > 0) { // now we draw the arc in the lower-left corner data.push('A' + arcParameter(bl, bl, 0, 0, 1, (x + 0), (y + height - bl))); } // next we go up data.push('V' + (y + tl)); if(tl > 0) { // now we draw the arc in the top-left corner data.push('A' + arcParameter(tl, tl, 0, 0, 1, (x + tl), (y + 0))); } // and we close the path data.push('Z'); return data.join(' '); }; MOUNT_CLASS_TO && (MOUNT_CLASS_TO.generatePathData = generatePathData); //export function findUpClassName(el: any, className: string): T; export function findUpClassName(el: any, className: string): HTMLElement { return el.closest('.' + className); /* if(el.classList.contains(className)) return el; // 03.02.2020 while(el.parentElement) { el = el.parentElement; if(el.classList.contains(className)) return el; } return null; */ } export function findUpTag(el: any, tag: string): HTMLElement { return el.closest(tag); /* if(el.tagName == tag) return el; // 03.02.2020 while(el.parentElement) { el = el.parentElement; if(el.tagName === tag) return el; } return null; */ } export function findUpAttribute(el: any, attribute: string): HTMLElement { return el.closest(`[${attribute}]`); /* if(el.getAttribute(attribute) != null) return el; // 03.02.2020 while(el.parentElement) { el = el.parentElement; if(el.getAttribute(attribute) != null) return el; } return null; */ } export function findUpAsChild(el: any, parent: any) { if(el.parentElement === parent) return el; while(el.parentElement) { el = el.parentElement; if(el.parentElement === parent) { return el; } } return null; } export function whichChild(elem: Node) { if(!elem.parentNode) { return -1; } let i = 0; // @ts-ignore while((elem = elem.previousElementSibling) != null) ++i; return i; }; export function fillPropertyValue(str: string) { let splitted = str.split(' '); if(splitted.length != 4) { if(!splitted[0]) splitted[0] = '0px'; for(let i = splitted.length; i < 4; ++i) { splitted[i] = splitted[i % 2] || splitted[0] || '0px'; } } return splitted; } export function calcImageInBox(imageW: number, imageH: number, boxW: number, boxH: number, noZoom = true) { if(imageW < boxW && imageH < boxH && noZoom) { return {w: imageW, h: imageH}; } let boxedImageW = boxW; let boxedImageH = boxH; if((imageW / imageH) > (boxW / boxH)) { boxedImageH = (imageH * boxW / imageW) | 0; } else { boxedImageW = (imageW * boxH / imageH) | 0; if(boxedImageW > boxW) { boxedImageH = (boxedImageH * boxW / boxedImageW) | 0; boxedImageW = boxW; } } // if (Config.Navigator.retina) { // imageW = Math.floor(imageW / 2) // imageH = Math.floor(imageH / 2) // } if(noZoom && boxedImageW >= imageW && boxedImageH >= imageH) { boxedImageW = imageW; boxedImageH = imageH; } return {w: boxedImageW, h: boxedImageH}; } MOUNT_CLASS_TO && (MOUNT_CLASS_TO.calcImageInBox = calcImageInBox); export function positionElementByIndex(element: HTMLElement, container: HTMLElement, pos: number) { const prevPos = element.parentElement === container ? whichChild(element) : -1; if(prevPos === pos) { return false; } else if(prevPos !== -1 && prevPos < pos) { // was higher pos += 1; } if(container.childElementCount > pos) { container.insertBefore(element, container.children[pos]); } else { container.append(element); } return true; } export function cancelSelection() { if(window.getSelection) { if(window.getSelection().empty) { // Chrome window.getSelection().empty(); } else if(window.getSelection().removeAllRanges) { // Firefox window.getSelection().removeAllRanges(); } // @ts-ignore } else if(document.selection) { // IE? // @ts-ignore document.selection.empty(); } } //(window as any).splitStringByLength = splitStringByLength; export function getSelectedText(): string { if(window.getSelection) { return window.getSelection().toString(); // @ts-ignore } else if(document.selection) { // @ts-ignore return document.selection.createRange().text; } return ''; } export function blurActiveElement() { if(document.activeElement && (document.activeElement as HTMLInputElement).blur) { (document.activeElement as HTMLInputElement).blur(); return true; } return false; } export function fixSafariStickyInput(input: HTMLElement) { input.style.transform = 'translateY(-99999px)'; /* input.style.position = 'fixed'; input.style.top = '-99999px'; input.style.left = '0'; */ input.focus(); setTimeout(() => { //fastSmoothScroll(findUpClassName(input, 'scrollable-y') || window as any, document.activeElement as HTMLElement, 'start', 4, undefined, FocusDirection.Static); /* input.style.position = ''; input.style.top = ''; */ input.style.transform = ''; //fastSmoothScroll(findUpClassName(input, 'scrollable-y') || window as any, document.activeElement as HTMLElement, 'start', 4, undefined, FocusDirection.Static); /* setTimeout(() => { fastSmoothScroll(findUpClassName(input, 'scrollable-y') || window as any, document.activeElement as HTMLElement, 'start', 4); }, 50); */ }, 0); } export const CLICK_EVENT_NAME: 'mousedown' | 'touchend' | 'click' = (isTouchSupported ? 'mousedown' : 'click') as any; export type AttachClickOptions = AddEventListenerOptions & Partial<{listenerSetter: ListenerSetter, touchMouseDown: true}>; export const attachClickEvent = (elem: HTMLElement, callback: (e: TouchEvent | MouseEvent) => void, options: AttachClickOptions = {}) => { const add = options.listenerSetter ? options.listenerSetter.add.bind(options.listenerSetter, elem) : elem.addEventListener.bind(elem); const remove = options.listenerSetter ? options.listenerSetter.removeManual.bind(options.listenerSetter, elem) : elem.removeEventListener.bind(elem); options.touchMouseDown = true; /* if(options.touchMouseDown && CLICK_EVENT_NAME === 'touchend') { add('mousedown', callback, options); } else if(CLICK_EVENT_NAME === 'touchend') { const o = {...options, once: true}; const onTouchStart = (e: TouchEvent) => { const onTouchMove = (e: TouchEvent) => { remove('touchmove', onTouchMove, o); remove('touchend', onTouchEnd, o); }; const onTouchEnd = (e: TouchEvent) => { remove('touchmove', onTouchMove, o); callback(e); if(options.once) { remove('touchstart', onTouchStart); } }; add('touchend', onTouchEnd, o); add('touchmove', onTouchMove, o); }; add('touchstart', onTouchStart); } else { add(CLICK_EVENT_NAME, callback, options); } */ add(CLICK_EVENT_NAME, callback, options); }; export const detachClickEvent = (elem: HTMLElement, callback: (e: TouchEvent | MouseEvent) => void, options?: AddEventListenerOptions) => { if(CLICK_EVENT_NAME == 'touchend') { elem.removeEventListener('touchstart', callback, options); } else { elem.removeEventListener(CLICK_EVENT_NAME, callback, options); } }; export const getSelectedNodes = () => { const nodes: Node[] = []; const selection = window.getSelection(); for(let i = 0; i < selection.rangeCount; ++i) { const range = selection.getRangeAt(i); let {startContainer, endContainer} = range; if(endContainer.nodeType != 3) endContainer = endContainer.firstChild; while(startContainer && startContainer != endContainer) { nodes.push(startContainer.nodeType == 3 ? startContainer : startContainer.firstChild); startContainer = startContainer.nextSibling; } if(nodes[nodes.length - 1] != endContainer) { nodes.push(endContainer); } } // * filter null's due to
return nodes.filter(node => !!node); }; export const isSelectionSingle = (input: Element = document.activeElement) => { const nodes = getSelectedNodes(); const parents = [...new Set(nodes.map(node => node.parentNode))]; const differentParents = parents.length > 1; let single = true; if(differentParents) { single = false; } else { const node = nodes[0]; if(node && node.parentNode != input && node.parentNode.parentNode != input) { single = false; } } return single; }; export const handleScrollSideEvent = (elem: HTMLElement, side: 'top' | 'bottom', callback: () => void, listenerSetter: ListenerSetter) => { if(isTouchSupported) { let lastY: number; const options = {passive: true}; listenerSetter.add(elem, 'touchstart', (e) => { if(e.touches.length > 1) { onTouchEnd(); return; } lastY = e.touches[0].clientY; listenerSetter.add(elem, 'touchmove', onTouchMove, options); listenerSetter.add(elem, 'touchend', onTouchEnd, options); }, options); const onTouchMove = (e: TouchEvent) => { const clientY = e.touches[0].clientY; const isDown = clientY < lastY; if(side == 'bottom' && isDown) callback(); else if(side == 'top' && !isDown) callback(); lastY = clientY; //alert('isDown: ' + !!isDown); }; const onTouchEnd = () => { listenerSetter.removeManual(elem, 'touchmove', onTouchMove, options); listenerSetter.removeManual(elem, 'touchend', onTouchEnd, options); }; } else { listenerSetter.add(elem, 'wheel', (e) => { const isDown = e.deltaY > 0; //this.log('wheel', e, isDown); if(side == 'bottom' && isDown) callback(); else if(side == 'top' && !isDown) callback(); }, {passive: true}); } }; export const getElementByPoint = (container: HTMLElement, verticalSide: 'top' | 'bottom', horizontalSide: 'center' | 'left'): HTMLElement => { const rect = container.getBoundingClientRect(); const x = horizontalSide === 'center' ? Math.ceil(rect.left + ((rect.right - rect.left) / 2) + 1) : Math.ceil(rect.left + 1); const y = verticalSide === 'bottom' ? Math.floor(rect.top + rect.height - 1) : Math.ceil(rect.top + 1); return document.elementFromPoint(x, y) as any; }; MOUNT_CLASS_TO && (MOUNT_CLASS_TO.getElementByPoint = getElementByPoint); export async function getFilesFromEvent(e: ClipboardEvent | DragEvent, onlyTypes = false): Promise { const files: any[] = []; const scanFiles = async(entry: any, item: DataTransferItem) => { if(entry.isDirectory) { const directoryReader = entry.createReader(); await new Promise((resolve, reject) => { directoryReader.readEntries(async(entries: any) => { for(const entry of entries) { await scanFiles(entry, item); } resolve(); }); }); } else if(entry) { if(onlyTypes) { files.push(entry.type); } else { const itemFile = item.getAsFile(); // * Safari can't handle entry.file with pasting const file = entry instanceof File ? entry : ( entry instanceof DataTransferItem ? entry.getAsFile() : await new Promise((resolve, reject) => entry.file(resolve, (err: any) => resolve(itemFile))) ); /* if(!onlyTypes) { console.log('getFilesFromEvent: got file', item, file); } */ if(!file) return; files.push(file); } } }; if(e instanceof DragEvent && e.dataTransfer.files && !e.dataTransfer.items) { for(let i = 0; i < e.dataTransfer.files.length; i++) { const file = e.dataTransfer.files[i]; files.push(onlyTypes ? file.type : file); } } else { // @ts-ignore const items = (e.dataTransfer || e.clipboardData || e.originalEvent.clipboardData).items; const promises: Promise[] = []; for(let i = 0; i < items.length; ++i) { const item: DataTransferItem = items[i]; if(item.kind === 'file') { const entry = (onlyTypes ? item : item.webkitGetAsEntry()) || item.getAsFile(); promises.push(scanFiles(entry, item)); } } await Promise.all(promises); } /* if(!onlyTypes) { console.log('getFilesFromEvent: got files:', e, files); } */ return files; } /* export function radiosHandleChange(inputs: HTMLInputElement[], onChange: (value: string) => void) { inputs.forEach(input => { input.addEventListener('change', () => { if(input.checked) { onChange(input.value); } }); }); } */ export function isSendShortcutPressed(e: KeyboardEvent) { if(e.key == 'Enter' && !isTouchSupported) { /* if(e.ctrlKey || e.metaKey) { this.messageInput.innerHTML += '
'; placeCaretAtEnd(this.message) return; } */ if(rootScope.settings.sendShortcut === 'enter') { if(e.shiftKey || e.ctrlKey || e.metaKey) { return; } return true; } else { const secondaryKey = isApple ? e.metaKey : e.ctrlKey; if(e.shiftKey || (isApple ? e.ctrlKey : e.metaKey)) { return; } if(secondaryKey) { return true; } } } return false; } export function reflowScrollableElement(element: HTMLElement) { element.style.display = 'none'; void element.offsetLeft; // reflow element.style.display = ''; } export function isSelectionEmpty(selection = window.getSelection()) { if(!selection || !selection.rangeCount) { return true; } const selectionRange = selection.getRangeAt(0); if(!selectionRange.toString() || !selectionRange.START_TO_END) { return true; } return false; }