@chayns-components/emoji-input 5.3.5 → 5.3.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/cjs/components/emoji-input/EmojiInput.js +61 -1
- package/lib/cjs/components/emoji-input/EmojiInput.js.map +1 -1
- package/lib/cjs/constants/emoji.js +6 -5
- package/lib/cjs/constants/emoji.js.map +1 -1
- package/lib/cjs/constants/regex.js +2 -1
- package/lib/cjs/constants/regex.js.map +1 -1
- package/lib/cjs/utils/emoji.js +9 -2
- package/lib/cjs/utils/emoji.js.map +1 -1
- package/lib/cjs/utils/insert.js +86 -1
- package/lib/cjs/utils/insert.js.map +1 -1
- package/lib/cjs/utils/text.js +5 -0
- package/lib/cjs/utils/text.js.map +1 -1
- package/lib/esm/components/emoji-input/EmojiInput.js +62 -2
- package/lib/esm/components/emoji-input/EmojiInput.js.map +1 -1
- package/lib/esm/constants/emoji.js +6 -5
- package/lib/esm/constants/emoji.js.map +1 -1
- package/lib/esm/constants/regex.js +1 -0
- package/lib/esm/constants/regex.js.map +1 -1
- package/lib/esm/utils/emoji.js +9 -2
- package/lib/esm/utils/emoji.js.map +1 -1
- package/lib/esm/utils/insert.js +83 -0
- package/lib/esm/utils/insert.js.map +1 -1
- package/lib/esm/utils/text.js +6 -1
- package/lib/esm/utils/text.js.map +1 -1
- package/lib/types/constants/regex.d.ts +1 -0
- package/lib/types/utils/emoji.d.ts +11 -1
- package/lib/types/utils/insert.d.ts +11 -0
- package/package.json +3 -3
package/lib/cjs/utils/insert.js
CHANGED
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.replaceText = exports.insertTextAtCursorPosition = void 0;
|
|
6
|
+
exports.revertAsciiSmileyConversion = exports.replaceText = exports.insertTextAtCursorPosition = void 0;
|
|
7
7
|
var _selection = require("./selection");
|
|
8
8
|
/**
|
|
9
9
|
* This function inserts the passed text at the correct position in the editor element. If the
|
|
@@ -109,4 +109,89 @@ const replaceText = ({
|
|
|
109
109
|
}
|
|
110
110
|
};
|
|
111
111
|
exports.replaceText = replaceText;
|
|
112
|
+
/** * Reverts the most recent ASCII-smiley-to-emoji conversion at the cursor * position. * * The emoji (and optionally the trigger character following it) is removed * and replaced by a `<span class="no-emoji-convert">{original}</span>` block. * The cursor is moved to the end of the inserted span. The span is then * protected against re-conversion by the existing `regAscii` rule that skips * the contents of any `<span>` element. * * @returns `true` if a revert was performed, `false` otherwise. */
|
|
113
|
+
const revertAsciiSmileyConversion = ({
|
|
114
|
+
editorElement,
|
|
115
|
+
original,
|
|
116
|
+
emoji,
|
|
117
|
+
shouldRemoveTriggerChar = true
|
|
118
|
+
}) => {
|
|
119
|
+
// Walk all text nodes and find the LAST one that contains the emoji.
|
|
120
|
+
// The just-converted emoji is the most recent occurrence; using the last
|
|
121
|
+
// match is correct because the conversion always inserts the emoji at
|
|
122
|
+
// the (former) cursor position and the cursor is at the end of that
|
|
123
|
+
// insertion when Backspace is pressed.
|
|
124
|
+
const walker = document.createTreeWalker(editorElement, NodeFilter.SHOW_TEXT);
|
|
125
|
+
let targetNode = null;
|
|
126
|
+
let targetIndex = -1;
|
|
127
|
+
let current = walker.nextNode();
|
|
128
|
+
while (current) {
|
|
129
|
+
var _current$nodeValue;
|
|
130
|
+
const idx = ((_current$nodeValue = current.nodeValue) === null || _current$nodeValue === void 0 ? void 0 : _current$nodeValue.lastIndexOf(emoji)) ?? -1;
|
|
131
|
+
if (idx !== -1) {
|
|
132
|
+
targetNode = current;
|
|
133
|
+
targetIndex = idx;
|
|
134
|
+
}
|
|
135
|
+
current = walker.nextNode();
|
|
136
|
+
}
|
|
137
|
+
if (!targetNode || targetIndex === -1) {
|
|
138
|
+
return false;
|
|
139
|
+
}
|
|
140
|
+
const nodeValue = targetNode.nodeValue ?? '';
|
|
141
|
+
const before = nodeValue.slice(0, targetIndex);
|
|
142
|
+
const afterEmoji = nodeValue.slice(targetIndex + emoji.length);
|
|
143
|
+
|
|
144
|
+
// Drop the single trigger char (space, ".", ",", "!", "?", ...) that
|
|
145
|
+
// immediately follows the emoji. If the trigger sits in a different
|
|
146
|
+
// text node or has already been removed, we silently keep the rest.
|
|
147
|
+
const after = shouldRemoveTriggerChar && afterEmoji.length > 0 ? afterEmoji.slice(1) : afterEmoji;
|
|
148
|
+
const parent = targetNode.parentNode;
|
|
149
|
+
if (!parent) {
|
|
150
|
+
return false;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// Build the protective span
|
|
154
|
+
const span = document.createElement('span');
|
|
155
|
+
span.className = 'no-emoji-convert';
|
|
156
|
+
span.textContent = original;
|
|
157
|
+
|
|
158
|
+
// Rewrite the original text node + insert span (+ remaining text node)
|
|
159
|
+
targetNode.nodeValue = before;
|
|
160
|
+
parent.insertBefore(span, targetNode.nextSibling);
|
|
161
|
+
|
|
162
|
+
// ALWAYS ensure a real text node exists directly after the span and place
|
|
163
|
+
// the cursor INSIDE it. Without this, contentEditable extends the span
|
|
164
|
+
// when the user keeps typing — which would protect every subsequent
|
|
165
|
+
// character from emoji conversion.
|
|
166
|
+
let cursorNode;
|
|
167
|
+
let cursorOffset;
|
|
168
|
+
if (after) {
|
|
169
|
+
cursorNode = document.createTextNode(after);
|
|
170
|
+
parent.insertBefore(cursorNode, span.nextSibling);
|
|
171
|
+
cursorOffset = 0;
|
|
172
|
+
} else {
|
|
173
|
+
// Insert a zero-width space that is stripped by convertHTMLToText
|
|
174
|
+
// (line 73 in text.ts), so it stays invisible to consumers.
|
|
175
|
+
cursorNode = document.createTextNode('\u200B');
|
|
176
|
+
parent.insertBefore(cursorNode, span.nextSibling);
|
|
177
|
+
cursorOffset = cursorNode.length;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// If the leading text node is now empty, remove it to keep the DOM tidy
|
|
181
|
+
if (before === '') {
|
|
182
|
+
parent.removeChild(targetNode);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// Move cursor into the text node that sits OUTSIDE the protection span
|
|
186
|
+
const selection = window.getSelection();
|
|
187
|
+
if (selection) {
|
|
188
|
+
const range = document.createRange();
|
|
189
|
+
range.setStart(cursorNode, cursorOffset);
|
|
190
|
+
range.setEnd(cursorNode, cursorOffset);
|
|
191
|
+
selection.removeAllRanges();
|
|
192
|
+
selection.addRange(range);
|
|
193
|
+
}
|
|
194
|
+
return true;
|
|
195
|
+
};
|
|
196
|
+
exports.revertAsciiSmileyConversion = revertAsciiSmileyConversion;
|
|
112
197
|
//# sourceMappingURL=insert.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"insert.js","names":["_selection","require","insertTextAtCursorPosition","editorElement","text","shouldUseSavedSelection","restoreSelection","selection","window","getSelection","saveSelection","anchorNode","contains","range","getRangeAt","parts","split","firstPart","shift","textNodes","map","part","document","createTextNode","deleteContents","nodeType","Node","TEXT_NODE","nodeValue","slice","startOffset","moveSelectionOffset","length","textNode","insertBefore","childNodes","anchorOffset","before","appendChild","textNodeIndex","Array","from","indexOf","setChildIndex","brElement","createElement","insertNode","setEndAfter","setStartAfter","forEach","index","collapse","removeAllRanges","addRange","innerText","exports","replaceText","searchText","pasteText","options","rangeToReplace","findAndSelectText"],"sources":["../../../src/utils/insert.ts"],"sourcesContent":["import {\n findAndSelectText,\n moveSelectionOffset,\n restoreSelection,\n saveSelection,\n setChildIndex,\n type ReplaceTextOptions as IReplaceTextOptions,\n} from './selection';\n\ninterface InsertTextAtCursorPositionOptions {\n editorElement: HTMLDivElement;\n text: string;\n shouldUseSavedSelection?: boolean;\n}\n\n/**\n * This function inserts the passed text at the correct position in the editor element. If the\n * element has the focus, the new emoji is inserted at the cursor position. If not, the emoji\n * will be appended to the back of the input field content.\n *\n * In addition, this function also sets the cursor to the correct position when the input field\n * has the focus. For this purpose, the current position of the cursor or a selection is read to\n * calculate the cursor position after inserting the text.\n *\n * @param {Object} options - Object with element and text to insert\n * @param {HTMLDivElement} options.editorElement - Element to insert text into\n * @param {string} options.text - Text to insert into element\n */\nexport const insertTextAtCursorPosition = ({\n editorElement,\n text,\n shouldUseSavedSelection = false,\n}: InsertTextAtCursorPositionOptions) => {\n if (shouldUseSavedSelection) {\n restoreSelection(editorElement);\n }\n\n const selection = window.getSelection();\n\n saveSelection(editorElement);\n\n if (selection?.anchorNode && editorElement.contains(selection.anchorNode)) {\n let range = selection.getRangeAt(0);\n\n const parts = text.split(/\\r\\n|\\r|\\n/);\n\n const firstPart = parts.shift();\n\n const textNodes = parts.map((part) => document.createTextNode(part));\n\n range.deleteContents();\n\n if (firstPart) {\n if (selection.anchorNode.nodeType === Node.TEXT_NODE) {\n const { nodeValue } = selection.anchorNode;\n\n if (typeof nodeValue === 'string') {\n selection.anchorNode.nodeValue =\n nodeValue.slice(0, range.startOffset) +\n firstPart +\n nodeValue.slice(range.startOffset);\n\n moveSelectionOffset(firstPart.length);\n }\n } else if (selection.anchorNode === editorElement) {\n const textNode = document.createTextNode(firstPart);\n\n // Inserts the text node before the node at the anchor offset.\n // If that node doesn't exist, the text node is appended to the editor, as a fallback. I'm not sure if there is any case where this would happen.\n const insertBefore = editorElement.childNodes[selection.anchorOffset];\n if (insertBefore) {\n insertBefore.before(textNode);\n } else {\n editorElement.appendChild(textNode);\n }\n\n const textNodeIndex = Array.from(editorElement.childNodes).indexOf(textNode);\n\n moveSelectionOffset(firstPart.length);\n setChildIndex(textNodeIndex);\n }\n }\n\n restoreSelection(editorElement);\n\n if (textNodes.length > 0) {\n range = selection.getRangeAt(0);\n\n let brElement = document.createElement('br');\n\n range.insertNode(brElement);\n range.setEndAfter(brElement);\n range.setStartAfter(brElement);\n\n textNodes.forEach((textNode, index) => {\n range.insertNode(textNode);\n range.setEndAfter(textNode);\n range.setStartAfter(textNode);\n\n if (index !== textNodes.length - 1) {\n brElement = document.createElement('br');\n\n range.insertNode(brElement);\n range.setEndAfter(brElement);\n range.setStartAfter(brElement);\n }\n });\n\n range.collapse(false);\n\n selection.removeAllRanges();\n selection.addRange(range);\n }\n } else {\n // eslint-disable-next-line no-param-reassign\n editorElement.innerText += text;\n }\n};\n\nexport interface ReplaceTextOptions {\n editorElement: HTMLDivElement;\n searchText: string;\n pasteText: string;\n options?: IReplaceTextOptions;\n}\n\nexport const replaceText = ({\n editorElement,\n searchText,\n pasteText,\n options,\n}: ReplaceTextOptions) => {\n const selection = window.getSelection();\n\n const rangeToReplace = findAndSelectText({ editorElement, searchText, options });\n\n if (rangeToReplace && selection) {\n selection.removeAllRanges();\n selection.addRange(rangeToReplace);\n\n insertTextAtCursorPosition({ editorElement, text: pasteText });\n }\n};\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"insert.js","names":["_selection","require","insertTextAtCursorPosition","editorElement","text","shouldUseSavedSelection","restoreSelection","selection","window","getSelection","saveSelection","anchorNode","contains","range","getRangeAt","parts","split","firstPart","shift","textNodes","map","part","document","createTextNode","deleteContents","nodeType","Node","TEXT_NODE","nodeValue","slice","startOffset","moveSelectionOffset","length","textNode","insertBefore","childNodes","anchorOffset","before","appendChild","textNodeIndex","Array","from","indexOf","setChildIndex","brElement","createElement","insertNode","setEndAfter","setStartAfter","forEach","index","collapse","removeAllRanges","addRange","innerText","exports","replaceText","searchText","pasteText","options","rangeToReplace","findAndSelectText","revertAsciiSmileyConversion","original","emoji","shouldRemoveTriggerChar","walker","createTreeWalker","NodeFilter","SHOW_TEXT","targetNode","targetIndex","current","nextNode","_current$nodeValue","idx","lastIndexOf","afterEmoji","after","parent","parentNode","span","className","textContent","nextSibling","cursorNode","cursorOffset","removeChild","createRange","setStart","setEnd"],"sources":["../../../src/utils/insert.ts"],"sourcesContent":["import {\n findAndSelectText,\n moveSelectionOffset,\n restoreSelection,\n saveSelection,\n setChildIndex,\n type ReplaceTextOptions as IReplaceTextOptions,\n} from './selection';\n\ninterface InsertTextAtCursorPositionOptions {\n editorElement: HTMLDivElement;\n text: string;\n shouldUseSavedSelection?: boolean;\n}\n\n/**\n * This function inserts the passed text at the correct position in the editor element. If the\n * element has the focus, the new emoji is inserted at the cursor position. If not, the emoji\n * will be appended to the back of the input field content.\n *\n * In addition, this function also sets the cursor to the correct position when the input field\n * has the focus. For this purpose, the current position of the cursor or a selection is read to\n * calculate the cursor position after inserting the text.\n *\n * @param {Object} options - Object with element and text to insert\n * @param {HTMLDivElement} options.editorElement - Element to insert text into\n * @param {string} options.text - Text to insert into element\n */\nexport const insertTextAtCursorPosition = ({\n editorElement,\n text,\n shouldUseSavedSelection = false,\n}: InsertTextAtCursorPositionOptions) => {\n if (shouldUseSavedSelection) {\n restoreSelection(editorElement);\n }\n\n const selection = window.getSelection();\n\n saveSelection(editorElement);\n\n if (selection?.anchorNode && editorElement.contains(selection.anchorNode)) {\n let range = selection.getRangeAt(0);\n\n const parts = text.split(/\\r\\n|\\r|\\n/);\n\n const firstPart = parts.shift();\n\n const textNodes = parts.map((part) => document.createTextNode(part));\n\n range.deleteContents();\n\n if (firstPart) {\n if (selection.anchorNode.nodeType === Node.TEXT_NODE) {\n const { nodeValue } = selection.anchorNode;\n\n if (typeof nodeValue === 'string') {\n selection.anchorNode.nodeValue =\n nodeValue.slice(0, range.startOffset) +\n firstPart +\n nodeValue.slice(range.startOffset);\n\n moveSelectionOffset(firstPart.length);\n }\n } else if (selection.anchorNode === editorElement) {\n const textNode = document.createTextNode(firstPart);\n\n // Inserts the text node before the node at the anchor offset.\n // If that node doesn't exist, the text node is appended to the editor, as a fallback. I'm not sure if there is any case where this would happen.\n const insertBefore = editorElement.childNodes[selection.anchorOffset];\n if (insertBefore) {\n insertBefore.before(textNode);\n } else {\n editorElement.appendChild(textNode);\n }\n\n const textNodeIndex = Array.from(editorElement.childNodes).indexOf(textNode);\n\n moveSelectionOffset(firstPart.length);\n setChildIndex(textNodeIndex);\n }\n }\n\n restoreSelection(editorElement);\n\n if (textNodes.length > 0) {\n range = selection.getRangeAt(0);\n\n let brElement = document.createElement('br');\n\n range.insertNode(brElement);\n range.setEndAfter(brElement);\n range.setStartAfter(brElement);\n\n textNodes.forEach((textNode, index) => {\n range.insertNode(textNode);\n range.setEndAfter(textNode);\n range.setStartAfter(textNode);\n\n if (index !== textNodes.length - 1) {\n brElement = document.createElement('br');\n\n range.insertNode(brElement);\n range.setEndAfter(brElement);\n range.setStartAfter(brElement);\n }\n });\n\n range.collapse(false);\n\n selection.removeAllRanges();\n selection.addRange(range);\n }\n } else {\n // eslint-disable-next-line no-param-reassign\n editorElement.innerText += text;\n }\n};\n\nexport interface ReplaceTextOptions {\n editorElement: HTMLDivElement;\n searchText: string;\n pasteText: string;\n options?: IReplaceTextOptions;\n}\n\nexport const replaceText = ({\n editorElement,\n searchText,\n pasteText,\n options,\n}: ReplaceTextOptions) => {\n const selection = window.getSelection();\n\n const rangeToReplace = findAndSelectText({ editorElement, searchText, options });\n\n if (rangeToReplace && selection) {\n selection.removeAllRanges();\n selection.addRange(rangeToReplace);\n\n insertTextAtCursorPosition({ editorElement, text: pasteText });\n }\n};\n\ninterface RevertAsciiSmileyConversionOptions {\n editorElement: HTMLDivElement;\n /** The original ASCII smiley to restore (e.g. \":-)\"). */\n original: string;\n /** The Unicode emoji that is currently in the DOM and should be replaced. */\n emoji: string;\n /** * Whether to also remove the single character immediately following the * emoji (typically the trigger whitespace / punctuation that caused the * conversion). Defaults to `true` to mirror common autocorrect UX * (Word, IntelliJ, ...). */\n shouldRemoveTriggerChar?: boolean;\n}\n\n/** * Reverts the most recent ASCII-smiley-to-emoji conversion at the cursor * position. * * The emoji (and optionally the trigger character following it) is removed * and replaced by a `<span class=\"no-emoji-convert\">{original}</span>` block. * The cursor is moved to the end of the inserted span. The span is then * protected against re-conversion by the existing `regAscii` rule that skips * the contents of any `<span>` element. * * @returns `true` if a revert was performed, `false` otherwise. */\nexport const revertAsciiSmileyConversion = ({\n editorElement,\n original,\n emoji,\n shouldRemoveTriggerChar = true,\n}: RevertAsciiSmileyConversionOptions): boolean => {\n // Walk all text nodes and find the LAST one that contains the emoji.\n // The just-converted emoji is the most recent occurrence; using the last\n // match is correct because the conversion always inserts the emoji at\n // the (former) cursor position and the cursor is at the end of that\n // insertion when Backspace is pressed.\n const walker = document.createTreeWalker(editorElement, NodeFilter.SHOW_TEXT);\n\n let targetNode: Text | null = null;\n let targetIndex = -1;\n\n let current = walker.nextNode() as Text | null;\n while (current) {\n const idx = current.nodeValue?.lastIndexOf(emoji) ?? -1;\n if (idx !== -1) {\n targetNode = current;\n targetIndex = idx;\n }\n current = walker.nextNode() as Text | null;\n }\n\n if (!targetNode || targetIndex === -1) {\n return false;\n }\n\n const nodeValue = targetNode.nodeValue ?? '';\n const before = nodeValue.slice(0, targetIndex);\n const afterEmoji = nodeValue.slice(targetIndex + emoji.length);\n\n // Drop the single trigger char (space, \".\", \",\", \"!\", \"?\", ...) that\n // immediately follows the emoji. If the trigger sits in a different\n // text node or has already been removed, we silently keep the rest.\n const after =\n shouldRemoveTriggerChar && afterEmoji.length > 0 ? afterEmoji.slice(1) : afterEmoji;\n\n const parent = targetNode.parentNode;\n if (!parent) {\n return false;\n }\n\n // Build the protective span\n const span = document.createElement('span');\n span.className = 'no-emoji-convert';\n span.textContent = original;\n\n // Rewrite the original text node + insert span (+ remaining text node)\n targetNode.nodeValue = before;\n parent.insertBefore(span, targetNode.nextSibling);\n\n // ALWAYS ensure a real text node exists directly after the span and place\n // the cursor INSIDE it. Without this, contentEditable extends the span\n // when the user keeps typing — which would protect every subsequent\n // character from emoji conversion.\n let cursorNode: Text;\n let cursorOffset: number;\n\n if (after) {\n cursorNode = document.createTextNode(after);\n parent.insertBefore(cursorNode, span.nextSibling);\n cursorOffset = 0;\n } else {\n // Insert a zero-width space that is stripped by convertHTMLToText\n // (line 73 in text.ts), so it stays invisible to consumers.\n cursorNode = document.createTextNode('\\u200B');\n parent.insertBefore(cursorNode, span.nextSibling);\n cursorOffset = cursorNode.length;\n }\n\n // If the leading text node is now empty, remove it to keep the DOM tidy\n if (before === '') {\n parent.removeChild(targetNode);\n }\n\n // Move cursor into the text node that sits OUTSIDE the protection span\n const selection = window.getSelection();\n if (selection) {\n const range = document.createRange();\n range.setStart(cursorNode, cursorOffset);\n range.setEnd(cursorNode, cursorOffset);\n selection.removeAllRanges();\n selection.addRange(range);\n }\n\n return true;\n};\n"],"mappings":"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","ignoreList":[]}
|
package/lib/cjs/utils/text.js
CHANGED
|
@@ -27,6 +27,11 @@ const convertHTMLToText = (text, {
|
|
|
27
27
|
preserveSpaces = false
|
|
28
28
|
} = {}) => {
|
|
29
29
|
let result = text;
|
|
30
|
+
|
|
31
|
+
// Unwrap "no-emoji-convert" protection spans so the plain-text
|
|
32
|
+
// representation does not leak the marker HTML to consumers.
|
|
33
|
+
// The protection itself is only relevant inside the live editor DOM.
|
|
34
|
+
result = result.replace(_regex.HTML_NO_EMOJI_REGEX, '$1');
|
|
30
35
|
result = result.replace(_regex.HTML_A_TAG_REGEX, '$1').replace(_regex.HTML_BOLD_REGEX, '[b]$1[/b]').replace(_regex.HTML_LC_MENTION_REGEX, '[lc_mention id="$1"]$2[/lc_mention]').replace(_regex.HTML_NER_IGNORE_REGEX, '[nerIgnore]$1[/nerIgnore]').replace(_regex.HTML_NER_REPLACE_REGEX, (_, prefix, type, value, entity) => {
|
|
31
36
|
const prefixAttr = prefix ? `prefix="${prefix}" ` : '';
|
|
32
37
|
return `[nerReplace ${prefixAttr}type="${type}" value="${value}"]${entity}[/nerReplace]`;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"text.js","names":["_regex","require","_emoji","convertTextToHTML","text","element","document","createElement","style","position","opacity","contentEditable","innerText","body","appendChild","result","innerHTML","removeChild","unescapeHTML","replace","HTML_A_TAG_REGEX","BB_LC_MENTION_REGEX","BB_NER_IGNORE_REGEX","BB_NER_REPLACE_REGEX","_","prefix","type","value","entity","prefixAttr","exports","convertHTMLToText","preserveSpaces","HTML_BOLD_REGEX","HTML_LC_MENTION_REGEX","HTML_NER_IGNORE_REGEX","HTML_NER_REPLACE_REGEX","escapeHTML","getElementTextLength","textLength","outerHTML","length","e"],"sources":["../../../src/utils/text.ts"],"sourcesContent":["import {\n BB_LC_MENTION_REGEX,\n BB_NER_IGNORE_REGEX,\n BB_NER_REPLACE_REGEX,\n HTML_A_TAG_REGEX,\n HTML_BOLD_REGEX,\n HTML_LC_MENTION_REGEX,\n HTML_NER_IGNORE_REGEX,\n HTML_NER_REPLACE_REGEX,\n} from '../constants/regex';\nimport { escapeHTML, unescapeHTML } from './emoji';\n\nexport const convertTextToHTML = (text: string) => {\n const element = document.createElement('div');\n\n element.style.position = 'absolute';\n element.style.opacity = '0';\n\n element.contentEditable = 'true';\n element.innerText = text;\n\n document.body.appendChild(element);\n\n let result = element.innerHTML;\n\n document.body.removeChild(element);\n\n result = unescapeHTML(result);\n\n result = result\n .replace(HTML_A_TAG_REGEX, '$1')\n .replace(\n BB_LC_MENTION_REGEX,\n '<lc_mention contenteditable=\"false\" id=\"$1\"><span>@</span>$2</lc_mention>',\n )\n .replace(BB_NER_IGNORE_REGEX, '<nerIgnore contenteditable=\"false\">$1</nerIgnore>')\n .replace(\n BB_NER_REPLACE_REGEX,\n (_, prefix: string | undefined, type: string, value: string, entity: string) => {\n const prefixAttr = prefix ? `prefix=\"${prefix}\" ` : '';\n\n return `<nerReplace contenteditable=\"false\" ${prefixAttr}type=\"${type}\" value=\"${value}\">${entity}</nerReplace>`;\n },\n );\n\n return result;\n};\n\nexport const convertHTMLToText = (text: string, { preserveSpaces = false } = {}) => {\n let result = text;\n\n result = result\n .replace(HTML_A_TAG_REGEX, '$1')\n .replace(HTML_BOLD_REGEX, '[b]$1[/b]')\n .replace(HTML_LC_MENTION_REGEX, '[lc_mention id=\"$1\"]$2[/lc_mention]')\n .replace(HTML_NER_IGNORE_REGEX, '[nerIgnore]$1[/nerIgnore]')\n .replace(\n HTML_NER_REPLACE_REGEX,\n (_, prefix: string | undefined, type: string, value: string, entity: string) => {\n const prefixAttr = prefix ? `prefix=\"${prefix}\" ` : '';\n\n return `[nerReplace ${prefixAttr}type=\"${type}\" value=\"${value}\"]${entity}[/nerReplace]`;\n },\n );\n\n if (preserveSpaces) {\n return result\n .replace(/ /g, '\\u00A0') // non-breaking space\n .replace(/\\u200B/g, ''); // zero-width space (sichtbar gemacht)\n }\n\n // eslint-disable-next-line no-irregular-whitespace\n result = result.replace(//g, '');\n\n result = escapeHTML(result);\n\n const element = document.createElement('div');\n\n element.style.position = 'absolute';\n element.style.opacity = '0';\n\n element.contentEditable = 'true';\n element.innerHTML = result;\n\n document.body.appendChild(element);\n\n result = element.innerText;\n\n document.body.removeChild(element);\n\n return result;\n};\n\nexport const getElementTextLength = (element: Element) => {\n let textLength = 0;\n\n try {\n textLength = convertHTMLToText(element.outerHTML).length;\n } catch (e) {\n // Do nothing\n }\n\n return textLength;\n};\n"],"mappings":";;;;;;AAAA,IAAAA,MAAA,GAAAC,OAAA;
|
|
1
|
+
{"version":3,"file":"text.js","names":["_regex","require","_emoji","convertTextToHTML","text","element","document","createElement","style","position","opacity","contentEditable","innerText","body","appendChild","result","innerHTML","removeChild","unescapeHTML","replace","HTML_A_TAG_REGEX","BB_LC_MENTION_REGEX","BB_NER_IGNORE_REGEX","BB_NER_REPLACE_REGEX","_","prefix","type","value","entity","prefixAttr","exports","convertHTMLToText","preserveSpaces","HTML_NO_EMOJI_REGEX","HTML_BOLD_REGEX","HTML_LC_MENTION_REGEX","HTML_NER_IGNORE_REGEX","HTML_NER_REPLACE_REGEX","escapeHTML","getElementTextLength","textLength","outerHTML","length","e"],"sources":["../../../src/utils/text.ts"],"sourcesContent":["import {\n BB_LC_MENTION_REGEX,\n BB_NER_IGNORE_REGEX,\n BB_NER_REPLACE_REGEX,\n HTML_A_TAG_REGEX,\n HTML_BOLD_REGEX,\n HTML_LC_MENTION_REGEX,\n HTML_NER_IGNORE_REGEX,\n HTML_NER_REPLACE_REGEX,\n HTML_NO_EMOJI_REGEX,\n} from '../constants/regex';\nimport { escapeHTML, unescapeHTML } from './emoji';\n\nexport const convertTextToHTML = (text: string) => {\n const element = document.createElement('div');\n\n element.style.position = 'absolute';\n element.style.opacity = '0';\n\n element.contentEditable = 'true';\n element.innerText = text;\n\n document.body.appendChild(element);\n\n let result = element.innerHTML;\n\n document.body.removeChild(element);\n\n result = unescapeHTML(result);\n\n result = result\n .replace(HTML_A_TAG_REGEX, '$1')\n .replace(\n BB_LC_MENTION_REGEX,\n '<lc_mention contenteditable=\"false\" id=\"$1\"><span>@</span>$2</lc_mention>',\n )\n .replace(BB_NER_IGNORE_REGEX, '<nerIgnore contenteditable=\"false\">$1</nerIgnore>')\n .replace(\n BB_NER_REPLACE_REGEX,\n (_, prefix: string | undefined, type: string, value: string, entity: string) => {\n const prefixAttr = prefix ? `prefix=\"${prefix}\" ` : '';\n\n return `<nerReplace contenteditable=\"false\" ${prefixAttr}type=\"${type}\" value=\"${value}\">${entity}</nerReplace>`;\n },\n );\n\n return result;\n};\n\nexport const convertHTMLToText = (text: string, { preserveSpaces = false } = {}) => {\n let result = text;\n\n // Unwrap \"no-emoji-convert\" protection spans so the plain-text\n // representation does not leak the marker HTML to consumers.\n // The protection itself is only relevant inside the live editor DOM.\n result = result.replace(HTML_NO_EMOJI_REGEX, '$1');\n\n result = result\n .replace(HTML_A_TAG_REGEX, '$1')\n .replace(HTML_BOLD_REGEX, '[b]$1[/b]')\n .replace(HTML_LC_MENTION_REGEX, '[lc_mention id=\"$1\"]$2[/lc_mention]')\n .replace(HTML_NER_IGNORE_REGEX, '[nerIgnore]$1[/nerIgnore]')\n .replace(\n HTML_NER_REPLACE_REGEX,\n (_, prefix: string | undefined, type: string, value: string, entity: string) => {\n const prefixAttr = prefix ? `prefix=\"${prefix}\" ` : '';\n\n return `[nerReplace ${prefixAttr}type=\"${type}\" value=\"${value}\"]${entity}[/nerReplace]`;\n },\n );\n\n if (preserveSpaces) {\n return result\n .replace(/ /g, '\\u00A0') // non-breaking space\n .replace(/\\u200B/g, ''); // zero-width space (sichtbar gemacht)\n }\n\n // eslint-disable-next-line no-irregular-whitespace\n result = result.replace(//g, '');\n\n result = escapeHTML(result);\n\n const element = document.createElement('div');\n\n element.style.position = 'absolute';\n element.style.opacity = '0';\n\n element.contentEditable = 'true';\n element.innerHTML = result;\n\n document.body.appendChild(element);\n\n result = element.innerText;\n\n document.body.removeChild(element);\n\n return result;\n};\n\nexport const getElementTextLength = (element: Element) => {\n let textLength = 0;\n\n try {\n textLength = convertHTMLToText(element.outerHTML).length;\n } catch (e) {\n // Do nothing\n }\n\n return textLength;\n};\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -2,7 +2,7 @@ import { AreaContext, useIsTouch } from '@chayns-components/core';
|
|
|
2
2
|
import { AnimatePresence } from 'motion/react';
|
|
3
3
|
import React, { forwardRef, useCallback, useContext, useEffect, useImperativeHandle, useLayoutEffect, useMemo, useRef, useState } from 'react';
|
|
4
4
|
import { convertEmojisToUnicode, escapeHTML } from '../../utils/emoji';
|
|
5
|
-
import { insertTextAtCursorPosition, replaceText } from '../../utils/insert';
|
|
5
|
+
import { insertTextAtCursorPosition, replaceText, revertAsciiSmileyConversion } from '../../utils/insert';
|
|
6
6
|
import { getCharCodeThatWillBeDeleted, insertInvisibleCursorMarker, restoreSelection, saveSelection, insertPseudoMarker, insertCursorAtMarker, getCurrentCursorPosition, setCursorPositionByAbsIndex } from '../../utils/selection';
|
|
7
7
|
import { convertHTMLToText, convertTextToHTML } from '../../utils/text';
|
|
8
8
|
import EmojiPickerPopup from '../emoji-picker-popup/EmojiPickerPopup';
|
|
@@ -11,6 +11,7 @@ import PrefixElement from './prefix-element/PrefixElement';
|
|
|
11
11
|
import { loadEmojiShortList } from '../../utils/asyncEmojiData';
|
|
12
12
|
import { scrollCursorIntoView } from '../../utils/scroll';
|
|
13
13
|
import { useCursorPosition } from '../../hooks/cursor';
|
|
14
|
+
const MODIFIER_KEYS = new Set(['Shift', 'Control', 'Alt', 'Meta', 'CapsLock']);
|
|
14
15
|
const useIsomorphicLayoutEffect = typeof window !== 'undefined' ? useLayoutEffect : useEffect;
|
|
15
16
|
const EmojiInput = /*#__PURE__*/forwardRef(({
|
|
16
17
|
accessToken,
|
|
@@ -51,6 +52,9 @@ const EmojiInput = /*#__PURE__*/forwardRef(({
|
|
|
51
52
|
const hasPrefixChanged = useRef(false);
|
|
52
53
|
const shouldDeleteOneMoreBackwards = useRef(false);
|
|
53
54
|
const shouldDeleteOneMoreForwards = useRef(false);
|
|
55
|
+
|
|
56
|
+
/** * Tracks the most recently auto-converted ASCII smiley so the next * Backspace keystroke can revert it (Word/IntelliJ-style autocorrect undo). * The ref is invalidated as soon as the user presses any non-modifier key * other than Backspace (see `handleKeyDown`). */
|
|
57
|
+
const lastAsciiConversionRef = useRef(null);
|
|
54
58
|
const savedCursorPositionRef = useRef(0);
|
|
55
59
|
const valueRef = useRef(value);
|
|
56
60
|
useCursorPosition(editorRef, useCallback(position => {
|
|
@@ -84,7 +88,10 @@ const EmojiInput = /*#__PURE__*/forwardRef(({
|
|
|
84
88
|
if (!editorRef.current) {
|
|
85
89
|
return;
|
|
86
90
|
}
|
|
87
|
-
|
|
91
|
+
const conversions = [];
|
|
92
|
+
let newInnerHTML = convertEmojisToUnicode(html, emojiRegShortNames, emojiShortNames, {
|
|
93
|
+
onAsciiConversion: conv => conversions.push(conv)
|
|
94
|
+
});
|
|
88
95
|
newInnerHTML = convertTextToHTML(newInnerHTML);
|
|
89
96
|
if (newInnerHTML !== editorRef.current.innerHTML) {
|
|
90
97
|
saveSelection(editorRef.current, {
|
|
@@ -92,6 +99,21 @@ const EmojiInput = /*#__PURE__*/forwardRef(({
|
|
|
92
99
|
});
|
|
93
100
|
editorRef.current.innerHTML = newInnerHTML;
|
|
94
101
|
restoreSelection(editorRef.current);
|
|
102
|
+
|
|
103
|
+
// Remember the LAST ASCII conversion so Backspace can revert it.
|
|
104
|
+
// Older conversions are dropped on purpose: only the smiley the
|
|
105
|
+
// user just produced should be undoable.
|
|
106
|
+
if (conversions.length > 0) {
|
|
107
|
+
const lastConv = conversions[conversions.length - 1];
|
|
108
|
+
const pos = getCurrentCursorPosition(editorRef.current);
|
|
109
|
+
if (pos !== null) {
|
|
110
|
+
lastAsciiConversionRef.current = {
|
|
111
|
+
original: lastConv.original,
|
|
112
|
+
emoji: lastConv.emoji,
|
|
113
|
+
plainTextCursorPos: pos
|
|
114
|
+
};
|
|
115
|
+
}
|
|
116
|
+
}
|
|
95
117
|
}
|
|
96
118
|
}, [emojiRegShortNames, emojiShortNames]);
|
|
97
119
|
const handleBeforeInput = useCallback(event => {
|
|
@@ -168,6 +190,44 @@ const EmojiInput = /*#__PURE__*/forwardRef(({
|
|
|
168
190
|
event.stopPropagation();
|
|
169
191
|
return;
|
|
170
192
|
}
|
|
193
|
+
|
|
194
|
+
// --- Backspace-revert for the most recent auto-conversion ---
|
|
195
|
+
// If the user presses Backspace immediately after an ASCII smiley
|
|
196
|
+
// was auto-converted (and the cursor is still at the post-conversion
|
|
197
|
+
// position), we revert the emoji back to the original text and wrap
|
|
198
|
+
// it in a protection span so it does not get re-converted.
|
|
199
|
+
if (event.key === 'Backspace' && !event.ctrlKey && !event.metaKey && lastAsciiConversionRef.current && editorRef.current) {
|
|
200
|
+
const {
|
|
201
|
+
original,
|
|
202
|
+
emoji,
|
|
203
|
+
plainTextCursorPos
|
|
204
|
+
} = lastAsciiConversionRef.current;
|
|
205
|
+
const currentPos = getCurrentCursorPosition(editorRef.current);
|
|
206
|
+
if (currentPos === plainTextCursorPos) {
|
|
207
|
+
event.preventDefault();
|
|
208
|
+
event.stopPropagation();
|
|
209
|
+
const didRevert = revertAsciiSmileyConversion({
|
|
210
|
+
editorElement: editorRef.current,
|
|
211
|
+
original,
|
|
212
|
+
emoji
|
|
213
|
+
});
|
|
214
|
+
lastAsciiConversionRef.current = null;
|
|
215
|
+
if (didRevert) {
|
|
216
|
+
// Notify React + downstream consumers via a synthetic input event
|
|
217
|
+
const newEvent = new Event('input', {
|
|
218
|
+
bubbles: true
|
|
219
|
+
});
|
|
220
|
+
editorRef.current.dispatchEvent(newEvent);
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
} else {
|
|
224
|
+
// Cursor moved away from the just-converted emoji -> drop tracker
|
|
225
|
+
lastAsciiConversionRef.current = null;
|
|
226
|
+
}
|
|
227
|
+
} else if (!MODIFIER_KEYS.has(event.key)) {
|
|
228
|
+
// Any other actual keystroke invalidates the revert window
|
|
229
|
+
lastAsciiConversionRef.current = null;
|
|
230
|
+
}
|
|
171
231
|
if (event.key === 'Enter' && isPopupVisible) {
|
|
172
232
|
event.preventDefault();
|
|
173
233
|
return;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"EmojiInput.js","names":["AreaContext","useIsTouch","AnimatePresence","React","forwardRef","useCallback","useContext","useEffect","useImperativeHandle","useLayoutEffect","useMemo","useRef","useState","convertEmojisToUnicode","escapeHTML","insertTextAtCursorPosition","replaceText","getCharCodeThatWillBeDeleted","insertInvisibleCursorMarker","restoreSelection","saveSelection","insertPseudoMarker","insertCursorAtMarker","getCurrentCursorPosition","setCursorPositionByAbsIndex","convertHTMLToText","convertTextToHTML","EmojiPickerPopup","StyledEmojiInput","StyledEmojiInputContent","StyledEmojiInputLabel","StyledEmojiInputRightWrapper","StyledMotionEmojiInputEditor","StyledMotionEmojiInputProgress","PrefixElement","loadEmojiShortList","scrollCursorIntoView","useCursorPosition","useIsomorphicLayoutEffect","window","EmojiInput","accessToken","height","inputId","isDisabled","maxHeight","onBlur","onFocus","onInput","onKeyDown","onPrefixElementRemove","onPopupVisibilityChange","personId","placeholder","popupAlignment","prefixElement","rightElement","shouldHidePlaceholderOnFocus","shouldPreventEmojiPicker","value","onCursorPositionChange","ref","isTouch","plainTextValue","setPlainTextValue","hasFocus","setHasFocus","progressDuration","setProgressDuration","labelWidth","setLabelWidth","isPopupVisible","setIsPopupVisible","isPrefixAnimationFinished","setIsPrefixAnimationFinished","prefixElementWidth","setPrefixElementWidth","emojiShortNames","setEmojiShortNames","emojiRegShortNames","setEmojiRegShortNames","areaProvider","editorRef","prefixElementRef","hasPrefixRendered","hasPrefixChanged","shouldDeleteOneMoreBackwards","shouldDeleteOneMoreForwards","savedCursorPositionRef","valueRef","position","current","shouldChangeColor","then","shortNameList","regShortnames","handleUpdateHTML","html","newInnerHTML","innerHTML","shouldIgnoreEmptyTextNodes","handleBeforeInput","event","preventDefault","stopPropagation","data","type","nativeEvent","includes","text","editorElement","newEvent","Event","bubbles","dispatchEvent","handleInput","document","execCommand","handleKeyDown","key","isPropagationStopped","requestAnimationFrame","charCodeThatWillBeDeleted","handlePopupVisibility","isVisible","handlePaste","clipboardData","getData","handleDrop","dataTransfer","handlePopupSelect","emoji","shouldUseSavedSelection","convertedText","replace","convertedPrefix","length","handleInsertTextAtCursorPosition","handleReplaceText","searchText","pasteText","options","handleStartProgress","duration","handleStopProgress","handleSetCursorPosition","resolvedPosition","focus","updatedPosition","startProgress","stopProgress","blur","setCursorPosition","handlePreventLoseFocus","element","target","classList","contains","parentElement","body","addEventListener","removeEventListener","shouldShowPlaceholder","isJustPrefixElement","shouldRenderPlaceholder","handleFocus","handleBlur","offsetWidth","undefined","handleResize","resizeObserver","ResizeObserver","observe","disconnect","blurElement","activeElement","createElement","$isDisabled","$shouldChangeColor","initial","animate","width","exit","opacity","transition","ease","className","minHeight","contentEditable","id","onBeforeInput","onPaste","onDrop","$shouldShowContent","$maxWidth","$offsetWidth","onSelect","displayName"],"sources":["../../../../src/components/emoji-input/EmojiInput.tsx"],"sourcesContent":["import { AreaContext, useIsTouch } from '@chayns-components/core';\nimport { AnimatePresence } from 'motion/react';\nimport React, {\n ChangeEvent,\n ClipboardEvent,\n CSSProperties,\n FocusEvent,\n FocusEventHandler,\n forwardRef,\n KeyboardEvent as TmpKeyboardEvent,\n KeyboardEventHandler,\n ReactElement,\n ReactNode,\n useCallback,\n useContext,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useMemo,\n useRef,\n useState,\n type FormEvent,\n} from 'react';\nimport type { PopupAlignment } from '../../constants/alignment';\nimport { convertEmojisToUnicode, escapeHTML } from '../../utils/emoji';\nimport { insertTextAtCursorPosition, replaceText } from '../../utils/insert';\nimport {\n getCharCodeThatWillBeDeleted,\n insertInvisibleCursorMarker,\n restoreSelection,\n saveSelection,\n type ReplaceTextOptions,\n insertPseudoMarker,\n insertCursorAtMarker,\n getCurrentCursorPosition,\n setCursorPositionByAbsIndex,\n} from '../../utils/selection';\nimport { convertHTMLToText, convertTextToHTML } from '../../utils/text';\nimport EmojiPickerPopup from '../emoji-picker-popup/EmojiPickerPopup';\nimport {\n StyledEmojiInput,\n StyledEmojiInputContent,\n StyledEmojiInputLabel,\n StyledEmojiInputRightWrapper,\n StyledMotionEmojiInputEditor,\n StyledMotionEmojiInputProgress,\n} from './EmojiInput.styles';\nimport PrefixElement from './prefix-element/PrefixElement';\nimport { loadEmojiShortList } from '../../utils/asyncEmojiData';\nimport { scrollCursorIntoView } from '../../utils/scroll';\nimport { useCursorPosition } from '../../hooks/cursor';\n\nconst useIsomorphicLayoutEffect = typeof window !== 'undefined' ? useLayoutEffect : useEffect;\n\nexport type EmojiInputProps = {\n /**\n * Access token of the logged-in user. Is needed to load and save the history of the emojis.\n */\n accessToken?: string;\n /**\n * Sets the height of the input field to a fixed value. If this value is not set, the component will use the needed height until the maximum height is reached.\n */\n height?: CSSProperties['height'];\n /**\n * HTML id of the input element\n */\n inputId?: string;\n /**\n * Disables the input so that it cannot be changed anymore\n */\n isDisabled?: boolean;\n /**\n * Sets the maximum height of the input field.\n */\n maxHeight?: CSSProperties['maxHeight'];\n /**\n * Function that is executed when the input field loses focus.\n */\n onBlur?: FocusEventHandler<HTMLDivElement>;\n /**\n * Function to be executed when the cursor position is changed.\n */\n onCursorPositionChange?: (position: number) => void;\n /**\n * Function that is executed when the input field gets the focus.\n */\n onFocus?: FocusEventHandler<HTMLDivElement>;\n /**\n * Function that is executed when the text of the input changes. In addition to the original\n * event, the original text is returned as second parameter, in which the internally used HTML\n * elements have been converted back to BB codes.\n */\n onInput?: (event: ChangeEvent<HTMLDivElement>, originalText: string) => void;\n /**\n * Function that is executed when a key is pressed down.\n */\n onKeyDown?: KeyboardEventHandler<HTMLDivElement>;\n /**\n * Function to be executed if the prefixElement is removed.\n */\n onPrefixElementRemove?: () => void;\n /**\n * Function that is executed when the visibility of the popup changes.\n * @param {boolean} isVisible - Whether the popup is visible or not\n */\n onPopupVisibilityChange?: (isVisible: boolean) => void;\n /**\n * Person id of the logged-in user. Is needed to load and save the history of the emojis.\n */\n personId?: string;\n /**\n * Placeholder for the input field\n */\n placeholder?: string | ReactElement;\n /**\n * Sets the alignment of the popup to a fixed value. If this value is not set, the component\n * calculates the best position on its own. Use the imported 'PopupAlignment' enum to set this\n * value.\n */\n popupAlignment?: PopupAlignment;\n /**\n * Element that is rendered before the input field but the placeholder is still visible.\n */\n prefixElement?: string;\n /**\n * Element that is rendered inside the EmojiInput on the right side.\n */\n rightElement?: ReactNode;\n /**\n * Whether the placeholder should be shown after the input has focus.\n */\n shouldHidePlaceholderOnFocus?: boolean;\n /**\n * Prevents the EmojiPickerPopup icon from being displayed\n */\n shouldPreventEmojiPicker?: boolean;\n /**\n * The plain text value of the input field. Instead of HTML elements BB codes must be used at\n * this point. These are then converted by the input field into corresponding HTML elements.\n */\n value: string;\n};\n\nexport interface EmojiInputRef {\n insertTextAtCursorPosition: (text: string) => void;\n replaceText: (searchText: string, replaceText: string, options?: ReplaceTextOptions) => void;\n startProgress: (durationInSeconds: number) => void;\n stopProgress: () => void;\n focus: () => void;\n blur: () => void;\n setCursorPosition: (position?: number) => void;\n}\n\nconst EmojiInput = forwardRef<EmojiInputRef, EmojiInputProps>(\n (\n {\n accessToken,\n height,\n inputId,\n isDisabled,\n maxHeight = '190px',\n onBlur,\n onFocus,\n onInput,\n onKeyDown,\n onPrefixElementRemove,\n onPopupVisibilityChange,\n personId,\n placeholder,\n popupAlignment,\n prefixElement,\n rightElement,\n shouldHidePlaceholderOnFocus = false,\n shouldPreventEmojiPicker,\n value,\n onCursorPositionChange,\n },\n ref,\n ) => {\n const isTouch = useIsTouch();\n const [plainTextValue, setPlainTextValue] = useState(value);\n const [hasFocus, setHasFocus] = useState(false);\n const [progressDuration, setProgressDuration] = useState(0);\n const [labelWidth, setLabelWidth] = useState(0);\n const [isPopupVisible, setIsPopupVisible] = useState(false);\n const [isPrefixAnimationFinished, setIsPrefixAnimationFinished] = useState(!prefixElement);\n const [prefixElementWidth, setPrefixElementWidth] = useState<number | undefined>();\n const [emojiShortNames, setEmojiShortNames] = useState<{ [p: string]: string }>({});\n const [emojiRegShortNames, setEmojiRegShortNames] = useState<RegExp>(/./);\n\n const areaProvider = useContext(AreaContext);\n\n const editorRef = useRef<HTMLDivElement>(null);\n const prefixElementRef = useRef<HTMLDivElement>(null);\n const hasPrefixRendered = useRef(false);\n const hasPrefixChanged = useRef(false);\n const shouldDeleteOneMoreBackwards = useRef(false);\n const shouldDeleteOneMoreForwards = useRef(false);\n\n const savedCursorPositionRef = useRef<number>(0);\n\n const valueRef = useRef(value);\n\n useCursorPosition(\n editorRef,\n useCallback(\n (position: number) => {\n savedCursorPositionRef.current = position;\n\n if (typeof onCursorPositionChange === 'function') {\n onCursorPositionChange(position);\n }\n },\n [onCursorPositionChange],\n ),\n { isDisabled },\n );\n\n const shouldChangeColor = useMemo(\n () => areaProvider.shouldChangeColor ?? false,\n [areaProvider.shouldChangeColor],\n );\n\n useEffect(() => {\n void loadEmojiShortList().then(({ shortNameList, regShortnames }) => {\n setEmojiShortNames(shortNameList);\n setEmojiRegShortNames(regShortnames);\n });\n }, []);\n\n /**\n * This function updates the content of the 'contentEditable' element if the new text is\n * different from the previous content. So this is only true if, for example, a text like \":-)\"\n * has been replaced to the corresponding emoji.\n *\n * When updating the HTML, the current cursor position is saved before replacing the content, so\n * that it can be set again afterward.\n */\n const handleUpdateHTML = useCallback(\n (html: string) => {\n if (!editorRef.current) {\n return;\n }\n\n let newInnerHTML = convertEmojisToUnicode(\n html,\n emojiRegShortNames,\n emojiShortNames,\n );\n\n newInnerHTML = convertTextToHTML(newInnerHTML);\n\n if (newInnerHTML !== editorRef.current.innerHTML) {\n saveSelection(editorRef.current, { shouldIgnoreEmptyTextNodes: true });\n\n editorRef.current.innerHTML = newInnerHTML;\n\n restoreSelection(editorRef.current);\n }\n },\n [emojiRegShortNames, emojiShortNames],\n );\n\n const handleBeforeInput = useCallback(\n (event: FormEvent<HTMLDivElement>) => {\n if (!editorRef.current) {\n return;\n }\n\n if (isDisabled) {\n event.preventDefault();\n event.stopPropagation();\n\n return;\n }\n\n const { data, type } = event.nativeEvent as InputEvent;\n\n if (type === 'textInput' && data && data.includes('\\n')) {\n event.preventDefault();\n event.stopPropagation();\n\n const text = convertEmojisToUnicode(data, emojiRegShortNames, emojiShortNames);\n\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function handles the 'input' events of the 'contentEditable' element and also passes the\n * respective event up accordingly if the 'onInput' property is a function.\n */\n const handleInput = useCallback(\n (event: ChangeEvent<HTMLDivElement>) => {\n if (!editorRef.current) {\n return;\n }\n\n if (isDisabled) {\n event.stopPropagation();\n event.preventDefault();\n }\n\n if (shouldDeleteOneMoreBackwards.current) {\n shouldDeleteOneMoreBackwards.current = false;\n shouldDeleteOneMoreForwards.current = false;\n\n event.preventDefault();\n event.stopPropagation();\n\n // Remove content and set cursor to the right position\n insertInvisibleCursorMarker();\n\n return;\n }\n\n if (shouldDeleteOneMoreForwards.current) {\n shouldDeleteOneMoreBackwards.current = false;\n shouldDeleteOneMoreForwards.current = false;\n\n event.preventDefault();\n event.stopPropagation();\n\n // noinspection JSDeprecatedSymbols\n document.execCommand('forwardDelete', false);\n\n return;\n }\n\n handleUpdateHTML(editorRef.current.innerHTML);\n\n const text = convertHTMLToText(editorRef.current.innerHTML);\n\n setPlainTextValue(text);\n\n if (typeof onInput === 'function') {\n onInput(event, text);\n }\n\n insertCursorAtMarker(editorRef);\n },\n [handleUpdateHTML, isDisabled, onInput],\n );\n\n const handleKeyDown = useCallback(\n (event: TmpKeyboardEvent<HTMLDivElement>) => {\n if (isDisabled) {\n event.preventDefault();\n event.stopPropagation();\n\n return;\n }\n\n if (event.key === 'Enter' && isPopupVisible) {\n event.preventDefault();\n\n return;\n }\n\n if (typeof onKeyDown === 'function') {\n onKeyDown(event);\n }\n\n if (event.key === 'Enter' && !event.isPropagationStopped() && editorRef.current) {\n event.preventDefault();\n\n // noinspection JSDeprecatedSymbols\n document.execCommand('insertLineBreak', false);\n }\n\n if (event.key === 'Enter') {\n requestAnimationFrame(() => {\n if (editorRef.current) scrollCursorIntoView(editorRef.current);\n });\n }\n\n if (\n event.key === 'Backspace' ||\n event.key === 'Delete' ||\n event.key === 'Unidentified'\n ) {\n const charCodeThatWillBeDeleted = getCharCodeThatWillBeDeleted(event);\n\n if (charCodeThatWillBeDeleted === 8203) {\n if (event.key === 'Backspace' || event.key === 'Unidentified') {\n shouldDeleteOneMoreBackwards.current = true;\n } else {\n shouldDeleteOneMoreForwards.current = true;\n }\n }\n }\n },\n [isDisabled, isPopupVisible, onKeyDown],\n );\n\n const handlePopupVisibility = useCallback(\n (isVisible: boolean) => {\n setIsPopupVisible(isVisible);\n\n if (editorRef.current && isVisible) {\n saveSelection(editorRef.current);\n }\n\n if (typeof onPopupVisibilityChange === 'function') {\n onPopupVisibilityChange(isVisible);\n }\n },\n [onPopupVisibilityChange],\n );\n\n /**\n * This function prevents formatting from being adopted when texts are inserted. To do this, the\n * plain text is read from the event after the default behavior has been prevented. The plain\n * text is then inserted at the correct position in the input field using document.execCommand('insertHTML')\n */\n const handlePaste = useCallback(\n (event: ClipboardEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n if (isDisabled) {\n event.stopPropagation();\n\n return;\n }\n\n // This ensures, that only the copied text is inserted and not its HTML formatting.\n let text = event.clipboardData.getData('text/plain');\n\n text = convertEmojisToUnicode(text, emojiRegShortNames, emojiShortNames);\n\n /* This ensures, that valid HTML in the inserted text is not interpreted as such. e.g. if the user\n pasted the text '<b>test</b>' (not as formatted html), the <b> tags need to be escaped, to\n prevent it from being interpreted as html. */\n text = escapeHTML(text);\n\n // Insert an invisible control character at the end of the text to place the cursor in the correct position after insertion.\n if (text.includes('\\n')) {\n text += '\\u200C';\n }\n\n // This deprecated function is used, because it causes the inserted content to be added to the undo stack.\n // If the text were to be inserted directly into the 'innerHTML' of the editor element, the undo stack would not be updated.\n // In that case on CTRL+Z the inserted text would not be removed.\n document.execCommand('insertHTML', false, text);\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function prevents formatting from being adopted when texts are dropped. To do this, the\n * plain text is read from the event after the default behavior has been prevented. The plain\n * text is then inserted at the correct position in the input field using document.execCommand('insertHTML')\n */\n const handleDrop = useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n if (isDisabled) {\n event.stopPropagation();\n\n return;\n }\n\n // This ensures, that only the dropped text is inserted and not its HTML formatting.\n let text = event.dataTransfer?.getData('text');\n\n if (!text) {\n return;\n }\n\n text = convertEmojisToUnicode(text, emojiRegShortNames, emojiShortNames);\n\n /* This ensures, that valid HTML in the inserted text is not interpreted as such. e.g. if the user\n drops the text '<b>test</b>' (not as formatted html), the <b> tags need to be escaped, to\n prevent it from being interpreted as html. */\n text = escapeHTML(text);\n\n // This deprecated function is used, because it causes the inserted content to be added to the undo stack.\n // If the text were to be inserted directly into the 'innerHTML' of the editor element, the undo stack would not be updated.\n // In that case on CTRL+Z the inserted text would not be removed.\n document.execCommand('insertHTML', false, text);\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function uses the 'insertTextAtCursorPosition' function to insert the emoji at the\n * correct position in the editor element.\n *\n * At the end an 'input' event is dispatched, so that the function 'handleInput' is triggered,\n * which in turn executes the 'onInput' function from the props. So this serves to ensure that\n * the event is also passed through to the top when inserting via the popup.\n */\n const handlePopupSelect = useCallback((emoji: string) => {\n if (editorRef.current) {\n insertTextAtCursorPosition({\n editorElement: editorRef.current,\n text: emoji,\n shouldUseSavedSelection: true,\n });\n\n const event = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(event);\n }\n }, []);\n\n useEffect(() => {\n if (typeof onPrefixElementRemove !== 'function') {\n return;\n }\n\n if (!hasPrefixRendered.current) {\n return;\n }\n\n const convertedText = convertHTMLToText(editorRef.current?.innerHTML ?? '').replace(\n ' ',\n ' ',\n );\n const convertedPrefix = prefixElement && prefixElement.replace(' ', ' ');\n\n if (\n (convertedPrefix &&\n convertedText.includes(convertedPrefix) &&\n convertedText.length > convertedPrefix.length) ||\n convertedPrefix === convertedText\n ) {\n return;\n }\n\n if (hasPrefixChanged.current) {\n hasPrefixChanged.current = false;\n\n return;\n }\n\n onPrefixElementRemove();\n hasPrefixRendered.current = false;\n }, [onPrefixElementRemove, plainTextValue.length, prefixElement]);\n\n useEffect(() => {\n if (typeof prefixElement === 'string') {\n hasPrefixChanged.current = true;\n }\n }, [prefixElement]);\n\n useEffect(() => {\n if (value !== plainTextValue) {\n setPlainTextValue(value);\n\n handleUpdateHTML(value);\n }\n }, [handleUpdateHTML, plainTextValue, value]);\n\n // This effect is used to call the 'handleUpdateHTML' function once after the component has been\n // rendered. This is necessary because the 'contentEditable' element otherwise does not display\n // the HTML content correctly when the component is rendered for the first time.\n useIsomorphicLayoutEffect(() => {\n handleUpdateHTML(valueRef.current);\n }, [handleUpdateHTML]);\n\n const handleInsertTextAtCursorPosition = useCallback((text: string) => {\n if (editorRef.current) {\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n }, []);\n\n const handleReplaceText = useCallback(\n (searchText: string, pasteText: string, options?: ReplaceTextOptions) => {\n if (editorRef.current) {\n replaceText({\n editorElement: editorRef.current,\n searchText,\n pasteText,\n options,\n });\n\n insertPseudoMarker();\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [],\n );\n\n const handleStartProgress = useCallback((duration: number) => {\n setProgressDuration(duration);\n }, []);\n\n const handleStopProgress = useCallback(() => {\n setProgressDuration(0);\n }, []);\n\n const handleSetCursorPosition = useCallback((position?: number) => {\n if (!editorRef.current) {\n return;\n }\n\n const resolvedPosition = position ?? savedCursorPositionRef.current;\n\n savedCursorPositionRef.current = resolvedPosition;\n\n editorRef.current.focus();\n\n setCursorPositionByAbsIndex({\n editorElement: editorRef.current,\n position: resolvedPosition,\n });\n\n const updatedPosition = getCurrentCursorPosition(editorRef.current);\n if (typeof updatedPosition === 'number') {\n savedCursorPositionRef.current = updatedPosition;\n }\n }, []);\n\n useImperativeHandle(\n ref,\n () => ({\n insertTextAtCursorPosition: handleInsertTextAtCursorPosition,\n replaceText: handleReplaceText,\n startProgress: handleStartProgress,\n stopProgress: handleStopProgress,\n focus: () => editorRef.current?.focus(),\n blur: () => editorRef.current?.blur(),\n setCursorPosition: handleSetCursorPosition,\n }),\n [\n handleInsertTextAtCursorPosition,\n handleReplaceText,\n handleStartProgress,\n handleStopProgress,\n handleSetCursorPosition,\n ],\n );\n\n useEffect(() => {\n /**\n * This function ensures that the input field does not lose focus when the popup is opened\n * or an emoji is selected in it. For this purpose the corresponding elements get the class\n * 'prevent-lose-focus'.\n *\n * The class can also be set to any other elements that should also not cause the input\n * field to lose focus.\n */\n const handlePreventLoseFocus = (event: MouseEvent) => {\n const element = event.target as Element;\n\n if (\n element.classList.contains('prevent-lose-focus') ||\n element.parentElement?.classList.contains('prevent-lose-focus') ||\n element.parentElement?.parentElement?.classList.contains('prevent-lose-focus')\n ) {\n event.preventDefault();\n event.stopPropagation();\n }\n };\n\n document.body.addEventListener('mousedown', handlePreventLoseFocus);\n\n return () => {\n document.body.removeEventListener('mousedown', handlePreventLoseFocus);\n };\n }, []);\n\n const shouldShowPlaceholder = useMemo(() => {\n if (!isPrefixAnimationFinished) {\n return false;\n }\n\n const isJustPrefixElement =\n prefixElement && convertTextToHTML(prefixElement) === editorRef.current?.innerHTML;\n\n const shouldRenderPlaceholder =\n (prefixElement && !plainTextValue) ||\n (prefixElement ? prefixElementWidth && prefixElementWidth > 0 : true);\n\n switch (true) {\n case (!plainTextValue || isJustPrefixElement) &&\n shouldHidePlaceholderOnFocus &&\n !hasFocus:\n case (!plainTextValue || isJustPrefixElement) && !shouldHidePlaceholderOnFocus:\n return shouldRenderPlaceholder;\n case (!plainTextValue || isJustPrefixElement) &&\n shouldHidePlaceholderOnFocus &&\n hasFocus:\n return false;\n default:\n return false;\n }\n }, [\n isPrefixAnimationFinished,\n hasFocus,\n plainTextValue,\n prefixElement,\n shouldHidePlaceholderOnFocus,\n prefixElementWidth,\n ]);\n\n useEffect(() => {\n if (prefixElement) {\n setIsPrefixAnimationFinished(false);\n }\n }, [prefixElement]);\n\n const handleFocus = (event: FocusEvent<HTMLDivElement>) => {\n if (typeof onFocus === 'function' && !isDisabled) {\n onFocus(event);\n }\n\n setHasFocus(true);\n };\n\n const handleBlur = (event: FocusEvent<HTMLDivElement>) => {\n if (typeof onBlur === 'function' && !isDisabled) {\n onBlur(event);\n }\n\n setHasFocus(false);\n };\n\n useEffect(() => {\n if (editorRef.current && prefixElement) {\n const text = convertEmojisToUnicode(\n prefixElement,\n emojiRegShortNames,\n emojiShortNames,\n );\n\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n handleUpdateHTML(prefixElement);\n hasPrefixRendered.current = true;\n }\n }, [emojiRegShortNames, emojiShortNames, handleUpdateHTML, prefixElement]);\n\n useEffect(() => {\n if (\n prefixElementRef.current &&\n prefixElement &&\n convertTextToHTML(prefixElement) === editorRef.current?.innerHTML\n ) {\n setPrefixElementWidth(prefixElementRef.current.offsetWidth + 2);\n } else {\n setPrefixElementWidth(undefined);\n }\n }, [plainTextValue, prefixElement]);\n\n useEffect(() => {\n const handleResize = () => {\n if (editorRef.current) {\n setLabelWidth(editorRef.current.offsetWidth);\n }\n };\n\n const resizeObserver = new ResizeObserver(handleResize);\n\n if (editorRef.current) {\n resizeObserver.observe(editorRef.current);\n }\n\n return () => {\n resizeObserver.disconnect();\n };\n }, []);\n\n useEffect(() => {\n const blurElement = () => {\n if (\n editorRef.current &&\n document.activeElement === editorRef.current &&\n isDisabled\n ) {\n editorRef.current.blur();\n }\n };\n\n document.addEventListener('focus', blurElement, true);\n\n return () => {\n document.removeEventListener('focus', blurElement, true);\n };\n }, [isDisabled]);\n\n return (\n <StyledEmojiInput $isDisabled={isDisabled} $shouldChangeColor={shouldChangeColor}>\n <AnimatePresence initial>\n {progressDuration > 0 && (\n <StyledMotionEmojiInputProgress\n animate={{ width: '100%' }}\n exit={{ opacity: 0 }}\n initial={{ opacity: 1, width: '0%' }}\n transition={{\n width: {\n ease: 'linear',\n duration: progressDuration,\n },\n opacity: {\n type: 'tween',\n duration: 0.3,\n },\n }}\n />\n )}\n </AnimatePresence>\n <StyledEmojiInputContent>\n {prefixElement && (\n <PrefixElement\n key={prefixElement}\n element={prefixElement}\n prefixElementRef={prefixElementRef}\n setIsPrefixAnimationFinished={setIsPrefixAnimationFinished}\n />\n )}\n <StyledMotionEmojiInputEditor\n className=\"chayns-scrollbar\"\n animate={{ maxHeight: height ?? maxHeight, minHeight: height ?? '26px' }}\n contentEditable\n id={inputId}\n onBeforeInput={handleBeforeInput}\n onBlur={handleBlur}\n onFocus={handleFocus}\n onInput={handleInput}\n onKeyDown={handleKeyDown}\n onPaste={handlePaste}\n onDrop={handleDrop}\n ref={editorRef}\n $shouldShowContent={isPrefixAnimationFinished}\n transition={{ type: 'tween', duration: 0.2 }}\n />\n\n {shouldShowPlaceholder && (\n <StyledEmojiInputLabel\n $maxWidth={labelWidth}\n $offsetWidth={prefixElementWidth}\n >\n {placeholder}\n </StyledEmojiInputLabel>\n )}\n {!isTouch && !shouldPreventEmojiPicker && (\n <EmojiPickerPopup\n accessToken={accessToken}\n onSelect={handlePopupSelect}\n onPopupVisibilityChange={handlePopupVisibility}\n personId={personId}\n />\n )}\n </StyledEmojiInputContent>\n {rightElement && (\n <StyledEmojiInputRightWrapper>{rightElement}</StyledEmojiInputRightWrapper>\n )}\n </StyledEmojiInput>\n );\n },\n);\n\nEmojiInput.displayName = 'EmojiInput';\n\nexport default EmojiInput;\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"EmojiInput.js","names":["AreaContext","useIsTouch","AnimatePresence","React","forwardRef","useCallback","useContext","useEffect","useImperativeHandle","useLayoutEffect","useMemo","useRef","useState","convertEmojisToUnicode","escapeHTML","insertTextAtCursorPosition","replaceText","revertAsciiSmileyConversion","getCharCodeThatWillBeDeleted","insertInvisibleCursorMarker","restoreSelection","saveSelection","insertPseudoMarker","insertCursorAtMarker","getCurrentCursorPosition","setCursorPositionByAbsIndex","convertHTMLToText","convertTextToHTML","EmojiPickerPopup","StyledEmojiInput","StyledEmojiInputContent","StyledEmojiInputLabel","StyledEmojiInputRightWrapper","StyledMotionEmojiInputEditor","StyledMotionEmojiInputProgress","PrefixElement","loadEmojiShortList","scrollCursorIntoView","useCursorPosition","MODIFIER_KEYS","Set","useIsomorphicLayoutEffect","window","EmojiInput","accessToken","height","inputId","isDisabled","maxHeight","onBlur","onFocus","onInput","onKeyDown","onPrefixElementRemove","onPopupVisibilityChange","personId","placeholder","popupAlignment","prefixElement","rightElement","shouldHidePlaceholderOnFocus","shouldPreventEmojiPicker","value","onCursorPositionChange","ref","isTouch","plainTextValue","setPlainTextValue","hasFocus","setHasFocus","progressDuration","setProgressDuration","labelWidth","setLabelWidth","isPopupVisible","setIsPopupVisible","isPrefixAnimationFinished","setIsPrefixAnimationFinished","prefixElementWidth","setPrefixElementWidth","emojiShortNames","setEmojiShortNames","emojiRegShortNames","setEmojiRegShortNames","areaProvider","editorRef","prefixElementRef","hasPrefixRendered","hasPrefixChanged","shouldDeleteOneMoreBackwards","shouldDeleteOneMoreForwards","lastAsciiConversionRef","savedCursorPositionRef","valueRef","position","current","shouldChangeColor","then","shortNameList","regShortnames","handleUpdateHTML","html","conversions","newInnerHTML","onAsciiConversion","conv","push","innerHTML","shouldIgnoreEmptyTextNodes","length","lastConv","pos","original","emoji","plainTextCursorPos","handleBeforeInput","event","preventDefault","stopPropagation","data","type","nativeEvent","includes","text","editorElement","newEvent","Event","bubbles","dispatchEvent","handleInput","document","execCommand","handleKeyDown","key","ctrlKey","metaKey","currentPos","didRevert","has","isPropagationStopped","requestAnimationFrame","charCodeThatWillBeDeleted","handlePopupVisibility","isVisible","handlePaste","clipboardData","getData","handleDrop","dataTransfer","handlePopupSelect","shouldUseSavedSelection","convertedText","replace","convertedPrefix","handleInsertTextAtCursorPosition","handleReplaceText","searchText","pasteText","options","handleStartProgress","duration","handleStopProgress","handleSetCursorPosition","resolvedPosition","focus","updatedPosition","startProgress","stopProgress","blur","setCursorPosition","handlePreventLoseFocus","element","target","classList","contains","parentElement","body","addEventListener","removeEventListener","shouldShowPlaceholder","isJustPrefixElement","shouldRenderPlaceholder","handleFocus","handleBlur","offsetWidth","undefined","handleResize","resizeObserver","ResizeObserver","observe","disconnect","blurElement","activeElement","createElement","$isDisabled","$shouldChangeColor","initial","animate","width","exit","opacity","transition","ease","className","minHeight","contentEditable","id","onBeforeInput","onPaste","onDrop","$shouldShowContent","$maxWidth","$offsetWidth","onSelect","displayName"],"sources":["../../../../src/components/emoji-input/EmojiInput.tsx"],"sourcesContent":["import { AreaContext, useIsTouch } from '@chayns-components/core';\nimport { AnimatePresence } from 'motion/react';\nimport React, {\n ChangeEvent,\n ClipboardEvent,\n CSSProperties,\n FocusEvent,\n FocusEventHandler,\n forwardRef,\n KeyboardEvent as TmpKeyboardEvent,\n KeyboardEventHandler,\n ReactElement,\n ReactNode,\n useCallback,\n useContext,\n useEffect,\n useImperativeHandle,\n useLayoutEffect,\n useMemo,\n useRef,\n useState,\n type FormEvent,\n} from 'react';\nimport type { PopupAlignment } from '../../constants/alignment';\nimport { convertEmojisToUnicode, escapeHTML } from '../../utils/emoji';\nimport {\n insertTextAtCursorPosition,\n replaceText,\n revertAsciiSmileyConversion,\n} from '../../utils/insert';\nimport {\n getCharCodeThatWillBeDeleted,\n insertInvisibleCursorMarker,\n restoreSelection,\n saveSelection,\n type ReplaceTextOptions,\n insertPseudoMarker,\n insertCursorAtMarker,\n getCurrentCursorPosition,\n setCursorPositionByAbsIndex,\n} from '../../utils/selection';\nimport { convertHTMLToText, convertTextToHTML } from '../../utils/text';\nimport EmojiPickerPopup from '../emoji-picker-popup/EmojiPickerPopup';\nimport {\n StyledEmojiInput,\n StyledEmojiInputContent,\n StyledEmojiInputLabel,\n StyledEmojiInputRightWrapper,\n StyledMotionEmojiInputEditor,\n StyledMotionEmojiInputProgress,\n} from './EmojiInput.styles';\nimport PrefixElement from './prefix-element/PrefixElement';\nimport { loadEmojiShortList } from '../../utils/asyncEmojiData';\nimport { scrollCursorIntoView } from '../../utils/scroll';\nimport { useCursorPosition } from '../../hooks/cursor';\n\nconst MODIFIER_KEYS = new Set(['Shift', 'Control', 'Alt', 'Meta', 'CapsLock']);\n\nconst useIsomorphicLayoutEffect = typeof window !== 'undefined' ? useLayoutEffect : useEffect;\n\nexport type EmojiInputProps = {\n /**\n * Access token of the logged-in user. Is needed to load and save the history of the emojis.\n */\n accessToken?: string;\n /**\n * Sets the height of the input field to a fixed value. If this value is not set, the component will use the needed height until the maximum height is reached.\n */\n height?: CSSProperties['height'];\n /**\n * HTML id of the input element\n */\n inputId?: string;\n /**\n * Disables the input so that it cannot be changed anymore\n */\n isDisabled?: boolean;\n /**\n * Sets the maximum height of the input field.\n */\n maxHeight?: CSSProperties['maxHeight'];\n /**\n * Function that is executed when the input field loses focus.\n */\n onBlur?: FocusEventHandler<HTMLDivElement>;\n /**\n * Function to be executed when the cursor position is changed.\n */\n onCursorPositionChange?: (position: number) => void;\n /**\n * Function that is executed when the input field gets the focus.\n */\n onFocus?: FocusEventHandler<HTMLDivElement>;\n /**\n * Function that is executed when the text of the input changes. In addition to the original\n * event, the original text is returned as second parameter, in which the internally used HTML\n * elements have been converted back to BB codes.\n */\n onInput?: (event: ChangeEvent<HTMLDivElement>, originalText: string) => void;\n /**\n * Function that is executed when a key is pressed down.\n */\n onKeyDown?: KeyboardEventHandler<HTMLDivElement>;\n /**\n * Function to be executed if the prefixElement is removed.\n */\n onPrefixElementRemove?: () => void;\n /**\n * Function that is executed when the visibility of the popup changes.\n * @param {boolean} isVisible - Whether the popup is visible or not\n */\n onPopupVisibilityChange?: (isVisible: boolean) => void;\n /**\n * Person id of the logged-in user. Is needed to load and save the history of the emojis.\n */\n personId?: string;\n /**\n * Placeholder for the input field\n */\n placeholder?: string | ReactElement;\n /**\n * Sets the alignment of the popup to a fixed value. If this value is not set, the component\n * calculates the best position on its own. Use the imported 'PopupAlignment' enum to set this\n * value.\n */\n popupAlignment?: PopupAlignment;\n /**\n * Element that is rendered before the input field but the placeholder is still visible.\n */\n prefixElement?: string;\n /**\n * Element that is rendered inside the EmojiInput on the right side.\n */\n rightElement?: ReactNode;\n /**\n * Whether the placeholder should be shown after the input has focus.\n */\n shouldHidePlaceholderOnFocus?: boolean;\n /**\n * Prevents the EmojiPickerPopup icon from being displayed\n */\n shouldPreventEmojiPicker?: boolean;\n /**\n * The plain text value of the input field. Instead of HTML elements BB codes must be used at\n * this point. These are then converted by the input field into corresponding HTML elements.\n */\n value: string;\n};\n\nexport interface EmojiInputRef {\n insertTextAtCursorPosition: (text: string) => void;\n replaceText: (searchText: string, replaceText: string, options?: ReplaceTextOptions) => void;\n startProgress: (durationInSeconds: number) => void;\n stopProgress: () => void;\n focus: () => void;\n blur: () => void;\n setCursorPosition: (position?: number) => void;\n}\n\nconst EmojiInput = forwardRef<EmojiInputRef, EmojiInputProps>(\n (\n {\n accessToken,\n height,\n inputId,\n isDisabled,\n maxHeight = '190px',\n onBlur,\n onFocus,\n onInput,\n onKeyDown,\n onPrefixElementRemove,\n onPopupVisibilityChange,\n personId,\n placeholder,\n popupAlignment,\n prefixElement,\n rightElement,\n shouldHidePlaceholderOnFocus = false,\n shouldPreventEmojiPicker,\n value,\n onCursorPositionChange,\n },\n ref,\n ) => {\n const isTouch = useIsTouch();\n const [plainTextValue, setPlainTextValue] = useState(value);\n const [hasFocus, setHasFocus] = useState(false);\n const [progressDuration, setProgressDuration] = useState(0);\n const [labelWidth, setLabelWidth] = useState(0);\n const [isPopupVisible, setIsPopupVisible] = useState(false);\n const [isPrefixAnimationFinished, setIsPrefixAnimationFinished] = useState(!prefixElement);\n const [prefixElementWidth, setPrefixElementWidth] = useState<number | undefined>();\n const [emojiShortNames, setEmojiShortNames] = useState<{ [p: string]: string }>({});\n const [emojiRegShortNames, setEmojiRegShortNames] = useState<RegExp>(/./);\n\n const areaProvider = useContext(AreaContext);\n\n const editorRef = useRef<HTMLDivElement>(null);\n const prefixElementRef = useRef<HTMLDivElement>(null);\n const hasPrefixRendered = useRef(false);\n const hasPrefixChanged = useRef(false);\n const shouldDeleteOneMoreBackwards = useRef(false);\n const shouldDeleteOneMoreForwards = useRef(false);\n\n /** * Tracks the most recently auto-converted ASCII smiley so the next * Backspace keystroke can revert it (Word/IntelliJ-style autocorrect undo). * The ref is invalidated as soon as the user presses any non-modifier key * other than Backspace (see `handleKeyDown`). */\n const lastAsciiConversionRef = useRef<{\n original: string;\n emoji: string;\n plainTextCursorPos: number;\n } | null>(null);\n\n const savedCursorPositionRef = useRef<number>(0);\n\n const valueRef = useRef(value);\n\n useCursorPosition(\n editorRef,\n useCallback(\n (position: number) => {\n savedCursorPositionRef.current = position;\n\n if (typeof onCursorPositionChange === 'function') {\n onCursorPositionChange(position);\n }\n },\n [onCursorPositionChange],\n ),\n { isDisabled },\n );\n\n const shouldChangeColor = useMemo(\n () => areaProvider.shouldChangeColor ?? false,\n [areaProvider.shouldChangeColor],\n );\n\n useEffect(() => {\n void loadEmojiShortList().then(({ shortNameList, regShortnames }) => {\n setEmojiShortNames(shortNameList);\n setEmojiRegShortNames(regShortnames);\n });\n }, []);\n\n /**\n * This function updates the content of the 'contentEditable' element if the new text is\n * different from the previous content. So this is only true if, for example, a text like \":-)\"\n * has been replaced to the corresponding emoji.\n *\n * When updating the HTML, the current cursor position is saved before replacing the content, so\n * that it can be set again afterward.\n */\n const handleUpdateHTML = useCallback(\n (html: string) => {\n if (!editorRef.current) {\n return;\n }\n\n const conversions: { original: string; emoji: string }[] = [];\n\n let newInnerHTML = convertEmojisToUnicode(\n html,\n emojiRegShortNames,\n emojiShortNames,\n {\n onAsciiConversion: (conv) => conversions.push(conv),\n },\n );\n\n newInnerHTML = convertTextToHTML(newInnerHTML);\n\n if (newInnerHTML !== editorRef.current.innerHTML) {\n saveSelection(editorRef.current, { shouldIgnoreEmptyTextNodes: true });\n\n editorRef.current.innerHTML = newInnerHTML;\n\n restoreSelection(editorRef.current);\n\n // Remember the LAST ASCII conversion so Backspace can revert it.\n // Older conversions are dropped on purpose: only the smiley the\n // user just produced should be undoable.\n if (conversions.length > 0) {\n const lastConv = conversions[conversions.length - 1];\n const pos = getCurrentCursorPosition(editorRef.current);\n if (pos !== null) {\n lastAsciiConversionRef.current = {\n original: lastConv!.original,\n emoji: lastConv!.emoji,\n plainTextCursorPos: pos,\n };\n }\n }\n }\n },\n [emojiRegShortNames, emojiShortNames],\n );\n\n const handleBeforeInput = useCallback(\n (event: FormEvent<HTMLDivElement>) => {\n if (!editorRef.current) {\n return;\n }\n\n if (isDisabled) {\n event.preventDefault();\n event.stopPropagation();\n\n return;\n }\n\n const { data, type } = event.nativeEvent as InputEvent;\n\n if (type === 'textInput' && data && data.includes('\\n')) {\n event.preventDefault();\n event.stopPropagation();\n\n const text = convertEmojisToUnicode(data, emojiRegShortNames, emojiShortNames);\n\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function handles the 'input' events of the 'contentEditable' element and also passes the\n * respective event up accordingly if the 'onInput' property is a function.\n */\n const handleInput = useCallback(\n (event: ChangeEvent<HTMLDivElement>) => {\n if (!editorRef.current) {\n return;\n }\n\n if (isDisabled) {\n event.stopPropagation();\n event.preventDefault();\n }\n\n if (shouldDeleteOneMoreBackwards.current) {\n shouldDeleteOneMoreBackwards.current = false;\n shouldDeleteOneMoreForwards.current = false;\n\n event.preventDefault();\n event.stopPropagation();\n\n // Remove content and set cursor to the right position\n insertInvisibleCursorMarker();\n\n return;\n }\n\n if (shouldDeleteOneMoreForwards.current) {\n shouldDeleteOneMoreBackwards.current = false;\n shouldDeleteOneMoreForwards.current = false;\n\n event.preventDefault();\n event.stopPropagation();\n\n // noinspection JSDeprecatedSymbols\n document.execCommand('forwardDelete', false);\n\n return;\n }\n\n handleUpdateHTML(editorRef.current.innerHTML);\n\n const text = convertHTMLToText(editorRef.current.innerHTML);\n\n setPlainTextValue(text);\n\n if (typeof onInput === 'function') {\n onInput(event, text);\n }\n\n insertCursorAtMarker(editorRef);\n },\n [handleUpdateHTML, isDisabled, onInput],\n );\n\n const handleKeyDown = useCallback(\n (event: TmpKeyboardEvent<HTMLDivElement>) => {\n if (isDisabled) {\n event.preventDefault();\n event.stopPropagation();\n\n return;\n }\n\n // --- Backspace-revert for the most recent auto-conversion ---\n // If the user presses Backspace immediately after an ASCII smiley\n // was auto-converted (and the cursor is still at the post-conversion\n // position), we revert the emoji back to the original text and wrap\n // it in a protection span so it does not get re-converted.\n if (\n event.key === 'Backspace' &&\n !event.ctrlKey &&\n !event.metaKey &&\n lastAsciiConversionRef.current &&\n editorRef.current\n ) {\n const { original, emoji, plainTextCursorPos } = lastAsciiConversionRef.current;\n const currentPos = getCurrentCursorPosition(editorRef.current);\n\n if (currentPos === plainTextCursorPos) {\n event.preventDefault();\n event.stopPropagation();\n\n const didRevert = revertAsciiSmileyConversion({\n editorElement: editorRef.current,\n original,\n emoji,\n });\n\n lastAsciiConversionRef.current = null;\n\n if (didRevert) {\n // Notify React + downstream consumers via a synthetic input event\n const newEvent = new Event('input', { bubbles: true });\n editorRef.current.dispatchEvent(newEvent);\n return;\n }\n } else {\n // Cursor moved away from the just-converted emoji -> drop tracker\n lastAsciiConversionRef.current = null;\n }\n } else if (!MODIFIER_KEYS.has(event.key)) {\n // Any other actual keystroke invalidates the revert window\n lastAsciiConversionRef.current = null;\n }\n\n if (event.key === 'Enter' && isPopupVisible) {\n event.preventDefault();\n\n return;\n }\n\n if (typeof onKeyDown === 'function') {\n onKeyDown(event);\n }\n\n if (event.key === 'Enter' && !event.isPropagationStopped() && editorRef.current) {\n event.preventDefault();\n\n // noinspection JSDeprecatedSymbols\n document.execCommand('insertLineBreak', false);\n }\n\n if (event.key === 'Enter') {\n requestAnimationFrame(() => {\n if (editorRef.current) scrollCursorIntoView(editorRef.current);\n });\n }\n\n if (\n event.key === 'Backspace' ||\n event.key === 'Delete' ||\n event.key === 'Unidentified'\n ) {\n const charCodeThatWillBeDeleted = getCharCodeThatWillBeDeleted(event);\n\n if (charCodeThatWillBeDeleted === 8203) {\n if (event.key === 'Backspace' || event.key === 'Unidentified') {\n shouldDeleteOneMoreBackwards.current = true;\n } else {\n shouldDeleteOneMoreForwards.current = true;\n }\n }\n }\n },\n [isDisabled, isPopupVisible, onKeyDown],\n );\n\n const handlePopupVisibility = useCallback(\n (isVisible: boolean) => {\n setIsPopupVisible(isVisible);\n\n if (editorRef.current && isVisible) {\n saveSelection(editorRef.current);\n }\n\n if (typeof onPopupVisibilityChange === 'function') {\n onPopupVisibilityChange(isVisible);\n }\n },\n [onPopupVisibilityChange],\n );\n\n /**\n * This function prevents formatting from being adopted when texts are inserted. To do this, the\n * plain text is read from the event after the default behavior has been prevented. The plain\n * text is then inserted at the correct position in the input field using document.execCommand('insertHTML')\n */\n const handlePaste = useCallback(\n (event: ClipboardEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n if (isDisabled) {\n event.stopPropagation();\n\n return;\n }\n\n // This ensures, that only the copied text is inserted and not its HTML formatting.\n let text = event.clipboardData.getData('text/plain');\n\n text = convertEmojisToUnicode(text, emojiRegShortNames, emojiShortNames);\n\n /* This ensures, that valid HTML in the inserted text is not interpreted as such. e.g. if the user\n pasted the text '<b>test</b>' (not as formatted html), the <b> tags need to be escaped, to\n prevent it from being interpreted as html. */\n text = escapeHTML(text);\n\n // Insert an invisible control character at the end of the text to place the cursor in the correct position after insertion.\n if (text.includes('\\n')) {\n text += '\\u200C';\n }\n\n // This deprecated function is used, because it causes the inserted content to be added to the undo stack.\n // If the text were to be inserted directly into the 'innerHTML' of the editor element, the undo stack would not be updated.\n // In that case on CTRL+Z the inserted text would not be removed.\n document.execCommand('insertHTML', false, text);\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function prevents formatting from being adopted when texts are dropped. To do this, the\n * plain text is read from the event after the default behavior has been prevented. The plain\n * text is then inserted at the correct position in the input field using document.execCommand('insertHTML')\n */\n const handleDrop = useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n if (isDisabled) {\n event.stopPropagation();\n\n return;\n }\n\n // This ensures, that only the dropped text is inserted and not its HTML formatting.\n let text = event.dataTransfer?.getData('text');\n\n if (!text) {\n return;\n }\n\n text = convertEmojisToUnicode(text, emojiRegShortNames, emojiShortNames);\n\n /* This ensures, that valid HTML in the inserted text is not interpreted as such. e.g. if the user\n drops the text '<b>test</b>' (not as formatted html), the <b> tags need to be escaped, to\n prevent it from being interpreted as html. */\n text = escapeHTML(text);\n\n // This deprecated function is used, because it causes the inserted content to be added to the undo stack.\n // If the text were to be inserted directly into the 'innerHTML' of the editor element, the undo stack would not be updated.\n // In that case on CTRL+Z the inserted text would not be removed.\n document.execCommand('insertHTML', false, text);\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [emojiRegShortNames, emojiShortNames, isDisabled],\n );\n\n /**\n * This function uses the 'insertTextAtCursorPosition' function to insert the emoji at the\n * correct position in the editor element.\n *\n * At the end an 'input' event is dispatched, so that the function 'handleInput' is triggered,\n * which in turn executes the 'onInput' function from the props. So this serves to ensure that\n * the event is also passed through to the top when inserting via the popup.\n */\n const handlePopupSelect = useCallback((emoji: string) => {\n if (editorRef.current) {\n insertTextAtCursorPosition({\n editorElement: editorRef.current,\n text: emoji,\n shouldUseSavedSelection: true,\n });\n\n const event = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(event);\n }\n }, []);\n\n useEffect(() => {\n if (typeof onPrefixElementRemove !== 'function') {\n return;\n }\n\n if (!hasPrefixRendered.current) {\n return;\n }\n\n const convertedText = convertHTMLToText(editorRef.current?.innerHTML ?? '').replace(\n ' ',\n ' ',\n );\n const convertedPrefix = prefixElement && prefixElement.replace(' ', ' ');\n\n if (\n (convertedPrefix &&\n convertedText.includes(convertedPrefix) &&\n convertedText.length > convertedPrefix.length) ||\n convertedPrefix === convertedText\n ) {\n return;\n }\n\n if (hasPrefixChanged.current) {\n hasPrefixChanged.current = false;\n\n return;\n }\n\n onPrefixElementRemove();\n hasPrefixRendered.current = false;\n }, [onPrefixElementRemove, plainTextValue.length, prefixElement]);\n\n useEffect(() => {\n if (typeof prefixElement === 'string') {\n hasPrefixChanged.current = true;\n }\n }, [prefixElement]);\n\n useEffect(() => {\n if (value !== plainTextValue) {\n setPlainTextValue(value);\n\n handleUpdateHTML(value);\n }\n }, [handleUpdateHTML, plainTextValue, value]);\n\n // This effect is used to call the 'handleUpdateHTML' function once after the component has been\n // rendered. This is necessary because the 'contentEditable' element otherwise does not display\n // the HTML content correctly when the component is rendered for the first time.\n useIsomorphicLayoutEffect(() => {\n handleUpdateHTML(valueRef.current);\n }, [handleUpdateHTML]);\n\n const handleInsertTextAtCursorPosition = useCallback((text: string) => {\n if (editorRef.current) {\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n }, []);\n\n const handleReplaceText = useCallback(\n (searchText: string, pasteText: string, options?: ReplaceTextOptions) => {\n if (editorRef.current) {\n replaceText({\n editorElement: editorRef.current,\n searchText,\n pasteText,\n options,\n });\n\n insertPseudoMarker();\n\n const newEvent = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(newEvent);\n }\n },\n [],\n );\n\n const handleStartProgress = useCallback((duration: number) => {\n setProgressDuration(duration);\n }, []);\n\n const handleStopProgress = useCallback(() => {\n setProgressDuration(0);\n }, []);\n\n const handleSetCursorPosition = useCallback((position?: number) => {\n if (!editorRef.current) {\n return;\n }\n\n const resolvedPosition = position ?? savedCursorPositionRef.current;\n\n savedCursorPositionRef.current = resolvedPosition;\n\n editorRef.current.focus();\n\n setCursorPositionByAbsIndex({\n editorElement: editorRef.current,\n position: resolvedPosition,\n });\n\n const updatedPosition = getCurrentCursorPosition(editorRef.current);\n if (typeof updatedPosition === 'number') {\n savedCursorPositionRef.current = updatedPosition;\n }\n }, []);\n\n useImperativeHandle(\n ref,\n () => ({\n insertTextAtCursorPosition: handleInsertTextAtCursorPosition,\n replaceText: handleReplaceText,\n startProgress: handleStartProgress,\n stopProgress: handleStopProgress,\n focus: () => editorRef.current?.focus(),\n blur: () => editorRef.current?.blur(),\n setCursorPosition: handleSetCursorPosition,\n }),\n [\n handleInsertTextAtCursorPosition,\n handleReplaceText,\n handleStartProgress,\n handleStopProgress,\n handleSetCursorPosition,\n ],\n );\n\n useEffect(() => {\n /**\n * This function ensures that the input field does not lose focus when the popup is opened\n * or an emoji is selected in it. For this purpose the corresponding elements get the class\n * 'prevent-lose-focus'.\n *\n * The class can also be set to any other elements that should also not cause the input\n * field to lose focus.\n */\n const handlePreventLoseFocus = (event: MouseEvent) => {\n const element = event.target as Element;\n\n if (\n element.classList.contains('prevent-lose-focus') ||\n element.parentElement?.classList.contains('prevent-lose-focus') ||\n element.parentElement?.parentElement?.classList.contains('prevent-lose-focus')\n ) {\n event.preventDefault();\n event.stopPropagation();\n }\n };\n\n document.body.addEventListener('mousedown', handlePreventLoseFocus);\n\n return () => {\n document.body.removeEventListener('mousedown', handlePreventLoseFocus);\n };\n }, []);\n\n const shouldShowPlaceholder = useMemo(() => {\n if (!isPrefixAnimationFinished) {\n return false;\n }\n\n const isJustPrefixElement =\n prefixElement && convertTextToHTML(prefixElement) === editorRef.current?.innerHTML;\n\n const shouldRenderPlaceholder =\n (prefixElement && !plainTextValue) ||\n (prefixElement ? prefixElementWidth && prefixElementWidth > 0 : true);\n\n switch (true) {\n case (!plainTextValue || isJustPrefixElement) &&\n shouldHidePlaceholderOnFocus &&\n !hasFocus:\n case (!plainTextValue || isJustPrefixElement) && !shouldHidePlaceholderOnFocus:\n return shouldRenderPlaceholder;\n case (!plainTextValue || isJustPrefixElement) &&\n shouldHidePlaceholderOnFocus &&\n hasFocus:\n return false;\n default:\n return false;\n }\n }, [\n isPrefixAnimationFinished,\n hasFocus,\n plainTextValue,\n prefixElement,\n shouldHidePlaceholderOnFocus,\n prefixElementWidth,\n ]);\n\n useEffect(() => {\n if (prefixElement) {\n setIsPrefixAnimationFinished(false);\n }\n }, [prefixElement]);\n\n const handleFocus = (event: FocusEvent<HTMLDivElement>) => {\n if (typeof onFocus === 'function' && !isDisabled) {\n onFocus(event);\n }\n\n setHasFocus(true);\n };\n\n const handleBlur = (event: FocusEvent<HTMLDivElement>) => {\n if (typeof onBlur === 'function' && !isDisabled) {\n onBlur(event);\n }\n\n setHasFocus(false);\n };\n\n useEffect(() => {\n if (editorRef.current && prefixElement) {\n const text = convertEmojisToUnicode(\n prefixElement,\n emojiRegShortNames,\n emojiShortNames,\n );\n\n insertTextAtCursorPosition({ editorElement: editorRef.current, text });\n\n handleUpdateHTML(prefixElement);\n hasPrefixRendered.current = true;\n }\n }, [emojiRegShortNames, emojiShortNames, handleUpdateHTML, prefixElement]);\n\n useEffect(() => {\n if (\n prefixElementRef.current &&\n prefixElement &&\n convertTextToHTML(prefixElement) === editorRef.current?.innerHTML\n ) {\n setPrefixElementWidth(prefixElementRef.current.offsetWidth + 2);\n } else {\n setPrefixElementWidth(undefined);\n }\n }, [plainTextValue, prefixElement]);\n\n useEffect(() => {\n const handleResize = () => {\n if (editorRef.current) {\n setLabelWidth(editorRef.current.offsetWidth);\n }\n };\n\n const resizeObserver = new ResizeObserver(handleResize);\n\n if (editorRef.current) {\n resizeObserver.observe(editorRef.current);\n }\n\n return () => {\n resizeObserver.disconnect();\n };\n }, []);\n\n useEffect(() => {\n const blurElement = () => {\n if (\n editorRef.current &&\n document.activeElement === editorRef.current &&\n isDisabled\n ) {\n editorRef.current.blur();\n }\n };\n\n document.addEventListener('focus', blurElement, true);\n\n return () => {\n document.removeEventListener('focus', blurElement, true);\n };\n }, [isDisabled]);\n\n return (\n <StyledEmojiInput $isDisabled={isDisabled} $shouldChangeColor={shouldChangeColor}>\n <AnimatePresence initial>\n {progressDuration > 0 && (\n <StyledMotionEmojiInputProgress\n animate={{ width: '100%' }}\n exit={{ opacity: 0 }}\n initial={{ opacity: 1, width: '0%' }}\n transition={{\n width: {\n ease: 'linear',\n duration: progressDuration,\n },\n opacity: {\n type: 'tween',\n duration: 0.3,\n },\n }}\n />\n )}\n </AnimatePresence>\n <StyledEmojiInputContent>\n {prefixElement && (\n <PrefixElement\n key={prefixElement}\n element={prefixElement}\n prefixElementRef={prefixElementRef}\n setIsPrefixAnimationFinished={setIsPrefixAnimationFinished}\n />\n )}\n <StyledMotionEmojiInputEditor\n className=\"chayns-scrollbar\"\n animate={{ maxHeight: height ?? maxHeight, minHeight: height ?? '26px' }}\n contentEditable\n id={inputId}\n onBeforeInput={handleBeforeInput}\n onBlur={handleBlur}\n onFocus={handleFocus}\n onInput={handleInput}\n onKeyDown={handleKeyDown}\n onPaste={handlePaste}\n onDrop={handleDrop}\n ref={editorRef}\n $shouldShowContent={isPrefixAnimationFinished}\n transition={{ type: 'tween', duration: 0.2 }}\n />\n\n {shouldShowPlaceholder && (\n <StyledEmojiInputLabel\n $maxWidth={labelWidth}\n $offsetWidth={prefixElementWidth}\n >\n {placeholder}\n </StyledEmojiInputLabel>\n )}\n {!isTouch && !shouldPreventEmojiPicker && (\n <EmojiPickerPopup\n accessToken={accessToken}\n onSelect={handlePopupSelect}\n onPopupVisibilityChange={handlePopupVisibility}\n personId={personId}\n />\n )}\n </StyledEmojiInputContent>\n {rightElement && (\n <StyledEmojiInputRightWrapper>{rightElement}</StyledEmojiInputRightWrapper>\n )}\n </StyledEmojiInput>\n );\n },\n);\n\nEmojiInput.displayName = 'EmojiInput';\n\nexport default EmojiInput;\n"],"mappings":"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","ignoreList":[]}
|