open-webui/src/lib/components/common/RichTextInput.svelte

349 lines
9.6 KiB
Svelte
Raw Normal View History

2024-10-18 21:55:39 +00:00
<script lang="ts">
2024-11-21 06:46:51 +00:00
import { marked } from 'marked';
import TurndownService from 'turndown';
2024-11-24 04:31:33 +00:00
const turndownService = new TurndownService({
2024-11-30 22:15:08 +00:00
codeBlockStyle: 'fenced',
headingStyle: 'atx'
2024-11-24 04:31:33 +00:00
});
turndownService.escape = (string) => string;
2024-11-21 06:46:51 +00:00
import { onMount, onDestroy } from 'svelte';
2024-10-19 05:40:15 +00:00
import { createEventDispatcher } from 'svelte';
const eventDispatch = createEventDispatcher();
2024-10-18 21:55:39 +00:00
2024-11-29 07:22:53 +00:00
import { EditorState, Plugin, PluginKey, TextSelection } from 'prosemirror-state';
import { Decoration, DecorationSet } from 'prosemirror-view';
2024-10-18 21:55:39 +00:00
2024-11-21 06:46:51 +00:00
import { Editor } from '@tiptap/core';
2024-11-29 07:22:53 +00:00
import { AIAutocompletion } from './RichTextInput/AutoCompletion.js';
2024-11-21 06:56:26 +00:00
import CodeBlockLowlight from '@tiptap/extension-code-block-lowlight';
2024-11-21 06:46:51 +00:00
import Placeholder from '@tiptap/extension-placeholder';
import Highlight from '@tiptap/extension-highlight';
import Typography from '@tiptap/extension-typography';
import StarterKit from '@tiptap/starter-kit';
2024-11-21 06:56:26 +00:00
import { all, createLowlight } from 'lowlight';
import { PASTED_TEXT_CHARACTER_LIMIT } from '$lib/constants';
2024-10-18 21:55:39 +00:00
2024-11-21 06:56:26 +00:00
// create a lowlight instance with all languages loaded
const lowlight = createLowlight(all);
2024-10-18 21:55:39 +00:00
export let className = 'input-prose';
export let placeholder = 'Type here...';
2024-11-21 06:46:51 +00:00
export let value = '';
export let id = '';
2024-10-18 21:55:39 +00:00
2024-11-30 23:05:08 +00:00
export let preserveBreaks = false;
2024-11-29 08:16:49 +00:00
export let generateAutoCompletion: Function = async () => null;
2024-11-29 07:22:53 +00:00
export let autocomplete = false;
2024-11-21 06:46:51 +00:00
export let messageInput = false;
export let shiftEnter = false;
export let largeTextAsFile = false;
2024-10-18 21:55:39 +00:00
2024-11-21 06:46:51 +00:00
let element;
let editor;
2024-10-18 21:55:39 +00:00
2024-11-22 05:32:19 +00:00
const options = {
throwOnError: false
};
2024-11-21 06:46:51 +00:00
// Function to find the next template in the document
2024-10-19 07:23:59 +00:00
function findNextTemplate(doc, from = 0) {
const patterns = [
{ start: '[', end: ']' },
{ 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;
}
2024-11-21 06:46:51 +00:00
// Function to select the next template in the document
2024-10-19 07:23:59 +00:00
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;
}
2024-10-19 10:15:40 +00:00
return false;
2024-10-19 07:23:59 +00:00
}
2024-11-21 06:46:51 +00:00
export const setContent = (content) => {
editor.commands.setContent(content);
};
2024-10-25 18:51:49 +00:00
2024-11-21 06:46:51 +00:00
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);
}
};
2024-10-18 21:55:39 +00:00
2024-11-22 06:20:57 +00:00
onMount(async () => {
2024-11-30 22:15:08 +00:00
console.log(value);
2024-11-30 23:05:08 +00:00
if (preserveBreaks) {
turndownService.addRule('preserveBreaks', {
filter: 'br', // Target <br> elements
replacement: function (content) {
return '<br/>';
}
});
}
2024-11-22 06:20:57 +00:00
async function tryParse(value, attempts = 3, interval = 100) {
try {
// Try parsing the value
2024-11-30 23:05:08 +00:00
return marked.parse(value.replaceAll(`\n<br/>`, `<br/>`), {
breaks: false
});
2024-11-22 06:20:57 +00:00
} 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
}
2024-11-22 06:15:04 +00:00
}
2024-11-22 06:20:57 +00:00
// Usage example
let content = await tryParse(value);
2024-11-21 06:46:51 +00:00
editor = new Editor({
element: element,
2024-11-21 06:56:26 +00:00
extensions: [
StarterKit,
CodeBlockLowlight.configure({
lowlight
}),
Highlight,
Typography,
2024-11-29 07:22:53 +00:00
Placeholder.configure({ placeholder }),
2024-11-29 07:26:09 +00:00
...(autocomplete
? [
AIAutocompletion.configure({
generateCompletion: async (text) => {
if (text.trim().length === 0) {
return null;
}
2024-11-29 08:16:49 +00:00
const suggestion = await generateAutoCompletion(text).catch(() => null);
if (!suggestion || suggestion.trim().length === 0) {
return null;
}
return suggestion;
2024-11-29 07:26:09 +00:00
}
})
]
: [])
2024-11-21 06:56:26 +00:00
],
2024-11-22 05:32:19 +00:00
content: content,
2024-11-30 23:44:04 +00:00
autofocus: messageInput ? true : false,
2024-11-21 06:46:51 +00:00
onTransaction: () => {
// force re-render so `editor.isActive` works as expected
editor = editor;
2024-11-30 23:05:08 +00:00
const newValue = turndownService.turndown(
preserveBreaks ? editor.getHTML().replace(/<p><\/p>/g, '<br/>') : editor.getHTML()
);
2024-11-21 06:46:51 +00:00
if (value !== newValue) {
2024-11-24 04:31:33 +00:00
value = newValue;
2024-11-24 07:57:05 +00:00
2024-11-30 22:15:08 +00:00
// check if the node is paragraph as well
if (editor.isActive('paragraph')) {
if (value === '') {
editor.commands.clearContent();
}
2024-11-24 07:57:05 +00:00
}
2024-11-21 06:46:51 +00:00
}
},
editorProps: {
attributes: { id },
handleDOMEvents: {
focus: (view, event) => {
eventDispatch('focus', { event });
return false;
2024-10-19 05:56:04 +00:00
},
2024-11-26 06:43:34 +00:00
keyup: (view, event) => {
eventDispatch('keyup', { event });
2024-10-19 05:56:04 +00:00
return false;
},
2024-11-21 06:46:51 +00:00
keydown: (view, event) => {
2024-11-30 23:44:04 +00:00
if (messageInput) {
// Handle Tab Key
if (event.key === 'Tab') {
const handled = selectNextTemplate(view.state, view.dispatch);
if (handled) {
event.preventDefault();
return true;
}
2024-11-21 06:46:51 +00:00
}
2024-10-18 21:55:39 +00:00
2024-11-21 06:56:26 +00:00
if (event.key === 'Enter') {
2024-11-21 07:14:06 +00:00
// Check if the current selection is inside a structured block (like codeBlock or list)
2024-11-21 06:56:26 +00:00
const { state } = view;
const { $head } = state.selection;
2024-11-21 07:14:06 +00:00
// 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;
2024-11-21 06:56:26 +00:00
}
}
2024-11-21 06:46:51 +00:00
// Handle shift + Enter for a line break
if (shiftEnter) {
2024-11-26 06:43:34 +00:00
if (event.key === 'Enter' && event.shiftKey && !event.ctrlKey && !event.metaKey) {
2024-11-21 06:46:51 +00:00
editor.commands.setHardBreak(); // Insert a hard break
2024-11-21 06:56:26 +00:00
view.dispatch(view.state.tr.scrollIntoView()); // Move viewport to the cursor
2024-11-21 06:46:51 +00:00
event.preventDefault();
return true;
}
}
2024-10-25 18:51:49 +00:00
}
2024-11-21 06:46:51 +00:00
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;
}
2024-10-19 06:54:35 +00:00
2024-11-21 06:46:51 +00:00
// Check if the pasted content contains image files
const hasImageFile = Array.from(event.clipboardData.files).some((file) =>
file.type.startsWith('image/')
);
2024-10-19 06:54:35 +00:00
2024-11-21 06:46:51 +00:00
// 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;
}
2024-10-19 06:54:35 +00:00
}
2024-11-21 06:46:51 +00:00
// 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;
2024-10-19 21:56:30 +00:00
}
2024-10-19 05:40:15 +00:00
}
2024-11-21 06:46:51 +00:00
}
2024-10-18 21:55:39 +00:00
});
2024-10-19 07:23:59 +00:00
2024-11-30 23:44:04 +00:00
if (messageInput) {
selectTemplate();
}
2024-11-21 06:46:51 +00:00
});
2024-10-18 21:55:39 +00:00
onDestroy(() => {
2024-11-21 06:46:51 +00:00
if (editor) {
editor.destroy();
}
2024-10-18 21:55:39 +00:00
});
2024-11-21 06:46:51 +00:00
// Update the editor content if the external `value` changes
2024-11-30 23:05:08 +00:00
$: if (
editor &&
value !==
turndownService.turndown(
preserveBreaks ? editor.getHTML().replace(/<p><\/p>/g, '<br/>') : editor.getHTML()
)
) {
editor.commands.setContent(
marked.parse(value.replaceAll(`\n<br/>`, `<br/>`), {
breaks: false
})
); // Update editor content
2024-11-21 06:46:51 +00:00
selectTemplate();
}
2024-10-18 21:55:39 +00:00
</script>
2024-11-21 06:46:51 +00:00
<div bind:this={element} class="relative w-full min-w-full h-full min-h-fit {className}" />