@zsviczian/excalidraw 0.10.0-fixjumpingtext-1 → 0.10.0-fixjumpingtext-2

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.
@@ -2550,7 +2550,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
2550
2550
  /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
2551
2551
 
2552
2552
  "use strict";
2553
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"textWysiwyg\": () => (/* binding */ textWysiwyg)\n/* harmony export */ });\n/* harmony import */ var _keys__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../keys */ \"../../keys.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../scene/Scene */ \"../../scene/Scene.ts\");\n/* harmony import */ var _typeChecks__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _mutateElement__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _textElement__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../actions/actionProperties */ \"../../actions/actionProperties.tsx\");\n\r\n\r\n\r\n\r\n\r\n\r\n\r\n\r\nconst normalizeText = (text) => {\r\n return (text\r\n // replace tabs with spaces so they render and measure correctly\r\n .replace(/\\t/g, \" \")\r\n // normalize newlines\r\n .replace(/\\r?\\n|\\r/g, \"\\n\"));\r\n};\r\nconst getTransform = (width, height, angle, appState, maxWidth, maxHeight) => {\r\n const { zoom } = appState;\r\n const degree = (180 * angle) / Math.PI;\r\n let translateX = ((width) * (zoom.value - 1)) / 2;\r\n let translateY = ((height) * (zoom.value - 1)) / 2;\r\n if (width > maxWidth && zoom.value !== 1) {\r\n translateX = ((maxWidth) * (zoom.value - 1)) / 2;\r\n }\r\n if (height > maxHeight && zoom.value !== 1) {\r\n translateY = ((maxHeight) * (zoom.value - 1)) / 2;\r\n }\r\n return `translate(${translateX}px, ${translateY}px) scale(${zoom.value}) rotate(${degree}deg)`;\r\n};\r\nconst textWysiwyg = ({ id, appState, onChange, onSubmit, getViewportCoords, element, canvas, excalidrawContainer, app, }) => {\r\n const textPropertiesUpdated = (updatedElement, editable) => {\r\n const currentFont = editable.style.fontFamily.replaceAll('\"', \"\");\r\n if ((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontFamilyString)({ fontFamily: updatedElement.fontFamily }) !==\r\n currentFont) {\r\n return true;\r\n }\r\n if (`${updatedElement.fontSize}px` !== editable.style.fontSize) {\r\n return true;\r\n }\r\n return false;\r\n };\r\n let originalContainerHeight;\r\n let approxLineHeight = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element));\r\n const updateWysiwygStyle = () => {\r\n var _a;\r\n const updatedElement = (_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(id);\r\n if (updatedElement && (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updatedElement)) {\r\n let coordX = updatedElement.x;\r\n let coordY = updatedElement.y;\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(updatedElement);\r\n let maxWidth = updatedElement.width;\r\n let maxHeight = updatedElement.height;\r\n let width = updatedElement.width;\r\n // Set to element height by default since thats\r\n // what is going to be used for unbounded text\r\n let height = updatedElement.height;\r\n if (container && updatedElement.containerId) {\r\n const propertiesUpdated = textPropertiesUpdated(updatedElement, editable);\r\n // using editor.style.height to get the accurate height of text editor\r\n const editorHeight = Number(editable.style.height.slice(0, -2));\r\n if (editorHeight > 0) {\r\n height = editorHeight;\r\n }\r\n if (propertiesUpdated) {\r\n approxLineHeight = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement));\r\n originalContainerHeight = container.height;\r\n // update height of the editor after properties updated\r\n height = updatedElement.height;\r\n }\r\n if (!originalContainerHeight) {\r\n originalContainerHeight = container.height;\r\n }\r\n maxWidth = container.width - _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING * 2;\r\n maxHeight = container.height - _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING * 2;\r\n width = maxWidth;\r\n // The coordinates of text box set a distance of\r\n // 30px to preserve padding\r\n coordX = container.x + _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING;\r\n // autogrow container height if text exceeds\r\n if (height > maxHeight) {\r\n const diff = Math.min(height - maxHeight, approxLineHeight);\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: container.height + diff });\r\n return;\r\n }\r\n else if (\r\n // autoshrink container height until original container height\r\n // is reached when text is removed\r\n container.height > originalContainerHeight &&\r\n height < maxHeight) {\r\n const diff = Math.min(maxHeight - height, approxLineHeight);\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: container.height - diff });\r\n }\r\n // Start pushing text upward until a diff of 30px (padding)\r\n // is reached\r\n else {\r\n // vertically center align the text\r\n coordY = container.y + container.height / 2 - height / 2;\r\n }\r\n }\r\n const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\r\n const { textAlign } = updatedElement;\r\n editable.value = updatedElement.originalText;\r\n const lines = updatedElement.originalText.split(\"\\n\");\r\n const lineHeight = updatedElement.containerId\r\n ? approxLineHeight\r\n : updatedElement.height / lines.length;\r\n if (!container) {\r\n maxWidth =\r\n (appState.offsetLeft + appState.width - viewportX - 8) /\r\n appState.zoom.value -\r\n // margin-right of parent if any\r\n Number(getComputedStyle(excalidrawContainer === null || excalidrawContainer === void 0 ? void 0 : excalidrawContainer.parentNode).marginRight.slice(0, -2));\r\n }\r\n // Make sure text editor height doesn't go beyond viewport\r\n const editorMaxHeight = (appState.height -\r\n viewportY) /\r\n appState.zoom.value;\r\n const angle = container ? container.angle : updatedElement.angle;\r\n Object.assign(editable.style, {\r\n font: (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement),\r\n // must be defined *after* font ¯\\_(ツ)_/¯\r\n lineHeight: `${lineHeight}px`,\r\n width: `${width}px`,\r\n height: `${height}px`,\r\n left: `${viewportX}px`,\r\n top: `${viewportY}px`,\r\n transform: getTransform(width, height, angle, appState, maxWidth, editorMaxHeight),\r\n textAlign,\r\n color: updatedElement.strokeColor,\r\n opacity: updatedElement.opacity / 100,\r\n filter: \"var(--theme-filter)\",\r\n maxWidth: `${maxWidth}px`,\r\n maxHeight: `${editorMaxHeight}px`,\r\n });\r\n // For some reason updating font attribute doesn't set font family\r\n // hence updating font family explicitly for test environment\r\n if ((0,_utils__WEBPACK_IMPORTED_MODULE_1__.isTestEnv)()) {\r\n editable.style.fontFamily = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontFamilyString)(updatedElement);\r\n }\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(updatedElement, { x: coordX, y: coordY });\r\n }\r\n };\r\n const editable = document.createElement(\"textarea\");\r\n editable.dir = \"auto\";\r\n editable.tabIndex = 0;\r\n editable.dataset.type = \"wysiwyg\";\r\n // prevent line wrapping on Safari\r\n editable.wrap = \"off\";\r\n editable.classList.add(\"excalidraw-wysiwyg\");\r\n let whiteSpace = \"pre\";\r\n let wordBreak = \"normal\";\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isBoundToContainer)(element)) {\r\n whiteSpace = \"pre-wrap\";\r\n wordBreak = \"break-word\";\r\n }\r\n Object.assign(editable.style, {\r\n position: \"absolute\",\r\n display: \"inline-block\",\r\n minHeight: \"1em\",\r\n backfaceVisibility: \"hidden\",\r\n margin: 0,\r\n padding: 0,\r\n border: 0,\r\n outline: 0,\r\n resize: \"none\",\r\n background: \"transparent\",\r\n overflow: \"hidden\",\r\n // must be specified because in dark mode canvas creates a stacking context\r\n zIndex: \"var(--zIndex-wysiwyg)\",\r\n wordBreak,\r\n // prevent line wrapping (`whitespace: nowrap` doesn't work on FF)\r\n whiteSpace,\r\n overflowWrap: \"break-word\",\r\n });\r\n updateWysiwygStyle();\r\n if (onChange) {\r\n editable.oninput = () => {\r\n // using scrollHeight here since we need to calculate\r\n // number of lines so cannot use editable.style.height\r\n // as that gets updated below\r\n const lines = editable.scrollHeight / approxLineHeight;\r\n // auto increase height only when lines > 1 so its\r\n // measured correctly and vertically alignes for\r\n // first line as well as setting height to \"auto\"\r\n // doubles the height as soon as user starts typing\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isBoundToContainer)(element) && lines > 1) {\r\n let height = \"auto\";\r\n if (lines === 2) {\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(element);\r\n const actualLineCount = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.wrapText)(editable.value, (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element), container.width).split(\"\\n\").length;\r\n // This is browser behaviour when setting height to \"auto\"\r\n // It sets the height needed for 2 lines even if actual\r\n // line count is 1 as mentioned above as well\r\n // hence reducing the height by half if actual line count is 1\r\n // so single line aligns vertically when deleting\r\n if (actualLineCount === 1) {\r\n height = `${editable.scrollHeight / 2}px`;\r\n }\r\n }\r\n editable.style.height = height;\r\n editable.style.height = `${editable.scrollHeight}px`;\r\n }\r\n onChange(normalizeText(editable.value));\r\n };\r\n }\r\n editable.onkeydown = (event) => {\r\n event.stopPropagation();\r\n if (_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionDecreaseFontSize.keyTest(event)) {\r\n app.actionManager.executeAction(_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionDecreaseFontSize);\r\n }\r\n else if (_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionIncreaseFontSize.keyTest(event)) {\r\n app.actionManager.executeAction(_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionIncreaseFontSize);\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.ESCAPE) {\r\n event.preventDefault();\r\n submittedViaKeyboard = true;\r\n handleSubmit();\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.ENTER && event[_keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.CTRL_OR_CMD]) {\r\n event.preventDefault();\r\n if (event.isComposing || event.keyCode === 229) {\r\n return;\r\n }\r\n submittedViaKeyboard = true;\r\n handleSubmit();\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.TAB ||\r\n (event[_keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.CTRL_OR_CMD] &&\r\n (event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_LEFT ||\r\n event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_RIGHT))) {\r\n event.preventDefault();\r\n if (event.shiftKey || event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_LEFT) {\r\n outdent();\r\n }\r\n else {\r\n indent();\r\n }\r\n // We must send an input event to resize the element\r\n editable.dispatchEvent(new Event(\"input\"));\r\n }\r\n };\r\n const TAB_SIZE = 4;\r\n const TAB = \" \".repeat(TAB_SIZE);\r\n const RE_LEADING_TAB = new RegExp(`^ {1,${TAB_SIZE}}`);\r\n const indent = () => {\r\n const { selectionStart, selectionEnd } = editable;\r\n const linesStartIndices = getSelectedLinesStartIndices();\r\n let value = editable.value;\r\n linesStartIndices.forEach((startIndex) => {\r\n const startValue = value.slice(0, startIndex);\r\n const endValue = value.slice(startIndex);\r\n value = `${startValue}${TAB}${endValue}`;\r\n });\r\n editable.value = value;\r\n editable.selectionStart = selectionStart + TAB_SIZE;\r\n editable.selectionEnd = selectionEnd + TAB_SIZE * linesStartIndices.length;\r\n };\r\n const outdent = () => {\r\n const { selectionStart, selectionEnd } = editable;\r\n const linesStartIndices = getSelectedLinesStartIndices();\r\n const removedTabs = [];\r\n let value = editable.value;\r\n linesStartIndices.forEach((startIndex) => {\r\n const tabMatch = value\r\n .slice(startIndex, startIndex + TAB_SIZE)\r\n .match(RE_LEADING_TAB);\r\n if (tabMatch) {\r\n const startValue = value.slice(0, startIndex);\r\n const endValue = value.slice(startIndex + tabMatch[0].length);\r\n // Delete a tab from the line\r\n value = `${startValue}${endValue}`;\r\n removedTabs.push(startIndex);\r\n }\r\n });\r\n editable.value = value;\r\n if (removedTabs.length) {\r\n if (selectionStart > removedTabs[removedTabs.length - 1]) {\r\n editable.selectionStart = Math.max(selectionStart - TAB_SIZE, removedTabs[removedTabs.length - 1]);\r\n }\r\n else {\r\n // If the cursor is before the first tab removed, ex:\r\n // Line| #1\r\n // Line #2\r\n // Lin|e #3\r\n // we should reset the selectionStart to his initial value.\r\n editable.selectionStart = selectionStart;\r\n }\r\n editable.selectionEnd = Math.max(editable.selectionStart, selectionEnd - TAB_SIZE * removedTabs.length);\r\n }\r\n };\r\n /**\r\n * @returns indeces of start positions of selected lines, in reverse order\r\n */\r\n const getSelectedLinesStartIndices = () => {\r\n let { selectionStart, selectionEnd, value } = editable;\r\n // chars before selectionStart on the same line\r\n const startOffset = value.slice(0, selectionStart).match(/[^\\n]*$/)[0]\r\n .length;\r\n // put caret at the start of the line\r\n selectionStart = selectionStart - startOffset;\r\n const selected = value.slice(selectionStart, selectionEnd);\r\n return selected\r\n .split(\"\\n\")\r\n .reduce((startIndices, line, idx, lines) => startIndices.concat(idx\r\n ? // curr line index is prev line's start + prev line's length + \\n\r\n startIndices[idx - 1] + lines[idx - 1].length + 1\r\n : // first selected line\r\n selectionStart), [])\r\n .reverse();\r\n };\r\n const stopEvent = (event) => {\r\n event.preventDefault();\r\n event.stopPropagation();\r\n };\r\n // using a state variable instead of passing it to the handleSubmit callback\r\n // so that we don't need to create separate a callback for event handlers\r\n let submittedViaKeyboard = false;\r\n const handleSubmit = () => {\r\n var _a, _b;\r\n // cleanup must be run before onSubmit otherwise when app blurs the wysiwyg\r\n // it'd get stuck in an infinite loop of blur→onSubmit after we re-focus the\r\n // wysiwyg on update\r\n cleanup();\r\n const updateElement = (_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.id);\r\n if (!updateElement) {\r\n return;\r\n }\r\n let text = editable.value;\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(updateElement);\r\n if (container) {\r\n text = updateElement.text;\r\n if (editable.value) {\r\n const boundTextElementId = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getBoundTextElementId)(container);\r\n if (!boundTextElementId || boundTextElementId !== element.id) {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, {\r\n boundElements: (container.boundElements || []).concat({\r\n type: \"text\",\r\n id: element.id,\r\n }),\r\n });\r\n }\r\n }\r\n else {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, {\r\n boundElements: (_b = container.boundElements) === null || _b === void 0 ? void 0 : _b.filter((ele) => !(0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(ele)),\r\n });\r\n }\r\n }\r\n onSubmit({\r\n text,\r\n viaKeyboard: submittedViaKeyboard,\r\n originalText: editable.value,\r\n });\r\n };\r\n const cleanup = () => {\r\n if (isDestroyed) {\r\n return;\r\n }\r\n isDestroyed = true;\r\n // remove events to ensure they don't late-fire\r\n editable.onblur = null;\r\n editable.oninput = null;\r\n editable.onkeydown = null;\r\n if (observer) {\r\n observer.disconnect();\r\n }\r\n window.removeEventListener(\"resize\", updateWysiwygStyle);\r\n window.removeEventListener(\"wheel\", stopEvent, true);\r\n window.removeEventListener(\"pointerdown\", onPointerDown);\r\n window.removeEventListener(\"pointerup\", bindBlurEvent);\r\n window.removeEventListener(\"blur\", handleSubmit);\r\n unbindUpdate();\r\n editable.remove();\r\n };\r\n const bindBlurEvent = (event) => {\r\n window.removeEventListener(\"pointerup\", bindBlurEvent);\r\n // Deferred so that the pointerdown that initiates the wysiwyg doesn't\r\n // trigger the blur on ensuing pointerup.\r\n // Also to handle cases such as picking a color which would trigger a blur\r\n // in that same tick.\r\n const target = event === null || event === void 0 ? void 0 : event.target;\r\n const isTargetColorPicker = target instanceof HTMLInputElement &&\r\n target.closest(\".color-picker-input\") &&\r\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(target);\r\n setTimeout(() => {\r\n editable.onblur = handleSubmit;\r\n if (target && isTargetColorPicker) {\r\n target.onblur = () => {\r\n editable.focus();\r\n };\r\n }\r\n // case: clicking on the same property → no change → no update → no focus\r\n if (!isTargetColorPicker) {\r\n editable.focus();\r\n }\r\n });\r\n };\r\n // prevent blur when changing properties from the menu\r\n const onPointerDown = (event) => {\r\n const isTargetColorPicker = event.target instanceof HTMLInputElement &&\r\n event.target.closest(\".color-picker-input\") &&\r\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(event.target);\r\n if (((event.target instanceof HTMLElement ||\r\n event.target instanceof SVGElement) &&\r\n event.target.closest(`.${_constants__WEBPACK_IMPORTED_MODULE_4__.CLASSES.SHAPE_ACTIONS_MENU}`) &&\r\n !(0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(event.target)) ||\r\n isTargetColorPicker) {\r\n editable.onblur = null;\r\n window.addEventListener(\"pointerup\", bindBlurEvent);\r\n // handle edge-case where pointerup doesn't fire e.g. due to user\r\n // alt-tabbing away\r\n window.addEventListener(\"blur\", handleSubmit);\r\n }\r\n };\r\n // handle updates of textElement properties of editing element\r\n const unbindUpdate = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element).addCallback(() => {\r\n var _a;\r\n updateWysiwygStyle();\r\n const isColorPickerActive = !!((_a = document.activeElement) === null || _a === void 0 ? void 0 : _a.closest(\".color-picker-input\"));\r\n if (!isColorPickerActive) {\r\n editable.focus();\r\n }\r\n });\r\n // ---------------------------------------------------------------------------\r\n let isDestroyed = false;\r\n // select on init (focusing is done separately inside the bindBlurEvent()\r\n // because we need it to happen *after* the blur event from `pointerdown`)\r\n editable.select();\r\n bindBlurEvent();\r\n // reposition wysiwyg in case of canvas is resized. Using ResizeObserver\r\n // is preferred so we catch changes from host, where window may not resize.\r\n let observer = null;\r\n if (canvas && \"ResizeObserver\" in window) {\r\n observer = new window.ResizeObserver(() => {\r\n updateWysiwygStyle();\r\n });\r\n observer.observe(canvas);\r\n }\r\n else {\r\n window.addEventListener(\"resize\", updateWysiwygStyle);\r\n }\r\n window.addEventListener(\"pointerdown\", onPointerDown);\r\n window.addEventListener(\"wheel\", stopEvent, {\r\n passive: false,\r\n capture: true,\r\n });\r\n excalidrawContainer === null || excalidrawContainer === void 0 ? void 0 : excalidrawContainer.querySelector(\".excalidraw-textEditorContainer\").appendChild(editable);\r\n};\r\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../element/textWysiwyg.tsx.js","mappings":"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","sources":["webpack:///../../element/textWysiwyg.tsx?8f4d"],"sourcesContent":["import { CODES, KEYS } from \"../keys\";\nimport {\n  isWritableElement,\n  getFontString,\n  getFontFamilyString,\n  isTestEnv,\n} from \"../utils\";\nimport Scene from \"../scene/Scene\";\nimport { isBoundToContainer, isTextElement } from \"./typeChecks\";\nimport { CLASSES, BOUND_TEXT_PADDING } from \"../constants\";\nimport {\n  ExcalidrawElement,\n  ExcalidrawTextElement,\n  ExcalidrawLinearElement,\n} from \"./types\";\nimport { AppState } from \"../types\";\nimport { mutateElement } from \"./mutateElement\";\nimport {\n  getApproxLineHeight,\n  getBoundTextElementId,\n  getContainerElement,\n  wrapText,\n} from \"./textElement\";\nimport {\n  actionDecreaseFontSize,\n  actionIncreaseFontSize,\n} from \"../actions/actionProperties\";\nimport App from \"../components/App\";\n\nconst normalizeText = (text: string) => {\n  return (\n    text\n      // replace tabs with spaces so they render and measure correctly\n      .replace(/\\t/g, \"        \")\n      // normalize newlines\n      .replace(/\\r?\\n|\\r/g, \"\\n\")\n  );\n};\n\nconst getTransform = (\n  width: number,\n  height: number,\n  angle: number,\n  appState: AppState,\n  maxWidth: number,\n  maxHeight: number,\n) => {\n  const { zoom } = appState;\n  const degree = (180 * angle) / Math.PI;\n  let translateX = ((width) * (zoom.value - 1)) / 2;\n  let translateY = ((height) * (zoom.value - 1)) / 2;\n  if (width > maxWidth && zoom.value !== 1) {\n    translateX = ((maxWidth) * (zoom.value - 1)) / 2;\n  }\n  if (height > maxHeight && zoom.value !== 1) {\n    translateY = ((maxHeight) * (zoom.value - 1)) / 2;\n  }\n  return `translate(${translateX}px, ${translateY}px) scale(${zoom.value}) rotate(${degree}deg)`;\n};\n\nexport const textWysiwyg = ({\n  id,\n  appState,\n  onChange,\n  onSubmit,\n  getViewportCoords,\n  element,\n  canvas,\n  excalidrawContainer,\n  app,\n}: {\n  id: ExcalidrawElement[\"id\"];\n  appState: AppState;\n  onChange?: (text: string) => void;\n  onSubmit: (data: {\n    text: string;\n    viaKeyboard: boolean;\n    originalText: string;\n  }) => void;\n  getViewportCoords: (x: number, y: number) => [number, number];\n  element: ExcalidrawTextElement;\n  canvas: HTMLCanvasElement | null;\n  excalidrawContainer: HTMLDivElement | null;\n  app: App;\n}) => {\n  const textPropertiesUpdated = (\n    updatedElement: ExcalidrawTextElement,\n    editable: HTMLTextAreaElement,\n  ) => {\n    const currentFont = editable.style.fontFamily.replaceAll('\"', \"\");\n    if (\n      getFontFamilyString({ fontFamily: updatedElement.fontFamily }) !==\n      currentFont\n    ) {\n      return true;\n    }\n    if (`${updatedElement.fontSize}px` !== editable.style.fontSize) {\n      return true;\n    }\n    return false;\n  };\n  let originalContainerHeight: number;\n  let approxLineHeight = getApproxLineHeight(getFontString(element));\n\n  const updateWysiwygStyle = () => {\n    const updatedElement = Scene.getScene(element)?.getElement(id);\n    if (updatedElement && isTextElement(updatedElement)) {\n      let coordX = updatedElement.x;\n      let coordY = updatedElement.y;\n      const container = getContainerElement(updatedElement);\n      let maxWidth = updatedElement.width;\n\n      let maxHeight = updatedElement.height;\n      let width = updatedElement.width;\n      // Set to element height by default since thats\n      // what is going to be used for unbounded text\n      let height = updatedElement.height;\n      if (container && updatedElement.containerId) {\n        const propertiesUpdated = textPropertiesUpdated(\n          updatedElement,\n          editable,\n        );\n        // using editor.style.height to get the accurate height of text editor\n        const editorHeight = Number(editable.style.height.slice(0, -2));\n        if (editorHeight > 0) {\n          height = editorHeight;\n        }\n        if (propertiesUpdated) {\n          approxLineHeight = getApproxLineHeight(getFontString(updatedElement));\n\n          originalContainerHeight = container.height;\n\n          // update height of the editor after properties updated\n          height = updatedElement.height;\n        }\n        if (!originalContainerHeight) {\n          originalContainerHeight = container.height;\n        }\n        maxWidth = container.width - BOUND_TEXT_PADDING * 2;\n        maxHeight = container.height - BOUND_TEXT_PADDING * 2;\n        width = maxWidth;\n        // The coordinates of text box set a distance of\n        // 30px to preserve padding\n        coordX = container.x + BOUND_TEXT_PADDING;\n        // autogrow container height if text exceeds\n        if (height > maxHeight) {\n          const diff = Math.min(height - maxHeight, approxLineHeight);\n          mutateElement(container, { height: container.height + diff });\n          return;\n        } else if (\n          // autoshrink container height until original container height\n          // is reached when text is removed\n          container.height > originalContainerHeight &&\n          height < maxHeight\n        ) {\n          const diff = Math.min(maxHeight - height, approxLineHeight);\n          mutateElement(container, { height: container.height - diff });\n        }\n        // Start pushing text upward until a diff of 30px (padding)\n        // is reached\n        else {\n          // vertically center align the text\n          coordY = container.y + container.height / 2 - height / 2;\n        }\n      }\n      const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\n      const { textAlign } = updatedElement;\n      editable.value = updatedElement.originalText;\n      const lines = updatedElement.originalText.split(\"\\n\");\n      const lineHeight = updatedElement.containerId\n        ? approxLineHeight\n        : updatedElement.height / lines.length;\n      if (!container) {\n        maxWidth =\n          (appState.offsetLeft + appState.width - viewportX - 8) /\n            appState.zoom.value -\n          // margin-right of parent if any\n          Number(\n            getComputedStyle(\n              excalidrawContainer?.parentNode as Element,\n            ).marginRight.slice(0, -2),\n          );\n      }\n\n      // Make sure text editor height doesn't go beyond viewport\n      const editorMaxHeight =\n        (appState.height -\n          viewportY) /\n        appState.zoom.value;\n      const angle = container ? container.angle : updatedElement.angle;\n      Object.assign(editable.style, {\n        font: getFontString(updatedElement),\n        // must be defined *after* font ¯\\_(ツ)_/¯\n        lineHeight: `${lineHeight}px`,\n        width: `${width}px`,\n        height: `${height}px`,\n        left: `${viewportX}px`,\n        top: `${viewportY}px`,\n        transform: getTransform(\n          width,\n          height,\n          angle,\n          appState,\n          maxWidth,\n          editorMaxHeight,\n        ),\n        textAlign,\n        color: updatedElement.strokeColor,\n        opacity: updatedElement.opacity / 100,\n        filter: \"var(--theme-filter)\",\n        maxWidth: `${maxWidth}px`,\n        maxHeight: `${editorMaxHeight}px`,\n      });\n      // For some reason updating font attribute doesn't set font family\n      // hence updating font family explicitly for test environment\n      if (isTestEnv()) {\n        editable.style.fontFamily = getFontFamilyString(updatedElement);\n      }\n      mutateElement(updatedElement, { x: coordX, y: coordY });\n    }\n  };\n\n  const editable = document.createElement(\"textarea\");\n\n  editable.dir = \"auto\";\n  editable.tabIndex = 0;\n  editable.dataset.type = \"wysiwyg\";\n  // prevent line wrapping on Safari\n  editable.wrap = \"off\";\n  editable.classList.add(\"excalidraw-wysiwyg\");\n\n  let whiteSpace = \"pre\";\n  let wordBreak = \"normal\";\n\n  if (isBoundToContainer(element)) {\n    whiteSpace = \"pre-wrap\";\n    wordBreak = \"break-word\";\n  }\n  Object.assign(editable.style, {\n    position: \"absolute\",\n    display: \"inline-block\",\n    minHeight: \"1em\",\n    backfaceVisibility: \"hidden\",\n    margin: 0,\n    padding: 0,\n    border: 0,\n    outline: 0,\n    resize: \"none\",\n    background: \"transparent\",\n    overflow: \"hidden\",\n    // must be specified because in dark mode canvas creates a stacking context\n    zIndex: \"var(--zIndex-wysiwyg)\",\n    wordBreak,\n    // prevent line wrapping (`whitespace: nowrap` doesn't work on FF)\n    whiteSpace,\n    overflowWrap: \"break-word\",\n  });\n  updateWysiwygStyle();\n\n  if (onChange) {\n    editable.oninput = () => {\n      // using scrollHeight here since we need to calculate\n      // number of lines so cannot use editable.style.height\n      // as that gets updated below\n      const lines = editable.scrollHeight / approxLineHeight;\n      // auto increase height only when lines  > 1 so its\n      // measured correctly and vertically alignes for\n      // first line as well as setting height to \"auto\"\n      // doubles the height as soon as user starts typing\n      if (isBoundToContainer(element) && lines > 1) {\n        let height = \"auto\";\n\n        if (lines === 2) {\n          const container = getContainerElement(element);\n          const actualLineCount = wrapText(\n            editable.value,\n            getFontString(element),\n            container!.width,\n          ).split(\"\\n\").length;\n\n          // This is browser behaviour when setting height to \"auto\"\n          // It sets the height needed for 2 lines even if actual\n          // line count is 1 as mentioned above as well\n          // hence reducing the height by half if actual line count is 1\n          // so single line aligns vertically when deleting\n          if (actualLineCount === 1) {\n            height = `${editable.scrollHeight / 2}px`;\n          }\n        }\n        editable.style.height = height;\n        editable.style.height = `${editable.scrollHeight}px`;\n      }\n      onChange(normalizeText(editable.value));\n    };\n  }\n\n  editable.onkeydown = (event) => {\n    event.stopPropagation();\n\n    if (actionDecreaseFontSize.keyTest(event)) {\n      app.actionManager.executeAction(actionDecreaseFontSize);\n    } else if (actionIncreaseFontSize.keyTest(event)) {\n      app.actionManager.executeAction(actionIncreaseFontSize);\n    } else if (event.key === KEYS.ESCAPE) {\n      event.preventDefault();\n      submittedViaKeyboard = true;\n      handleSubmit();\n    } else if (event.key === KEYS.ENTER && event[KEYS.CTRL_OR_CMD]) {\n      event.preventDefault();\n      if (event.isComposing || event.keyCode === 229) {\n        return;\n      }\n      submittedViaKeyboard = true;\n      handleSubmit();\n    } else if (\n      event.key === KEYS.TAB ||\n      (event[KEYS.CTRL_OR_CMD] &&\n        (event.code === CODES.BRACKET_LEFT ||\n          event.code === CODES.BRACKET_RIGHT))\n    ) {\n      event.preventDefault();\n      if (event.shiftKey || event.code === CODES.BRACKET_LEFT) {\n        outdent();\n      } else {\n        indent();\n      }\n      // We must send an input event to resize the element\n      editable.dispatchEvent(new Event(\"input\"));\n    }\n  };\n\n  const TAB_SIZE = 4;\n  const TAB = \" \".repeat(TAB_SIZE);\n  const RE_LEADING_TAB = new RegExp(`^ {1,${TAB_SIZE}}`);\n  const indent = () => {\n    const { selectionStart, selectionEnd } = editable;\n    const linesStartIndices = getSelectedLinesStartIndices();\n\n    let value = editable.value;\n    linesStartIndices.forEach((startIndex: number) => {\n      const startValue = value.slice(0, startIndex);\n      const endValue = value.slice(startIndex);\n\n      value = `${startValue}${TAB}${endValue}`;\n    });\n\n    editable.value = value;\n\n    editable.selectionStart = selectionStart + TAB_SIZE;\n    editable.selectionEnd = selectionEnd + TAB_SIZE * linesStartIndices.length;\n  };\n\n  const outdent = () => {\n    const { selectionStart, selectionEnd } = editable;\n    const linesStartIndices = getSelectedLinesStartIndices();\n    const removedTabs: number[] = [];\n\n    let value = editable.value;\n    linesStartIndices.forEach((startIndex) => {\n      const tabMatch = value\n        .slice(startIndex, startIndex + TAB_SIZE)\n        .match(RE_LEADING_TAB);\n\n      if (tabMatch) {\n        const startValue = value.slice(0, startIndex);\n        const endValue = value.slice(startIndex + tabMatch[0].length);\n\n        // Delete a tab from the line\n        value = `${startValue}${endValue}`;\n        removedTabs.push(startIndex);\n      }\n    });\n\n    editable.value = value;\n\n    if (removedTabs.length) {\n      if (selectionStart > removedTabs[removedTabs.length - 1]) {\n        editable.selectionStart = Math.max(\n          selectionStart - TAB_SIZE,\n          removedTabs[removedTabs.length - 1],\n        );\n      } else {\n        // If the cursor is before the first tab removed, ex:\n        // Line| #1\n        //     Line #2\n        // Lin|e #3\n        // we should reset the selectionStart to his initial value.\n        editable.selectionStart = selectionStart;\n      }\n      editable.selectionEnd = Math.max(\n        editable.selectionStart,\n        selectionEnd - TAB_SIZE * removedTabs.length,\n      );\n    }\n  };\n\n  /**\n   * @returns indeces of start positions of selected lines, in reverse order\n   */\n  const getSelectedLinesStartIndices = () => {\n    let { selectionStart, selectionEnd, value } = editable;\n\n    // chars before selectionStart on the same line\n    const startOffset = value.slice(0, selectionStart).match(/[^\\n]*$/)![0]\n      .length;\n    // put caret at the start of the line\n    selectionStart = selectionStart - startOffset;\n\n    const selected = value.slice(selectionStart, selectionEnd);\n\n    return selected\n      .split(\"\\n\")\n      .reduce(\n        (startIndices, line, idx, lines) =>\n          startIndices.concat(\n            idx\n              ? // curr line index is prev line's start + prev line's length + \\n\n                startIndices[idx - 1] + lines[idx - 1].length + 1\n              : // first selected line\n                selectionStart,\n          ),\n        [] as number[],\n      )\n      .reverse();\n  };\n\n  const stopEvent = (event: Event) => {\n    event.preventDefault();\n    event.stopPropagation();\n  };\n\n  // using a state variable instead of passing it to the handleSubmit callback\n  // so that we don't need to create separate a callback for event handlers\n  let submittedViaKeyboard = false;\n  const handleSubmit = () => {\n    // cleanup must be run before onSubmit otherwise when app blurs the wysiwyg\n    // it'd get stuck in an infinite loop of blur→onSubmit after we re-focus the\n    // wysiwyg on update\n    cleanup();\n    const updateElement = Scene.getScene(element)?.getElement(\n      element.id,\n    ) as ExcalidrawTextElement;\n    if (!updateElement) {\n      return;\n    }\n    let text = editable.value;\n    const container = getContainerElement(updateElement);\n\n    if (container) {\n      text = updateElement.text;\n      if (editable.value) {\n        const boundTextElementId = getBoundTextElementId(container);\n        if (!boundTextElementId || boundTextElementId !== element.id) {\n          mutateElement(container, {\n            boundElements: (container.boundElements || []).concat({\n              type: \"text\",\n              id: element.id,\n            }),\n          });\n        }\n      } else {\n        mutateElement(container, {\n          boundElements: container.boundElements?.filter(\n            (ele) =>\n              !isTextElement(\n                ele as ExcalidrawTextElement | ExcalidrawLinearElement,\n              ),\n          ),\n        });\n      }\n    }\n\n    onSubmit({\n      text,\n      viaKeyboard: submittedViaKeyboard,\n      originalText: editable.value,\n    });\n  };\n\n  const cleanup = () => {\n    if (isDestroyed) {\n      return;\n    }\n    isDestroyed = true;\n    // remove events to ensure they don't late-fire\n    editable.onblur = null;\n    editable.oninput = null;\n    editable.onkeydown = null;\n\n    if (observer) {\n      observer.disconnect();\n    }\n\n    window.removeEventListener(\"resize\", updateWysiwygStyle);\n    window.removeEventListener(\"wheel\", stopEvent, true);\n    window.removeEventListener(\"pointerdown\", onPointerDown);\n    window.removeEventListener(\"pointerup\", bindBlurEvent);\n    window.removeEventListener(\"blur\", handleSubmit);\n\n    unbindUpdate();\n\n    editable.remove();\n  };\n\n  const bindBlurEvent = (event?: MouseEvent) => {\n    window.removeEventListener(\"pointerup\", bindBlurEvent);\n    // Deferred so that the pointerdown that initiates the wysiwyg doesn't\n    // trigger the blur on ensuing pointerup.\n    // Also to handle cases such as picking a color which would trigger a blur\n    // in that same tick.\n    const target = event?.target;\n\n    const isTargetColorPicker =\n      target instanceof HTMLInputElement &&\n      target.closest(\".color-picker-input\") &&\n      isWritableElement(target);\n\n    setTimeout(() => {\n      editable.onblur = handleSubmit;\n      if (target && isTargetColorPicker) {\n        target.onblur = () => {\n          editable.focus();\n        };\n      }\n      // case: clicking on the same property → no change → no update → no focus\n      if (!isTargetColorPicker) {\n        editable.focus();\n      }\n    });\n  };\n\n  // prevent blur when changing properties from the menu\n  const onPointerDown = (event: MouseEvent) => {\n    const isTargetColorPicker =\n      event.target instanceof HTMLInputElement &&\n      event.target.closest(\".color-picker-input\") &&\n      isWritableElement(event.target);\n    if (\n      ((event.target instanceof HTMLElement ||\n        event.target instanceof SVGElement) &&\n        event.target.closest(`.${CLASSES.SHAPE_ACTIONS_MENU}`) &&\n        !isWritableElement(event.target)) ||\n      isTargetColorPicker\n    ) {\n      editable.onblur = null;\n      window.addEventListener(\"pointerup\", bindBlurEvent);\n      // handle edge-case where pointerup doesn't fire e.g. due to user\n      // alt-tabbing away\n      window.addEventListener(\"blur\", handleSubmit);\n    }\n  };\n\n  // handle updates of textElement properties of editing element\n  const unbindUpdate = Scene.getScene(element)!.addCallback(() => {\n    updateWysiwygStyle();\n    const isColorPickerActive = !!document.activeElement?.closest(\n      \".color-picker-input\",\n    );\n    if (!isColorPickerActive) {\n      editable.focus();\n    }\n  });\n\n  // ---------------------------------------------------------------------------\n\n  let isDestroyed = false;\n\n  // select on init (focusing is done separately inside the bindBlurEvent()\n  // because we need it to happen *after* the blur event from `pointerdown`)\n  editable.select();\n  bindBlurEvent();\n\n  // reposition wysiwyg in case of canvas is resized. Using ResizeObserver\n  // is preferred so we catch changes from host, where window may not resize.\n  let observer: ResizeObserver | null = null;\n  if (canvas && \"ResizeObserver\" in window) {\n    observer = new window.ResizeObserver(() => {\n      updateWysiwygStyle();\n    });\n    observer.observe(canvas);\n  } else {\n    window.addEventListener(\"resize\", updateWysiwygStyle);\n  }\n\n  window.addEventListener(\"pointerdown\", onPointerDown);\n  window.addEventListener(\"wheel\", stopEvent, {\n    passive: false,\n    capture: true,\n  });\n  excalidrawContainer\n    ?.querySelector(\".excalidraw-textEditorContainer\")!\n    .appendChild(editable);\n};\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../element/textWysiwyg.tsx\n");
2553
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"textWysiwyg\": () => (/* binding */ textWysiwyg)\n/* harmony export */ });\n/* harmony import */ var _keys__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../keys */ \"../../keys.ts\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../utils */ \"../../utils.ts\");\n/* harmony import */ var _scene_Scene__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../scene/Scene */ \"../../scene/Scene.ts\");\n/* harmony import */ var _typeChecks__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./typeChecks */ \"../../element/typeChecks.ts\");\n/* harmony import */ var _constants__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../constants */ \"../../constants.ts\");\n/* harmony import */ var _mutateElement__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./mutateElement */ \"../../element/mutateElement.ts\");\n/* harmony import */ var _textElement__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./textElement */ \"../../element/textElement.ts\");\n/* harmony import */ var _actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../actions/actionProperties */ \"../../actions/actionProperties.tsx\");\n\r\n\r\n\r\n\r\n\r\n\r\n\r\n\r\nconst normalizeText = (text) => {\r\n return (text\r\n // replace tabs with spaces so they render and measure correctly\r\n .replace(/\\t/g, \" \")\r\n // normalize newlines\r\n .replace(/\\r?\\n|\\r/g, \"\\n\"));\r\n};\r\nconst getTransform = (width, height, angle, appState, maxWidth, maxHeight) => {\r\n const { zoom } = appState;\r\n const degree = (180 * angle) / Math.PI;\r\n let translateX = ((width) * (zoom.value - 1)) / 2;\r\n let translateY = ((height) * (zoom.value - 1)) / 2;\r\n if (width > maxWidth && zoom.value !== 1) {\r\n translateX = ((maxWidth) * (zoom.value - 1)) / 2;\r\n }\r\n if (height > maxHeight && zoom.value !== 1) {\r\n translateY = ((maxHeight) * (zoom.value - 1)) / 2;\r\n }\r\n return `translate(${translateX}px, ${translateY}px) scale(${zoom.value}) rotate(${degree}deg)`;\r\n};\r\nconst textWysiwyg = ({ id, appState, onChange, onSubmit, getViewportCoords, element, canvas, excalidrawContainer, app, }) => {\r\n const textPropertiesUpdated = (updatedElement, editable) => {\r\n const currentFont = editable.style.fontFamily.replaceAll('\"', \"\");\r\n if ((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontFamilyString)({ fontFamily: updatedElement.fontFamily }) !==\r\n currentFont) {\r\n return true;\r\n }\r\n if (`${updatedElement.fontSize}px` !== editable.style.fontSize) {\r\n return true;\r\n }\r\n return false;\r\n };\r\n let originalContainerHeight;\r\n let approxLineHeight = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element));\r\n const updateWysiwygStyle = () => {\r\n var _a;\r\n const updatedElement = (_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(id);\r\n if (updatedElement && (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updatedElement)) {\r\n let coordX = updatedElement.x;\r\n let coordY = updatedElement.y;\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(updatedElement);\r\n let maxWidth = updatedElement.width;\r\n let maxHeight = updatedElement.height;\r\n let width = updatedElement.width;\r\n // Set to element height by default since thats\r\n // what is going to be used for unbounded text\r\n let height = updatedElement.height;\r\n if (container && updatedElement.containerId) {\r\n const propertiesUpdated = textPropertiesUpdated(updatedElement, editable);\r\n // using editor.style.height to get the accurate height of text editor\r\n const editorHeight = Number(editable.style.height.slice(0, -2));\r\n if (editorHeight > 0) {\r\n height = editorHeight;\r\n }\r\n if (propertiesUpdated) {\r\n approxLineHeight = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement));\r\n originalContainerHeight = container.height;\r\n // update height of the editor after properties updated\r\n height = updatedElement.height;\r\n }\r\n if (!originalContainerHeight) {\r\n originalContainerHeight = container.height;\r\n }\r\n maxWidth = container.width - _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING * 2;\r\n maxHeight = container.height - _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING * 2;\r\n width = maxWidth;\r\n // The coordinates of text box set a distance of\r\n // 30px to preserve padding\r\n coordX = container.x + _constants__WEBPACK_IMPORTED_MODULE_4__.BOUND_TEXT_PADDING;\r\n // autogrow container height if text exceeds\r\n if (height > maxHeight) {\r\n const diff = Math.min(height - maxHeight, approxLineHeight);\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: container.height + diff });\r\n return;\r\n }\r\n else if (\r\n // autoshrink container height until original container height\r\n // is reached when text is removed\r\n container.height > originalContainerHeight &&\r\n height < maxHeight) {\r\n const diff = Math.min(maxHeight - height, approxLineHeight);\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: container.height - diff });\r\n }\r\n // Start pushing text upward until a diff of 30px (padding)\r\n // is reached\r\n else {\r\n // vertically center align the text\r\n coordY = container.y + container.height / 2 - height / 2;\r\n }\r\n }\r\n const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\r\n const { textAlign } = updatedElement;\r\n editable.value = updatedElement.originalText;\r\n const lines = updatedElement.originalText.split(\"\\n\");\r\n const lineHeight = updatedElement.containerId\r\n ? approxLineHeight\r\n : updatedElement.height / lines.length;\r\n if (!container) {\r\n maxWidth =\r\n (appState.width - viewportX - 8) /\r\n appState.zoom.value -\r\n // margin-right of parent if any\r\n Number(getComputedStyle(excalidrawContainer === null || excalidrawContainer === void 0 ? void 0 : excalidrawContainer.parentNode).marginRight.slice(0, -2));\r\n }\r\n // Make sure text editor height doesn't go beyond viewport\r\n const editorMaxHeight = (appState.height -\r\n viewportY) /\r\n appState.zoom.value;\r\n const angle = container ? container.angle : updatedElement.angle;\r\n Object.assign(editable.style, {\r\n font: (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement),\r\n // must be defined *after* font ¯\\_(ツ)_/¯\r\n lineHeight: `${lineHeight}px`,\r\n width: `${width}px`,\r\n height: `${height}px`,\r\n left: `${viewportX}px`,\r\n top: `${viewportY}px`,\r\n transform: getTransform(width, height, angle, appState, maxWidth, editorMaxHeight),\r\n textAlign,\r\n color: updatedElement.strokeColor,\r\n opacity: updatedElement.opacity / 100,\r\n filter: \"var(--theme-filter)\",\r\n maxWidth: `${maxWidth}px`,\r\n maxHeight: `${editorMaxHeight}px`,\r\n });\r\n // For some reason updating font attribute doesn't set font family\r\n // hence updating font family explicitly for test environment\r\n if ((0,_utils__WEBPACK_IMPORTED_MODULE_1__.isTestEnv)()) {\r\n editable.style.fontFamily = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontFamilyString)(updatedElement);\r\n }\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(updatedElement, { x: coordX, y: coordY });\r\n }\r\n };\r\n const editable = document.createElement(\"textarea\");\r\n editable.dir = \"auto\";\r\n editable.tabIndex = 0;\r\n editable.dataset.type = \"wysiwyg\";\r\n // prevent line wrapping on Safari\r\n editable.wrap = \"off\";\r\n editable.classList.add(\"excalidraw-wysiwyg\");\r\n let whiteSpace = \"pre\";\r\n let wordBreak = \"normal\";\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isBoundToContainer)(element)) {\r\n whiteSpace = \"pre-wrap\";\r\n wordBreak = \"break-word\";\r\n }\r\n Object.assign(editable.style, {\r\n position: \"absolute\",\r\n display: \"inline-block\",\r\n minHeight: \"1em\",\r\n backfaceVisibility: \"hidden\",\r\n margin: 0,\r\n padding: 0,\r\n border: 0,\r\n outline: 0,\r\n resize: \"none\",\r\n background: \"transparent\",\r\n overflow: \"hidden\",\r\n // must be specified because in dark mode canvas creates a stacking context\r\n zIndex: \"var(--zIndex-wysiwyg)\",\r\n wordBreak,\r\n // prevent line wrapping (`whitespace: nowrap` doesn't work on FF)\r\n whiteSpace,\r\n overflowWrap: \"break-word\",\r\n });\r\n updateWysiwygStyle();\r\n if (onChange) {\r\n editable.oninput = () => {\r\n // using scrollHeight here since we need to calculate\r\n // number of lines so cannot use editable.style.height\r\n // as that gets updated below\r\n const lines = editable.scrollHeight / approxLineHeight;\r\n // auto increase height only when lines > 1 so its\r\n // measured correctly and vertically alignes for\r\n // first line as well as setting height to \"auto\"\r\n // doubles the height as soon as user starts typing\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isBoundToContainer)(element) && lines > 1) {\r\n let height = \"auto\";\r\n if (lines === 2) {\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(element);\r\n const actualLineCount = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.wrapText)(editable.value, (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element), container.width).split(\"\\n\").length;\r\n // This is browser behaviour when setting height to \"auto\"\r\n // It sets the height needed for 2 lines even if actual\r\n // line count is 1 as mentioned above as well\r\n // hence reducing the height by half if actual line count is 1\r\n // so single line aligns vertically when deleting\r\n if (actualLineCount === 1) {\r\n height = `${editable.scrollHeight / 2}px`;\r\n }\r\n }\r\n editable.style.height = height;\r\n editable.style.height = `${editable.scrollHeight}px`;\r\n }\r\n onChange(normalizeText(editable.value));\r\n };\r\n }\r\n editable.onkeydown = (event) => {\r\n event.stopPropagation();\r\n if (_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionDecreaseFontSize.keyTest(event)) {\r\n app.actionManager.executeAction(_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionDecreaseFontSize);\r\n }\r\n else if (_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionIncreaseFontSize.keyTest(event)) {\r\n app.actionManager.executeAction(_actions_actionProperties__WEBPACK_IMPORTED_MODULE_7__.actionIncreaseFontSize);\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.ESCAPE) {\r\n event.preventDefault();\r\n submittedViaKeyboard = true;\r\n handleSubmit();\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.ENTER && event[_keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.CTRL_OR_CMD]) {\r\n event.preventDefault();\r\n if (event.isComposing || event.keyCode === 229) {\r\n return;\r\n }\r\n submittedViaKeyboard = true;\r\n handleSubmit();\r\n }\r\n else if (event.key === _keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.TAB ||\r\n (event[_keys__WEBPACK_IMPORTED_MODULE_0__.KEYS.CTRL_OR_CMD] &&\r\n (event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_LEFT ||\r\n event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_RIGHT))) {\r\n event.preventDefault();\r\n if (event.shiftKey || event.code === _keys__WEBPACK_IMPORTED_MODULE_0__.CODES.BRACKET_LEFT) {\r\n outdent();\r\n }\r\n else {\r\n indent();\r\n }\r\n // We must send an input event to resize the element\r\n editable.dispatchEvent(new Event(\"input\"));\r\n }\r\n };\r\n const TAB_SIZE = 4;\r\n const TAB = \" \".repeat(TAB_SIZE);\r\n const RE_LEADING_TAB = new RegExp(`^ {1,${TAB_SIZE}}`);\r\n const indent = () => {\r\n const { selectionStart, selectionEnd } = editable;\r\n const linesStartIndices = getSelectedLinesStartIndices();\r\n let value = editable.value;\r\n linesStartIndices.forEach((startIndex) => {\r\n const startValue = value.slice(0, startIndex);\r\n const endValue = value.slice(startIndex);\r\n value = `${startValue}${TAB}${endValue}`;\r\n });\r\n editable.value = value;\r\n editable.selectionStart = selectionStart + TAB_SIZE;\r\n editable.selectionEnd = selectionEnd + TAB_SIZE * linesStartIndices.length;\r\n };\r\n const outdent = () => {\r\n const { selectionStart, selectionEnd } = editable;\r\n const linesStartIndices = getSelectedLinesStartIndices();\r\n const removedTabs = [];\r\n let value = editable.value;\r\n linesStartIndices.forEach((startIndex) => {\r\n const tabMatch = value\r\n .slice(startIndex, startIndex + TAB_SIZE)\r\n .match(RE_LEADING_TAB);\r\n if (tabMatch) {\r\n const startValue = value.slice(0, startIndex);\r\n const endValue = value.slice(startIndex + tabMatch[0].length);\r\n // Delete a tab from the line\r\n value = `${startValue}${endValue}`;\r\n removedTabs.push(startIndex);\r\n }\r\n });\r\n editable.value = value;\r\n if (removedTabs.length) {\r\n if (selectionStart > removedTabs[removedTabs.length - 1]) {\r\n editable.selectionStart = Math.max(selectionStart - TAB_SIZE, removedTabs[removedTabs.length - 1]);\r\n }\r\n else {\r\n // If the cursor is before the first tab removed, ex:\r\n // Line| #1\r\n // Line #2\r\n // Lin|e #3\r\n // we should reset the selectionStart to his initial value.\r\n editable.selectionStart = selectionStart;\r\n }\r\n editable.selectionEnd = Math.max(editable.selectionStart, selectionEnd - TAB_SIZE * removedTabs.length);\r\n }\r\n };\r\n /**\r\n * @returns indeces of start positions of selected lines, in reverse order\r\n */\r\n const getSelectedLinesStartIndices = () => {\r\n let { selectionStart, selectionEnd, value } = editable;\r\n // chars before selectionStart on the same line\r\n const startOffset = value.slice(0, selectionStart).match(/[^\\n]*$/)[0]\r\n .length;\r\n // put caret at the start of the line\r\n selectionStart = selectionStart - startOffset;\r\n const selected = value.slice(selectionStart, selectionEnd);\r\n return selected\r\n .split(\"\\n\")\r\n .reduce((startIndices, line, idx, lines) => startIndices.concat(idx\r\n ? // curr line index is prev line's start + prev line's length + \\n\r\n startIndices[idx - 1] + lines[idx - 1].length + 1\r\n : // first selected line\r\n selectionStart), [])\r\n .reverse();\r\n };\r\n const stopEvent = (event) => {\r\n event.preventDefault();\r\n event.stopPropagation();\r\n };\r\n // using a state variable instead of passing it to the handleSubmit callback\r\n // so that we don't need to create separate a callback for event handlers\r\n let submittedViaKeyboard = false;\r\n const handleSubmit = () => {\r\n var _a, _b;\r\n // cleanup must be run before onSubmit otherwise when app blurs the wysiwyg\r\n // it'd get stuck in an infinite loop of blur→onSubmit after we re-focus the\r\n // wysiwyg on update\r\n cleanup();\r\n const updateElement = (_a = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element)) === null || _a === void 0 ? void 0 : _a.getElement(element.id);\r\n if (!updateElement) {\r\n return;\r\n }\r\n let text = editable.value;\r\n const container = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getContainerElement)(updateElement);\r\n if (container) {\r\n text = updateElement.text;\r\n if (editable.value) {\r\n const boundTextElementId = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getBoundTextElementId)(container);\r\n if (!boundTextElementId || boundTextElementId !== element.id) {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, {\r\n boundElements: (container.boundElements || []).concat({\r\n type: \"text\",\r\n id: element.id,\r\n }),\r\n });\r\n }\r\n }\r\n else {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, {\r\n boundElements: (_b = container.boundElements) === null || _b === void 0 ? void 0 : _b.filter((ele) => !(0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(ele)),\r\n });\r\n }\r\n }\r\n onSubmit({\r\n text,\r\n viaKeyboard: submittedViaKeyboard,\r\n originalText: editable.value,\r\n });\r\n };\r\n const cleanup = () => {\r\n if (isDestroyed) {\r\n return;\r\n }\r\n isDestroyed = true;\r\n // remove events to ensure they don't late-fire\r\n editable.onblur = null;\r\n editable.oninput = null;\r\n editable.onkeydown = null;\r\n if (observer) {\r\n observer.disconnect();\r\n }\r\n window.removeEventListener(\"resize\", updateWysiwygStyle);\r\n window.removeEventListener(\"wheel\", stopEvent, true);\r\n window.removeEventListener(\"pointerdown\", onPointerDown);\r\n window.removeEventListener(\"pointerup\", bindBlurEvent);\r\n window.removeEventListener(\"blur\", handleSubmit);\r\n unbindUpdate();\r\n editable.remove();\r\n };\r\n const bindBlurEvent = (event) => {\r\n window.removeEventListener(\"pointerup\", bindBlurEvent);\r\n // Deferred so that the pointerdown that initiates the wysiwyg doesn't\r\n // trigger the blur on ensuing pointerup.\r\n // Also to handle cases such as picking a color which would trigger a blur\r\n // in that same tick.\r\n const target = event === null || event === void 0 ? void 0 : event.target;\r\n const isTargetColorPicker = target instanceof HTMLInputElement &&\r\n target.closest(\".color-picker-input\") &&\r\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(target);\r\n setTimeout(() => {\r\n editable.onblur = handleSubmit;\r\n if (target && isTargetColorPicker) {\r\n target.onblur = () => {\r\n editable.focus();\r\n };\r\n }\r\n // case: clicking on the same property → no change → no update → no focus\r\n if (!isTargetColorPicker) {\r\n editable.focus();\r\n }\r\n });\r\n };\r\n // prevent blur when changing properties from the menu\r\n const onPointerDown = (event) => {\r\n const isTargetColorPicker = event.target instanceof HTMLInputElement &&\r\n event.target.closest(\".color-picker-input\") &&\r\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(event.target);\r\n if (((event.target instanceof HTMLElement ||\r\n event.target instanceof SVGElement) &&\r\n event.target.closest(`.${_constants__WEBPACK_IMPORTED_MODULE_4__.CLASSES.SHAPE_ACTIONS_MENU}`) &&\r\n !(0,_utils__WEBPACK_IMPORTED_MODULE_1__.isWritableElement)(event.target)) ||\r\n isTargetColorPicker) {\r\n editable.onblur = null;\r\n window.addEventListener(\"pointerup\", bindBlurEvent);\r\n // handle edge-case where pointerup doesn't fire e.g. due to user\r\n // alt-tabbing away\r\n window.addEventListener(\"blur\", handleSubmit);\r\n }\r\n };\r\n // handle updates of textElement properties of editing element\r\n const unbindUpdate = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(element).addCallback(() => {\r\n var _a;\r\n updateWysiwygStyle();\r\n const isColorPickerActive = !!((_a = document.activeElement) === null || _a === void 0 ? void 0 : _a.closest(\".color-picker-input\"));\r\n if (!isColorPickerActive) {\r\n editable.focus();\r\n }\r\n });\r\n // ---------------------------------------------------------------------------\r\n let isDestroyed = false;\r\n // select on init (focusing is done separately inside the bindBlurEvent()\r\n // because we need it to happen *after* the blur event from `pointerdown`)\r\n editable.select();\r\n bindBlurEvent();\r\n // reposition wysiwyg in case of canvas is resized. Using ResizeObserver\r\n // is preferred so we catch changes from host, where window may not resize.\r\n let observer = null;\r\n if (canvas && \"ResizeObserver\" in window) {\r\n observer = new window.ResizeObserver(() => {\r\n updateWysiwygStyle();\r\n });\r\n observer.observe(canvas);\r\n }\r\n else {\r\n window.addEventListener(\"resize\", updateWysiwygStyle);\r\n }\r\n window.addEventListener(\"pointerdown\", onPointerDown);\r\n window.addEventListener(\"wheel\", stopEvent, {\r\n passive: false,\r\n capture: true,\r\n });\r\n excalidrawContainer === null || excalidrawContainer === void 0 ? void 0 : excalidrawContainer.querySelector(\".excalidraw-textEditorContainer\").appendChild(editable);\r\n};\r\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"../../element/textWysiwyg.tsx.js","mappings":"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","sources":["webpack:///../../element/textWysiwyg.tsx?8f4d"],"sourcesContent":["import { CODES, KEYS } from \"../keys\";\nimport {\n  isWritableElement,\n  getFontString,\n  getFontFamilyString,\n  isTestEnv,\n} from \"../utils\";\nimport Scene from \"../scene/Scene\";\nimport { isBoundToContainer, isTextElement } from \"./typeChecks\";\nimport { CLASSES, BOUND_TEXT_PADDING } from \"../constants\";\nimport {\n  ExcalidrawElement,\n  ExcalidrawTextElement,\n  ExcalidrawLinearElement,\n} from \"./types\";\nimport { AppState } from \"../types\";\nimport { mutateElement } from \"./mutateElement\";\nimport {\n  getApproxLineHeight,\n  getBoundTextElementId,\n  getContainerElement,\n  wrapText,\n} from \"./textElement\";\nimport {\n  actionDecreaseFontSize,\n  actionIncreaseFontSize,\n} from \"../actions/actionProperties\";\nimport App from \"../components/App\";\n\nconst normalizeText = (text: string) => {\n  return (\n    text\n      // replace tabs with spaces so they render and measure correctly\n      .replace(/\\t/g, \"        \")\n      // normalize newlines\n      .replace(/\\r?\\n|\\r/g, \"\\n\")\n  );\n};\n\nconst getTransform = (\n  width: number,\n  height: number,\n  angle: number,\n  appState: AppState,\n  maxWidth: number,\n  maxHeight: number,\n) => {\n  const { zoom } = appState;\n  const degree = (180 * angle) / Math.PI;\n  let translateX = ((width) * (zoom.value - 1)) / 2;\n  let translateY = ((height) * (zoom.value - 1)) / 2;\n  if (width > maxWidth && zoom.value !== 1) {\n    translateX = ((maxWidth) * (zoom.value - 1)) / 2;\n  }\n  if (height > maxHeight && zoom.value !== 1) {\n    translateY = ((maxHeight) * (zoom.value - 1)) / 2;\n  }\n  return `translate(${translateX}px, ${translateY}px) scale(${zoom.value}) rotate(${degree}deg)`;\n};\n\nexport const textWysiwyg = ({\n  id,\n  appState,\n  onChange,\n  onSubmit,\n  getViewportCoords,\n  element,\n  canvas,\n  excalidrawContainer,\n  app,\n}: {\n  id: ExcalidrawElement[\"id\"];\n  appState: AppState;\n  onChange?: (text: string) => void;\n  onSubmit: (data: {\n    text: string;\n    viaKeyboard: boolean;\n    originalText: string;\n  }) => void;\n  getViewportCoords: (x: number, y: number) => [number, number];\n  element: ExcalidrawTextElement;\n  canvas: HTMLCanvasElement | null;\n  excalidrawContainer: HTMLDivElement | null;\n  app: App;\n}) => {\n  const textPropertiesUpdated = (\n    updatedElement: ExcalidrawTextElement,\n    editable: HTMLTextAreaElement,\n  ) => {\n    const currentFont = editable.style.fontFamily.replaceAll('\"', \"\");\n    if (\n      getFontFamilyString({ fontFamily: updatedElement.fontFamily }) !==\n      currentFont\n    ) {\n      return true;\n    }\n    if (`${updatedElement.fontSize}px` !== editable.style.fontSize) {\n      return true;\n    }\n    return false;\n  };\n  let originalContainerHeight: number;\n  let approxLineHeight = getApproxLineHeight(getFontString(element));\n\n  const updateWysiwygStyle = () => {\n    const updatedElement = Scene.getScene(element)?.getElement(id);\n    if (updatedElement && isTextElement(updatedElement)) {\n      let coordX = updatedElement.x;\n      let coordY = updatedElement.y;\n      const container = getContainerElement(updatedElement);\n      let maxWidth = updatedElement.width;\n\n      let maxHeight = updatedElement.height;\n      let width = updatedElement.width;\n      // Set to element height by default since thats\n      // what is going to be used for unbounded text\n      let height = updatedElement.height;\n      if (container && updatedElement.containerId) {\n        const propertiesUpdated = textPropertiesUpdated(\n          updatedElement,\n          editable,\n        );\n        // using editor.style.height to get the accurate height of text editor\n        const editorHeight = Number(editable.style.height.slice(0, -2));\n        if (editorHeight > 0) {\n          height = editorHeight;\n        }\n        if (propertiesUpdated) {\n          approxLineHeight = getApproxLineHeight(getFontString(updatedElement));\n\n          originalContainerHeight = container.height;\n\n          // update height of the editor after properties updated\n          height = updatedElement.height;\n        }\n        if (!originalContainerHeight) {\n          originalContainerHeight = container.height;\n        }\n        maxWidth = container.width - BOUND_TEXT_PADDING * 2;\n        maxHeight = container.height - BOUND_TEXT_PADDING * 2;\n        width = maxWidth;\n        // The coordinates of text box set a distance of\n        // 30px to preserve padding\n        coordX = container.x + BOUND_TEXT_PADDING;\n        // autogrow container height if text exceeds\n        if (height > maxHeight) {\n          const diff = Math.min(height - maxHeight, approxLineHeight);\n          mutateElement(container, { height: container.height + diff });\n          return;\n        } else if (\n          // autoshrink container height until original container height\n          // is reached when text is removed\n          container.height > originalContainerHeight &&\n          height < maxHeight\n        ) {\n          const diff = Math.min(maxHeight - height, approxLineHeight);\n          mutateElement(container, { height: container.height - diff });\n        }\n        // Start pushing text upward until a diff of 30px (padding)\n        // is reached\n        else {\n          // vertically center align the text\n          coordY = container.y + container.height / 2 - height / 2;\n        }\n      }\n      const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\n      const { textAlign } = updatedElement;\n      editable.value = updatedElement.originalText;\n      const lines = updatedElement.originalText.split(\"\\n\");\n      const lineHeight = updatedElement.containerId\n        ? approxLineHeight\n        : updatedElement.height / lines.length;\n      if (!container) {\n        maxWidth =\n          (appState.width - viewportX - 8) /\n            appState.zoom.value -\n          // margin-right of parent if any\n          Number(\n            getComputedStyle(\n              excalidrawContainer?.parentNode as Element,\n            ).marginRight.slice(0, -2),\n          );\n      }\n\n      // Make sure text editor height doesn't go beyond viewport\n      const editorMaxHeight =\n        (appState.height -\n          viewportY) /\n        appState.zoom.value;\n      const angle = container ? container.angle : updatedElement.angle;\n      Object.assign(editable.style, {\n        font: getFontString(updatedElement),\n        // must be defined *after* font ¯\\_(ツ)_/¯\n        lineHeight: `${lineHeight}px`,\n        width: `${width}px`,\n        height: `${height}px`,\n        left: `${viewportX}px`,\n        top: `${viewportY}px`,\n        transform: getTransform(\n          width,\n          height,\n          angle,\n          appState,\n          maxWidth,\n          editorMaxHeight,\n        ),\n        textAlign,\n        color: updatedElement.strokeColor,\n        opacity: updatedElement.opacity / 100,\n        filter: \"var(--theme-filter)\",\n        maxWidth: `${maxWidth}px`,\n        maxHeight: `${editorMaxHeight}px`,\n      });\n      // For some reason updating font attribute doesn't set font family\n      // hence updating font family explicitly for test environment\n      if (isTestEnv()) {\n        editable.style.fontFamily = getFontFamilyString(updatedElement);\n      }\n      mutateElement(updatedElement, { x: coordX, y: coordY });\n    }\n  };\n\n  const editable = document.createElement(\"textarea\");\n\n  editable.dir = \"auto\";\n  editable.tabIndex = 0;\n  editable.dataset.type = \"wysiwyg\";\n  // prevent line wrapping on Safari\n  editable.wrap = \"off\";\n  editable.classList.add(\"excalidraw-wysiwyg\");\n\n  let whiteSpace = \"pre\";\n  let wordBreak = \"normal\";\n\n  if (isBoundToContainer(element)) {\n    whiteSpace = \"pre-wrap\";\n    wordBreak = \"break-word\";\n  }\n  Object.assign(editable.style, {\n    position: \"absolute\",\n    display: \"inline-block\",\n    minHeight: \"1em\",\n    backfaceVisibility: \"hidden\",\n    margin: 0,\n    padding: 0,\n    border: 0,\n    outline: 0,\n    resize: \"none\",\n    background: \"transparent\",\n    overflow: \"hidden\",\n    // must be specified because in dark mode canvas creates a stacking context\n    zIndex: \"var(--zIndex-wysiwyg)\",\n    wordBreak,\n    // prevent line wrapping (`whitespace: nowrap` doesn't work on FF)\n    whiteSpace,\n    overflowWrap: \"break-word\",\n  });\n  updateWysiwygStyle();\n\n  if (onChange) {\n    editable.oninput = () => {\n      // using scrollHeight here since we need to calculate\n      // number of lines so cannot use editable.style.height\n      // as that gets updated below\n      const lines = editable.scrollHeight / approxLineHeight;\n      // auto increase height only when lines  > 1 so its\n      // measured correctly and vertically alignes for\n      // first line as well as setting height to \"auto\"\n      // doubles the height as soon as user starts typing\n      if (isBoundToContainer(element) && lines > 1) {\n        let height = \"auto\";\n\n        if (lines === 2) {\n          const container = getContainerElement(element);\n          const actualLineCount = wrapText(\n            editable.value,\n            getFontString(element),\n            container!.width,\n          ).split(\"\\n\").length;\n\n          // This is browser behaviour when setting height to \"auto\"\n          // It sets the height needed for 2 lines even if actual\n          // line count is 1 as mentioned above as well\n          // hence reducing the height by half if actual line count is 1\n          // so single line aligns vertically when deleting\n          if (actualLineCount === 1) {\n            height = `${editable.scrollHeight / 2}px`;\n          }\n        }\n        editable.style.height = height;\n        editable.style.height = `${editable.scrollHeight}px`;\n      }\n      onChange(normalizeText(editable.value));\n    };\n  }\n\n  editable.onkeydown = (event) => {\n    event.stopPropagation();\n\n    if (actionDecreaseFontSize.keyTest(event)) {\n      app.actionManager.executeAction(actionDecreaseFontSize);\n    } else if (actionIncreaseFontSize.keyTest(event)) {\n      app.actionManager.executeAction(actionIncreaseFontSize);\n    } else if (event.key === KEYS.ESCAPE) {\n      event.preventDefault();\n      submittedViaKeyboard = true;\n      handleSubmit();\n    } else if (event.key === KEYS.ENTER && event[KEYS.CTRL_OR_CMD]) {\n      event.preventDefault();\n      if (event.isComposing || event.keyCode === 229) {\n        return;\n      }\n      submittedViaKeyboard = true;\n      handleSubmit();\n    } else if (\n      event.key === KEYS.TAB ||\n      (event[KEYS.CTRL_OR_CMD] &&\n        (event.code === CODES.BRACKET_LEFT ||\n          event.code === CODES.BRACKET_RIGHT))\n    ) {\n      event.preventDefault();\n      if (event.shiftKey || event.code === CODES.BRACKET_LEFT) {\n        outdent();\n      } else {\n        indent();\n      }\n      // We must send an input event to resize the element\n      editable.dispatchEvent(new Event(\"input\"));\n    }\n  };\n\n  const TAB_SIZE = 4;\n  const TAB = \" \".repeat(TAB_SIZE);\n  const RE_LEADING_TAB = new RegExp(`^ {1,${TAB_SIZE}}`);\n  const indent = () => {\n    const { selectionStart, selectionEnd } = editable;\n    const linesStartIndices = getSelectedLinesStartIndices();\n\n    let value = editable.value;\n    linesStartIndices.forEach((startIndex: number) => {\n      const startValue = value.slice(0, startIndex);\n      const endValue = value.slice(startIndex);\n\n      value = `${startValue}${TAB}${endValue}`;\n    });\n\n    editable.value = value;\n\n    editable.selectionStart = selectionStart + TAB_SIZE;\n    editable.selectionEnd = selectionEnd + TAB_SIZE * linesStartIndices.length;\n  };\n\n  const outdent = () => {\n    const { selectionStart, selectionEnd } = editable;\n    const linesStartIndices = getSelectedLinesStartIndices();\n    const removedTabs: number[] = [];\n\n    let value = editable.value;\n    linesStartIndices.forEach((startIndex) => {\n      const tabMatch = value\n        .slice(startIndex, startIndex + TAB_SIZE)\n        .match(RE_LEADING_TAB);\n\n      if (tabMatch) {\n        const startValue = value.slice(0, startIndex);\n        const endValue = value.slice(startIndex + tabMatch[0].length);\n\n        // Delete a tab from the line\n        value = `${startValue}${endValue}`;\n        removedTabs.push(startIndex);\n      }\n    });\n\n    editable.value = value;\n\n    if (removedTabs.length) {\n      if (selectionStart > removedTabs[removedTabs.length - 1]) {\n        editable.selectionStart = Math.max(\n          selectionStart - TAB_SIZE,\n          removedTabs[removedTabs.length - 1],\n        );\n      } else {\n        // If the cursor is before the first tab removed, ex:\n        // Line| #1\n        //     Line #2\n        // Lin|e #3\n        // we should reset the selectionStart to his initial value.\n        editable.selectionStart = selectionStart;\n      }\n      editable.selectionEnd = Math.max(\n        editable.selectionStart,\n        selectionEnd - TAB_SIZE * removedTabs.length,\n      );\n    }\n  };\n\n  /**\n   * @returns indeces of start positions of selected lines, in reverse order\n   */\n  const getSelectedLinesStartIndices = () => {\n    let { selectionStart, selectionEnd, value } = editable;\n\n    // chars before selectionStart on the same line\n    const startOffset = value.slice(0, selectionStart).match(/[^\\n]*$/)![0]\n      .length;\n    // put caret at the start of the line\n    selectionStart = selectionStart - startOffset;\n\n    const selected = value.slice(selectionStart, selectionEnd);\n\n    return selected\n      .split(\"\\n\")\n      .reduce(\n        (startIndices, line, idx, lines) =>\n          startIndices.concat(\n            idx\n              ? // curr line index is prev line's start + prev line's length + \\n\n                startIndices[idx - 1] + lines[idx - 1].length + 1\n              : // first selected line\n                selectionStart,\n          ),\n        [] as number[],\n      )\n      .reverse();\n  };\n\n  const stopEvent = (event: Event) => {\n    event.preventDefault();\n    event.stopPropagation();\n  };\n\n  // using a state variable instead of passing it to the handleSubmit callback\n  // so that we don't need to create separate a callback for event handlers\n  let submittedViaKeyboard = false;\n  const handleSubmit = () => {\n    // cleanup must be run before onSubmit otherwise when app blurs the wysiwyg\n    // it'd get stuck in an infinite loop of blur→onSubmit after we re-focus the\n    // wysiwyg on update\n    cleanup();\n    const updateElement = Scene.getScene(element)?.getElement(\n      element.id,\n    ) as ExcalidrawTextElement;\n    if (!updateElement) {\n      return;\n    }\n    let text = editable.value;\n    const container = getContainerElement(updateElement);\n\n    if (container) {\n      text = updateElement.text;\n      if (editable.value) {\n        const boundTextElementId = getBoundTextElementId(container);\n        if (!boundTextElementId || boundTextElementId !== element.id) {\n          mutateElement(container, {\n            boundElements: (container.boundElements || []).concat({\n              type: \"text\",\n              id: element.id,\n            }),\n          });\n        }\n      } else {\n        mutateElement(container, {\n          boundElements: container.boundElements?.filter(\n            (ele) =>\n              !isTextElement(\n                ele as ExcalidrawTextElement | ExcalidrawLinearElement,\n              ),\n          ),\n        });\n      }\n    }\n\n    onSubmit({\n      text,\n      viaKeyboard: submittedViaKeyboard,\n      originalText: editable.value,\n    });\n  };\n\n  const cleanup = () => {\n    if (isDestroyed) {\n      return;\n    }\n    isDestroyed = true;\n    // remove events to ensure they don't late-fire\n    editable.onblur = null;\n    editable.oninput = null;\n    editable.onkeydown = null;\n\n    if (observer) {\n      observer.disconnect();\n    }\n\n    window.removeEventListener(\"resize\", updateWysiwygStyle);\n    window.removeEventListener(\"wheel\", stopEvent, true);\n    window.removeEventListener(\"pointerdown\", onPointerDown);\n    window.removeEventListener(\"pointerup\", bindBlurEvent);\n    window.removeEventListener(\"blur\", handleSubmit);\n\n    unbindUpdate();\n\n    editable.remove();\n  };\n\n  const bindBlurEvent = (event?: MouseEvent) => {\n    window.removeEventListener(\"pointerup\", bindBlurEvent);\n    // Deferred so that the pointerdown that initiates the wysiwyg doesn't\n    // trigger the blur on ensuing pointerup.\n    // Also to handle cases such as picking a color which would trigger a blur\n    // in that same tick.\n    const target = event?.target;\n\n    const isTargetColorPicker =\n      target instanceof HTMLInputElement &&\n      target.closest(\".color-picker-input\") &&\n      isWritableElement(target);\n\n    setTimeout(() => {\n      editable.onblur = handleSubmit;\n      if (target && isTargetColorPicker) {\n        target.onblur = () => {\n          editable.focus();\n        };\n      }\n      // case: clicking on the same property → no change → no update → no focus\n      if (!isTargetColorPicker) {\n        editable.focus();\n      }\n    });\n  };\n\n  // prevent blur when changing properties from the menu\n  const onPointerDown = (event: MouseEvent) => {\n    const isTargetColorPicker =\n      event.target instanceof HTMLInputElement &&\n      event.target.closest(\".color-picker-input\") &&\n      isWritableElement(event.target);\n    if (\n      ((event.target instanceof HTMLElement ||\n        event.target instanceof SVGElement) &&\n        event.target.closest(`.${CLASSES.SHAPE_ACTIONS_MENU}`) &&\n        !isWritableElement(event.target)) ||\n      isTargetColorPicker\n    ) {\n      editable.onblur = null;\n      window.addEventListener(\"pointerup\", bindBlurEvent);\n      // handle edge-case where pointerup doesn't fire e.g. due to user\n      // alt-tabbing away\n      window.addEventListener(\"blur\", handleSubmit);\n    }\n  };\n\n  // handle updates of textElement properties of editing element\n  const unbindUpdate = Scene.getScene(element)!.addCallback(() => {\n    updateWysiwygStyle();\n    const isColorPickerActive = !!document.activeElement?.closest(\n      \".color-picker-input\",\n    );\n    if (!isColorPickerActive) {\n      editable.focus();\n    }\n  });\n\n  // ---------------------------------------------------------------------------\n\n  let isDestroyed = false;\n\n  // select on init (focusing is done separately inside the bindBlurEvent()\n  // because we need it to happen *after* the blur event from `pointerdown`)\n  editable.select();\n  bindBlurEvent();\n\n  // reposition wysiwyg in case of canvas is resized. Using ResizeObserver\n  // is preferred so we catch changes from host, where window may not resize.\n  let observer: ResizeObserver | null = null;\n  if (canvas && \"ResizeObserver\" in window) {\n    observer = new window.ResizeObserver(() => {\n      updateWysiwygStyle();\n    });\n    observer.observe(canvas);\n  } else {\n    window.addEventListener(\"resize\", updateWysiwygStyle);\n  }\n\n  window.addEventListener(\"pointerdown\", onPointerDown);\n  window.addEventListener(\"wheel\", stopEvent, {\n    passive: false,\n    capture: true,\n  });\n  excalidrawContainer\n    ?.querySelector(\".excalidraw-textEditorContainer\")!\n    .appendChild(editable);\n};\n"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///../../element/textWysiwyg.tsx\n");
2554
2554
 
