{"version":3,"file":"_baseClone-70a2689c.js","sources":["../../src/contexts/EditorContext.tsx","../../src/utils/rect.ts","../../src/utils/dom.ts","../../src/components/editor/chrome/ChromeGrid.tsx","../../src/utils/element.ts","../../src/components/Dot.tsx","../../src/components/FormError.tsx","../../src/components/FieldsetField.tsx","../../src/contexts/VersionContext.tsx","../../src/components/ButtonGroupSwitch.tsx","../../../node_modules/@babel/runtime/helpers/interopRequireDefault.js","../../../node_modules/refractor/lang/jsx.js","../../../node_modules/react-syntax-highlighter/dist/cjs/languages/prism/jsx.js","../../../node_modules/refractor/lang/markup.js","../../../node_modules/react-syntax-highlighter/dist/cjs/languages/prism/markup.js","../../../node_modules/@babel/runtime/helpers/objectWithoutPropertiesLoose.js","../../../node_modules/@babel/runtime/helpers/objectWithoutProperties.js","../../../node_modules/@babel/runtime/helpers/arrayLikeToArray.js","../../../node_modules/@babel/runtime/helpers/arrayWithoutHoles.js","../../../node_modules/@babel/runtime/helpers/iterableToArray.js","../../../node_modules/@babel/runtime/helpers/unsupportedIterableToArray.js","../../../node_modules/@babel/runtime/helpers/nonIterableSpread.js","../../../node_modules/@babel/runtime/helpers/toConsumableArray.js","../../../node_modules/@babel/runtime/helpers/typeof.js","../../../node_modules/@babel/runtime/helpers/toPrimitive.js","../../../node_modules/@babel/runtime/helpers/toPropertyKey.js","../../../node_modules/@babel/runtime/helpers/defineProperty.js","../../../node_modules/react-syntax-highlighter/dist/cjs/create-element.js","../../../node_modules/react-syntax-highlighter/dist/cjs/checkForListedLanguage.js","../../../node_modules/react-syntax-highlighter/dist/cjs/highlight.js","../../../node_modules/xtend/immutable.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/schema.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/merge.js","../../../node_modules/hastscript/node_modules/property-information/normalize.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/info.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/types.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/defined-info.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/create.js","../../../node_modules/hastscript/node_modules/property-information/lib/xlink.js","../../../node_modules/hastscript/node_modules/property-information/lib/xml.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/case-sensitive-transform.js","../../../node_modules/hastscript/node_modules/property-information/lib/util/case-insensitive-transform.js","../../../node_modules/hastscript/node_modules/property-information/lib/xmlns.js","../../../node_modules/hastscript/node_modules/property-information/lib/aria.js","../../../node_modules/hastscript/node_modules/property-information/lib/html.js","../../../node_modules/hastscript/node_modules/property-information/html.js","../../../node_modules/hastscript/node_modules/property-information/find.js","../../../node_modules/hast-util-parse-selector/index.js","../../../node_modules/hastscript/node_modules/space-separated-tokens/index.js","../../../node_modules/hastscript/node_modules/comma-separated-tokens/index.js","../../../node_modules/hastscript/factory.js","../../../node_modules/hastscript/html.js","../../../node_modules/hastscript/index.js","../../../node_modules/is-decimal/index.js","../../../node_modules/is-hexadecimal/index.js","../../../node_modules/is-alphabetical/index.js","../../../node_modules/is-alphanumerical/index.js","../../../node_modules/parse-entities/decode-entity.browser.js","../../../node_modules/parse-entities/index.js","../../../node_modules/refractor/node_modules/prismjs/components/prism-core.js","../../../node_modules/refractor/lang/css.js","../../../node_modules/refractor/lang/clike.js","../../../node_modules/refractor/lang/javascript.js","../../../node_modules/refractor/core.js","../../../node_modules/react-syntax-highlighter/dist/cjs/prism-light.js","../../../node_modules/react-syntax-highlighter/dist/cjs/styles/prism/material-oceanic.js","../../../node_modules/@babel/runtime/helpers/esm/objectSpread2.js","../../../node_modules/@babel/runtime/helpers/esm/arrayWithHoles.js","../../../node_modules/@babel/runtime/helpers/esm/iterableToArrayLimit.js","../../../node_modules/@babel/runtime/helpers/esm/arrayLikeToArray.js","../../../node_modules/@babel/runtime/helpers/esm/unsupportedIterableToArray.js","../../../node_modules/@babel/runtime/helpers/esm/nonIterableRest.js","../../../node_modules/@babel/runtime/helpers/esm/slicedToArray.js","../../../node_modules/@babel/runtime/helpers/esm/objectWithoutProperties.js","../../../node_modules/react-select/dist/useStateManager-7e1e8489.esm.js","../../../node_modules/@babel/runtime/helpers/esm/classCallCheck.js","../../../node_modules/@babel/runtime/helpers/esm/createClass.js","../../../node_modules/@babel/runtime/helpers/esm/inherits.js","../../../node_modules/@babel/runtime/helpers/esm/getPrototypeOf.js","../../../node_modules/@babel/runtime/helpers/esm/isNativeReflectConstruct.js","../../../node_modules/@babel/runtime/helpers/esm/assertThisInitialized.js","../../../node_modules/@babel/runtime/helpers/esm/possibleConstructorReturn.js","../../../node_modules/@babel/runtime/helpers/esm/createSuper.js","../../../node_modules/@babel/runtime/helpers/esm/arrayWithoutHoles.js","../../../node_modules/@babel/runtime/helpers/esm/iterableToArray.js","../../../node_modules/@babel/runtime/helpers/esm/nonIterableSpread.js","../../../node_modules/@babel/runtime/helpers/esm/toConsumableArray.js","../../../node_modules/@babel/runtime/helpers/esm/taggedTemplateLiteral.js","../../../node_modules/@floating-ui/dom/dist/floating-ui.dom.browser.mjs","../../../node_modules/use-isomorphic-layout-effect/dist/use-isomorphic-layout-effect.browser.esm.js","../../../node_modules/react-select/dist/index-a86253bb.esm.js","../../../node_modules/memoize-one/dist/memoize-one.esm.js","../../../node_modules/react-select/dist/Select-40119e12.esm.js","../../../node_modules/react-select/dist/react-select.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/breakpoint-utils/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/shared-utils/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/object-utils/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/icon/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/react-context/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/node_modules/@chakra-ui/react-children-utils/dist/index.esm.js","../../../node_modules/@chakra-ui/layout/dist/index.esm.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/containers.js","../../../node_modules/@chakra-ui/spinner/dist/index.esm.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/control.js","../../../node_modules/chakra-react-select/dist/esm/utils.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/input.js","../../../node_modules/@chakra-ui/menu/dist/chunk-I3AUOXDN.mjs","../../../node_modules/chakra-react-select/dist/esm/chakra-components/menu.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/multi-value.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/placeholder.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/single-value.js","../../../node_modules/chakra-react-select/dist/esm/chakra-components/index.js","../../../node_modules/chakra-react-select/dist/esm/use-chakra-select-props.js","../../../node_modules/chakra-react-select/dist/esm/select.js","../../../node_modules/react-select/dist/useCreatable-36230047.esm.js","../../../node_modules/react-select/creatable/dist/react-select-creatable.esm.js","../../../node_modules/chakra-react-select/dist/esm/creatable-select.js","../../src/components/Select.tsx","../../src/components/editor/version/VersionEmbed.tsx","../../src/hooks/useStylesUpToDateConfirmationAction.ts","../../../node_modules/lodash-es/_freeGlobal.js","../../../node_modules/lodash-es/_root.js","../../../node_modules/lodash-es/_Symbol.js","../../../node_modules/lodash-es/_getRawTag.js","../../../node_modules/lodash-es/_objectToString.js","../../../node_modules/lodash-es/_baseGetTag.js","../../../node_modules/lodash-es/isObjectLike.js","../../../node_modules/lodash-es/isSymbol.js","../../../node_modules/lodash-es/isArray.js","../../../node_modules/lodash-es/_trimmedEndIndex.js","../../../node_modules/lodash-es/_baseTrim.js","../../../node_modules/lodash-es/isObject.js","../../../node_modules/lodash-es/toNumber.js","../../../node_modules/lodash-es/isFunction.js","../../../node_modules/lodash-es/_coreJsData.js","../../../node_modules/lodash-es/_isMasked.js","../../../node_modules/lodash-es/_toSource.js","../../../node_modules/lodash-es/_baseIsNative.js","../../../node_modules/lodash-es/_getValue.js","../../../node_modules/lodash-es/_getNative.js","../../../node_modules/lodash-es/_WeakMap.js","../../../node_modules/lodash-es/_baseCreate.js","../../../node_modules/lodash-es/_copyArray.js","../../../node_modules/lodash-es/_defineProperty.js","../../../node_modules/lodash-es/_arrayEach.js","../../../node_modules/lodash-es/_isIndex.js","../../../node_modules/lodash-es/_baseAssignValue.js","../../../node_modules/lodash-es/eq.js","../../../node_modules/lodash-es/_assignValue.js","../../../node_modules/lodash-es/_copyObject.js","../../../node_modules/lodash-es/isLength.js","../../../node_modules/lodash-es/isArrayLike.js","../../../node_modules/lodash-es/_isPrototype.js","../../../node_modules/lodash-es/_baseTimes.js","../../../node_modules/lodash-es/_baseIsArguments.js","../../../node_modules/lodash-es/isArguments.js","../../../node_modules/lodash-es/stubFalse.js","../../../node_modules/lodash-es/isBuffer.js","../../../node_modules/lodash-es/_baseIsTypedArray.js","../../../node_modules/lodash-es/_baseUnary.js","../../../node_modules/lodash-es/_nodeUtil.js","../../../node_modules/lodash-es/isTypedArray.js","../../../node_modules/lodash-es/_arrayLikeKeys.js","../../../node_modules/lodash-es/_overArg.js","../../../node_modules/lodash-es/_nativeKeys.js","../../../node_modules/lodash-es/_baseKeys.js","../../../node_modules/lodash-es/keys.js","../../../node_modules/lodash-es/_nativeKeysIn.js","../../../node_modules/lodash-es/_baseKeysIn.js","../../../node_modules/lodash-es/keysIn.js","../../../node_modules/lodash-es/_nativeCreate.js","../../../node_modules/lodash-es/_hashClear.js","../../../node_modules/lodash-es/_hashDelete.js","../../../node_modules/lodash-es/_hashGet.js","../../../node_modules/lodash-es/_hashHas.js","../../../node_modules/lodash-es/_hashSet.js","../../../node_modules/lodash-es/_Hash.js","../../../node_modules/lodash-es/_listCacheClear.js","../../../node_modules/lodash-es/_assocIndexOf.js","../../../node_modules/lodash-es/_listCacheDelete.js","../../../node_modules/lodash-es/_listCacheGet.js","../../../node_modules/lodash-es/_listCacheHas.js","../../../node_modules/lodash-es/_listCacheSet.js","../../../node_modules/lodash-es/_ListCache.js","../../../node_modules/lodash-es/_Map.js","../../../node_modules/lodash-es/_mapCacheClear.js","../../../node_modules/lodash-es/_isKeyable.js","../../../node_modules/lodash-es/_getMapData.js","../../../node_modules/lodash-es/_mapCacheDelete.js","../../../node_modules/lodash-es/_mapCacheGet.js","../../../node_modules/lodash-es/_mapCacheHas.js","../../../node_modules/lodash-es/_mapCacheSet.js","../../../node_modules/lodash-es/_MapCache.js","../../../node_modules/lodash-es/_arrayPush.js","../../../node_modules/lodash-es/_getPrototype.js","../../../node_modules/lodash-es/isPlainObject.js","../../../node_modules/lodash-es/_stackClear.js","../../../node_modules/lodash-es/_stackDelete.js","../../../node_modules/lodash-es/_stackGet.js","../../../node_modules/lodash-es/_stackHas.js","../../../node_modules/lodash-es/_stackSet.js","../../../node_modules/lodash-es/_Stack.js","../../../node_modules/lodash-es/_baseAssign.js","../../../node_modules/lodash-es/_baseAssignIn.js","../../../node_modules/lodash-es/_cloneBuffer.js","../../../node_modules/lodash-es/_arrayFilter.js","../../../node_modules/lodash-es/stubArray.js","../../../node_modules/lodash-es/_getSymbols.js","../../../node_modules/lodash-es/_copySymbols.js","../../../node_modules/lodash-es/_getSymbolsIn.js","../../../node_modules/lodash-es/_copySymbolsIn.js","../../../node_modules/lodash-es/_baseGetAllKeys.js","../../../node_modules/lodash-es/_getAllKeys.js","../../../node_modules/lodash-es/_getAllKeysIn.js","../../../node_modules/lodash-es/_DataView.js","../../../node_modules/lodash-es/_Promise.js","../../../node_modules/lodash-es/_Set.js","../../../node_modules/lodash-es/_getTag.js","../../../node_modules/lodash-es/_initCloneArray.js","../../../node_modules/lodash-es/_Uint8Array.js","../../../node_modules/lodash-es/_cloneArrayBuffer.js","../../../node_modules/lodash-es/_cloneDataView.js","../../../node_modules/lodash-es/_cloneRegExp.js","../../../node_modules/lodash-es/_cloneSymbol.js","../../../node_modules/lodash-es/_cloneTypedArray.js","../../../node_modules/lodash-es/_initCloneByTag.js","../../../node_modules/lodash-es/_initCloneObject.js","../../../node_modules/lodash-es/_baseIsMap.js","../../../node_modules/lodash-es/isMap.js","../../../node_modules/lodash-es/_baseIsSet.js","../../../node_modules/lodash-es/isSet.js","../../../node_modules/lodash-es/_baseClone.js"],"sourcesContent":["import type * as CK from '@sitecore-feaas/ckeditor5'\nimport { Style, VersionModel } from '@sitecore-feaas/sdk'\nimport { ContextType } from 'react'\nimport type { ChromeRefs } from '../components/editor/chrome/ChromeContext.js'\nimport type { PickerProps } from '../components/picker/Picker.js'\nimport { createContextProvider } from '../hooks/useContextProvider.js'\nimport { Measurement } from '../utils/dom.js'\nimport { Rect } from '../utils/rect.js'\n\nexport const [EditorContext, useEditorContextProvider] = createContextProvider(() => {\n return {\n editor: null as CK.Editor,\n editorClass: null as typeof CK.Editor,\n node: null as HTMLDivElement,\n root: null as HTMLElement,\n\n status: 'unloaded' as 'unloaded' | 'loading' | 'initialized' | 'ready',\n savingStatus: 'synchronized' as 'saving' | 'synchronized' | 'saved' | 'waiting',\n activeVersionId: null as string,\n isFocused: false,\n isDirty: false,\n isHiddenDisplayed: true,\n isDataDisplayed: true,\n isDataRepeated: true,\n isChromeVisible: false,\n isAutosaveEnabled: true,\n hasData: true,\n hasRepeating: true,\n isArchivedDisplayed: false,\n sidebarDialog: null as string,\n sidebarMode: null as PickerProps['mode'],\n current: null as CK.Editor['current'],\n context: null as Style.Context,\n\n // addedElement: null as CK.ModelElement,\n\n blur: null as () => void,\n focus: null as (id?: string) => void,\n setActiveVersionId: null as (rootName?: string) => void,\n\n onChromeBeforeMeasure: null as (refs: ChromeRefs) => void,\n onChromeMeasureElement: (element: HTMLElement, rect?: Rect) => {\n if (rect) return rect\n const domRect = element.getBoundingClientRect()\n return {\n left: domRect.left,\n top: domRect.top,\n width: domRect.width,\n height: domRect.height\n }\n },\n onChromeMeasure: null as (measurements: Record, refs: ChromeRefs) => void,\n onChromePosition: null as (positions: Record) => void,\n onContentChange: null as (content?: string) => void,\n\n onPlacementStart: null as (element: CK.ModelElement) => void,\n onVersionMove: null as (id: VersionModel['id'], before: VersionModel['id']) => void\n }\n}, '__FEAASEditorContext')\n\nexport type EditorContext = ContextType[0]\nexport type EditorContextSetter = ContextType[1]\n","export type Rect = {\n top: number\n left: number\n width: number\n height: number\n}\nexport function intersectRect(r1: Rect, r2: Rect): Rect {\n var x = Math.max(r1.left, r2.left)\n var y = Math.max(r1.top, r2.top)\n var xx = Math.min(r1.left + r1.width, r2.left + r2.width)\n var yy = Math.min(r1.top + r1.height, r2.top + r2.height)\n return { top: y, left: x, width: Math.max(0, xx - x), height: Math.max(0, yy - y) }\n}\n\nexport function unionRect(r1: Rect, r2: Rect) {\n if (!r2) return { ...r1 }\n if (!r1) return { ...r2 }\n var x = Math.min(r1.left, r2.left)\n var y = Math.min(r1.top, r2.top)\n var xx = Math.max(r1.left + r1.width, r2.left + r2.width)\n var yy = Math.max(r1.top + r1.height, r2.top + r2.height)\n return { top: y, left: x, width: xx - x, height: yy - y }\n}\n\nfunction dist(x1: number, y1: number, x2: number, y2: number) {\n return Math.sqrt(Math.pow(x2 - x1, 2) + Math.pow(y2 - y1, 2))\n}\n\nexport function distanceRect(r1: Rect, r2: Rect) {\n const left = r2.width + r2.left < r1.left\n const right = r1.width + r1.left < r2.left\n const bottom = r2.height + r2.top < r1.top\n const top = r1.height + r1.top < r2.top\n if (top && left) return dist(r1.left, r1.height + r1.top, r2.width + r2.left, r2.top)\n else if (left && bottom) return dist(r1.left, r1.top, r2.width + r2.left, r2.height + r2.top)\n else if (bottom && right) return dist(r1.width + r1.left, r1.top, r2.left, r2.height + r2.top)\n else if (right && top) return dist(r1.width + r1.left, r1.height + r1.top, r2.left, r2.top)\n else if (left) return r1.left - (r2.width + r2.left)\n else if (right) return r2.left - (r1.width + r1.left)\n else if (bottom) return r1.top - (r2.height + r2.top)\n else if (top) return r2.top - (r1.height + r1.top)\n else return 0\n}\n\nexport function getRectStyle(rect: Rect, usePosition = false) {\n return rect != null && rect.width && rect.height\n ? usePosition\n ? {\n left: rect.left,\n top: rect.top,\n width: rect.width,\n height: rect.height\n }\n : {\n transform: `translate(${rect.left}px, ${rect.top}px)`,\n width: rect.width,\n height: rect.height\n }\n : { ...rect, transform: 'translate(-1000px, -1000px)' }\n}\n\nexport function offsetRect(rect: Rect, offset: number) {\n return {\n left: rect.left + offset,\n top: rect.top + offset,\n width: rect.width - offset * 2,\n height: rect.height - offset * 2\n }\n}\n\nexport interface Point {\n left: number\n top: number\n}\nexport function distanceRectToPoint(rect: Rect, p: Point) {\n var dx = Math.max(rect.left - p.left, 0, p.left - (rect.left + rect.width))\n var dy = Math.max(rect.top - p.top, 0, p.top - (rect.top + rect.height))\n return Math.sqrt(dx * dx + dy * dy)\n}\n\nexport function isPointInsideRectangle(point: Point, rect: Rect) {\n return (\n rect.left <= point.left &&\n point.left <= rect.left + rect.width &&\n rect.top <= point.top &&\n point.top <= rect.top + rect.height\n )\n}\n\nexport type Line = [start: Point, end: Point]\nexport function getDistanceFromPointToLine(point: Point, [from, to]: Line) {\n var len2 = Math.pow(to.left - from.left, 2) + Math.pow(to.top - from.top, 2)\n\n if (len2 === 0) {\n return Math.sqrt(Math.pow(point.left - from.left, 2) + Math.pow(point.top - from.top, 2))\n }\n\n var t = Math.max(\n 0,\n Math.min(\n 1,\n ((point.left - from.left) * (to.left - from.left) + (point.top - from.top) * (to.top - from.top)) / len2\n )\n )\n\n var closestX = from.left + t * (to.left - from.left)\n var closestY = from.top + t * (to.top - from.top)\n\n return Math.sqrt(Math.pow(point.left - closestX, 2) + Math.pow(point.top - closestY, 2))\n}\n\nexport function getDistanceFromPointToRect(point: Point, rect: Rect) {\n // Find the closest point on the rectangle to the point\n let closestX = Math.max(rect.left, Math.min(point.left, rect.left + rect.width))\n let closestY = Math.max(rect.top, Math.min(point.top, rect.top + rect.height))\n\n // Calculate the distance between the closest point and the point\n let distanceX = point.left - closestX\n let distanceY = point.top - closestY\n let distance = Math.sqrt(distanceX * distanceX + distanceY * distanceY)\n\n return distance\n}\n\nexport function getDistanceFromPointToPoint(point: Point, other: Point) {\n return Math.sqrt(Math.pow(point.left - other.left, 2) + Math.pow(point.top - other.top, 2))\n}\n\nexport function getLineSegmentCenter(line: Line) {\n return {\n left: line[0].left + (line[1].left - line[0].left) / 2,\n top: line[0].top + (line[1].top - line[0].top) / 2\n }\n}\n\nexport function getRectCenter(rect: Rect) {\n return {\n left: rect.left + rect.width / 2,\n top: rect.top + rect.height / 2\n }\n}\n\nexport function getRoundedRectPath(\n x: number,\n y: number,\n width: number,\n height: number,\n topLeftRadius: number,\n topRightRadius: number,\n bottomLeftRadius: number,\n bottomRightRadius: number\n): string {\n const minSize = Math.min(width, height)\n topLeftRadius = Math.min(topLeftRadius, minSize / 2)\n topRightRadius = Math.min(topRightRadius, minSize / 2)\n bottomLeftRadius = Math.min(bottomLeftRadius, minSize / 2)\n bottomRightRadius = Math.min(bottomRightRadius, minSize / 2)\n\n const path =\n `M ${x + topLeftRadius},${y} ` +\n `L ${x + width - topRightRadius},${y} ` +\n `Q ${x + width},${y} ${x + width},${y + topRightRadius} ` +\n `L ${x + width},${y + height - bottomRightRadius} ` +\n `Q ${x + width},${y + height} ${x + width - bottomRightRadius},${y + height} ` +\n `L ${x + bottomLeftRadius},${y + height} ` +\n `Q ${x},${y + height} ${x},${y + height - bottomLeftRadius} ` +\n `L ${x},${y + topLeftRadius} ` +\n `Q ${x},${y} ${x + topLeftRadius},${y} ` +\n `Z`\n\n return path\n}\n","import { intersectRect, Rect } from './rect.js'\n\nexport function getActiveElement(doc: Document | ShadowRoot = document): HTMLElement {\n if (doc.activeElement.shadowRoot) {\n return getActiveElement(doc.activeElement.shadowRoot)\n } else {\n return doc.activeElement as HTMLElement\n }\n}\n\nvar willScroll: ReturnType\nexport function scrollForTogglingCard(element: HTMLElement) {\n clearTimeout(willScroll)\n\n const wrapper = document.querySelector('#overflow-wrapper') as HTMLElement\n const modalContent = document.querySelector('#entity-modal-body-wrapper')\n if (!wrapper) return\n const footer = document.querySelector('#fixed-footer') as HTMLElement\n\n const expansion = element.parentNode.querySelector('.entity-overlay-lower, .entity-overlay-form') as HTMLElement\n const entity = (expansion as HTMLElement) || element\n const scrollOffset = 24\n const headerHeight = 60\n const footerHeight = footer?.offsetHeight || 0\n const expansionBottom = entity.getBoundingClientRect().bottom\n const scrollableBottom =\n wrapper.getBoundingClientRect().bottom - (parseInt(window.getComputedStyle(wrapper).paddingBottom) || 0)\n const overflow = expansionBottom + scrollOffset - scrollableBottom\n const target = expansion || element\n const targetHeight = target.getBoundingClientRect().height\n const targetBottom = target.getBoundingClientRect().bottom\n const screenBottom = modalContent ? modalContent.getBoundingClientRect().bottom : window.innerHeight\n wrapper.style.transition = 'none'\n wrapper.style.paddingBottom = Math.max(0, overflow) + 'px'\n if (expansion) {\n const el = document.createElement('div')\n el.className = 'spacer'\n el.style.position = 'absolute'\n el.style.top = '100%'\n el.style.width = scrollOffset + 'px'\n el.style.height = scrollOffset + 'px'\n expansion.appendChild(el)\n }\n\n if (element.getBoundingClientRect().top < headerHeight || targetHeight > window.innerHeight - headerHeight) {\n element.style.scrollMargin = headerHeight + scrollOffset + 'px'\n element.scrollIntoView({\n block: 'start',\n behavior: 'smooth'\n })\n } else if (targetBottom > screenBottom) {\n target.style.scrollMargin = scrollOffset + (modalContent ? 0 : footerHeight) + 'px'\n target.scrollIntoView({\n block: 'end',\n behavior: 'smooth'\n })\n }\n return () => {\n expansion?.querySelector('.spacer')?.remove()\n wrapper.style.transition = ''\n wrapper.style.paddingBottom = '0px'\n willScroll = setTimeout(() => {\n // hack to make browser re-set its scroll position target.\n // otherwise if page shrinks, browser still considers its position past the limit\n document.getElementById('root').scrollBy({\n top: -0.01\n })\n }, 500)\n }\n}\n\nexport function isInDocument(element: ParentNode) {\n var currentElement = element\n while (currentElement && currentElement.parentNode) {\n const parent = currentElement.parentNode\n if (parent.nodeType == Node.DOCUMENT_NODE) {\n return true\n } else if (parent instanceof ShadowRoot) {\n currentElement = parent.host\n } else {\n currentElement = parent\n }\n }\n return false\n}\n\nexport function documentPositionComparator(a: HTMLElement, b: HTMLElement) {\n if (a === b) {\n return 0\n }\n var position = a.compareDocumentPosition(b)\n if (position & Node.DOCUMENT_POSITION_FOLLOWING || position & Node.DOCUMENT_POSITION_CONTAINED_BY) {\n return -1\n } else if (position & Node.DOCUMENT_POSITION_PRECEDING || position & Node.DOCUMENT_POSITION_CONTAINS) {\n return 1\n } else {\n return 0\n }\n}\n\nexport function getClippedRectangle(\n element: HTMLElement,\n getBoundingClientRect: (element: HTMLElement, rect?: Rect) => Rect,\n box = getBoundingClientRect(element) as Rect\n) {\n box = { top: box.top, left: box.left, width: box.width, height: box.height }\n for (let p = element; p; p = p.parentElement) {\n if (p.classList.contains('editor-content')) {\n break\n }\n if (window.getComputedStyle(p).overflow != 'visible') {\n box = intersectRect(box, getBoundingClientRect(p) as Rect)\n }\n }\n return box\n}\n\nexport interface Measurement {\n rect: Rect\n parentRect: Rect\n rects: Rect[]\n element: HTMLElement\n styles: CSSStyleDeclaration\n}\n\nexport function measureElement(\n element: HTMLElement,\n getBoundingClientRect?: (element: HTMLElement, rect?: Rect) => Rect,\n initialRectangle?: Rect,\n clip = true\n): Measurement {\n if (!element || !isInDocument(element)) return null\n const rect = clip\n ? getClippedRectangle(element, getBoundingClientRect, initialRectangle)\n : getBoundingClientRect(element)\n const parentRect = getClippedRectangle(element.parentElement, getBoundingClientRect, initialRectangle)\n return {\n element,\n parentRect,\n rect: rect,\n rects: Array.from(element.getClientRects()).map((r) =>\n getClippedRectangle(element, getBoundingClientRect, getBoundingClientRect(element, r))\n ),\n styles: window.getComputedStyle(element)\n }\n}\n","const purple = 'var(--chakra-colors-primary-300)'\nimport { Box, Icon, IconButton } from '@chakra-ui/react'\nimport type * as CK from '@sitecore-feaas/ckeditor5'\nimport { mdiArrowUpDown, mdiChevronDown } from '@mdi/js'\nimport { cloneDeep, CSS, isDeepEquals, mergeDeep, SDKModel, Style, StylesheetModel } from '@sitecore-feaas/sdk'\nimport { useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react'\nimport {\n Point,\n Rect,\n offsetRect,\n isPointInsideRectangle,\n getDistanceFromPointToPoint,\n getDistanceFromPointToLine,\n getLineSegmentCenter,\n getDistanceFromPointToRect\n} from '../../../utils/rect.js'\nimport { useSDK } from '../../../hooks/useData.js'\n\ntype Line = [start: Point, end: Point]\n\nconst cursors = {\n ne: 'nesw-resize',\n sw: 'nesw-resize',\n se: 'nwse-resize',\n nw: 'nwse-resize',\n n: 'ns-resize',\n s: 'ns-resize',\n w: 'ew-resize',\n e: 'ew-resize',\n c: 'grab',\n add: 'cell'\n}\ntype Anchors = {\n ne: Point\n nw: Point\n se: Point\n sw: Point\n c: Point\n n: Line\n s: Line\n w: Line\n e: Line\n add?: null\n}\ntype Anchor = {\n index: number\n distance: number\n inside: boolean\n type: keyof Anchors\n point: Point\n}\ntype Interaction = {\n anchor: Anchor\n point: Point\n type: 'resizing' | 'moving' | 'adding'\n}\nexport interface Grid extends Style.Rule<'grid'> {}\nexport class Grid {\n constructor(rule?: Style.Rule<'grid'>) {\n if (rule) this.set(rule)\n }\n set(rule: Partial>) {\n Object.assign(this, rule)\n return this\n }\n snapshotted: Grid\n snapshot() {\n this.snapshotted = this.clone()\n return this.snapshotted\n }\n\n clone(): Grid {\n return new Grid(cloneDeep(this))\n }\n\n static forSections(\n stylesheet: StylesheetModel,\n grids: Grid[],\n sections: CK.ModelElement[],\n customStyles: Style.Rule[]\n ) {\n return sections.map((section) => {\n const gridStyle =\n Style.Set.getContextElementAspect(stylesheet.rules, section, 'grid', customStyles) ||\n Style.Rule({ type: 'grid' })\n const grid = grids.find((grid) => grid.context == section) || new Grid()\n grid.set(gridStyle)\n grid.context = section\n return grid\n })\n }\n\n getHiddenProps(): string[] {\n return ['library', 'libraryId', 'stylesheet']\n }\n columnWidth: number\n offsetLeft: number\n offsetTop: number\n paddingTop: number\n paddingBottom: number\n paddingLeft: number\n paddingRight: number\n rows: number[] = []\n columns: number[] = []\n rowGap: number\n columnGap: number\n spacing: Style.Rule<'spacing'>\n outer: Rect\n inner: Rect\n\n rowHeight: number\n\n context: Style.Context\n commit(isFinal: boolean) {}\n render(isFinal: boolean) {}\n\n compute(rect: Rect, style: CSSStyleDeclaration, point: Point) {\n this.outer = rect\n this.offsetLeft = point.left\n this.offsetTop = point.top\n this.paddingLeft = parseFloat(style['paddingLeft'])\n this.paddingRight = parseFloat(style['paddingRight'])\n this.paddingTop = parseFloat(style['paddingTop'])\n this.paddingBottom = parseFloat(style['paddingBottom'])\n this.inner = {\n top: rect.top + this.paddingTop,\n left: rect.left + this.paddingLeft,\n width: rect.width - this.paddingRight - this.paddingLeft,\n height: rect.height - this.paddingBottom - this.paddingTop\n }\n this.columnGap = parseFloat(style['columnGap'])\n this.rowGap = parseFloat(style['rowGap'])\n this.rows = style['gridTemplateRows'].split(/\\s/g).map((v) => parseFloat(v))\n this.columns = style['gridTemplateColumns'].split(/\\s/g).map((v) => parseFloat(v))\n }\n\n getCell({ left, top }: Point, bias: Anchor['type'] = 'c'): Style.Grid.Cell {\n left -= this.paddingLeft - this.columnGap / 2\n top -= this.paddingTop - this.rowGap / 2\n if (bias.includes('n')) top++\n if (bias.includes('s')) top--\n if (bias.includes('w')) left++\n if (bias.includes('e')) left--\n for (var row = 0; row < this.rows.length; row++) {\n const oy = this.rowGap + this.rows[row]\n if (top > oy) {\n top -= oy\n } else {\n break\n }\n }\n for (var column = 0; column < this.columns.length; column++) {\n const ox = this.columnGap + this.columns[column]\n if (left > ox) {\n left -= ox\n } else {\n break\n }\n }\n\n column++\n row++\n\n return { column: Math.min(this.columns.length, column), row: Math.min(this.rows.length, row) }\n }\n\n getSpan(from: Point, to: Point) {\n to ||= from\n const min = {\n left: Math.min(from.left, to.left),\n top: Math.min(from.top, to.top)\n }\n const max = {\n left: Math.max(from.left, to.left),\n top: Math.max(from.top, to.top)\n }\n const m = this.getCell(max)\n return {\n from: this.getCell(min),\n to: {\n row: m.row + 1,\n column: m.column + 1\n }\n }\n }\n\n getColumnIndex(column: number) {\n if (column < 0) {\n return this.columns.length + column + 2\n }\n return column\n }\n getRowIndex(row: number) {\n if (row < 0) {\n return this.rows.length + row + 2\n }\n return row\n }\n\n getCellPoint(cell: Style.Grid.Cell) {\n const c = this.getColumnIndex(cell.column)\n const r = this.getRowIndex(cell.row)\n var left = this.paddingLeft\n var top = this.paddingTop\n for (var column = 0; column < Math.min(c - 1, this.columns.length); column++) {\n left += this.columnGap + this.columns[column]\n }\n for (var row = 0; row < Math.min(r - 1, this.rows.length); row++) {\n top += this.rowGap + this.rows[row] * Math.min(1, r - 1 - row)\n }\n return { top, left }\n }\n\n getSpanRect(from: Style.Grid.Cell, to: Style.Grid.Cell): Rect {\n const f = this.getCellPoint(from)\n const t = this.getCellPoint(to)\n return {\n ...f,\n width: t.left - f.left - this.columnGap,\n height: t.top - f.top - this.rowGap\n }\n }\n\n getItemsInCell(cell: Style.Grid.Cell) {\n return this.props.items.filter((item) => {\n const i = this.getNormalizedItem(item)\n return (\n i.from.column <= cell.column &&\n cell.column <= i.to.column - 1 &&\n i.from.row <= cell.row &&\n cell.row <= i.to.row - 1\n )\n })\n }\n\n getOverlappingItems(cell: Style.Grid.Item) {\n const c = this.getNormalizedItem(cell)\n return this.props.items.filter((item) => {\n const i = this.getNormalizedItem(item)\n if (\n c.from.column >= i.to.column ||\n c.to.column <= i.from.column ||\n c.from.row >= i.to.row ||\n c.to.row <= i.from.row\n ) {\n return false\n }\n return true\n })\n }\n\n getRectAnchors(rect: Rect): Anchors {\n return {\n nw: { top: rect.top, left: rect.left },\n ne: { top: rect.top, left: rect.left + rect.width },\n sw: { top: rect.top + rect.height, left: rect.left },\n se: { top: rect.top + rect.height, left: rect.left + rect.width },\n c: null, //{ top: rect.top + rect.height / 2, left: rect.left + rect.width / 2 },\n n: [\n { top: rect.top, left: rect.left },\n { top: rect.top, left: rect.left + rect.width }\n ],\n s: [\n { top: rect.top + rect.height, left: rect.left },\n { top: rect.top + rect.height, left: rect.left + rect.width }\n ],\n w: [\n { top: rect.top, left: rect.left },\n { top: rect.top + rect.height, left: rect.left }\n ],\n e: [\n { top: rect.top, left: rect.left + rect.width },\n { top: rect.top + rect.height, left: rect.left + rect.width }\n ]\n }\n }\n getClosestAnchor(point: Point, rect: Rect, target: 'edge' | 'corner' | null) {\n const anchors = this.getRectAnchors(rect)\n const inside = isPointInsideRectangle(point, rect)\n return Object.keys(anchors)\n .map((type: keyof Anchors) => {\n const anchor = anchors[type as keyof Anchors]\n if (!anchor) return\n if ('top' in anchor) {\n if (target && target != 'corner') return\n var distance = getDistanceFromPointToPoint(point, anchor)\n var p = anchor\n } else {\n if (target && target == 'corner') return\n var p = getLineSegmentCenter(anchor)\n var distance = getDistanceFromPointToLine(point, anchor)\n }\n return { type, distance, inside, point: p }\n })\n .filter(Boolean)\n .sort((a, b) => a.distance - b.distance)[0]\n }\n\n getAnchor(point: Point, preferIndex?: number) {\n const threshold = preferIndex == null ? 5 : 5\n const rects = this.props.items.map(({ from, to }) => this.getSpanRect(from, to))\n const corner = rects\n .map((anchor, index) => ({ index, ...this.getClosestAnchor(point, anchor, 'corner') }))\n .filter((anchor) => {\n return anchor.inside ? anchor.distance < threshold : anchor.distance < 5\n })\n const edge = rects\n .map((anchor, index) => ({ index, ...this.getClosestAnchor(point, anchor, 'edge') }))\n .filter((anchor) => {\n return anchor.inside ? anchor.distance < threshold : anchor.distance < 5\n })\n\n const match = rects\n .map((rect, index) => {\n return isPointInsideRectangle(point, rect)\n ? {\n index: index,\n type: 'c' as const,\n inside: true,\n point: point,\n // select smaller rect under the cursor\n distance: getDistanceFromPointToRect(point, rect) || rect.width * rect.height\n //getDistanceFromPointToPoint(point, { top: rect.top + rect.height / 2, left: rect.left + rect.width / 2 })\n }\n : null\n })\n .filter(Boolean)\n const groups = [corner, edge, match]\n const candidates = groups.flat().sort((a, b) => {\n return (\n Number(b.index == preferIndex) - Number(a.index == preferIndex) ||\n groups.findIndex((g) => g.includes(a)) - groups.findIndex((g) => g.includes(b)) ||\n a.distance - b.distance ||\n Number(b.inside) - Number(a.inside) ||\n getDistanceFromPointToRect(point, rects[a.index]) - getDistanceFromPointToRect(point, rects[b.index])\n )\n })\n\n return candidates[0]\n }\n\n filterAnchor(anchor: Anchor, preferIndex: number) {\n if (preferIndex != null && (anchor?.type == 'c' || anchor?.type == 'add')) return null\n return anchor\n }\n\n getDenormalizedItem(item: Style.Grid.Item): Style.Grid.Item {\n const center = Math.floor((this.props.rows.length - 1) / 2)\n return {\n ...item,\n to: {\n column: item.to.column,\n row: item.to.row > center ? item.to.row - this.props.rows.length - 2 : item.to.row\n }\n }\n }\n getNormalizedItem(old: Style.Grid.Item) {\n return {\n ...old,\n ...this.getNormalizedSpan(old)\n }\n }\n getNormalizedSpan(old: Style.Grid.CellSpan) {\n const from = {\n column: Math.min(old.from.column, old.to.column),\n row: Math.min(old.from.row, old.to.row)\n }\n const to = {\n column: Math.max(old.from.column, old.to.column),\n row: Math.max(old.from.row, old.to.row)\n }\n return {\n from: {\n column: this.getColumnIndex(from.column),\n row: this.getRowIndex(from.row)\n },\n to: {\n column: this.getColumnIndex(to.column),\n row: this.getRowIndex(to.row)\n }\n }\n }\n\n clipPoint(point: Point) {\n return {\n left: Math.max(0, Math.min(this.outer.width, point.left - this.outer.left)),\n top: Math.max(0, Math.min(this.outer.height, point.top - this.outer.top))\n }\n }\n\n interacting = {} as Interaction\n\n interact(interaction: Partial) {\n Object.assign(this.interacting, interaction)\n return true\n }\n\n hoveredAnchor: Anchor\n preventAdding: boolean\n\n matchAnchor(point: Point, preferIndex?: number) {\n return this.filterAnchor(\n this.getAnchor(this.clipPoint(point), preferIndex) ||\n (isPointInsideRectangle(point, this.inner) && !this.preventAdding\n ? {\n type: 'add',\n index: this.props.items.length,\n distance: 0,\n inside: false,\n point: null\n }\n : null),\n preferIndex\n )\n }\n\n recognize(point: Point, preferIndex?: number) {\n if (this.interacting.anchor) return this\n if (getDistanceFromPointToPoint(point, this.interacting.point) < 20) {\n return\n }\n if (this.interacting.type == 'resizing') {\n this.interacting.anchor = {\n type: 'n',\n point: null,\n index: null,\n distance: 0,\n inside: true\n }\n } else {\n this.interacting.anchor = this.matchAnchor(this.interacting.point, preferIndex)\n if (!this.interacting.anchor) return\n }\n this.snapshot()\n this.rowHeight = this.getRowHeight()\n return this.set({\n props: {\n ...this.props,\n rows: this.rows.map((m, i) => ({\n min: { unit: 'px' as const, value: this.rows[i] },\n max: { unit: 'px' as const, value: this.rows[i] }\n }))\n }\n })\n }\n\n getRowHeight() {\n return (this.props.rows.find((r) => !this.isRowExpandable(r)) || this.props.rows[0])?.min?.value ?? 24\n }\n\n modify(point: Point) {\n var props = JSON.stringify(this.props)\n if (this.interacting.type == 'resizing') {\n const delta = this.interacting.point.top - point.top\n const rowHeight = this.getRowHeight() + this.rowGap\n const deltaRows = -Math.floor(delta / rowHeight)\n this.setRowNumber(this.snapshotted.props.rows.length + deltaRows)\n } else {\n if (this.interacting.anchor?.type == 'add') {\n this.addItem(this.clipPoint(point), this.clipPoint(this.interacting.point))\n } else {\n this.resizeItem(this.clipPoint(point), this.clipPoint(this.interacting.point), this.interacting.anchor)\n }\n }\n return JSON.stringify(this.props) != props\n }\n\n isRowExpandable(row: Style.Grid.CellSize) {\n return row.max == null || row.max.unit == 'fr'\n }\n\n modifiedRows: number[] = []\n setRowNumber(rowNumber: number, snapshot = this.snapshotted || this) {\n const deltaRows = rowNumber - snapshot.props.rows.length\n const rowHeight = this.getRowHeight()\n const centerRow = (snapshot.props.rows.length - 1) / 2 + 1\n const before = snapshot.props.rows.slice(0, centerRow)\n const after = snapshot.props.rows.slice(centerRow)\n const middle: Style.Grid.CellSize[] = []\n const items = snapshot.props.items.map((item) => this.getNormalizedItem(item))\n\n const centeredExpansion = [Math.floor(centerRow), Math.ceil(centerRow)].some((index) => {\n return this.isRowExpandable(snapshot.props.rows[index - 1])\n })\n this.modifiedRows = []\n const delta = Math.max(-(snapshot.props.rows.length - 1), deltaRows)\n if (deltaRows > 0) {\n for (var i = 0; i < deltaRows; i++) {\n this.modifiedRows.push(before.length + i)\n middle.push(\n Style.Grid.CellSize({\n min: rowHeight,\n max: rowHeight\n })\n )\n }\n } else if (deltaRows < 0) {\n const removedAbove = Math.floor(-delta / 2)\n const removedBelow = Math.ceil(-delta / 2)\n before.splice(before.length - removedAbove)\n after.splice(0, removedBelow)\n }\n for (var item of items) {\n if (item.from.row > centerRow) {\n item.from.row += delta\n }\n if (item.to.row >= centerRow + 1) {\n item.to.row += delta\n }\n item.from.row = Math.max(1, Math.min(snapshot.props.rows.length + delta - 1, item.from.row))\n item.to.row = Math.max(item.from.row + 1, Math.min(snapshot.props.rows.length + delta + 1, item.to.row))\n }\n const rows = before.concat(middle).concat(after)\n\n if (centeredExpansion) {\n for (var r = 0; r < rows.length; r++) {\n const row = { ...rows[r] }\n if (this.isRowExpandable(row)) row.max = { unit: row.min.unit, value: row.min.value }\n if (r == Math.floor(rows.length / 2)) {\n row.max = null\n }\n rows[r] = row\n }\n }\n\n return this.set({\n props: {\n ...this.props,\n items: this.props.items.map((item, index) =>\n this.getNormalizedItem({\n order: items[index].order,\n from: {\n column: item.from.column,\n row: items[index].from.row\n },\n to: {\n column: item.to.column,\n row: items[index].to.row\n }\n })\n ),\n rows: rows\n }\n })\n }\n\n setRowHeight(length: Style.Length) {\n return this.set({\n props: {\n ...this.props,\n rows: this.props.rows.map((r) => ({\n min: length,\n max: this.isRowExpandable(r) ? r.max : length\n }))\n }\n })\n }\n\n modifiedColumns: number[] = []\n setColumnNumber(columnNumber: number, snapshot = this.snapshotted || this) {\n const deltaColumns = columnNumber - snapshot.props.columns.length\n const centerColumn = (snapshot.props.columns.length - 1) / 2 + 1\n const before = snapshot.props.columns.slice(0, centerColumn)\n const after = snapshot.props.columns.slice(centerColumn)\n const middle: Style.Grid.CellSize[] = []\n const items = snapshot.props.items.map((item) => this.getNormalizedItem(item))\n this.modifiedColumns = []\n const delta = Math.max(-(snapshot.props.columns.length - 1), deltaColumns)\n if (deltaColumns > 0) {\n for (var i = 0; i < deltaColumns; i++) {\n this.modifiedColumns.push(before.length + i)\n middle.push(\n Style.Grid.CellSize({\n min: 0,\n max: { unit: 'fr', value: 1 }\n })\n )\n }\n } else if (deltaColumns < 0) {\n const removedAbove = Math.floor(-delta / 2)\n const removedBelow = Math.ceil(-delta / 2)\n before.splice(before.length - removedAbove)\n after.splice(0, removedBelow)\n }\n for (var item of items) {\n if (item.from.column > centerColumn) {\n item.from.column += delta\n }\n if (item.to.column >= centerColumn + 1) {\n item.to.column += delta\n }\n item.from.column = Math.max(1, Math.min(snapshot.props.columns.length + delta - 1, item.from.column))\n item.to.column = Math.max(\n item.from.column + 1,\n Math.min(snapshot.props.columns.length + delta + 1, item.to.column)\n )\n }\n const columns = before.concat(middle).concat(after)\n\n return this.set({\n props: {\n ...this.props,\n items: this.props.items.map((item, index) =>\n this.getNormalizedItem({\n order: items[index].order,\n from: {\n column: items[index].from.column,\n row: item.from.row\n },\n to: {\n column: items[index].to.column,\n row: item.to.row\n }\n })\n ),\n columns: columns\n }\n })\n }\n\n getExpandableRows() {\n return (\n !this.interacting.type || this.interacting.type == 'resizing' || (this.interacting.type && !this.snapshotted)\n ? this.props.rows\n : this.snapshotted.props.rows\n )\n .map((r, i) => {\n return this.isRowExpandable(r) ? i : null\n })\n .filter((i) => i != null)\n }\n\n addItem(point: Point, start: Point) {\n const item = {\n ...this.getSpan(point, start),\n order: 0\n }\n const index =\n this.snapshotted.props.items.length == this.props.items.length\n ? this.props.items.length\n : this.props.items.length - 1\n return this.set({\n props: {\n ...this.props,\n items: this.snapshotted.props.items\n .slice(0, index)\n .concat(item)\n .map((item) => {\n return this.getNormalizedItem(item)\n })\n }\n })\n }\n\n resizeItem(point: Point, start: Point, anchor: Anchor) {\n const old = this.getNormalizedItem(this.snapshotted.props.items[anchor.index])\n const from = this.getCell(anchor.point, anchor.type)\n const to = this.getCell(point, anchor.type)\n const delta = { column: to.column - from.column, row: to.row - from.row }\n var item = old\n\n const dir = anchor.type\n\n for (var c = 0; c != delta.column; ) {\n const nc = c + (delta.column > 0 ? 1 : -1)\n if (\n (old.to.column + nc == old.from.column && dir.match(/e/)) ||\n (old.to.column == old.from.column + nc && dir.match(/w/)) ||\n (old.from.column + nc < 1 && !dir.includes('e')) ||\n (old.to.column + nc > this.props.columns.length + 1 && !dir.includes('w'))\n )\n break\n c = nc\n }\n for (var r = 0; r != delta.row; ) {\n const nr = r + (delta.row > 0 ? 1 : -1)\n if (\n (old.to.row + nr == old.from.row && dir.match(/s/)) ||\n (old.to.row == old.from.row + nr && dir.match(/n/)) ||\n (old.from.row + nr < 1 && !dir.includes('s')) ||\n (old.to.row + nr > this.props.rows.length + 1 && !dir.includes('n'))\n )\n break\n r = nr\n }\n\n if (dir == 'c' || dir == 'n' || dir == 'nw' || dir == 'ne') {\n item.from.row += r\n }\n if (dir == 'c' || dir == 's' || dir == 'sw' || dir == 'se') {\n item.to.row += r\n }\n if (dir == 'c' || dir == 'e' || dir == 'ne' || dir == 'se') {\n item.to.column += c\n }\n if (dir == 'c' || dir == 'w' || dir == 'nw' || dir == 'sw') {\n item.from.column += c\n }\n return this.set({\n props: {\n ...this.props,\n items: this.snapshotted.props.items\n .slice(0, Math.max(0, anchor.index))\n .concat(item)\n .concat(this.snapshotted.props.items.slice(anchor.index + 1))\n }\n })\n }\n\n reset() {\n this.interacting = {} as Interaction\n }\n\n abort() {\n this.reset()\n }\n\n finalize() {\n this.modifiedRows = []\n const expandableRows = this.getExpandableRows()\n return this.set({\n props: {\n ...this.props,\n rows: this.props.rows.map((row, r) => {\n const isEmpty = this.props.columns.every((col, c) => {\n return this.getItemsInCell({ row: r + 1, column: c + 1 }).length == 0\n })\n const isExpandable = expandableRows.includes(r)\n return {\n min: {\n value: this.rowHeight,\n unit: 'px'\n },\n max: isExpandable\n ? null /*(isEmpty ? { unit: 'fr', value: 0.000001 } : { unit: 'fr', value: 1 })*/\n : row.max\n }\n })\n }\n })\n }\n\n getCacheChecksum() {\n return JSON.stringify([\n this.props.items,\n this.columns,\n this.rows,\n this.rowGap,\n this.columnGap,\n this.outer,\n this.inner,\n this.props.rows\n ])\n }\n}\n\nfunction Span({\n grid,\n from,\n to,\n offset = 0,\n corners,\n isBackground,\n isHovered: isHovered,\n id,\n ...props\n}: Style.Grid.CellSpan & {\n isHovered: boolean\n isBackground: boolean\n grid: Grid\n corners: string[]\n offset?: number\n} & Record) {\n const rect = offsetRect(grid.getSpanRect(from, to), offset)\n const c = 6\n const p = 12\n return (\n <>\n \n \n\n \n\n \n\n \n \n \n \n \n \n \n \n \n <>\n \n \n \n \n \n \n \n \n \n \n )\n}\nexport function ChromeGrid({\n grid: givenGrid,\n activeIndex,\n index,\n isVisible,\n isActive,\n isDisabled,\n pointer\n}: {\n index?: number\n activeIndex?: number\n grid: Grid\n isVisible: boolean\n isActive: boolean\n isDisabled: boolean\n pointer?: Point\n}) {\n const sdk = useSDK()\n const [, setAnchor] = useState(null)\n const [hovered, setHovered] = useState(null)\n const ref = useRef<{ grid: Grid; activeIndex: number }>({\n grid: null,\n activeIndex: null\n })\n if (givenGrid) ref.current.grid = givenGrid\n const grid = ref.current.grid\n ref.current.activeIndex = activeIndex\n\n const interaction = grid?.interacting\n const currentAnchor = grid?.interacting.anchor || grid?.hoveredAnchor\n const cursor = isVisible\n ? interaction.type == 'moving' && interaction.anchor?.type == 'c'\n ? 'grabbing'\n : currentAnchor\n ? cursors[currentAnchor.type as keyof typeof cursors]\n : ''\n : ''\n\n const currentIndex = currentAnchor?.index\n const offset = 1.5\n const allCorners = ['ne', 'nw', 'se', 'sw']\n\n sdk.log('grid rendered', grid, grid.interacting, isDisabled)\n\n useEffect(() => {\n var timeout: ReturnType\n function recognizeAnchor(pointer: Point) {\n const grid = ref.current.grid\n if (!grid.inner) return\n if (!grid.interacting.type) {\n const isInside = isPointInsideRectangle(pointer, offsetRect(grid.outer, -10))\n const hoveredAnchor = isInside ? grid.matchAnchor(pointer, ref.current.activeIndex) : null\n if (grid.hoveredAnchor?.type != hoveredAnchor?.type || grid.hoveredAnchor?.index != hoveredAnchor?.index)\n grid.hoveredAnchor = hoveredAnchor\n const hoveredCell = !grid.interacting.anchor && isInside ? grid.getCell(grid.clipPoint(pointer)) : null\n\n if (isActive) setHovered((hovered) => (isDeepEquals(hoveredCell, hovered) ? hovered : hoveredCell))\n setAnchor((anchor) => (isDeepEquals(grid.hoveredAnchor, anchor) ? anchor : grid.hoveredAnchor))\n }\n }\n const onPointerDown = ({ target, clientX: left, clientY: top }: PointerEvent) => {\n clearTimeout(timeout)\n if ((target as HTMLElement).closest('.ui')) return\n recognizeAnchor(pointer || { top, left })\n }\n const onPointerMove = ({ clientX: left, clientY: top }: PointerEvent) => {\n clearTimeout(timeout)\n timeout = setTimeout(\n () => {\n recognizeAnchor(pointer || { top, left })\n },\n grid.interacting.type ? 1 : 32\n )\n }\n const onPointerUp = (e: PointerEvent) => {\n setTimeout(() => onPointerMove(e), 100)\n }\n const onSelectStart = (e: Event) => {\n const grid = ref.current.grid\n const target = e.target as HTMLElement\n if ((grid.hoveredAnchor || grid.interacting.anchor) && ref.current.activeIndex != null) {\n console.log('prevent selection because of anchor', e, grid.hoveredAnchor)\n e.preventDefault()\n }\n }\n const onDragStart = (e: Event) => {\n const grid = ref.current.grid\n if (grid.hoveredAnchor) {\n console.log('prevent drag because of anchor', grid.hoveredAnchor)\n e.preventDefault()\n }\n }\n clearTimeout(timeout)\n if (!isDisabled) {\n document.addEventListener('pointermove', onPointerMove)\n document.addEventListener('pointerdown', onPointerDown, { capture: true })\n document.addEventListener('pointerup', onPointerUp)\n document.addEventListener('selectstart', onSelectStart, { capture: true })\n document.addEventListener('selectionchange', onSelectStart, { capture: true })\n document.addEventListener('dragstart', onDragStart, { capture: true })\n }\n return () => {\n setHovered(null)\n setAnchor(null)\n document.removeEventListener('pointermove', onPointerMove)\n document.removeEventListener('pointerdown', onPointerDown, { capture: true })\n document.removeEventListener('pointerup', onPointerUp)\n document.removeEventListener('selectstart', onSelectStart, { capture: true })\n document.removeEventListener('selectionchange', onSelectStart, { capture: true })\n document.removeEventListener('dragstart', onDragStart, { capture: true })\n }\n }, [isActive, isDisabled, activeIndex])\n\n useEffect(() => {\n document.body.style.cursor = cursor\n return () => {\n document.body.style.cursor = ''\n }\n }, [isActive, cursor])\n\n const button = useMemo(() => {\n return (\n \n \n \n }\n />\n )\n }, [])\n\n return useMemo(() => {\n if (!grid || !grid.outer) return\n\n const boxes = grid.props.items\n\n const strokeWidth = grid.rowGap == 0 || grid.columnGap == 0 ? 0.5 : 1\n\n const middleLine = grid.getCellPoint({ row: grid.rows.length / 2 + 1, column: 0 })\n var y = grid.paddingTop\n const expandableRows = grid.getExpandableRows()\n const cells = grid.rows.map((height, r) => {\n const oy = height + grid.rowGap\n y += oy\n var x = grid.paddingLeft\n return grid.columns.map((width, c) => {\n const ox = width + grid.columnGap\n x += ox\n return (\n \n )\n })\n })\n return (\n <>\n \n \n \n \n {cells}\n \n \n {boxes\n .slice()\n .sort((a, b) => {\n return Number(currentIndex == boxes.indexOf(a)) - Number(currentIndex == boxes.indexOf(b))\n })\n .map((b) => {\n const i = boxes.indexOf(b)\n return (\n c.includes(grid.interacting.anchor.type))\n : []\n }\n id={index + '-' + i}\n key={i}\n isBackground={grid.interacting.anchor?.index != i}\n isHovered={currentIndex == i}\n />\n )\n })}\n {grid.interacting.type == 'resizing' && grid.modifiedRows.length == 0 && (\n \n )}\n \n \n \n {button}\n \n \n \n )\n }, [grid.hoveredAnchor?.type, grid.hoveredAnchor?.index, isActive, currentIndex, grid.getCacheChecksum()])\n}\n","import {\n mdiArrowExpand,\n mdiArrowExpandHorizontal,\n mdiBorderNoneVariant,\n mdiButtonPointer,\n mdiCalendar,\n mdiCardMultipleOutline,\n mdiCodeTags,\n mdiCreditCardChipOutline,\n mdiFileDocument,\n mdiFormatHeader1,\n mdiFormatHeader2,\n mdiFormatHeader3,\n mdiFormatHeader4,\n mdiFormatHeader5,\n mdiFormatHeader6,\n mdiFormatListBulleted,\n mdiFormatParagraph,\n mdiFormatQuoteOpen,\n mdiImageOutline,\n mdiLink,\n mdiTable,\n mdiTextureBox,\n mdiToyBrick,\n mdiToyBrickOutline,\n mdiVariable,\n mdiVideoOutline,\n mdiViewDashboardOutline\n} from '@mdi/js'\nimport type * as CK from '@sitecore-feaas/ckeditor5'\nimport { Style } from '@sitecore-feaas/sdk'\n\nconst form =\n 'M22 4H3a.5.5 0 0 0-.5.5v6a.5.5 0 0 0 .5.5h19a.5.5 0 0 0 .5-.5v-6A.5.5 0 0 0 22 4ZM14 13H3a.5.5 0 0 0-.5.5v6a.5.5 0 0 0 .5.5h11a.5.5 0 0 0 .5-.5v-6a.5.5 0 0 0-.5-.5Z'\n\nexport function getIconPathByElementName(type: string) {\n switch (type) {\n case '$root':\n return mdiFileDocument\n case 'paragraph':\n return mdiFormatParagraph\n case 'heading1':\n return mdiFormatHeader1\n case 'heading2':\n return mdiFormatHeader2\n case 'heading3':\n return mdiFormatHeader3\n case 'heading4':\n return mdiFormatHeader4\n case 'heading5':\n return mdiFormatHeader5\n case 'heading6':\n return mdiFormatHeader6\n case 'listItem':\n return mdiFormatListBulleted\n case 'card':\n return mdiCardMultipleOutline\n case 'section':\n return mdiViewDashboardOutline\n case 'container':\n return mdiBorderNoneVariant\n case 'blockquote':\n return mdiFormatQuoteOpen\n case 'button':\n return mdiButtonPointer\n case 'inline':\n return mdiCreditCardChipOutline\n case 'media':\n //return mdiMultimedia\n case 'image':\n return mdiImageOutline\n case 'video':\n return mdiVideoOutline\n case 'link':\n return mdiLink\n case 'var':\n return mdiVariable\n case 'table':\n return mdiTable\n case 'embed':\n return mdiCodeTags\n case 'component':\n return mdiToyBrickOutline\n case 'block':\n return mdiTextureBox\n case 'spacer':\n return mdiArrowExpandHorizontal\n case 'time':\n return mdiCalendar\n case 'form':\n return form\n }\n}\n\nexport function getContextIcon(context: CK.ModelElement) {\n const definition = Style.Context.getDefinition(context)\n return getIconPathByElementName(definition.name)\n}\n","import { Box, BoxProps } from '@chakra-ui/react'\n\nexport const Dot = (props: BoxProps) => (\n \n  \n \n)\n","import { Icon, Flex, Text } from '@chakra-ui/react'\nimport { mdiAlertCircleOutline } from '@mdi/js'\n\nconst FormError = ({ error }: { error?: string }) => {\n return (\n \n \n \n \n \n {error}\n \n \n )\n}\nexport default FormError\n","import { forwardRef, MutableRefObject, ReactElement, ReactNode } from 'react'\nimport { Box, Flex, FormControl, FormControlProps, FormLabel } from '@chakra-ui/react'\nimport { Dot } from './Dot.js'\nimport FormError from './FormError.js'\n\nconst FieldsetField = forwardRef(\n (\n {\n children,\n label,\n isChanged,\n required,\n htmlFor,\n isDisabled,\n error,\n extraProps,\n className,\n labelProps,\n ...props\n }: {\n label?: string | ReactElement\n isChanged?: boolean\n required?: boolean\n children?: ReactNode\n htmlFor?: string\n isDisabled?: boolean\n error?: string\n className?: string\n labelProps?: { [key: string]: any }\n extraProps?: { [key: string]: any }\n } & Omit,\n ref: MutableRefObject\n ) => (\n \n {extraProps?.labelRight && children}\n\n {(label || required || isChanged) && (\n \n \n {label}\n\n {required && (\n \n *\n \n )}\n\n {isChanged && }\n \n \n )}\n\n {!extraProps?.labelRight && children}\n\n {error && }\n \n )\n)\n\nexport default FieldsetField\n","import { Style, VersionModel } from '@sitecore-feaas/sdk'\nimport { ContextType } from 'react'\nimport { createContextProvider } from '../hooks/useContextProvider.js'\nimport type * as CK from '@sitecore-feaas/ckeditor5'\n\nexport const [VersionContext, useVersionContextProvider] = createContextProvider(() => {\n return {\n error: null as Error,\n node: null as HTMLElement,\n breakpoint: null as Style.Rule<'breakpoint'>,\n snapshotStatus: null as VersionModel['status'],\n needsWrite: false,\n version: null as VersionModel,\n openBar: null as 'breakpoint' | 'status'\n }\n}, '__FEAASVersionContext')\nexport type VersionContext = ContextType[0]\nexport type VersionContextSetter = ContextType[1]\n\nexport function saveVersionData(editor: CK.Editor, version: VersionModel) {\n return version.saveData({\n view: editor.getData({ rootName: version.id }),\n model: ''\n })\n}\nexport function commitVersionData(editor: CK.Editor, version: VersionModel) {\n return version.commitData({\n view: editor.getData({ rootName: version.id }),\n model: ''\n })\n}\n","import { ButtonGroup, ButtonGroupProps } from '@chakra-ui/react'\nimport { Children, cloneElement, ReactElement, ReactNode } from 'react'\n\ninterface Props {\n value?: T\n onChange: (index: T) => void\n activeProps?: any\n children?: ReactNode\n inactiveProps?: any\n}\n\nconst ButtonGroupSwitch = (props: Props & Omit) => {\n const {\n value,\n onChange,\n activeProps = { bg: 'primary.100', variant: 'subtle' },\n inactiveProps = { variant: 'minimal' },\n ...more\n } = props\n\n return (\n \n {Children.map(props.children, (child, i) => {\n const reactChild = child as ReactElement\n function getValue() {\n return 'value' in reactChild.props ? reactChild.props.value : i\n }\n return (\n reactChild &&\n cloneElement(\n reactChild,\n getValue() === value\n ? { onClick: () => onChange(getValue()), ...activeProps, 'data-activated': true }\n : { onClick: () => onChange(getValue()), ...inactiveProps }\n )\n )\n })}\n \n )\n}\n\nexport default ButtonGroupSwitch\n","function _interopRequireDefault(obj) {\n return obj && obj.__esModule ? obj : {\n \"default\": obj\n };\n}\nmodule.exports = _interopRequireDefault, module.exports.__esModule = true, module.exports[\"default\"] = module.exports;","'use strict'\n\nmodule.exports = jsx\njsx.displayName = 'jsx'\njsx.aliases = []\nfunction jsx(Prism) {\n ;(function (Prism) {\n var javascript = Prism.util.clone(Prism.languages.javascript)\n var space = /(?:\\s|\\/\\/.*(?!.)|\\/\\*(?:[^*]|\\*(?!\\/))\\*\\/)/.source\n var braces = /(?:\\{(?:\\{(?:\\{[^{}]*\\}|[^{}])*\\}|[^{}])*\\})/.source\n var spread = /(?:\\{*\\.{3}(?:[^{}]|)*\\})/.source\n /**\n * @param {string} source\n * @param {string} [flags]\n */\n function re(source, flags) {\n source = source\n .replace(//g, function () {\n return space\n })\n .replace(//g, function () {\n return braces\n })\n .replace(//g, function () {\n return spread\n })\n return RegExp(source, flags)\n }\n spread = re(spread).source\n Prism.languages.jsx = Prism.languages.extend('markup', javascript)\n Prism.languages.jsx.tag.pattern = re(\n /<\\/?(?:[\\w.:-]+(?:+(?:[\\w.:$-]+(?:=(?:\"(?:\\\\[\\s\\S]|[^\\\\\"])*\"|'(?:\\\\[\\s\\S]|[^\\\\'])*'|[^\\s{'\"/>=]+|))?|))**\\/?)?>/\n .source\n )\n Prism.languages.jsx.tag.inside['tag'].pattern = /^<\\/?[^\\s>\\/]*/\n Prism.languages.jsx.tag.inside['attr-value'].pattern =\n /=(?!\\{)(?:\"(?:\\\\[\\s\\S]|[^\\\\\"])*\"|'(?:\\\\[\\s\\S]|[^\\\\'])*'|[^\\s'\">]+)/\n Prism.languages.jsx.tag.inside['tag'].inside['class-name'] =\n /^[A-Z]\\w*(?:\\.[A-Z]\\w*)*$/\n Prism.languages.jsx.tag.inside['comment'] = javascript['comment']\n Prism.languages.insertBefore(\n 'inside',\n 'attr-name',\n {\n spread: {\n pattern: re(//.source),\n inside: Prism.languages.jsx\n }\n },\n Prism.languages.jsx.tag\n )\n Prism.languages.insertBefore(\n 'inside',\n 'special-attr',\n {\n script: {\n // Allow for two levels of nesting\n pattern: re(/=/.source),\n alias: 'language-javascript',\n inside: {\n 'script-punctuation': {\n pattern: /^=(?=\\{)/,\n alias: 'punctuation'\n },\n rest: Prism.languages.jsx\n }\n }\n },\n Prism.languages.jsx.tag\n ) // The following will handle plain text inside tags\n var stringifyToken = function (token) {\n if (!token) {\n return ''\n }\n if (typeof token === 'string') {\n return token\n }\n if (typeof token.content === 'string') {\n return token.content\n }\n return token.content.map(stringifyToken).join('')\n }\n var walkTokens = function (tokens) {\n var openedTags = []\n for (var i = 0; i < tokens.length; i++) {\n var token = tokens[i]\n var notTagNorBrace = false\n if (typeof token !== 'string') {\n if (\n token.type === 'tag' &&\n token.content[0] &&\n token.content[0].type === 'tag'\n ) {\n // We found a tag, now find its kind\n if (token.content[0].content[0].content === ' 0 &&\n openedTags[openedTags.length - 1].tagName ===\n stringifyToken(token.content[0].content[1])\n ) {\n // Pop matching opening tag\n openedTags.pop()\n }\n } else {\n if (token.content[token.content.length - 1].content === '/>') {\n // Autoclosed tag, ignore\n } else {\n // Opening tag\n openedTags.push({\n tagName: stringifyToken(token.content[0].content[1]),\n openedBraces: 0\n })\n }\n }\n } else if (\n openedTags.length > 0 &&\n token.type === 'punctuation' &&\n token.content === '{'\n ) {\n // Here we might have entered a JSX context inside a tag\n openedTags[openedTags.length - 1].openedBraces++\n } else if (\n openedTags.length > 0 &&\n openedTags[openedTags.length - 1].openedBraces > 0 &&\n token.type === 'punctuation' &&\n token.content === '}'\n ) {\n // Here we might have left a JSX context inside a tag\n openedTags[openedTags.length - 1].openedBraces--\n } else {\n notTagNorBrace = true\n }\n }\n if (notTagNorBrace || typeof token === 'string') {\n if (\n openedTags.length > 0 &&\n openedTags[openedTags.length - 1].openedBraces === 0\n ) {\n // Here we are inside a tag, and not inside a JSX context.\n // That's plain text: drop any tokens matched.\n var plainText = stringifyToken(token) // And merge text with adjacent text\n if (\n i < tokens.length - 1 &&\n (typeof tokens[i + 1] === 'string' ||\n tokens[i + 1].type === 'plain-text')\n ) {\n plainText += stringifyToken(tokens[i + 1])\n tokens.splice(i + 1, 1)\n }\n if (\n i > 0 &&\n (typeof tokens[i - 1] === 'string' ||\n tokens[i - 1].type === 'plain-text')\n ) {\n plainText = stringifyToken(tokens[i - 1]) + plainText\n tokens.splice(i - 1, 1)\n i--\n }\n tokens[i] = new Prism.Token(\n 'plain-text',\n plainText,\n null,\n plainText\n )\n }\n }\n if (token.content && typeof token.content !== 'string') {\n walkTokens(token.content)\n }\n }\n }\n Prism.hooks.add('after-tokenize', function (env) {\n if (env.language !== 'jsx' && env.language !== 'tsx') {\n return\n }\n walkTokens(env.tokens)\n })\n })(Prism)\n}\n","\"use strict\";\n\nvar _interopRequireDefault = require(\"@babel/runtime/helpers/interopRequireDefault\");\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports[\"default\"] = void 0;\n\nvar _jsx = _interopRequireDefault(require(\"refractor/lang/jsx.js\"));\n\n;\nvar _default = _jsx[\"default\"];\nexports[\"default\"] = _default;","'use strict'\n\nmodule.exports = markup\nmarkup.displayName = 'markup'\nmarkup.aliases = ['html', 'mathml', 'svg', 'xml', 'ssml', 'atom', 'rss']\nfunction markup(Prism) {\n Prism.languages.markup = {\n comment: {\n pattern: //,\n greedy: true\n },\n prolog: {\n pattern: /<\\?[\\s\\S]+?\\?>/,\n greedy: true\n },\n doctype: {\n // https://www.w3.org/TR/xml/#NT-doctypedecl\n pattern:\n /\"'[\\]]|\"[^\"]*\"|'[^']*')+(?:\\[(?:[^<\"'\\]]|\"[^\"]*\"|'[^']*'|<(?!!--)|)*\\]\\s*)?>/i,\n greedy: true,\n inside: {\n 'internal-subset': {\n pattern: /(^[^\\[]*\\[)[\\s\\S]+(?=\\]>$)/,\n lookbehind: true,\n greedy: true,\n inside: null // see below\n },\n string: {\n pattern: /\"[^\"]*\"|'[^']*'/,\n greedy: true\n },\n punctuation: /^$|[[\\]]/,\n 'doctype-tag': /^DOCTYPE/i,\n name: /[^\\s<>'\"]+/\n }\n },\n cdata: {\n pattern: //i,\n greedy: true\n },\n tag: {\n pattern:\n /<\\/?(?!\\d)[^\\s>\\/=$<%]+(?:\\s(?:\\s*[^\\s>\\/=]+(?:\\s*=\\s*(?:\"[^\"]*\"|'[^']*'|[^\\s'\">=]+(?=[\\s>]))|(?=[\\s/>])))+)?\\s*\\/?>/,\n greedy: true,\n inside: {\n tag: {\n pattern: /^<\\/?[^\\s>\\/]+/,\n inside: {\n punctuation: /^<\\/?/,\n namespace: /^[^\\s>\\/:]+:/\n }\n },\n 'special-attr': [],\n 'attr-value': {\n pattern: /=\\s*(?:\"[^\"]*\"|'[^']*'|[^\\s'\">=]+)/,\n inside: {\n punctuation: [\n {\n pattern: /^=/,\n alias: 'attr-equals'\n },\n /\"|'/\n ]\n }\n },\n punctuation: /\\/?>/,\n 'attr-name': {\n pattern: /[^\\s>\\/]+/,\n inside: {\n namespace: /^[^\\s>\\/:]+:/\n }\n }\n }\n },\n entity: [\n {\n pattern: /&[\\da-z]{1,8};/i,\n alias: 'named-entity'\n },\n /&#x?[\\da-f]{1,8};/i\n ]\n }\n Prism.languages.markup['tag'].inside['attr-value'].inside['entity'] =\n Prism.languages.markup['entity']\n Prism.languages.markup['doctype'].inside['internal-subset'].inside =\n Prism.languages.markup // Plugin to make entity title show the real entity, idea by Roman Komarov\n Prism.hooks.add('wrap', function (env) {\n if (env.type === 'entity') {\n env.attributes['title'] = env.content.value.replace(/&/, '&')\n }\n })\n Object.defineProperty(Prism.languages.markup.tag, 'addInlined', {\n /**\n * Adds an inlined language to markup.\n *\n * An example of an inlined language is CSS with `