2023-08-03 07:44:52 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import type Quill from 'quill';
|
|
|
|
import Delta from 'quill-delta';
|
|
|
|
import type { EmojiData } from '../../components/emoji/lib';
|
|
|
|
import {
|
|
|
|
convertShortName,
|
|
|
|
convertShortNameToData,
|
|
|
|
} from '../../components/emoji/lib';
|
|
|
|
|
|
|
|
type AutoSubstituteAsciiEmojisOptions = {
|
|
|
|
skinTone: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
const emojiMap: Record<string, string> = {
|
|
|
|
':-)': 'slightly_smiling_face',
|
|
|
|
':-(': 'slightly_frowning_face',
|
2023-12-18 23:22:46 +00:00
|
|
|
':-D': 'grinning',
|
|
|
|
':-*': 'kissing_heart',
|
2023-08-03 07:44:52 +00:00
|
|
|
':-P': 'stuck_out_tongue',
|
2023-12-18 23:22:46 +00:00
|
|
|
':-p': 'stuck_out_tongue',
|
2023-08-03 07:44:52 +00:00
|
|
|
":'(": 'cry',
|
2023-12-18 23:22:46 +00:00
|
|
|
':-\\': 'confused',
|
2023-08-03 07:44:52 +00:00
|
|
|
';-)': 'wink',
|
|
|
|
'(Y)': '+1',
|
|
|
|
'(N)': '-1',
|
2023-12-18 23:22:46 +00:00
|
|
|
'(y)': '+1',
|
|
|
|
'(n)': '-1',
|
|
|
|
'<3': 'heart',
|
|
|
|
'^_^': 'grin',
|
|
|
|
'>_<': 'laughing',
|
2023-08-03 07:44:52 +00:00
|
|
|
};
|
|
|
|
|
2023-12-18 23:22:46 +00:00
|
|
|
function buildRegexp(obj: Record<string, string>): RegExp {
|
|
|
|
const sanitizedKeys = Object.keys(obj).map(x =>
|
|
|
|
x.replace(/([^a-zA-Z0-9])/g, '\\$1')
|
|
|
|
);
|
|
|
|
|
|
|
|
return new RegExp(`(${sanitizedKeys.join('|')})$`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const EMOJI_REGEXP = buildRegexp(emojiMap);
|
|
|
|
|
2023-08-03 07:44:52 +00:00
|
|
|
export class AutoSubstituteAsciiEmojis {
|
|
|
|
options: AutoSubstituteAsciiEmojisOptions;
|
|
|
|
|
|
|
|
quill: Quill;
|
|
|
|
|
|
|
|
constructor(quill: Quill, options: AutoSubstituteAsciiEmojisOptions) {
|
|
|
|
this.options = options;
|
|
|
|
this.quill = quill;
|
|
|
|
|
2023-12-18 23:22:46 +00:00
|
|
|
this.quill.on('text-change', (_now, _before, source) => {
|
|
|
|
if (source !== 'user') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// When pasting - Quill first updates contents with "user" source and only
|
|
|
|
// then updates the selection with "silent" source. This means that unless
|
|
|
|
// we wrap `onTextChange` with setTimeout - we are not going to see the
|
|
|
|
// updated cursor position.
|
|
|
|
setTimeout(() => this.onTextChange(), 0);
|
|
|
|
});
|
2023-08-03 07:44:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
onTextChange(): void {
|
2023-12-18 23:22:46 +00:00
|
|
|
if (!window.storage.get('autoConvertEmoji', false)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-03 07:44:52 +00:00
|
|
|
const range = this.quill.getSelection();
|
|
|
|
|
|
|
|
if (!range) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const [blot, index] = this.quill.getLeaf(range.index);
|
|
|
|
|
2023-12-18 23:22:46 +00:00
|
|
|
if (blot?.text == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const textBeforeCursor = blot.text.slice(0, index);
|
|
|
|
const match = textBeforeCursor.match(EMOJI_REGEXP);
|
|
|
|
if (match == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const [, textEmoji] = match;
|
|
|
|
const emojiName = emojiMap[textEmoji];
|
|
|
|
|
|
|
|
const emojiData = convertShortNameToData(emojiName, this.options.skinTone);
|
|
|
|
if (emojiData) {
|
|
|
|
this.insertEmoji(
|
|
|
|
emojiData,
|
|
|
|
range.index - textEmoji.length,
|
|
|
|
textEmoji.length,
|
|
|
|
textEmoji
|
|
|
|
);
|
2023-08-03 07:44:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-18 23:22:46 +00:00
|
|
|
insertEmoji(
|
|
|
|
emojiData: EmojiData,
|
|
|
|
index: number,
|
|
|
|
range: number,
|
|
|
|
source: string
|
|
|
|
): void {
|
2023-08-03 07:44:52 +00:00
|
|
|
const emoji = convertShortName(emojiData.short_name, this.options.skinTone);
|
2023-12-18 23:22:46 +00:00
|
|
|
const delta = new Delta()
|
|
|
|
.retain(index)
|
|
|
|
.delete(range)
|
|
|
|
.insert({
|
|
|
|
emoji: { value: emoji, source },
|
|
|
|
});
|
|
|
|
this.quill.updateContents(delta, 'api');
|
2023-08-03 07:44:52 +00:00
|
|
|
this.quill.setSelection(index + 1, 0);
|
|
|
|
}
|
|
|
|
}
|