@zsviczian/excalidraw 0.10.0-textfix-9 → 0.10.0-textfix-10
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.
|
@@ -2542,7 +2542,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac
|
|
|
2542
2542
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
2543
2543
|
|
|
2544
2544
|
"use strict";
|
|
2545
|
-
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\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, offsetTop, offsetLeft } = appState;\r\n const degree = (180 * angle) / Math.PI;\r\n // offsets must be multiplied by 2 to account for the division by 2 of\r\n // the whole expression afterwards\r\n let translateX = ((width - offsetLeft * 2) * (zoom.value - 1)) / 2;\r\n let translateY = ((height - offsetTop * 2) * (zoom.value - 1)) / 2;\r\n if (width > maxWidth && zoom.value !== 1) {\r\n translateX = ((maxWidth - offsetLeft * 2) * (zoom.value - 1)) / 2;\r\n }\r\n if (height > maxHeight && zoom.value !== 1) {\r\n translateY = ((maxHeight - offsetTop * 2) * (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, }) => {\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,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(element)\r\n ? (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element))\r\n : 0;\r\n const updateWysiwygStyle = () => {\r\n var _a, _b;\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 let container = (updatedElement === null || updatedElement === void 0 ? void 0 : updatedElement.containerId)\r\n ? _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updatedElement).getElement(updatedElement.containerId)\r\n : null;\r\n let maxWidth = updatedElement.width;\r\n let maxHeight = updatedElement.height;\r\n let width = updatedElement.width;\r\n const height = editable.scrollHeight === 0\r\n ? updatedElement.height\r\n : editable.scrollHeight;\r\n if (container && updatedElement.containerId) {\r\n const propertiesUpdated = textPropertiesUpdated(updatedElement, editable);\r\n if (propertiesUpdated) {\r\n const currentContainer = (_b = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updatedElement)) === null || _b === void 0 ? void 0 : _b.getElement(updatedElement.containerId);\r\n approxLineHeight = (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updatedElement)\r\n ? (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement))\r\n : 0;\r\n if (updatedElement.height > currentContainer.height - _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2) {\r\n const nextHeight = updatedElement.height + _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2;\r\n originalContainerHeight = nextHeight;\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: nextHeight });\r\n container = Object.assign(Object.assign({}, container), { height: nextHeight });\r\n }\r\n editable.style.height = `${updatedElement.height}px`;\r\n }\r\n if (!originalContainerHeight) {\r\n originalContainerHeight = container.height;\r\n }\r\n maxWidth = container.width - _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2;\r\n maxHeight = container.height - _constants__WEBPACK_IMPORTED_MODULE_4__.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__.PADDING;\r\n // autogrow container height if text exceeds\r\n if (editable.scrollHeight > maxHeight) {\r\n const diff = Math.min(editable.scrollHeight - 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 editable.scrollHeight < maxHeight) {\r\n const diff = Math.min(maxHeight - editable.scrollHeight, 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 const lines = editable.scrollHeight / approxLineHeight;\r\n if (lines > 1 || propertiesUpdated) {\r\n // vertically center align the text\r\n coordY =\r\n container.y + container.height / 2 - editable.scrollHeight / 2;\r\n }\r\n }\r\n }\r\n const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\r\n const { textAlign, angle } = updatedElement;\r\n editable.value = updatedElement.originalText || updatedElement.text;\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 + appState.offsetLeft - viewportX - 8) / 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 + appState.offsetTop - viewportY) / appState.zoom.value;\r\n console.log({ maxWidth, editorMaxHeight, width, height, viewportX, viewportY });\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 }\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 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 editable.style.height = \"auto\";\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 (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 wrappedText = \"\";\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updateElement) && (updateElement === null || updateElement === void 0 ? void 0 : updateElement.containerId)) {\r\n const container = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updateElement).getElement(updateElement.containerId);\r\n if (container) {\r\n wrappedText = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.wrapText)(editable.value, (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updateElement), container.width);\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updateElement) && updateElement.containerId) {\r\n const editorHeight = Number(editable.style.height.slice(0, -2));\r\n if (editable.value) {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(updateElement, {\r\n // vertically center align\r\n y: container.y + container.height / 2 - editorHeight / 2,\r\n height: editorHeight,\r\n width: Number(editable.style.width.slice(0, -2)),\r\n // preserve padding\r\n x: container.x + _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING,\r\n });\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) => ele.type !== \"text\"),\r\n });\r\n }\r\n }\r\n }\r\n }\r\n else {\r\n wrappedText = editable.value;\r\n }\r\n onSubmit({\r\n text: normalizeText(wrappedText),\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  sceneCoordsToViewportCoords,\n} from \"../utils\";\nimport Scene from \"../scene/Scene\";\nimport { isBoundToContainer, isTextElement } from \"./typeChecks\";\nimport { CLASSES, PADDING } from \"../constants\";\nimport {\n  ExcalidrawBindableElement,\n  ExcalidrawElement,\n  ExcalidrawTextElement,\n} from \"./types\";\nimport { AppState } from \"../types\";\nimport { mutateElement } from \"./mutateElement\";\nimport {\n  getApproxLineHeight,\n  getBoundTextElementId,\n  wrapText,\n} from \"./textElement\";\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, offsetTop, offsetLeft } = appState;\n  const degree = (180 * angle) / Math.PI;\n  // offsets must be multiplied by 2 to account for the division by 2 of\n  // the whole expression afterwards\n  let translateX = ((width - offsetLeft * 2) * (zoom.value - 1)) / 2;\n  let translateY = ((height - offsetTop * 2) * (zoom.value - 1)) / 2;\n  if (width > maxWidth && zoom.value !== 1) {\n    translateX = ((maxWidth - offsetLeft * 2) * (zoom.value - 1)) / 2\n  }\n  if (height > maxHeight && zoom.value !== 1) {\n    translateY = ((maxHeight - offsetTop * 2) * (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}: {\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: ExcalidrawElement;\n  canvas: HTMLCanvasElement | null;\n  excalidrawContainer: HTMLDivElement | null;\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 = isTextElement(element)\n    ? getApproxLineHeight(getFontString(element))\n    : 0;\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      let container = updatedElement?.containerId\n        ? Scene.getScene(updatedElement)!.getElement(updatedElement.containerId)\n        : null;\n      let maxWidth = updatedElement.width;\n\n      let maxHeight = updatedElement.height;\n      let width = updatedElement.width;\n      const height =\n        editable.scrollHeight === 0\n          ? updatedElement.height\n          : editable.scrollHeight;\n      if (container && updatedElement.containerId) {\n        const propertiesUpdated = textPropertiesUpdated(\n          updatedElement,\n          editable,\n        );\n\n        if (propertiesUpdated) {\n          const currentContainer = Scene.getScene(updatedElement)?.getElement(\n            updatedElement.containerId,\n          ) as ExcalidrawBindableElement;\n          approxLineHeight = isTextElement(updatedElement)\n            ? getApproxLineHeight(getFontString(updatedElement))\n            : 0;\n          if (updatedElement.height > currentContainer.height - PADDING * 2) {\n            const nextHeight = updatedElement.height + PADDING * 2;\n            originalContainerHeight = nextHeight;\n            mutateElement(container, { height: nextHeight });\n            container = { ...container, height: nextHeight };\n          }\n          editable.style.height = `${updatedElement.height}px`;\n        }\n        if (!originalContainerHeight) {\n          originalContainerHeight = container.height;\n        }\n        maxWidth = container.width - PADDING * 2;\n        maxHeight = container.height - PADDING * 2;\n        width = maxWidth;\n        // The coordinates of text box set a distance of\n        // 30px to preserve padding\n        coordX = container.x + PADDING;\n\n        // autogrow container height if text exceeds\n        if (editable.scrollHeight > maxHeight) {\n          const diff = Math.min(\n            editable.scrollHeight - maxHeight,\n            approxLineHeight,\n          );\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          editable.scrollHeight < maxHeight\n        ) {\n          const diff = Math.min(\n            maxHeight - editable.scrollHeight,\n            approxLineHeight,\n          );\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          const lines = editable.scrollHeight / approxLineHeight;\n\n          if (lines > 1 || propertiesUpdated) {\n            // vertically center align the text\n            coordY =\n              container.y + container.height / 2 - editable.scrollHeight / 2;\n          }\n        }\n      }\n      const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\n      const { textAlign, angle } = updatedElement;\n\n      editable.value = updatedElement.originalText || updatedElement.text;\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 + appState.offsetLeft - viewportX - 8) / 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 + appState.offsetTop - viewportY) / appState.zoom.value;\n\n      console.log({maxWidth,editorMaxHeight,width,height,viewportX,viewportY});\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    }\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      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        editable.style.height = \"auto\";\n        editable.style.height = `${editable.scrollHeight}px`;\n      }\n      onChange(normalizeText(editable.value));\n    };\n  }\n\n  editable.onkeydown = (event) => {\n    event.stopPropagation();\n    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(element.id);\n    if (!updateElement) {\n      return;\n    }\n    let wrappedText = \"\";\n    if (isTextElement(updateElement) && updateElement?.containerId) {\n      const container = Scene.getScene(updateElement)!.getElement(\n        updateElement.containerId,\n      ) as ExcalidrawBindableElement;\n\n      if (container) {\n        wrappedText = wrapText(\n          editable.value,\n          getFontString(updateElement),\n          container.width,\n        );\n        if (isTextElement(updateElement) && updateElement.containerId) {\n          const editorHeight = Number(editable.style.height.slice(0, -2));\n          if (editable.value) {\n            mutateElement(updateElement, {\n              // vertically center align\n              y: container.y + container.height / 2 - editorHeight / 2,\n              height: editorHeight,\n              width: Number(editable.style.width.slice(0, -2)),\n              // preserve padding\n              x: container.x + PADDING,\n            });\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) => ele.type !== \"text\",\n              ),\n            });\n          }\n        }\n      }\n    } else {\n      wrappedText = editable.value;\n    }\n\n    onSubmit({\n      text: normalizeText(wrappedText),\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");
|
|
2545
|
+
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\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, offsetTop, offsetLeft } = appState;\r\n const degree = (180 * angle) / Math.PI;\r\n // offsets must be multiplied by 2 to account for the division by 2 of\r\n // the whole expression afterwards\r\n let translateX = ((width - offsetLeft * 2) * (zoom.value - 1)) / 2;\r\n let translateY = ((height - offsetTop * 2) * (zoom.value - 1)) / 2;\r\n if (width > maxWidth && zoom.value !== 1) {\r\n translateX = ((maxWidth - offsetLeft * 2) * (zoom.value - 1)) / 2;\r\n }\r\n if (height > maxHeight && zoom.value !== 1) {\r\n translateY = ((maxHeight - offsetTop * 2) * (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, }) => {\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,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(element)\r\n ? (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(element))\r\n : 0;\r\n const updateWysiwygStyle = () => {\r\n var _a, _b;\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 let container = (updatedElement === null || updatedElement === void 0 ? void 0 : updatedElement.containerId)\r\n ? _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updatedElement).getElement(updatedElement.containerId)\r\n : null;\r\n let maxWidth = updatedElement.width;\r\n let maxHeight = updatedElement.height;\r\n let width = updatedElement.width;\r\n const height = editable.scrollHeight === 0\r\n ? updatedElement.height\r\n : editable.scrollHeight;\r\n if (container && updatedElement.containerId) {\r\n const propertiesUpdated = textPropertiesUpdated(updatedElement, editable);\r\n if (propertiesUpdated) {\r\n const currentContainer = (_b = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updatedElement)) === null || _b === void 0 ? void 0 : _b.getElement(updatedElement.containerId);\r\n approxLineHeight = (0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updatedElement)\r\n ? (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.getApproxLineHeight)((0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updatedElement))\r\n : 0;\r\n if (updatedElement.height > currentContainer.height - _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2) {\r\n const nextHeight = updatedElement.height + _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2;\r\n originalContainerHeight = nextHeight;\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(container, { height: nextHeight });\r\n container = Object.assign(Object.assign({}, container), { height: nextHeight });\r\n }\r\n editable.style.height = `${updatedElement.height}px`;\r\n }\r\n if (!originalContainerHeight) {\r\n originalContainerHeight = container.height;\r\n }\r\n maxWidth = container.width - _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING * 2;\r\n maxHeight = container.height - _constants__WEBPACK_IMPORTED_MODULE_4__.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__.PADDING;\r\n // autogrow container height if text exceeds\r\n if (editable.scrollHeight > maxHeight) {\r\n const diff = Math.min(editable.scrollHeight - 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 editable.scrollHeight < maxHeight) {\r\n const diff = Math.min(maxHeight - editable.scrollHeight, 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 const lines = editable.scrollHeight / approxLineHeight;\r\n if (lines > 1 || propertiesUpdated) {\r\n // vertically center align the text\r\n coordY =\r\n container.y + container.height / 2 - editable.scrollHeight / 2;\r\n }\r\n }\r\n }\r\n const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\r\n const { textAlign, angle } = updatedElement;\r\n editable.value = updatedElement.originalText || updatedElement.text;\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 + appState.offsetLeft - viewportX - 8) / 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 + appState.offsetTop - viewportY) / appState.zoom.value;\r\n console.log({\r\n maxWidth,\r\n editorMaxHeight,\r\n width,\r\n height,\r\n viewportX,\r\n viewportY,\r\n appStateWidth: appState.width,\r\n appStateHeight: appState.height,\r\n scrollX: appState.scrollX,\r\n scrollY: appState.scrollY,\r\n transform: getTransform(width, height, angle, appState, maxWidth, editorMaxHeight)\r\n });\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 }\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 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 editable.style.height = \"auto\";\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 (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 wrappedText = \"\";\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updateElement) && (updateElement === null || updateElement === void 0 ? void 0 : updateElement.containerId)) {\r\n const container = _scene_Scene__WEBPACK_IMPORTED_MODULE_2__[\"default\"].getScene(updateElement).getElement(updateElement.containerId);\r\n if (container) {\r\n wrappedText = (0,_textElement__WEBPACK_IMPORTED_MODULE_6__.wrapText)(editable.value, (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getFontString)(updateElement), container.width);\r\n if ((0,_typeChecks__WEBPACK_IMPORTED_MODULE_3__.isTextElement)(updateElement) && updateElement.containerId) {\r\n const editorHeight = Number(editable.style.height.slice(0, -2));\r\n if (editable.value) {\r\n (0,_mutateElement__WEBPACK_IMPORTED_MODULE_5__.mutateElement)(updateElement, {\r\n // vertically center align\r\n y: container.y + container.height / 2 - editorHeight / 2,\r\n height: editorHeight,\r\n width: Number(editable.style.width.slice(0, -2)),\r\n // preserve padding\r\n x: container.x + _constants__WEBPACK_IMPORTED_MODULE_4__.PADDING,\r\n });\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) => ele.type !== \"text\"),\r\n });\r\n }\r\n }\r\n }\r\n }\r\n else {\r\n wrappedText = editable.value;\r\n }\r\n onSubmit({\r\n text: normalizeText(wrappedText),\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  sceneCoordsToViewportCoords,\n} from \"../utils\";\nimport Scene from \"../scene/Scene\";\nimport { isBoundToContainer, isTextElement } from \"./typeChecks\";\nimport { CLASSES, PADDING } from \"../constants\";\nimport {\n  ExcalidrawBindableElement,\n  ExcalidrawElement,\n  ExcalidrawTextElement,\n} from \"./types\";\nimport { AppState } from \"../types\";\nimport { mutateElement } from \"./mutateElement\";\nimport {\n  getApproxLineHeight,\n  getBoundTextElementId,\n  wrapText,\n} from \"./textElement\";\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, offsetTop, offsetLeft } = appState;\n  const degree = (180 * angle) / Math.PI;\n  // offsets must be multiplied by 2 to account for the division by 2 of\n  // the whole expression afterwards\n  let translateX = ((width - offsetLeft * 2) * (zoom.value - 1)) / 2;\n  let translateY = ((height - offsetTop * 2) * (zoom.value - 1)) / 2;\n  if (width > maxWidth && zoom.value !== 1) {\n    translateX = ((maxWidth - offsetLeft * 2) * (zoom.value - 1)) / 2\n  }\n  if (height > maxHeight && zoom.value !== 1) {\n    translateY = ((maxHeight - offsetTop * 2) * (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}: {\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: ExcalidrawElement;\n  canvas: HTMLCanvasElement | null;\n  excalidrawContainer: HTMLDivElement | null;\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 = isTextElement(element)\n    ? getApproxLineHeight(getFontString(element))\n    : 0;\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      let container = updatedElement?.containerId\n        ? Scene.getScene(updatedElement)!.getElement(updatedElement.containerId)\n        : null;\n      let maxWidth = updatedElement.width;\n\n      let maxHeight = updatedElement.height;\n      let width = updatedElement.width;\n      const height =\n        editable.scrollHeight === 0\n          ? updatedElement.height\n          : editable.scrollHeight;\n      if (container && updatedElement.containerId) {\n        const propertiesUpdated = textPropertiesUpdated(\n          updatedElement,\n          editable,\n        );\n\n        if (propertiesUpdated) {\n          const currentContainer = Scene.getScene(updatedElement)?.getElement(\n            updatedElement.containerId,\n          ) as ExcalidrawBindableElement;\n          approxLineHeight = isTextElement(updatedElement)\n            ? getApproxLineHeight(getFontString(updatedElement))\n            : 0;\n          if (updatedElement.height > currentContainer.height - PADDING * 2) {\n            const nextHeight = updatedElement.height + PADDING * 2;\n            originalContainerHeight = nextHeight;\n            mutateElement(container, { height: nextHeight });\n            container = { ...container, height: nextHeight };\n          }\n          editable.style.height = `${updatedElement.height}px`;\n        }\n        if (!originalContainerHeight) {\n          originalContainerHeight = container.height;\n        }\n        maxWidth = container.width - PADDING * 2;\n        maxHeight = container.height - PADDING * 2;\n        width = maxWidth;\n        // The coordinates of text box set a distance of\n        // 30px to preserve padding\n        coordX = container.x + PADDING;\n\n        // autogrow container height if text exceeds\n        if (editable.scrollHeight > maxHeight) {\n          const diff = Math.min(\n            editable.scrollHeight - maxHeight,\n            approxLineHeight,\n          );\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          editable.scrollHeight < maxHeight\n        ) {\n          const diff = Math.min(\n            maxHeight - editable.scrollHeight,\n            approxLineHeight,\n          );\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          const lines = editable.scrollHeight / approxLineHeight;\n\n          if (lines > 1 || propertiesUpdated) {\n            // vertically center align the text\n            coordY =\n              container.y + container.height / 2 - editable.scrollHeight / 2;\n          }\n        }\n      }\n      const [viewportX, viewportY] = getViewportCoords(coordX, coordY);\n      const { textAlign, angle } = updatedElement;\n\n      editable.value = updatedElement.originalText || updatedElement.text;\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 + appState.offsetLeft - viewportX - 8) / 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 + appState.offsetTop - viewportY) / appState.zoom.value;\n\n      console.log({\n        maxWidth,\n        editorMaxHeight,\n        width,\n        height,\n        viewportX,\n        viewportY,\n        appStateWidth:appState.width,\n        appStateHeight:appState.height,\n        scrollX: appState.scrollX,\n        scrollY: appState.scrollY,\n        transform: getTransform(\n          width,\n          height,\n          angle,\n          appState,\n          maxWidth,\n          editorMaxHeight,\n        )\n      });\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    }\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      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        editable.style.height = \"auto\";\n        editable.style.height = `${editable.scrollHeight}px`;\n      }\n      onChange(normalizeText(editable.value));\n    };\n  }\n\n  editable.onkeydown = (event) => {\n    event.stopPropagation();\n    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(element.id);\n    if (!updateElement) {\n      return;\n    }\n    let wrappedText = \"\";\n    if (isTextElement(updateElement) && updateElement?.containerId) {\n      const container = Scene.getScene(updateElement)!.getElement(\n        updateElement.containerId,\n      ) as ExcalidrawBindableElement;\n\n      if (container) {\n        wrappedText = wrapText(\n          editable.value,\n          getFontString(updateElement),\n          container.width,\n        );\n        if (isTextElement(updateElement) && updateElement.containerId) {\n          const editorHeight = Number(editable.style.height.slice(0, -2));\n          if (editable.value) {\n            mutateElement(updateElement, {\n              // vertically center align\n              y: container.y + container.height / 2 - editorHeight / 2,\n              height: editorHeight,\n              width: Number(editable.style.width.slice(0, -2)),\n              // preserve padding\n              x: container.x + PADDING,\n            });\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) => ele.type !== \"text\",\n              ),\n            });\n          }\n        }\n      }\n    } else {\n      wrappedText = editable.value;\n    }\n\n    onSubmit({\n      text: normalizeText(wrappedText),\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");
|
|
2546
2546
|
|
|
2547
2547
|
/***/ }),
|
|
2548
2548
|
|
|
@@ -3103,7 +3103,7 @@ module.exports = JSON.parse('{"ar-SA":86,"bg-BG":59,"bn-BD":0,"ca-ES":78,"cs-CZ"
|
|
|
3103
3103
|
/***/ ((module) => {
|
|
3104
3104
|
|
|
3105
3105
|
"use strict";
|
|
3106
|
-
module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.10.0-textfix-
|
|
3106
|
+
module.exports = JSON.parse('{"name":"@zsviczian/excalidraw","version":"0.10.0-textfix-10","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.0","@babel/plugin-transform-arrow-functions":"7.16.0","@babel/plugin-transform-async-to-generator":"7.16.0","@babel/plugin-transform-runtime":"7.16.4","@babel/plugin-transform-typescript":"7.16.1","@babel/preset-env":"7.16.4","@babel/preset-react":"7.16.0","@babel/preset-typescript":"7.16.0","autoprefixer":"10.4.0","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.5","postcss-loader":"6.2.1","sass-loader":"12.4.0","terser-webpack-plugin":"5.2.5","ts-loader":"9.2.6","typescript":"4.5.3","webpack":"5.65.0","webpack-bundle-analyzer":"4.5.0","webpack-cli":"4.9.1"},"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"},"dependencies":{"dotenv":"10.0.0"}}');
|
|
3107
3107
|
|
|
3108
3108
|
/***/ })
|
|
3109
3109
|
|