@bbl-digital/snorre 4.2.57 → 4.2.58
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/bundle.js
CHANGED
@@ -40193,11 +40193,11 @@ to {top: 100vh;}
|
|
40193
40193
|
_quill.default.register(Font, true);
|
40194
40194
|
_quill.default.register(_LinkSanitizer.CustomLinkSanitizer, true);
|
40195
40195
|
var _ref = process.env.NODE_ENV === "production" ? {
|
40196
|
-
name: "
|
40197
|
-
styles: "display:flex;gap:
|
40196
|
+
name: "108zowc",
|
40197
|
+
styles: "display:flex;gap:4px;align-items:baseline"
|
40198
40198
|
} : {
|
40199
|
-
name: "
|
40200
|
-
styles: "display:flex;gap:2px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
40199
|
+
name: "1jvq0lt-WordCount",
|
40200
|
+
styles: "display:flex;gap:4px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
40201
40201
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
40202
40202
|
};
|
40203
40203
|
var _ref2 = process.env.NODE_ENV === "production" ? {
|
@@ -40205,7 +40205,7 @@ to {top: 100vh;}
|
|
40205
40205
|
styles: "margin-top:4px"
|
40206
40206
|
} : {
|
40207
40207
|
name: "14rbig3-QuillEditor",
|
40208
|
-
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
40208
|
+
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
40209
40209
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
40210
40210
|
};
|
40211
40211
|
const QuillEditor = ({
|
@@ -40318,7 +40318,7 @@ to {top: 100vh;}
|
|
40318
40318
|
return (0, _jsxRuntime.jsxs)("div", {
|
40319
40319
|
css: _ref,
|
40320
40320
|
children: [(0, _jsxRuntime.jsxs)(_Text.default, {
|
40321
|
-
children: [currentLength, "/", maxlenght]
|
40321
|
+
children: [currentLength, " / ", maxlenght]
|
40322
40322
|
}), (0, _jsxRuntime.jsxs)(_Text.default, {
|
40323
40323
|
small: true,
|
40324
40324
|
color: "textSecondary",
|
@@ -47,11 +47,11 @@ Font.whitelist = ['Source Sans Pro', 'sans-serif'];
|
|
47
47
|
_quill.default.register(Font, true);
|
48
48
|
_quill.default.register(_LinkSanitizer.CustomLinkSanitizer, true);
|
49
49
|
var _ref = process.env.NODE_ENV === "production" ? {
|
50
|
-
name: "
|
51
|
-
styles: "display:flex;gap:
|
50
|
+
name: "108zowc",
|
51
|
+
styles: "display:flex;gap:4px;align-items:baseline"
|
52
52
|
} : {
|
53
|
-
name: "
|
54
|
-
styles: "display:flex;gap:2px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
53
|
+
name: "1jvq0lt-WordCount",
|
54
|
+
styles: "display:flex;gap:4px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
55
55
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
56
56
|
};
|
57
57
|
var _ref2 = process.env.NODE_ENV === "production" ? {
|
@@ -59,7 +59,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
59
59
|
styles: "margin-top:4px"
|
60
60
|
} : {
|
61
61
|
name: "14rbig3-QuillEditor",
|
62
|
-
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
62
|
+
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
63
63
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
64
64
|
};
|
65
65
|
const QuillEditor = ({
|
@@ -172,7 +172,7 @@ const QuillEditor = ({
|
|
172
172
|
return (0, _jsxRuntime.jsxs)("div", {
|
173
173
|
css: _ref,
|
174
174
|
children: [(0, _jsxRuntime.jsxs)(_Text.default, {
|
175
|
-
children: [currentLength, "/", maxlenght]
|
175
|
+
children: [currentLength, " / ", maxlenght]
|
176
176
|
}), (0, _jsxRuntime.jsxs)(_Text.default, {
|
177
177
|
small: true,
|
178
178
|
color: "textSecondary",
|
@@ -47,11 +47,11 @@ Font.whitelist = ['Source Sans Pro', 'sans-serif'];
|
|
47
47
|
_quill.default.register(Font, true);
|
48
48
|
_quill.default.register(_LinkSanitizer.CustomLinkSanitizer, true);
|
49
49
|
var _ref = process.env.NODE_ENV === "production" ? {
|
50
|
-
name: "
|
51
|
-
styles: "display:flex;gap:
|
50
|
+
name: "108zowc",
|
51
|
+
styles: "display:flex;gap:4px;align-items:baseline"
|
52
52
|
} : {
|
53
|
-
name: "
|
54
|
-
styles: "display:flex;gap:2px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
53
|
+
name: "1jvq0lt-WordCount",
|
54
|
+
styles: "display:flex;gap:4px;align-items:baseline;label:WordCount;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AA0LW","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
55
55
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
56
56
|
};
|
57
57
|
var _ref2 = process.env.NODE_ENV === "production" ? {
|
@@ -59,7 +59,7 @@ var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
59
59
|
styles: "margin-top:4px"
|
60
60
|
} : {
|
61
61
|
name: "14rbig3-QuillEditor",
|
62
|
-
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '2px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength}/{maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
62
|
+
styles: "margin-top:4px;label:QuillEditor;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/packages/core/QuillEditor/index.tsx"],"names":[],"mappings":"AAqVa","file":"../../../src/packages/core/QuillEditor/index.tsx","sourcesContent":["/** @jsxImportSource @emotion/react */\nimport { useRef, useState, useEffect, useCallback } from 'react'\nimport Quill, { Range } from 'quill'\nimport { QuillEditorProps } from './models'\nimport { basicToolbar, richToolbar } from './config'\nimport { ErrorWrapper, Label, styles } from './styles'\nimport Text from '../Text'\nimport IconErrorOutline from '../../icons/General/IconErrorOutline'\nimport { useTheme } from '@emotion/react'\nimport IAppTheme from '../../app-shell/theme'\n\nimport { EmitterSource, Op } from 'quill/core'\nimport { CustomLinkSanitizer } from './classes/LinkSanitizer'\nimport DOMPurify from 'dompurify'\nimport 'quill/dist/quill.snow.css'\n\n// Colors\nQuill.register(Quill.import('attributors/style/color') as any, true)\n// Background colors\nQuill.register(Quill.import('attributors/style/background') as any, true)\n//Text direction\nQuill.register(Quill.import('attributors/style/direction') as any, true)\n//Alignment\nQuill.register(Quill.import('attributors/style/align') as any, true)\n// Image\nQuill.register('attributors/style/image', true)\n// Video\nQuill.register('attributors/style/video', true)\n// Strike\nQuill.register('attributors/style/strike', true)\n// Table\nQuill.register('attributors/style/table', true)\n// Underline\nQuill.register('attributors/style/underline', true)\n// Bold\nQuill.register('attributors/style/bold', true)\n// Text sizes\nconst Size = Quill.import('attributors/style/size') as any\nSize.whitelist = ['0.75em', '1em', '1.5em', '2.5em']\nQuill.register(Size, true)\n\n// Font\nconst Font = Quill.import('formats/font') as any\nFont.whitelist = ['Source Sans Pro', 'sans-serif']\nQuill.register(Font, true)\n\nQuill.register(CustomLinkSanitizer, true)\n\nconst QuillEditor: React.FC<QuillEditorProps> = ({\n  height,\n  width,\n  maxHeight,\n  minHeight = '150px',\n  maxWidth,\n  minWidth,\n  placeholder,\n  initialValue,\n  overrideValue,\n  label,\n  readOnly = false,\n  maxlenght = 0,\n  wordCount = true,\n  type = 'basic',\n  customModules = {\n    toolbar: basicToolbar,\n  },\n  formats = [\n    'header',\n    'size',\n    'bold',\n    'italic',\n    'underline',\n    'list',\n    'indent',\n    'link',\n    'image',\n    'align',\n  ],\n  bounds,\n  onChange,\n  onFocus,\n  onBlur,\n  invalidMessage,\n  className,\n  disabled,\n  pasteAsText,\n  resizable,\n}) => {\n  const [quill, setQuill] = useState<Quill | null>(null)\n  const isMounted = useRef(false)\n  const quillRef = useRef<Quill | null>(null)\n  const defaultValueRef = useRef<string | undefined>(initialValue)\n\n  const editorId = useRef<string>(\n    new Date().getTime() + (100 * Math.random()).toFixed(0)\n  )\n\n  const [currentLength, setCurrentLength] = useState(initialValue?.length || 0)\n  const isWordCount = wordCount && maxlenght > 0\n\n  const { alert } = useTheme()\n\n  const toolbars: Record<string, any> = {\n    basic: basicToolbar,\n    rich: richToolbar,\n    custom: customModules.toolbar,\n  }\n\n  const handleBackspaceWithIndentetList = useCallback(\n    (range: any, _context: any, quill: Quill | null) => {\n      if (!quill) return true\n\n      const [line, offset] = quill.getLine(range.index)\n      const formats = line!.formats()\n\n      if (formats.list && formats.indent) {\n        // Check if we're at the beginning of the line\n        if (range.index === range.index - offset) {\n          // Remove indent\n          const indent = parseInt(formats.indent, 10)\n          quill.format('indent', indent ? indent - 1 : false)\n          quill.format('list', formats.list)\n          return false\n        } else if (range.length === 0 && offset > 0) {\n          // If there's content and the cursor is not at the beginning,\n          // let Quill handle it normally\n          return true\n        } else {\n          // Remove indent\n          quill.format('indent', false)\n          quill.format('list', formats.list)\n          return false\n        }\n      }\n      return true\n    },\n    [quill]\n  )\n\n  const modules = {\n    toolbar: toolbars[type || 'basic'],\n    clipboard: {\n      matchVisual: false,\n      onPaste: (e: ClipboardEvent) => {\n        if (pasteAsText && quill) {\n          e.preventDefault()\n          const text = e.clipboardData?.getData('text/plain') || ''\n          const range = quill.getSelection()\n          if (range) {\n            quill.insertText(range.index, text)\n          }\n        }\n      },\n    },\n  }\n\n  const formatOutputHtml = useCallback((html: string) => {\n    let formattedHtml = html.replace(/<p[^>]*?><\\/p>/g, '<br />')\n    formattedHtml = formattedHtml.replace(/&nbsp;/g, ' ')\n\n    return formattedHtml\n  }, [])\n\n  const handleChange = useCallback(\n    (quill: Quill) => {\n      if (!quill) return\n\n      const html = DOMPurify.sanitize(quill.getSemanticHTML(), {\n        USE_PROFILES: { html: true },\n      })\n\n      const formattedHtml = formatOutputHtml(html)\n      const length = quill.getLength() - 1\n      if (length === 0) {\n        setCurrentLength(0)\n      } else {\n        setCurrentLength(formattedHtml.length)\n      }\n\n      onChange?.(formattedHtml, length)\n    },\n    [onChange, quill]\n  )\n\n  const WordCount = () => {\n    return (\n      <div css={{ display: 'flex', gap: '4px', alignItems: 'baseline' }}>\n        <Text>\n          {currentLength} / {maxlenght}\n        </Text>\n        <Text small color=\"textSecondary\">\n          (NB: Formatering er inkludert i maksgrensen){' '}\n        </Text>\n      </div>\n    )\n  }\n\n  useEffect(() => {\n    if (isMounted.current) return\n\n    quillRef.current = new Quill(`#editor${editorId.current}`, {\n      debug: false,\n      theme: 'snow',\n      bounds: bounds || `#editor${editorId.current}`,\n      modules: {\n        ...modules,\n        keyboard: {\n          scrollingContainer: null,\n        },\n      },\n      formats,\n      placeholder,\n      readOnly,\n    })\n    quillRef.current.keyboard.addBinding(\n      { key: 'Backspace' },\n      {\n        format: {\n          list: true,\n          indent: true,\n        },\n      },\n      (range: any, context: any) => {\n        return handleBackspaceWithIndentetList(\n          range,\n          context,\n          quillRef!.current\n        )\n      }\n    )\n\n    quillRef.current.root.style.touchAction = 'pan-y'\n    quillRef.current.root.style.overflowY = 'auto'\n    quillRef.current.disable()\n    quillRef.current.enable(true)\n    // Move the last binding in the list, to the top. Ensure that the custom binding is called first.\n    quillRef.current.keyboard.bindings['Backspace'].unshift(\n      quillRef.current.keyboard.bindings['Backspace'].pop() as any\n    )\n    quillRef.current.keyboard.bindings['Tab'].unshift(\n      quillRef.current.keyboard.bindings['Tab'].pop() as any\n    )\n    quillRef.current.keyboard.addBinding({ key: 'Escape' }, () => {\n      // Remove focus from editor\n      quillRef!.current!.blur()\n    })\n\n    if (defaultValueRef.current) {\n      const delta = quillRef.current.clipboard.convert({\n        html: defaultValueRef.current,\n      })\n      const formattedDelta = delta.ops.map((op: any) => {\n        if (op.attributes && op.attributes.indent) {\n          op.attributes.indent = parseInt(op.attributes.indent, 10)\n        }\n        return op\n      })\n\n      quillRef.current.setContents(formattedDelta)\n    }\n\n    if (onFocus || onBlur) {\n      quillRef.current.on(\n        Quill.events.SELECTION_CHANGE,\n        (range: Range, oldRange: Range, source: EmitterSource) => {\n          const hasFocus = range && !oldRange\n          const hasBlur = !range && oldRange\n          if (onFocus && hasFocus) onFocus(quillRef.current!.root.innerHTML)\n          if (onBlur && hasBlur) onBlur(quillRef.current!.root.innerHTML)\n        }\n      )\n    }\n    isMounted.current = true\n\n    setQuill(quillRef.current)\n\n    return () => {\n      quillRef.current = null\n    }\n  }, [quillRef])\n\n  useEffect(() => {\n    if (!quill) return\n    quill.on(Quill.events.TEXT_CHANGE, () => handleChange(quill))\n  }, [quill, handleChange])\n\n  useEffect(() => {\n    if (!quill) return\n\n    quill.clipboard.addMatcher(Node.ELEMENT_NODE, (node, delta) => {\n      if (!pasteAsText) return delta\n      let ops: Op[] = []\n      delta.ops.forEach((op) => {\n        if (op.insert && typeof op.insert === 'string') {\n          ops.push({\n            insert: op.insert,\n          })\n        }\n      })\n      delta.ops = ops\n      return delta\n    })\n  }, [quill, pasteAsText])\n\n  useEffect(() => {\n    if (quill && overrideValue !== undefined) {\n      const currentContent = quill.root.innerHTML\n      if (currentContent !== overrideValue) {\n        const delta = quill.clipboard.convert({ html: overrideValue })\n        quill.setContents(delta)\n      }\n    }\n  }, [quill, overrideValue])\n\n  useEffect(() => {\n    if (quill && (disabled || readOnly)) {\n      quill.enable(false)\n    } else if (quill) {\n      quill.enable(true)\n    }\n  }, [readOnly, quill])\n\n  return (\n    <div\n      className={className}\n      css={(theme: IAppTheme) => [\n        styles.default(\n          theme,\n          maxHeight,\n          minHeight,\n          width,\n          minWidth,\n          maxWidth,\n          height,\n          resizable\n        ),\n      ]}\n    >\n      {label && <Label>{label}</Label>}\n      <div id={`editor${editorId.current}`} />\n      {isWordCount && (\n        <div css={{ marginTop: '4px' }}>\n          <WordCount />\n        </div>\n      )}\n      {invalidMessage && (\n        <ErrorWrapper>\n          <Text color={alert}>\n            <IconErrorOutline size=\"16px\" color={alert} /> {invalidMessage}\n          </Text>\n        </ErrorWrapper>\n      )}\n    </div>\n  )\n}\n\nexport default QuillEditor\n"]} */",
|
63
63
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
64
64
|
};
|
65
65
|
const QuillEditor = ({
|
@@ -172,7 +172,7 @@ const QuillEditor = ({
|
|
172
172
|
return (0, _jsxRuntime.jsxs)("div", {
|
173
173
|
css: _ref,
|
174
174
|
children: [(0, _jsxRuntime.jsxs)(_Text.default, {
|
175
|
-
children: [currentLength, "/", maxlenght]
|
175
|
+
children: [currentLength, " / ", maxlenght]
|
176
176
|
}), (0, _jsxRuntime.jsxs)(_Text.default, {
|
177
177
|
small: true,
|
178
178
|
color: "textSecondary",
|