@chayns-components/emoji-input 5.0.0-beta.156 → 5.0.0-beta.157
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.
|
@@ -32,6 +32,8 @@ const EmojiInput = _ref => {
|
|
|
32
32
|
const [isMobile] = (0, _react.useState)((0, _environment.getIsMobile)());
|
|
33
33
|
const [plainTextValue, setPlainTextValue] = (0, _react.useState)(value);
|
|
34
34
|
const editorRef = (0, _react.useRef)(null);
|
|
35
|
+
const shouldDeleteOneMoreBackwards = (0, _react.useRef)(false);
|
|
36
|
+
const shouldDeleteOneMoreForwards = (0, _react.useRef)(false);
|
|
35
37
|
|
|
36
38
|
/**
|
|
37
39
|
* This function updates the content of the 'contentEditable' element if the new text is
|
|
@@ -64,6 +66,22 @@ const EmojiInput = _ref => {
|
|
|
64
66
|
if (!editorRef.current) {
|
|
65
67
|
return;
|
|
66
68
|
}
|
|
69
|
+
if (shouldDeleteOneMoreBackwards.current) {
|
|
70
|
+
shouldDeleteOneMoreBackwards.current = false;
|
|
71
|
+
shouldDeleteOneMoreForwards.current = false;
|
|
72
|
+
event.preventDefault();
|
|
73
|
+
event.stopPropagation();
|
|
74
|
+
document.execCommand('delete', false);
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
if (shouldDeleteOneMoreForwards.current) {
|
|
78
|
+
shouldDeleteOneMoreBackwards.current = false;
|
|
79
|
+
shouldDeleteOneMoreForwards.current = false;
|
|
80
|
+
event.preventDefault();
|
|
81
|
+
event.stopPropagation();
|
|
82
|
+
document.execCommand('forwardDelete', false);
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
67
85
|
handleUpdateHTML(editorRef.current.innerHTML);
|
|
68
86
|
const text = (0, _text.convertHTMLToText)(editorRef.current.innerHTML);
|
|
69
87
|
setPlainTextValue(text);
|
|
@@ -79,6 +97,16 @@ const EmojiInput = _ref => {
|
|
|
79
97
|
event.preventDefault();
|
|
80
98
|
document.execCommand('insertLineBreak', false);
|
|
81
99
|
}
|
|
100
|
+
if (event.key === 'Backspace' || event.key === 'Delete') {
|
|
101
|
+
const charCodeThatWillBeDeleted = (0, _selection.getCharCodeThatWillBeDeleted)(event);
|
|
102
|
+
if (charCodeThatWillBeDeleted === 8203) {
|
|
103
|
+
if (event.key === 'Backspace') {
|
|
104
|
+
shouldDeleteOneMoreBackwards.current = true;
|
|
105
|
+
} else {
|
|
106
|
+
shouldDeleteOneMoreForwards.current = true;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
82
110
|
}, [onKeyDown]);
|
|
83
111
|
|
|
84
112
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"EmojiInput.js","names":["_react","_interopRequireWildcard","require","_emoji","_environment","_insert","_selection","_text","_EmojiPickerPopup","_interopRequireDefault","_EmojiInput","obj","__esModule","default","_getRequireWildcardCache","nodeInterop","WeakMap","cacheBabelInterop","cacheNodeInterop","cache","has","get","newObj","hasPropertyDescriptor","Object","defineProperty","getOwnPropertyDescriptor","key","prototype","hasOwnProperty","call","desc","set","EmojiInput","_ref","accessToken","inputId","isDisabled","onInput","onKeyDown","onPopupVisibilityChange","personId","placeholder","popupAlignment","rightElement","value","isMobile","useState","getIsMobile","plainTextValue","setPlainTextValue","editorRef","useRef","handleUpdateHTML","useCallback","html","current","newInnerHTML","convertEmojisToUnicode","convertTextToHTML","innerHTML","saveSelection","shouldIgnoreEmptyTextNodes","restoreSelection","handleInput","event","text","convertHTMLToText","handleKeyDown","shiftKey","isPropagationStopped","preventDefault","document","execCommand","handlePaste","clipboardData","getData","insertTextAtCursorPosition","editorElement","newEvent","Event","bubbles","dispatchEvent","handlePopupSelect","emoji","useEffect","handlePreventLoseFocus","_element$parentElemen","_element$parentElemen2","_element$parentElemen3","element","target","classList","contains","parentElement","stopPropagation","body","addEventListener","removeEventListener","createElement","StyledEmojiInput","StyledEmojiInputContent","isRightElementGiven","StyledEmojiInputEditor","contentEditable","id","onPaste","ref","alignment","onSelect","StyledEmojiInputRightWrapper","displayName","_default","exports"],"sources":["../../../src/components/emoji-input/EmojiInput.tsx"],"sourcesContent":["import React, {\n ChangeEvent,\n ClipboardEvent,\n FC,\n KeyboardEvent,\n KeyboardEventHandler,\n ReactNode,\n useCallback,\n useEffect,\n useRef,\n useState,\n} from 'react';\nimport type { PopupAlignment } from '../../constants/alignment';\nimport { convertEmojisToUnicode } from '../../utils/emoji';\nimport { getIsMobile } from '../../utils/environment';\nimport { insertTextAtCursorPosition } from '../../utils/insert';\nimport { restoreSelection, saveSelection } from '../../utils/selection';\nimport { convertHTMLToText, convertTextToHTML } from '../../utils/text';\nimport EmojiPickerPopup from '../emoji-picker-popup/EmojiPickerPopup';\nimport {\n StyledEmojiInput,\n StyledEmojiInputContent,\n StyledEmojiInputEditor,\n StyledEmojiInputRightWrapper,\n} from './EmojiInput.styles';\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 * 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 * 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 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;\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 inside the EmojiInput on the right side.\n */\n rightElement?: ReactNode;\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\nconst EmojiInput: FC<EmojiInputProps> = ({\n accessToken,\n inputId,\n isDisabled,\n onInput,\n onKeyDown,\n onPopupVisibilityChange,\n personId,\n placeholder,\n popupAlignment,\n rightElement,\n value,\n}) => {\n const [isMobile] = useState(getIsMobile());\n const [plainTextValue, setPlainTextValue] = useState(value);\n\n const editorRef = useRef<HTMLDivElement>(null);\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((html: string) => {\n if (!editorRef.current) {\n return;\n }\n\n let newInnerHTML = convertEmojisToUnicode(html);\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\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 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 [handleUpdateHTML, onInput]\n );\n\n const handleKeyDown = useCallback(\n (event: KeyboardEvent<HTMLDivElement>) => {\n if (typeof onKeyDown === 'function') {\n onKeyDown(event);\n }\n\n if (\n event.key === 'Enter' &&\n !event.shiftKey &&\n !event.isPropagationStopped() &&\n editorRef.current\n ) {\n event.preventDefault();\n\n document.execCommand('insertLineBreak', false);\n }\n },\n [onKeyDown]\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 the\n * 'insertTextAtCursorPosition' function.\n */\n const handlePaste = useCallback((event: ClipboardEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n let text = event.clipboardData.getData('text/plain');\n\n text = convertEmojisToUnicode(text);\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\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({ editorElement: editorRef.current, text: emoji });\n\n const event = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(event);\n }\n }, []);\n\n useEffect(() => {\n if (value !== plainTextValue) {\n setPlainTextValue(value);\n\n handleUpdateHTML(value);\n }\n }, [handleUpdateHTML, plainTextValue, value]);\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 return (\n <StyledEmojiInput isDisabled={isDisabled}>\n <StyledEmojiInputContent isRightElementGiven={!!rightElement}>\n <StyledEmojiInputEditor\n contentEditable={!isDisabled}\n id={inputId}\n onInput={handleInput}\n onKeyDown={handleKeyDown}\n onPaste={handlePaste}\n placeholder={placeholder}\n ref={editorRef}\n />\n {!isMobile && (\n <EmojiPickerPopup\n accessToken={accessToken}\n alignment={popupAlignment}\n onSelect={handlePopupSelect}\n onPopupVisibilityChange={onPopupVisibilityChange}\n personId={personId}\n />\n )}\n </StyledEmojiInputContent>\n {rightElement && (\n <StyledEmojiInputRightWrapper>{rightElement}</StyledEmojiInputRightWrapper>\n )}\n </StyledEmojiInput>\n );\n};\n\nEmojiInput.displayName = 'EmojiInput';\n\nexport default EmojiInput;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"EmojiInput.js","names":["_react","_interopRequireWildcard","require","_emoji","_environment","_insert","_selection","_text","_EmojiPickerPopup","_interopRequireDefault","_EmojiInput","obj","__esModule","default","_getRequireWildcardCache","nodeInterop","WeakMap","cacheBabelInterop","cacheNodeInterop","cache","has","get","newObj","hasPropertyDescriptor","Object","defineProperty","getOwnPropertyDescriptor","key","prototype","hasOwnProperty","call","desc","set","EmojiInput","_ref","accessToken","inputId","isDisabled","onInput","onKeyDown","onPopupVisibilityChange","personId","placeholder","popupAlignment","rightElement","value","isMobile","useState","getIsMobile","plainTextValue","setPlainTextValue","editorRef","useRef","shouldDeleteOneMoreBackwards","shouldDeleteOneMoreForwards","handleUpdateHTML","useCallback","html","current","newInnerHTML","convertEmojisToUnicode","convertTextToHTML","innerHTML","saveSelection","shouldIgnoreEmptyTextNodes","restoreSelection","handleInput","event","preventDefault","stopPropagation","document","execCommand","text","convertHTMLToText","handleKeyDown","shiftKey","isPropagationStopped","charCodeThatWillBeDeleted","getCharCodeThatWillBeDeleted","handlePaste","clipboardData","getData","insertTextAtCursorPosition","editorElement","newEvent","Event","bubbles","dispatchEvent","handlePopupSelect","emoji","useEffect","handlePreventLoseFocus","_element$parentElemen","_element$parentElemen2","_element$parentElemen3","element","target","classList","contains","parentElement","body","addEventListener","removeEventListener","createElement","StyledEmojiInput","StyledEmojiInputContent","isRightElementGiven","StyledEmojiInputEditor","contentEditable","id","onPaste","ref","alignment","onSelect","StyledEmojiInputRightWrapper","displayName","_default","exports"],"sources":["../../../src/components/emoji-input/EmojiInput.tsx"],"sourcesContent":["import React, {\n ChangeEvent,\n ClipboardEvent,\n FC,\n KeyboardEvent,\n KeyboardEventHandler,\n ReactNode,\n useCallback,\n useEffect,\n useRef,\n useState,\n} from 'react';\nimport type { PopupAlignment } from '../../constants/alignment';\nimport { convertEmojisToUnicode } from '../../utils/emoji';\nimport { getIsMobile } from '../../utils/environment';\nimport { insertTextAtCursorPosition } from '../../utils/insert';\nimport {\n getCharCodeThatWillBeDeleted,\n restoreSelection,\n saveSelection,\n} from '../../utils/selection';\nimport { convertHTMLToText, convertTextToHTML } from '../../utils/text';\nimport EmojiPickerPopup from '../emoji-picker-popup/EmojiPickerPopup';\nimport {\n StyledEmojiInput,\n StyledEmojiInputContent,\n StyledEmojiInputEditor,\n StyledEmojiInputRightWrapper,\n} from './EmojiInput.styles';\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 * 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 * 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 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;\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 inside the EmojiInput on the right side.\n */\n rightElement?: ReactNode;\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\nconst EmojiInput: FC<EmojiInputProps> = ({\n accessToken,\n inputId,\n isDisabled,\n onInput,\n onKeyDown,\n onPopupVisibilityChange,\n personId,\n placeholder,\n popupAlignment,\n rightElement,\n value,\n}) => {\n const [isMobile] = useState(getIsMobile());\n const [plainTextValue, setPlainTextValue] = useState(value);\n\n const editorRef = useRef<HTMLDivElement>(null);\n\n const shouldDeleteOneMoreBackwards = useRef(false);\n const shouldDeleteOneMoreForwards = useRef(false);\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((html: string) => {\n if (!editorRef.current) {\n return;\n }\n\n let newInnerHTML = convertEmojisToUnicode(html);\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\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 (shouldDeleteOneMoreBackwards.current) {\n shouldDeleteOneMoreBackwards.current = false;\n shouldDeleteOneMoreForwards.current = false;\n\n event.preventDefault();\n event.stopPropagation();\n\n document.execCommand('delete', false);\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 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 [handleUpdateHTML, onInput]\n );\n\n const handleKeyDown = useCallback(\n (event: KeyboardEvent<HTMLDivElement>) => {\n if (typeof onKeyDown === 'function') {\n onKeyDown(event);\n }\n\n if (\n event.key === 'Enter' &&\n !event.shiftKey &&\n !event.isPropagationStopped() &&\n editorRef.current\n ) {\n event.preventDefault();\n\n document.execCommand('insertLineBreak', false);\n }\n\n if (event.key === 'Backspace' || event.key === 'Delete') {\n const charCodeThatWillBeDeleted = getCharCodeThatWillBeDeleted(event);\n\n if (charCodeThatWillBeDeleted === 8203) {\n if (event.key === 'Backspace') {\n shouldDeleteOneMoreBackwards.current = true;\n } else {\n shouldDeleteOneMoreForwards.current = true;\n }\n }\n }\n },\n [onKeyDown]\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 the\n * 'insertTextAtCursorPosition' function.\n */\n const handlePaste = useCallback((event: ClipboardEvent<HTMLDivElement>) => {\n if (editorRef.current) {\n event.preventDefault();\n\n let text = event.clipboardData.getData('text/plain');\n\n text = convertEmojisToUnicode(text);\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\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({ editorElement: editorRef.current, text: emoji });\n\n const event = new Event('input', { bubbles: true });\n\n editorRef.current.dispatchEvent(event);\n }\n }, []);\n\n useEffect(() => {\n if (value !== plainTextValue) {\n setPlainTextValue(value);\n\n handleUpdateHTML(value);\n }\n }, [handleUpdateHTML, plainTextValue, value]);\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 return (\n <StyledEmojiInput isDisabled={isDisabled}>\n <StyledEmojiInputContent isRightElementGiven={!!rightElement}>\n <StyledEmojiInputEditor\n contentEditable={!isDisabled}\n id={inputId}\n onInput={handleInput}\n onKeyDown={handleKeyDown}\n onPaste={handlePaste}\n placeholder={placeholder}\n ref={editorRef}\n />\n {!isMobile && (\n <EmojiPickerPopup\n accessToken={accessToken}\n alignment={popupAlignment}\n onSelect={handlePopupSelect}\n onPopupVisibilityChange={onPopupVisibilityChange}\n personId={personId}\n />\n )}\n </StyledEmojiInputContent>\n {rightElement && (\n <StyledEmojiInputRightWrapper>{rightElement}</StyledEmojiInputRightWrapper>\n )}\n </StyledEmojiInput>\n );\n};\n\nEmojiInput.displayName = 'EmojiInput';\n\nexport default EmojiInput;\n"],"mappings":"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"}
|
package/lib/utils/selection.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import type { KeyboardEvent } from 'react';
|
|
1
2
|
interface SaveSelectionOptions {
|
|
2
3
|
shouldIgnoreEmptyTextNodes?: boolean;
|
|
3
4
|
}
|
|
@@ -5,4 +6,12 @@ export declare const saveSelection: (element: HTMLDivElement, { shouldIgnoreEmpt
|
|
|
5
6
|
export declare const restoreSelection: (element: HTMLDivElement) => void;
|
|
6
7
|
export declare const moveSelectionOffset: (distance: number) => void;
|
|
7
8
|
export declare const setChildIndex: (index: number) => void;
|
|
9
|
+
/**
|
|
10
|
+
* This function returns the code of the character that will be removed by the KeyDown event in the
|
|
11
|
+
* next step, if the "Backspace" or "Delete" key was pressed and there is no selection of multiple
|
|
12
|
+
* characters.
|
|
13
|
+
*
|
|
14
|
+
* @param event - Keyboard event from "onKeyDown"
|
|
15
|
+
*/
|
|
16
|
+
export declare const getCharCodeThatWillBeDeleted: (event: KeyboardEvent<HTMLDivElement>) => number | null | undefined;
|
|
8
17
|
export {};
|
package/lib/utils/selection.js
CHANGED
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.setChildIndex = exports.saveSelection = exports.restoreSelection = exports.moveSelectionOffset = void 0;
|
|
6
|
+
exports.setChildIndex = exports.saveSelection = exports.restoreSelection = exports.moveSelectionOffset = exports.getCharCodeThatWillBeDeleted = void 0;
|
|
7
7
|
let childIndex = -1;
|
|
8
8
|
let endOffset = -1;
|
|
9
9
|
let startOffset = -1;
|
|
@@ -44,18 +44,32 @@ const restoreSelection = element => {
|
|
|
44
44
|
return;
|
|
45
45
|
}
|
|
46
46
|
if (typeof childNode.nodeValue !== 'string') {
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
47
|
+
if (childNode.nextSibling) {
|
|
48
|
+
childNode = childNode.nextSibling;
|
|
49
|
+
if (childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue) {
|
|
50
|
+
if (childNode.nodeValue.charCodeAt(0) === 8203) {
|
|
51
|
+
endOffset = 1;
|
|
52
|
+
startOffset = 1;
|
|
53
|
+
} else {
|
|
54
|
+
endOffset = 0;
|
|
55
|
+
startOffset = 0;
|
|
56
|
+
}
|
|
57
|
+
} else {
|
|
58
|
+
var _childNode$parentNode;
|
|
59
|
+
const textNode = document.createTextNode('\u200B');
|
|
60
|
+
(_childNode$parentNode = childNode.parentNode) === null || _childNode$parentNode === void 0 ? void 0 : _childNode$parentNode.insertBefore(textNode, childNode.nextSibling);
|
|
61
|
+
childNode = textNode;
|
|
62
|
+
endOffset = textNode.length;
|
|
63
|
+
startOffset = textNode.length;
|
|
64
|
+
}
|
|
65
|
+
} else {
|
|
66
|
+
var _childNode$parentNode2;
|
|
67
|
+
const textNode = document.createTextNode('\u200B');
|
|
68
|
+
(_childNode$parentNode2 = childNode.parentNode) === null || _childNode$parentNode2 === void 0 ? void 0 : _childNode$parentNode2.insertBefore(textNode, childNode.nextSibling);
|
|
69
|
+
childNode = textNode;
|
|
70
|
+
endOffset = textNode.length;
|
|
71
|
+
startOffset = textNode.length;
|
|
72
|
+
}
|
|
59
73
|
} else if (childNode.nodeValue && endOffset > childNode.nodeValue.length) {
|
|
60
74
|
if (childNode.nextSibling) {
|
|
61
75
|
childNode = childNode.nextSibling;
|
|
@@ -63,19 +77,12 @@ const restoreSelection = element => {
|
|
|
63
77
|
endOffset = childNode.nodeValue.length;
|
|
64
78
|
startOffset = childNode.nodeValue.length;
|
|
65
79
|
} else {
|
|
66
|
-
var _childNode$
|
|
80
|
+
var _childNode$parentNode3;
|
|
67
81
|
const textNode = document.createTextNode('\u200B');
|
|
68
|
-
|
|
69
|
-
childNode,
|
|
70
|
-
nextSibling: childNode.nextSibling,
|
|
71
|
-
nodeType: childNode.nodeType,
|
|
72
|
-
parentNode: childNode.parentNode,
|
|
73
|
-
textNode
|
|
74
|
-
});
|
|
75
|
-
(_childNode$parentNode2 = childNode.parentNode) === null || _childNode$parentNode2 === void 0 ? void 0 : _childNode$parentNode2.insertBefore(textNode, childNode.nextSibling);
|
|
82
|
+
(_childNode$parentNode3 = childNode.parentNode) === null || _childNode$parentNode3 === void 0 ? void 0 : _childNode$parentNode3.insertBefore(textNode, childNode.nextSibling);
|
|
76
83
|
childNode = textNode;
|
|
77
|
-
endOffset =
|
|
78
|
-
startOffset =
|
|
84
|
+
endOffset = textNode.length;
|
|
85
|
+
startOffset = textNode.length;
|
|
79
86
|
}
|
|
80
87
|
} else {
|
|
81
88
|
endOffset = childNode.nodeValue.length;
|
|
@@ -98,5 +105,46 @@ exports.moveSelectionOffset = moveSelectionOffset;
|
|
|
98
105
|
const setChildIndex = index => {
|
|
99
106
|
childIndex = index;
|
|
100
107
|
};
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* This function returns the code of the character that will be removed by the KeyDown event in the
|
|
111
|
+
* next step, if the "Backspace" or "Delete" key was pressed and there is no selection of multiple
|
|
112
|
+
* characters.
|
|
113
|
+
*
|
|
114
|
+
* @param event - Keyboard event from "onKeyDown"
|
|
115
|
+
*/
|
|
101
116
|
exports.setChildIndex = setChildIndex;
|
|
117
|
+
const getCharCodeThatWillBeDeleted = event => {
|
|
118
|
+
var _window$getSelection, _nodeValue$length, _nextSibling$nodeValu;
|
|
119
|
+
const range = (_window$getSelection = window.getSelection()) === null || _window$getSelection === void 0 ? void 0 : _window$getSelection.getRangeAt(0);
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* At this point the function is aborted if there is no selection range, several characters have
|
|
123
|
+
* been selected and therefore no single letter is removed or neither the "Backspace" nor the
|
|
124
|
+
* "Delete" key has been pressed.
|
|
125
|
+
*/
|
|
126
|
+
if (!range || range.endOffset !== range.startOffset || event.key !== 'Backspace' && event.key !== 'Delete') {
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
if (event.key === 'Backspace') {
|
|
130
|
+
var _previousSibling$node;
|
|
131
|
+
const {
|
|
132
|
+
nodeValue,
|
|
133
|
+
previousSibling
|
|
134
|
+
} = range.startContainer;
|
|
135
|
+
if (range.startOffset > 0) {
|
|
136
|
+
return nodeValue === null || nodeValue === void 0 ? void 0 : nodeValue.charCodeAt(range.startOffset - 1);
|
|
137
|
+
}
|
|
138
|
+
return previousSibling === null || previousSibling === void 0 ? void 0 : (_previousSibling$node = previousSibling.nodeValue) === null || _previousSibling$node === void 0 ? void 0 : _previousSibling$node.charCodeAt(previousSibling.nodeValue.length - 1);
|
|
139
|
+
}
|
|
140
|
+
const {
|
|
141
|
+
nextSibling,
|
|
142
|
+
nodeValue
|
|
143
|
+
} = range.endContainer;
|
|
144
|
+
if (range.endOffset < ((_nodeValue$length = nodeValue === null || nodeValue === void 0 ? void 0 : nodeValue.length) !== null && _nodeValue$length !== void 0 ? _nodeValue$length : 0)) {
|
|
145
|
+
return nodeValue === null || nodeValue === void 0 ? void 0 : nodeValue.charCodeAt(range.endOffset);
|
|
146
|
+
}
|
|
147
|
+
return nextSibling === null || nextSibling === void 0 ? void 0 : (_nextSibling$nodeValu = nextSibling.nodeValue) === null || _nextSibling$nodeValu === void 0 ? void 0 : _nextSibling$nodeValu.charCodeAt(0);
|
|
148
|
+
};
|
|
149
|
+
exports.getCharCodeThatWillBeDeleted = getCharCodeThatWillBeDeleted;
|
|
102
150
|
//# sourceMappingURL=selection.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"selection.js","names":["childIndex","endOffset","startOffset","saveSelection","element","shouldIgnoreEmptyTextNodes","arguments","length","undefined","selection","window","getSelection","anchorNode","range","getRangeAt","childNodesArray","Array","from","childNodes","filter","_ref","nodeType","nodeValue","Node","TEXT_NODE","indexOf","exports","restoreSelection","childNode","_childNode$parentNode","textNode","document","createTextNode","console","debug","nextSibling","parentNode","insertBefore","_childNode$parentNode2","createRange","setStart","setEnd","removeAllRanges","addRange","collapse","moveSelectionOffset","distance","setChildIndex","index"],"sources":["../../src/utils/selection.ts"],"sourcesContent":["let childIndex = -1;\nlet endOffset = -1;\nlet startOffset = -1;\n\ninterface SaveSelectionOptions {\n shouldIgnoreEmptyTextNodes?: boolean;\n}\n\nexport const saveSelection = (\n element: HTMLDivElement,\n { shouldIgnoreEmptyTextNodes }: SaveSelectionOptions = {}\n) => {\n const selection = window.getSelection();\n\n if (!selection) {\n return;\n }\n\n const { anchorNode } = selection;\n\n if (!anchorNode) {\n return;\n }\n\n const range = selection.getRangeAt(0);\n\n let childNodesArray = Array.from(element.childNodes);\n\n if (shouldIgnoreEmptyTextNodes) {\n childNodesArray = childNodesArray.filter(\n ({ nodeType, nodeValue }) =>\n nodeType !== Node.TEXT_NODE || (nodeValue !== '' && nodeValue !== '\\u200B')\n );\n }\n\n childIndex = childNodesArray.indexOf(anchorNode as ChildNode);\n\n endOffset = range.endOffset;\n startOffset = range.startOffset;\n};\n\nexport const restoreSelection = (element: HTMLDivElement) => {\n let childNode = element.childNodes[childIndex];\n\n const selection = window.getSelection();\n\n if (!childNode || !element || !selection) {\n return;\n }\n\n if (typeof childNode.nodeValue !== 'string') {\n const textNode = document.createTextNode('\\u200B');\n\n console.debug('[1] Will add ZeroWithSpace...', {\n childNode,\n nextSibling: childNode.nextSibling,\n parentNode: childNode.parentNode,\n textNode,\n });\n\n childNode.parentNode?.insertBefore(textNode, childNode.nextSibling);\n\n childNode = textNode;\n\n endOffset = 0;\n startOffset = 0;\n } else if (childNode.nodeValue && endOffset > childNode.nodeValue.length) {\n if (childNode.nextSibling) {\n childNode = childNode.nextSibling;\n\n if (childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue) {\n endOffset = childNode.nodeValue.length;\n startOffset = childNode.nodeValue.length;\n } else {\n const textNode = document.createTextNode('\\u200B');\n\n console.debug('[2] Will add ZeroWithSpace...', {\n childNode,\n nextSibling: childNode.nextSibling,\n nodeType: childNode.nodeType,\n parentNode: childNode.parentNode,\n textNode,\n });\n\n childNode.parentNode?.insertBefore(textNode, childNode.nextSibling);\n\n childNode = textNode;\n\n endOffset = 0;\n startOffset = 0;\n }\n } else {\n endOffset = childNode.nodeValue.length;\n startOffset = childNode.nodeValue.length;\n }\n }\n\n const range = document.createRange();\n\n range.setStart(childNode, startOffset);\n range.setEnd(childNode, endOffset);\n\n selection.removeAllRanges();\n selection.addRange(range);\n\n range.collapse(true);\n};\n\nexport const moveSelectionOffset = (distance: number) => {\n endOffset += distance;\n startOffset += distance;\n};\n\nexport const setChildIndex = (index: number) => {\n childIndex = index;\n};\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"selection.js","names":["childIndex","endOffset","startOffset","saveSelection","element","shouldIgnoreEmptyTextNodes","arguments","length","undefined","selection","window","getSelection","anchorNode","range","getRangeAt","childNodesArray","Array","from","childNodes","filter","_ref","nodeType","nodeValue","Node","TEXT_NODE","indexOf","exports","restoreSelection","childNode","nextSibling","charCodeAt","_childNode$parentNode","textNode","document","createTextNode","parentNode","insertBefore","_childNode$parentNode2","_childNode$parentNode3","createRange","setStart","setEnd","removeAllRanges","addRange","collapse","moveSelectionOffset","distance","setChildIndex","index","getCharCodeThatWillBeDeleted","event","_window$getSelection","_nodeValue$length","_nextSibling$nodeValu","key","_previousSibling$node","previousSibling","startContainer","endContainer"],"sources":["../../src/utils/selection.ts"],"sourcesContent":["import type { KeyboardEvent } from 'react';\n\nlet childIndex = -1;\nlet endOffset = -1;\nlet startOffset = -1;\n\ninterface SaveSelectionOptions {\n shouldIgnoreEmptyTextNodes?: boolean;\n}\n\nexport const saveSelection = (\n element: HTMLDivElement,\n { shouldIgnoreEmptyTextNodes }: SaveSelectionOptions = {}\n) => {\n const selection = window.getSelection();\n\n if (!selection) {\n return;\n }\n\n const { anchorNode } = selection;\n\n if (!anchorNode) {\n return;\n }\n\n const range = selection.getRangeAt(0);\n\n let childNodesArray = Array.from(element.childNodes);\n\n if (shouldIgnoreEmptyTextNodes) {\n childNodesArray = childNodesArray.filter(\n ({ nodeType, nodeValue }) =>\n nodeType !== Node.TEXT_NODE || (nodeValue !== '' && nodeValue !== '\\u200B')\n );\n }\n\n childIndex = childNodesArray.indexOf(anchorNode as ChildNode);\n\n endOffset = range.endOffset;\n startOffset = range.startOffset;\n};\n\nexport const restoreSelection = (element: HTMLDivElement) => {\n let childNode = element.childNodes[childIndex];\n\n const selection = window.getSelection();\n\n if (!childNode || !element || !selection) {\n return;\n }\n\n if (typeof childNode.nodeValue !== 'string') {\n if (childNode.nextSibling) {\n childNode = childNode.nextSibling;\n\n if (childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue) {\n if (childNode.nodeValue.charCodeAt(0) === 8203) {\n endOffset = 1;\n startOffset = 1;\n } else {\n endOffset = 0;\n startOffset = 0;\n }\n } else {\n const textNode = document.createTextNode('\\u200B');\n\n childNode.parentNode?.insertBefore(textNode, childNode.nextSibling);\n\n childNode = textNode;\n\n endOffset = textNode.length;\n startOffset = textNode.length;\n }\n } else {\n const textNode = document.createTextNode('\\u200B');\n\n childNode.parentNode?.insertBefore(textNode, childNode.nextSibling);\n\n childNode = textNode;\n\n endOffset = textNode.length;\n startOffset = textNode.length;\n }\n } else if (childNode.nodeValue && endOffset > childNode.nodeValue.length) {\n if (childNode.nextSibling) {\n childNode = childNode.nextSibling;\n\n if (childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue) {\n endOffset = childNode.nodeValue.length;\n startOffset = childNode.nodeValue.length;\n } else {\n const textNode = document.createTextNode('\\u200B');\n\n childNode.parentNode?.insertBefore(textNode, childNode.nextSibling);\n\n childNode = textNode;\n\n endOffset = textNode.length;\n startOffset = textNode.length;\n }\n } else {\n endOffset = childNode.nodeValue.length;\n startOffset = childNode.nodeValue.length;\n }\n }\n\n const range = document.createRange();\n\n range.setStart(childNode, startOffset);\n range.setEnd(childNode, endOffset);\n\n selection.removeAllRanges();\n selection.addRange(range);\n\n range.collapse(true);\n};\n\nexport const moveSelectionOffset = (distance: number) => {\n endOffset += distance;\n startOffset += distance;\n};\n\nexport const setChildIndex = (index: number) => {\n childIndex = index;\n};\n\n/**\n * This function returns the code of the character that will be removed by the KeyDown event in the\n * next step, if the \"Backspace\" or \"Delete\" key was pressed and there is no selection of multiple\n * characters.\n *\n * @param event - Keyboard event from \"onKeyDown\"\n */\nexport const getCharCodeThatWillBeDeleted = (event: KeyboardEvent<HTMLDivElement>) => {\n const range = window.getSelection()?.getRangeAt(0);\n\n /**\n * At this point the function is aborted if there is no selection range, several characters have\n * been selected and therefore no single letter is removed or neither the \"Backspace\" nor the\n * \"Delete\" key has been pressed.\n */\n if (\n !range ||\n range.endOffset !== range.startOffset ||\n (event.key !== 'Backspace' && event.key !== 'Delete')\n ) {\n return null;\n }\n\n if (event.key === 'Backspace') {\n const { nodeValue, previousSibling } = range.startContainer;\n\n if (range.startOffset > 0) {\n return nodeValue?.charCodeAt(range.startOffset - 1);\n }\n\n return previousSibling?.nodeValue?.charCodeAt(previousSibling.nodeValue.length - 1);\n }\n\n const { nextSibling, nodeValue } = range.endContainer;\n\n if (range.endOffset < (nodeValue?.length ?? 0)) {\n return nodeValue?.charCodeAt(range.endOffset);\n }\n\n return nextSibling?.nodeValue?.charCodeAt(0);\n};\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@chayns-components/emoji-input",
|
|
3
|
-
"version": "5.0.0-beta.
|
|
3
|
+
"version": "5.0.0-beta.157",
|
|
4
4
|
"description": "Input field that supports HTML elements and emojis",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"chayns",
|
|
@@ -68,5 +68,5 @@
|
|
|
68
68
|
"publishConfig": {
|
|
69
69
|
"access": "public"
|
|
70
70
|
},
|
|
71
|
-
"gitHead": "
|
|
71
|
+
"gitHead": "5cb4670cd4b014c26286f6d219b801d5f1f65d28"
|
|
72
72
|
}
|