123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 |
- <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}" />
|