Spaces:
Paused
Paused
| <script lang="ts"> | |
| import { marked } from 'marked'; | |
| import TurndownService from 'turndown'; | |
| const turndownService = new TurndownService({ | |
| codeBlockStyle: 'fenced', | |
| headingStyle: 'atx' | |
| }); | |
| turndownService.escape = (string) => string; | |
| import { onMount, onDestroy } from 'svelte'; | |
| import { createEventDispatcher } from 'svelte'; | |
| const eventDispatch = createEventDispatcher(); | |
| import { EditorState, Plugin, PluginKey, TextSelection } from 'prosemirror-state'; | |
| import { Decoration, DecorationSet } from 'prosemirror-view'; | |
| import { Editor } from '@tiptap/core'; | |
| import { AIAutocompletion } from './RichTextInput/AutoCompletion.js'; | |
| import CodeBlockLowlight from '@tiptap/extension-code-block-lowlight'; | |
| import Placeholder from '@tiptap/extension-placeholder'; | |
| import Highlight from '@tiptap/extension-highlight'; | |
| import Typography from '@tiptap/extension-typography'; | |
| import StarterKit from '@tiptap/starter-kit'; | |
| import { all, createLowlight } from 'lowlight'; | |
| import { PASTED_TEXT_CHARACTER_LIMIT } from '$lib/constants'; | |
| export let oncompositionstart = (e) => {}; | |
| export let oncompositionend = (e) => {}; | |
| // create a lowlight instance with all languages loaded | |
| const lowlight = createLowlight(all); | |
| export let className = 'input-prose'; | |
| export let placeholder = 'Type here...'; | |
| export let value = ''; | |
| export let id = ''; | |
| export let raw = false; | |
| export let preserveBreaks = false; | |
| export let generateAutoCompletion: Function = async () => null; | |
| export let autocomplete = false; | |
| export let messageInput = false; | |
| export let shiftEnter = false; | |
| export let largeTextAsFile = false; | |
| let element; | |
| let editor; | |
| const options = { | |
| throwOnError: false | |
| }; | |
| // Function to find the next template in the document | |
| function findNextTemplate(doc, from = 0) { | |
| const patterns = [{ start: '{{', end: '}}' }]; | |
| let result = null; | |
| doc.nodesBetween(from, doc.content.size, (node, pos) => { | |
| if (result) return false; // Stop if we've found a match | |
| if (node.isText) { | |
| const text = node.text; | |
| let index = Math.max(0, from - pos); | |
| while (index < text.length) { | |
| for (const pattern of patterns) { | |
| if (text.startsWith(pattern.start, index)) { | |
| const endIndex = text.indexOf(pattern.end, index + pattern.start.length); | |
| if (endIndex !== -1) { | |
| result = { | |
| from: pos + index, | |
| to: pos + endIndex + pattern.end.length | |
| }; | |
| return false; // Stop searching | |
| } | |
| } | |
| } | |
| index++; | |
| } | |
| } | |
| }); | |
| return result; | |
| } | |
| // Function to select the next template in the document | |
| function selectNextTemplate(state, dispatch) { | |
| const { doc, selection } = state; | |
| const from = selection.to; | |
| let template = findNextTemplate(doc, from); | |
| if (!template) { | |
| // If not found, search from the beginning | |
| template = findNextTemplate(doc, 0); | |
| } | |
| if (template) { | |
| if (dispatch) { | |
| const tr = state.tr.setSelection(TextSelection.create(doc, template.from, template.to)); | |
| dispatch(tr); | |
| } | |
| return true; | |
| } | |
| return false; | |
| } | |
| export const setContent = (content) => { | |
| editor.commands.setContent(content); | |
| }; | |
| const selectTemplate = () => { | |
| if (value !== '') { | |
| // After updating the state, try to find and select the next template | |
| setTimeout(() => { | |
| const templateFound = selectNextTemplate(editor.view.state, editor.view.dispatch); | |
| if (!templateFound) { | |
| // If no template found, set cursor at the end | |
| const endPos = editor.view.state.doc.content.size; | |
| editor.view.dispatch( | |
| editor.view.state.tr.setSelection(TextSelection.create(editor.view.state.doc, endPos)) | |
| ); | |
| } | |
| }, 0); | |
| } | |
| }; | |
| onMount(async () => { | |
| console.log(value); | |
| if (preserveBreaks) { | |
| turndownService.addRule('preserveBreaks', { | |
| filter: 'br', // Target <br> elements | |
| replacement: function (content) { | |
| return '<br/>'; | |
| } | |
| }); | |
| } | |
| let content = value; | |
| if (!raw) { | |
| async function tryParse(value, attempts = 3, interval = 100) { | |
| try { | |
| // Try parsing the value | |
| return marked.parse(value.replaceAll(`\n<br/>`, `<br/>`), { | |
| breaks: false | |
| }); | |
| } catch (error) { | |
| // If no attempts remain, fallback to plain text | |
| if (attempts <= 1) { | |
| return value; | |
| } | |
| // Wait for the interval, then retry | |
| await new Promise((resolve) => setTimeout(resolve, interval)); | |
| return tryParse(value, attempts - 1, interval); // Recursive call | |
| } | |
| } | |
| // Usage example | |
| content = await tryParse(value); | |
| } | |
| editor = new Editor({ | |
| element: element, | |
| extensions: [ | |
| StarterKit, | |
| CodeBlockLowlight.configure({ | |
| lowlight | |
| }), | |
| Highlight, | |
| Typography, | |
| Placeholder.configure({ placeholder }), | |
| ...(autocomplete | |
| ? [ | |
| AIAutocompletion.configure({ | |
| generateCompletion: async (text) => { | |
| if (text.trim().length === 0) { | |
| return null; | |
| } | |
| const suggestion = await generateAutoCompletion(text).catch(() => null); | |
| if (!suggestion || suggestion.trim().length === 0) { | |
| return null; | |
| } | |
| return suggestion; | |
| } | |
| }) | |
| ] | |
| : []) | |
| ], | |
| content: content, | |
| autofocus: messageInput ? true : false, | |
| onTransaction: () => { | |
| // force re-render so `editor.isActive` works as expected | |
| editor = editor; | |
| if (!raw) { | |
| let newValue = turndownService | |
| .turndown( | |
| editor | |
| .getHTML() | |
| .replace(/<p><\/p>/g, '<br/>') | |
| .replace(/ {2,}/g, (m) => m.replace(/ /g, '\u00a0')) | |
| ) | |
| .replace(/\u00a0/g, ' '); | |
| if (!preserveBreaks) { | |
| newValue = newValue.replace(/<br\/>/g, ''); | |
| } | |
| if (value !== newValue) { | |
| value = newValue; | |
| // check if the node is paragraph as well | |
| if (editor.isActive('paragraph')) { | |
| if (value === '') { | |
| editor.commands.clearContent(); | |
| } | |
| } | |
| } | |
| } else { | |
| value = editor.getHTML(); | |
| } | |
| }, | |
| editorProps: { | |
| attributes: { id }, | |
| handleDOMEvents: { | |
| compositionstart: (view, event) => { | |
| oncompositionstart(event); | |
| return false; | |
| }, | |
| compositionend: (view, event) => { | |
| oncompositionend(event); | |
| return false; | |
| }, | |
| focus: (view, event) => { | |
| eventDispatch('focus', { event }); | |
| return false; | |
| }, | |
| keyup: (view, event) => { | |
| eventDispatch('keyup', { event }); | |
| return false; | |
| }, | |
| keydown: (view, event) => { | |
| if (messageInput) { | |
| // Handle Tab Key | |
| if (event.key === 'Tab') { | |
| const handled = selectNextTemplate(view.state, view.dispatch); | |
| if (handled) { | |
| event.preventDefault(); | |
| return true; | |
| } | |
| } | |
| if (event.key === 'Enter') { | |
| // Check if the current selection is inside a structured block (like codeBlock or list) | |
| const { state } = view; | |
| const { $head } = state.selection; | |
| // Recursive function to check ancestors for specific node types | |
| function isInside(nodeTypes: string[]): boolean { | |
| let currentNode = $head; | |
| while (currentNode) { | |
| if (nodeTypes.includes(currentNode.parent.type.name)) { | |
| return true; | |
| } | |
| if (!currentNode.depth) break; // Stop if we reach the top | |
| currentNode = state.doc.resolve(currentNode.before()); // Move to the parent node | |
| } | |
| return false; | |
| } | |
| const isInCodeBlock = isInside(['codeBlock']); | |
| const isInList = isInside(['listItem', 'bulletList', 'orderedList']); | |
| const isInHeading = isInside(['heading']); | |
| if (isInCodeBlock || isInList || isInHeading) { | |
| // Let ProseMirror handle the normal Enter behavior | |
| return false; | |
| } | |
| } | |
| // Handle shift + Enter for a line break | |
| if (shiftEnter) { | |
| if (event.key === 'Enter' && event.shiftKey && !event.ctrlKey && !event.metaKey) { | |
| editor.commands.setHardBreak(); // Insert a hard break | |
| view.dispatch(view.state.tr.scrollIntoView()); // Move viewport to the cursor | |
| event.preventDefault(); | |
| return true; | |
| } | |
| } | |
| } | |
| eventDispatch('keydown', { event }); | |
| return false; | |
| }, | |
| paste: (view, event) => { | |
| if (event.clipboardData) { | |
| // Extract plain text from clipboard and paste it without formatting | |
| const plainText = event.clipboardData.getData('text/plain'); | |
| if (plainText) { | |
| if (largeTextAsFile) { | |
| if (plainText.length > PASTED_TEXT_CHARACTER_LIMIT) { | |
| // Dispatch paste event to parent component | |
| eventDispatch('paste', { event }); | |
| event.preventDefault(); | |
| return true; | |
| } | |
| } | |
| return false; | |
| } | |
| // Check if the pasted content contains image files | |
| const hasImageFile = Array.from(event.clipboardData.files).some((file) => | |
| file.type.startsWith('image/') | |
| ); | |
| // Check for image in dataTransfer items (for cases where files are not available) | |
| const hasImageItem = Array.from(event.clipboardData.items).some((item) => | |
| item.type.startsWith('image/') | |
| ); | |
| if (hasImageFile) { | |
| // If there's an image, dispatch the event to the parent | |
| eventDispatch('paste', { event }); | |
| event.preventDefault(); | |
| return true; | |
| } | |
| if (hasImageItem) { | |
| // If there's an image item, dispatch the event to the parent | |
| eventDispatch('paste', { event }); | |
| event.preventDefault(); | |
| return true; | |
| } | |
| } | |
| // For all other cases (text, formatted text, etc.), let ProseMirror handle it | |
| view.dispatch(view.state.tr.scrollIntoView()); // Move viewport to the cursor after pasting | |
| return false; | |
| } | |
| } | |
| } | |
| }); | |
| if (messageInput) { | |
| selectTemplate(); | |
| } | |
| }); | |
| onDestroy(() => { | |
| if (editor) { | |
| editor.destroy(); | |
| } | |
| }); | |
| // Update the editor content if the external `value` changes | |
| $: if ( | |
| editor && | |
| (raw | |
| ? value !== editor.getHTML() | |
| : value !== | |
| turndownService | |
| .turndown( | |
| (preserveBreaks | |
| ? editor.getHTML().replace(/<p><\/p>/g, '<br/>') | |
| : editor.getHTML() | |
| ).replace(/ {2,}/g, (m) => m.replace(/ /g, '\u00a0')) | |
| ) | |
| .replace(/\u00a0/g, ' ')) | |
| ) { | |
| if (raw) { | |
| editor.commands.setContent(value); | |
| } else { | |
| preserveBreaks | |
| ? editor.commands.setContent(value) | |
| : editor.commands.setContent( | |
| marked.parse(value.replaceAll(`\n<br/>`, `<br/>`), { | |
| breaks: false | |
| }) | |
| ); // Update editor content | |
| } | |
| selectTemplate(); | |
| } | |
| </script> | |
| <div bind:this={element} class="relative w-full min-w-full h-full min-h-fit {className}" /> | |