2555
2555
  /***/ }),
2556
2556
 
@@ -3111,7 +3111,7 @@ module.exports = JSON.parse('{"ar-SA":88,"bg-BG":62,"bn-BD":0,"ca-ES":97,"cs-CZ"
3111
3111
  /***/ ((module) => {
3112
3112
 
3113
3113
  "use strict";
3114
- module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.10.0-fixjumpingtext-1","main":"main.js","types":"types/packages/excalidraw/index.d.ts","files":["dist/*","types/*"],"publishConfig":{"access":"public"},"description":"Excalidraw as a React component","repository":"https://github.com/excalidraw/excalidraw","license":"MIT","keywords":["excalidraw","excalidraw-embed","react","npm","npm excalidraw"],"browserslist":{"production":[">0.2%","not dead","not ie <= 11","not op_mini all","not safari < 12","not kaios <= 2.5","not edge < 79","not chrome < 70","not and_uc < 13","not samsung < 10"],"development":["last 1 chrome version","last 1 firefox version","last 1 safari version"]},"peerDependencies":{"react":"^17.0.2","react-dom":"^17.0.2"},"devDependencies":{"@babel/core":"7.16.7","@babel/plugin-transform-arrow-functions":"7.16.7","@babel/plugin-transform-async-to-generator":"7.16.0","@babel/plugin-transform-runtime":"7.16.8","@babel/plugin-transform-typescript":"7.16.1","@babel/preset-env":"7.16.7","@babel/preset-react":"7.16.7","@babel/preset-typescript":"7.16.7","autoprefixer":"10.4.2","babel-loader":"8.2.3","babel-plugin-transform-class-properties":"6.24.1","cross-env":"7.0.3","css-loader":"6.5.1","mini-css-extract-plugin":"2.4.6","postcss-loader":"6.2.1","sass-loader":"12.4.0","terser-webpack-plugin":"5.3.0","ts-loader":"9.2.6","typescript":"4.5.4","webpack":"5.65.0","webpack-bundle-analyzer":"4.5.0","webpack-cli":"4.9.1","webpack-dev-server":"4.7.3","webpack-merge":"5.8.0"},"bugs":"https://github.com/excalidraw/excalidraw/issues","homepage":"https://github.com/excalidraw/excalidraw/tree/master/src/packages/excalidraw","scripts":{"gen:types":"tsc --project ../../../tsconfig-types.json","build:umd":"cross-env NODE_ENV=production webpack --config webpack.prod.config.js && cross-env NODE_ENV=development webpack --config webpack.dev.config.js && yarn gen:types","build:umd:withAnalyzer":"cross-env NODE_ENV=production ANALYZER=true webpack --config webpack.prod.config.js","pack":"yarn build:umd && yarn pack","start":"webpack serve --config webpack.dev-server.config.js "},"dependencies":{"dotenv":"10.0.0"}}');
3114
+ module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.10.0-fixjumpingtext-2","main":"main.js","types":"types/packages/excalidraw/index.d.ts","files":["dist/*","types/*"],"publishConfig":{"access":"public"},"description":"Excalidraw as a React component","repository":"https://github.com/excalidraw/excalidraw","license":"MIT","keywords":["excalidraw","excalidraw-embed","react","npm","npm excalidraw"],"browserslist":{"production":[">0.2%","not dead","not ie <= 11","not op_mini all","not safari < 12","not kaios <= 2.5","not edge < 79","not chrome < 70","not and_uc < 13","not samsung < 10"],"development":["last 1 chrome version","last 1 firefox version","last 1 safari version"]},"peerDependencies":{"react":"^17.0.2","react-dom":"^17.0.2"},"devDependencies":{"@babel/core":"7.16.7","@babel/plugin-transform-arrow-functions":"7.16.7","@babel/plugin-transform-async-to-generator":"7.16.0","@babel/plugin-transform-runtime":"7.16.8","@babel/plugin-transform-typescript":"7.16.1","@babel/preset-env":"7.16.7","@babel/preset-react":"7.16.7","@babel/preset-typescript":"7.16.7","autoprefixer":"10.4.2","babel-loader":"8.2.3","babel-plugin-transform-class-properties":"6.24.1","cross-env":"7.0.3","css-loader":"6.5.1","mini-css-extract-plugin":"2.4.6","postcss-loader":"6.2.1","sass-loader":"12.4.0","terser-webpack-plugin":"5.3.0","ts-loader":"9.2.6","typescript":"4.5.4","webpack":"5.65.0","webpack-bundle-analyzer":"4.5.0","webpack-cli":"4.9.1","webpack-dev-server":"4.7.3","webpack-merge":"5.8.0"},"bugs":"https://github.com/excalidraw/excalidraw/issues","homepage":"https://github.com/excalidraw/excalidraw/tree/master/src/packages/excalidraw","scripts":{"gen:types":"tsc --project ../../../tsconfig-types.json","build:umd":"cross-env NODE_ENV=production webpack --config webpack.prod.config.js && cross-env NODE_ENV=development webpack --config webpack.dev.config.js && yarn gen:types","build:umd:withAnalyzer":"cross-env NODE_ENV=production ANALYZER=true webpack --config webpack.prod.config.js","pack":"yarn build:umd && yarn pack","start":"webpack serve --config webpack.dev-server.config.js "},"dependencies":{"dotenv":"10.0.0"}}');
3115
3115
 
3116
3116
  /***/ })
3117
3117