// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import emojiRegex from 'emoji-regex'; import Delta from 'quill-delta'; import type { LeafBlot, DeltaOperation, AttributeMap } from 'quill'; import type Op from 'quill-delta/dist/Op'; import type { DisplayNode, DraftBodyRange, DraftBodyRanges, } from '../types/BodyRange'; import { BodyRange } from '../types/BodyRange'; import type { MentionBlot } from './mentions/blot'; import { isNewlineOnlyOp, QuillFormattingStyle } from './formatting/menu'; import { isNotNil } from '../util/isNotNil'; import type { AciString } from '../types/ServiceId'; export type MentionBlotValue = { uuid: AciString; title: string; }; export type FormattingBlotValue = { style: BodyRange.Style; }; export const isMentionBlot = (blot: LeafBlot): blot is MentionBlot => blot.value() && blot.value().mention; export const isFormatting = (blot: LeafBlot): blot is MentionBlot => blot.value() && blot.value().style; export type RetainOp = Op & { retain: number }; export type InsertOp = Op & { insert: { [V in K]: T } }; export type InsertMentionOp = InsertOp<'mention', MentionBlotValue>; export type InsertEmojiOp = InsertOp<'emoji', string>; export const isRetainOp = (op?: Op): op is RetainOp => op !== undefined && op.retain !== undefined; export const isSpecificInsertOp = (op: Op, type: string): boolean => { return ( op.insert !== undefined && typeof op.insert === 'object' && Object.hasOwnProperty.call(op.insert, type) ); }; export const isInsertEmojiOp = (op: Op): op is InsertEmojiOp => isSpecificInsertOp(op, 'emoji'); export const isInsertMentionOp = (op: Op): op is InsertMentionOp => isSpecificInsertOp(op, 'mention'); export const getTextFromOps = (ops: Array): string => ops .reduce((acc, op) => { if (typeof op.insert === 'string') { return acc + op.insert; } if (isInsertEmojiOp(op)) { return acc + op.insert.emoji; } if (isInsertMentionOp(op)) { return `${acc}@${op.insert.mention.title}`; } return acc; }, '') .trim(); const { BOLD, ITALIC, MONOSPACE, SPOILER, STRIKETHROUGH, NONE } = BodyRange.Style; function extractFormatRange({ bodyRanges, index, previousData, hasStyle, style, }: { bodyRanges: Array; index: number; previousData: { start: number } | undefined; hasStyle: boolean; style: BodyRange.Style; }) { if (hasStyle && !previousData) { return { start: index }; } if (!hasStyle && previousData) { const { start } = previousData; bodyRanges.push({ length: index - start, start, style, }); return undefined; } return previousData; } function extractAllFormats( bodyRanges: Array, formats: Record, index: number, op?: Op ): Record { const result = { ...formats }; const params = { bodyRanges, index, }; result[BOLD] = extractFormatRange({ ...params, style: BOLD, previousData: result[BOLD], hasStyle: op?.attributes?.[QuillFormattingStyle.bold], }); result[ITALIC] = extractFormatRange({ ...params, style: ITALIC, previousData: result[ITALIC], hasStyle: op?.attributes?.[QuillFormattingStyle.italic], }); result[MONOSPACE] = extractFormatRange({ ...params, style: MONOSPACE, previousData: result[MONOSPACE], hasStyle: op?.attributes?.[QuillFormattingStyle.monospace], }); result[SPOILER] = extractFormatRange({ ...params, style: SPOILER, previousData: result[SPOILER], hasStyle: op?.attributes?.[QuillFormattingStyle.spoiler], }); result[STRIKETHROUGH] = extractFormatRange({ ...params, style: STRIKETHROUGH, previousData: result[STRIKETHROUGH], hasStyle: op?.attributes?.[QuillFormattingStyle.strike], }); return result; } export const getTextAndRangesFromOps = ( ops: Array ): { text: string; bodyRanges: DraftBodyRanges } => { const startingBodyRanges: Array = []; let formats: Record = { [BOLD]: undefined, [ITALIC]: undefined, [MONOSPACE]: undefined, [SPOILER]: undefined, [STRIKETHROUGH]: undefined, [NONE]: undefined, }; const preTrimText = ops.reduce((acc, op) => { // We special-case all-newline ops because Quill doesn't apply styles to them if (isNewlineOnlyOp(op)) { return acc + op.insert; } // Start or finish format sections as needed formats = extractAllFormats(startingBodyRanges, formats, acc.length, op); if (typeof op.insert === 'string') { return acc + op.insert; } if (isInsertEmojiOp(op)) { return acc + op.insert.emoji; } if (isInsertMentionOp(op)) { startingBodyRanges.push({ length: 1, // The length of `\uFFFC` mentionUuid: op.insert.mention.uuid, replacementText: op.insert.mention.title, start: acc.length, }); return `${acc}\uFFFC`; } return acc; }, ''); // Close off any pending formats extractAllFormats(startingBodyRanges, formats, preTrimText.length); // Now repair bodyRanges after trimming const trimStart = preTrimText.trimStart(); const trimmedFromStart = preTrimText.length - trimStart.length; const text = trimStart.trimEnd(); const textLength = text.length; const bodyRanges = startingBodyRanges .map(startingRange => { let range = { ...startingRange, start: startingRange.start - trimmedFromStart, }; if (range.start >= text.length) { return null; } const underStartBy = -range.start; if (underStartBy > 0) { const length = range.length - underStartBy; if (length <= 0) { return null; } range = { ...range, start: 0, length, }; } const end = range.start + range.length; const overEndBy = end - textLength; if (overEndBy > 0) { range = { ...range, length: range.length - overEndBy, }; } return range; }) .filter(isNotNil); return { text, bodyRanges }; }; export const getBlotTextPartitions = ( blotText: string | undefined, index: number ): [string, string] => { const lowerCaseBlotText = (blotText || '').toLowerCase(); const leftLeafText = lowerCaseBlotText.substr(0, index); const rightLeafText = lowerCaseBlotText.substr(index); return [leftLeafText, rightLeafText]; }; export const matchBlotTextPartitions = ( blot: LeafBlot, index: number, leftRegExp: RegExp, rightRegExp?: RegExp ): Array => { const [leftText, rightText] = getBlotTextPartitions(blot.text, index); const leftMatch = leftRegExp.exec(leftText); let rightMatch = null; if (rightRegExp) { rightMatch = rightRegExp.exec(rightText); } return [leftMatch, rightMatch]; }; export const getDeltaToRestartMention = (ops: Array): Delta => { const changes = ops.reduce((acc, op): Array => { if (op.insert && typeof op.insert === 'string') { acc.push({ retain: op.insert.length }); } else { acc.push({ retain: 1 }); } return acc; }, Array()); changes.push({ delete: 1 }); changes.push({ insert: '@' }); return new Delta(changes); }; export const getDeltaToRemoveStaleMentions = ( ops: Array, memberUuids: Array ): Delta => { const newOps = ops.reduce((memo, op) => { if (op.insert) { if ( isInsertMentionOp(op) && !memberUuids.includes(op.insert.mention.uuid) ) { const deleteOp = { delete: 1 }; const textOp = { insert: `@${op.insert.mention.title}` }; return [...memo, deleteOp, textOp]; } if (typeof op.insert === 'string') { const retainStringOp = { retain: op.insert.length }; return [...memo, retainStringOp]; } const retainEmbedOp = { retain: 1 }; return [...memo, retainEmbedOp]; } return [...memo, op]; }, Array()); return new Delta(newOps); }; export const insertFormattingAndMentionsOps = ( nodes: ReadonlyArray ): ReadonlyArray => { let ops: Array = []; nodes.forEach(node => { const startingOp: Op = { insert: node.text, attributes: { [QuillFormattingStyle.bold]: node.isBold, [QuillFormattingStyle.italic]: node.isItalic, [QuillFormattingStyle.monospace]: node.isMonospace, [QuillFormattingStyle.spoiler]: node.isSpoiler, [QuillFormattingStyle.strike]: node.isStrikethrough, }, }; ops = ops.concat(insertMentionOps([startingOp], node.mentions)); }); return ops; }; export const insertMentionOps = ( incomingOps: Array, bodyRanges: DraftBodyRanges ): Array => { const ops = [...incomingOps]; const sortableBodyRanges: Array = bodyRanges.slice(); // Working backwards through bodyRanges (to avoid offsetting later mentions), // Shift off the op with the text to the left of the last mention, // Insert a mention based on the current bodyRange, // Unshift the mention and surrounding text to leave the ops ready for the next range sortableBodyRanges .sort((a, b) => b.start - a.start) .forEach(bodyRange => { if (!BodyRange.isMention(bodyRange)) { return; } const { start, length, mentionUuid, replacementText } = bodyRange; const op = ops.shift(); if (op) { const { insert, attributes } = op; if (typeof insert === 'string') { const left = insert.slice(0, start); const right = insert.slice(start + length); const mention = { uuid: mentionUuid, title: replacementText, }; ops.unshift({ insert: right, attributes }); ops.unshift({ insert: { mention }, attributes }); ops.unshift({ insert: left, attributes }); } else { ops.unshift(op); } } }); return ops; }; export const insertEmojiOps = ( incomingOps: ReadonlyArray, existingAttributes: AttributeMap ): Array => { return incomingOps.reduce((ops, op) => { if (typeof op.insert === 'string') { const text = op.insert; const { attributes } = op; const re = emojiRegex(); let index = 0; let match: RegExpExecArray | null; // eslint-disable-next-line no-cond-assign while ((match = re.exec(text))) { const [emoji] = match; ops.push({ insert: text.slice(index, match.index), attributes }); ops.push({ insert: { emoji }, attributes: { ...existingAttributes, ...attributes }, }); index = match.index + emoji.length; } ops.push({ insert: text.slice(index, text.length), attributes }); } else { ops.push(op); } return ops; }, [] as Array); };