@bendyline/squisq-editor-react 1.2.2 → 1.4.0
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/EditorContext.d.ts +65 -1
- package/dist/EditorContext.d.ts.map +1 -1
- package/dist/EditorContext.js +31 -4
- package/dist/EditorContext.js.map +1 -1
- package/dist/EditorShell.d.ts +112 -2
- package/dist/EditorShell.d.ts.map +1 -1
- package/dist/EditorShell.js +95 -11
- package/dist/EditorShell.js.map +1 -1
- package/dist/ImageNodeView.d.ts.map +1 -1
- package/dist/ImageNodeView.js +12 -2
- package/dist/ImageNodeView.js.map +1 -1
- package/dist/MediaBin.d.ts +12 -1
- package/dist/MediaBin.d.ts.map +1 -1
- package/dist/MediaBin.js +29 -4
- package/dist/MediaBin.js.map +1 -1
- package/dist/MentionExtension.d.ts +22 -0
- package/dist/MentionExtension.d.ts.map +1 -0
- package/dist/MentionExtension.js +242 -0
- package/dist/MentionExtension.js.map +1 -0
- package/dist/RawEditor.d.ts +8 -1
- package/dist/RawEditor.d.ts.map +1 -1
- package/dist/RawEditor.js +167 -30
- package/dist/RawEditor.js.map +1 -1
- package/dist/TemplateAnnotation.d.ts.map +1 -1
- package/dist/TemplateAnnotation.js +4 -2
- package/dist/TemplateAnnotation.js.map +1 -1
- package/dist/Toolbar.d.ts +7 -1
- package/dist/Toolbar.d.ts.map +1 -1
- package/dist/Toolbar.js +57 -18
- package/dist/Toolbar.js.map +1 -1
- package/dist/Tooltip.d.ts +10 -0
- package/dist/Tooltip.d.ts.map +1 -0
- package/dist/Tooltip.js +104 -0
- package/dist/Tooltip.js.map +1 -0
- package/dist/ViewSwitcher.d.ts +1 -1
- package/dist/ViewSwitcher.d.ts.map +1 -1
- package/dist/ViewSwitcher.js +10 -4
- package/dist/ViewSwitcher.js.map +1 -1
- package/dist/WysiwygEditor.d.ts +13 -2
- package/dist/WysiwygEditor.d.ts.map +1 -1
- package/dist/WysiwygEditor.js +239 -4
- package/dist/WysiwygEditor.js.map +1 -1
- package/dist/__tests__/detectMarkdown.test.d.ts +2 -0
- package/dist/__tests__/detectMarkdown.test.d.ts.map +1 -0
- package/dist/__tests__/detectMarkdown.test.js +69 -0
- package/dist/__tests__/detectMarkdown.test.js.map +1 -0
- package/dist/__tests__/fileKind.test.d.ts +2 -0
- package/dist/__tests__/fileKind.test.d.ts.map +1 -0
- package/dist/__tests__/fileKind.test.js +81 -0
- package/dist/__tests__/fileKind.test.js.map +1 -0
- package/dist/__tests__/mediaAttachmentFlow.test.d.ts +2 -0
- package/dist/__tests__/mediaAttachmentFlow.test.d.ts.map +1 -0
- package/dist/__tests__/mediaAttachmentFlow.test.js +99 -0
- package/dist/__tests__/mediaAttachmentFlow.test.js.map +1 -0
- package/dist/__tests__/tiptapBridge.test.js +49 -0
- package/dist/__tests__/tiptapBridge.test.js.map +1 -1
- package/dist/__tests__/tiptapImageRoundTrip.test.d.ts +2 -0
- package/dist/__tests__/tiptapImageRoundTrip.test.d.ts.map +1 -0
- package/dist/__tests__/tiptapImageRoundTrip.test.js +68 -0
- package/dist/__tests__/tiptapImageRoundTrip.test.js.map +1 -0
- package/dist/detectMarkdown.d.ts +20 -0
- package/dist/detectMarkdown.d.ts.map +1 -0
- package/dist/detectMarkdown.js +61 -0
- package/dist/detectMarkdown.js.map +1 -0
- package/dist/fileKind.d.ts +30 -0
- package/dist/fileKind.d.ts.map +1 -0
- package/dist/fileKind.js +123 -0
- package/dist/fileKind.js.map +1 -0
- package/dist/hooks/useFileDrop.d.ts.map +1 -1
- package/dist/hooks/useFileDrop.js +9 -7
- package/dist/hooks/useFileDrop.js.map +1 -1
- package/dist/index.d.ts +4 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4 -0
- package/dist/index.js.map +1 -1
- package/dist/mediaDragMime.d.ts +17 -0
- package/dist/mediaDragMime.d.ts.map +1 -0
- package/dist/mediaDragMime.js +22 -0
- package/dist/mediaDragMime.js.map +1 -0
- package/dist/tiptapBridge.d.ts.map +1 -1
- package/dist/tiptapBridge.js +99 -6
- package/dist/tiptapBridge.js.map +1 -1
- package/package.json +9 -7
- package/src/EditorContext.tsx +106 -3
- package/src/EditorShell.tsx +313 -21
- package/src/ImageNodeView.tsx +15 -2
- package/src/MediaBin.tsx +45 -4
- package/src/MentionExtension.tsx +258 -0
- package/src/RawEditor.tsx +193 -37
- package/src/TemplateAnnotation.ts +4 -2
- package/src/Toolbar.tsx +111 -48
- package/src/Tooltip.tsx +124 -0
- package/src/ViewSwitcher.tsx +15 -5
- package/src/WysiwygEditor.tsx +270 -5
- package/src/__tests__/detectMarkdown.test.ts +88 -0
- package/src/__tests__/fileKind.test.ts +96 -0
- package/src/__tests__/mediaAttachmentFlow.test.ts +110 -0
- package/src/__tests__/tiptapBridge.test.ts +58 -0
- package/src/__tests__/tiptapImageRoundTrip.test.ts +73 -0
- package/src/detectMarkdown.ts +62 -0
- package/src/fileKind.ts +134 -0
- package/src/hooks/useFileDrop.ts +10 -6
- package/src/index.ts +11 -0
- package/src/mediaDragMime.ts +32 -0
- package/src/styles/editor.css +214 -8
- package/src/tiptapBridge.ts +107 -6
package/dist/WysiwygEditor.js
CHANGED
|
@@ -10,7 +10,7 @@ import { jsx as _jsx } from "react/jsx-runtime";
|
|
|
10
10
|
* Includes extensions for GFM features: tables, task lists, strikethrough,
|
|
11
11
|
* and code blocks.
|
|
12
12
|
*/
|
|
13
|
-
import { useEffect, useRef } from 'react';
|
|
13
|
+
import { useEffect, useMemo, useRef } from 'react';
|
|
14
14
|
import { useEditor, EditorContent } from '@tiptap/react';
|
|
15
15
|
import StarterKit from '@tiptap/starter-kit';
|
|
16
16
|
import Table from '@tiptap/extension-table';
|
|
@@ -23,7 +23,10 @@ import Placeholder from '@tiptap/extension-placeholder';
|
|
|
23
23
|
import { HeadingWithTemplate } from './TemplateAnnotation';
|
|
24
24
|
import { ImageWithMediaProvider } from './ImageNodeView';
|
|
25
25
|
import { useEditorContext } from './EditorContext';
|
|
26
|
+
import { buildMentionExtension } from './MentionExtension';
|
|
26
27
|
import { markdownToTiptap, tiptapToMarkdown } from './tiptapBridge';
|
|
28
|
+
import { looksLikeMarkdown } from './detectMarkdown';
|
|
29
|
+
import { SQUISQ_MEDIA_MIME, parseSquisqMediaPayload } from './mediaDragMime';
|
|
27
30
|
// ── Frontmatter helpers ────────────────────────────────────────────
|
|
28
31
|
/** Regex matching a YAML frontmatter block at the start of the document. */
|
|
29
32
|
const FRONTMATTER_RE = /^---\r?\n([\s\S]*?)\r?\n---\r?\n?/;
|
|
@@ -34,17 +37,60 @@ function stripFrontmatter(md) {
|
|
|
34
37
|
return { body: md, frontmatter: '' };
|
|
35
38
|
return { body: md.slice(m[0].length), frontmatter: m[0] };
|
|
36
39
|
}
|
|
40
|
+
/**
|
|
41
|
+
* Rotating placeholder prompts shown when the editor is empty. One is
|
|
42
|
+
* picked at random per editor mount. Hosts can override by passing the
|
|
43
|
+
* `placeholder` prop with a fixed string.
|
|
44
|
+
*/
|
|
45
|
+
const EMPTY_PROMPTS = [
|
|
46
|
+
'Start typing your content, or drop images on top of me…',
|
|
47
|
+
'Write anything — paste markdown, drag in images, or just start typing…',
|
|
48
|
+
'Type away. Markdown syntax works too…',
|
|
49
|
+
'Chapter 1 begins here…',
|
|
50
|
+
'Once upon a time…',
|
|
51
|
+
'A blank page. Exciting, isn\u2019t it?',
|
|
52
|
+
'The first word is always the hardest…',
|
|
53
|
+
'Plot twist: this is where it all starts…',
|
|
54
|
+
'Write something the future you will thank you for…',
|
|
55
|
+
'Begin at the beginning…',
|
|
56
|
+
];
|
|
57
|
+
function pickEmptyPrompt() {
|
|
58
|
+
return EMPTY_PROMPTS[Math.floor(Math.random() * EMPTY_PROMPTS.length)];
|
|
59
|
+
}
|
|
37
60
|
/**
|
|
38
61
|
* Rich WYSIWYG markdown editor built on Tiptap (ProseMirror).
|
|
39
62
|
* Binds to the shared EditorContext for source synchronization.
|
|
40
63
|
*/
|
|
41
|
-
export function WysiwygEditor({ placeholder
|
|
42
|
-
const { markdownSource, setMarkdownSource, setTiptapEditor } = useEditorContext();
|
|
64
|
+
export function WysiwygEditor({ placeholder, className, submitOnEnter, readOnly = false, }) {
|
|
65
|
+
const { markdownSource, setMarkdownSource, setTiptapEditor, mediaProvider, mentionProvider } = useEditorContext();
|
|
66
|
+
// Keep a ref so the mention extension — created once at editor mount —
|
|
67
|
+
// always sees the latest provider. Swapping projects or gezels changes
|
|
68
|
+
// the provider without remounting the editor.
|
|
69
|
+
const mentionProviderRef = useRef(mentionProvider);
|
|
70
|
+
useEffect(() => {
|
|
71
|
+
mentionProviderRef.current = mentionProvider;
|
|
72
|
+
}, [mentionProvider]);
|
|
73
|
+
// Stable per mount: either the host-supplied string, or a random pick
|
|
74
|
+
// from EMPTY_PROMPTS. Re-renders don't reshuffle.
|
|
75
|
+
const resolvedPlaceholder = useMemo(() => placeholder ?? pickEmptyPrompt(), [placeholder]);
|
|
43
76
|
const isExternalUpdate = useRef(false);
|
|
44
77
|
const lastSourceRef = useRef(markdownSource);
|
|
78
|
+
// Keep a ref so the editor's drop/paste handlers (created once) always
|
|
79
|
+
// see the current MediaProvider without needing to recreate the editor.
|
|
80
|
+
const mediaProviderRef = useRef(mediaProvider);
|
|
81
|
+
useEffect(() => {
|
|
82
|
+
mediaProviderRef.current = mediaProvider;
|
|
83
|
+
}, [mediaProvider]);
|
|
45
84
|
// Preserve frontmatter across edits — hidden from WYSIWYG but prepended on save
|
|
46
85
|
const frontmatterRef = useRef(stripFrontmatter(markdownSource).frontmatter);
|
|
86
|
+
// Stash the latest submit callback so the editor's handleKeyDown (bound
|
|
87
|
+
// once at creation) always sees the current value.
|
|
88
|
+
const submitOnEnterRef = useRef(submitOnEnter);
|
|
89
|
+
useEffect(() => {
|
|
90
|
+
submitOnEnterRef.current = submitOnEnter;
|
|
91
|
+
}, [submitOnEnter]);
|
|
47
92
|
const editor = useEditor({
|
|
93
|
+
editable: !readOnly,
|
|
48
94
|
extensions: [
|
|
49
95
|
StarterKit.configure({
|
|
50
96
|
// Disable built-in heading; we use HeadingWithTemplate instead
|
|
@@ -61,7 +107,8 @@ export function WysiwygEditor({ placeholder = 'Start typing your markdown…', c
|
|
|
61
107
|
TaskList,
|
|
62
108
|
TaskItem.configure({ nested: true }),
|
|
63
109
|
ImageWithMediaProvider.configure({ inline: false }),
|
|
64
|
-
Placeholder.configure({ placeholder }),
|
|
110
|
+
Placeholder.configure({ placeholder: resolvedPlaceholder }),
|
|
111
|
+
buildMentionExtension(() => mentionProviderRef.current),
|
|
65
112
|
],
|
|
66
113
|
content: markdownToTiptap(stripFrontmatter(markdownSource).body),
|
|
67
114
|
onUpdate: ({ editor: ed }) => {
|
|
@@ -78,6 +125,84 @@ export function WysiwygEditor({ placeholder = 'Start typing your markdown…', c
|
|
|
78
125
|
class: 'squisq-wysiwyg-editor',
|
|
79
126
|
'data-testid': 'wysiwyg-editor',
|
|
80
127
|
},
|
|
128
|
+
// Chat-composer mode: Enter commits via submitOnEnter(), Cmd/Ctrl+Enter
|
|
129
|
+
// inserts a soft break. When no callback is set, fall through to Tiptap's
|
|
130
|
+
// normal behavior (Enter = paragraph break, Shift+Enter = soft break).
|
|
131
|
+
handleKeyDown: (view, event) => {
|
|
132
|
+
if (event.key !== 'Enter' || !submitOnEnterRef.current)
|
|
133
|
+
return false;
|
|
134
|
+
if (event.metaKey || event.ctrlKey) {
|
|
135
|
+
// User wants a newline. Insert a hard-break and stop propagation so
|
|
136
|
+
// we don't also create a new paragraph.
|
|
137
|
+
event.preventDefault();
|
|
138
|
+
view.dispatch(view.state.tr.replaceSelectionWith(view.state.schema.nodes.hardBreak.create()));
|
|
139
|
+
return true;
|
|
140
|
+
}
|
|
141
|
+
if (event.shiftKey) {
|
|
142
|
+
// Preserve the conventional Shift+Enter soft break.
|
|
143
|
+
return false;
|
|
144
|
+
}
|
|
145
|
+
// Plain Enter — submit.
|
|
146
|
+
event.preventDefault();
|
|
147
|
+
submitOnEnterRef.current();
|
|
148
|
+
return true;
|
|
149
|
+
},
|
|
150
|
+
// When the clipboard's plain-text payload looks like markdown source,
|
|
151
|
+
// convert it via tiptapBridge before pasting. This applies even when
|
|
152
|
+
// the clipboard also contains HTML (most rich-text sources do), since
|
|
153
|
+
// the markdown-looking text is usually what the user actually wants.
|
|
154
|
+
// Without this, pasted markdown shows up as literal "# Heading" text
|
|
155
|
+
// instead of becoming a real heading.
|
|
156
|
+
handlePaste: (view, event) => {
|
|
157
|
+
const clipboard = event.clipboardData;
|
|
158
|
+
if (!clipboard)
|
|
159
|
+
return false;
|
|
160
|
+
// Image files in the clipboard → upload via MediaProvider and insert
|
|
161
|
+
const imageFiles = filesFromClipboard(clipboard);
|
|
162
|
+
if (imageFiles.length > 0 && mediaProviderRef.current) {
|
|
163
|
+
event.preventDefault();
|
|
164
|
+
uploadAndInsertImages(view, imageFiles, mediaProviderRef.current);
|
|
165
|
+
return true;
|
|
166
|
+
}
|
|
167
|
+
const text = clipboard.getData('text/plain');
|
|
168
|
+
if (!text || !looksLikeMarkdown(text))
|
|
169
|
+
return false;
|
|
170
|
+
const html = markdownToTiptap(text);
|
|
171
|
+
if (!html)
|
|
172
|
+
return false;
|
|
173
|
+
event.preventDefault();
|
|
174
|
+
view.pasteHTML(html);
|
|
175
|
+
return true;
|
|
176
|
+
},
|
|
177
|
+
// When image files are dropped onto the editor, upload them via the
|
|
178
|
+
// MediaProvider and insert <img> nodes referencing the relative paths.
|
|
179
|
+
// Also handles drags from the MediaBin, which reference existing
|
|
180
|
+
// entries via a custom MIME type and skip the upload step.
|
|
181
|
+
// Falls through to default handling for non-image drops or when no
|
|
182
|
+
// MediaProvider is available.
|
|
183
|
+
handleDrop: (view, event, _slice, _moved) => {
|
|
184
|
+
const dt = event.dataTransfer;
|
|
185
|
+
if (!dt)
|
|
186
|
+
return false;
|
|
187
|
+
// In-app drag from the MediaBin — insert without uploading
|
|
188
|
+
const squisqRaw = dt.getData(SQUISQ_MEDIA_MIME);
|
|
189
|
+
if (squisqRaw) {
|
|
190
|
+
const payload = parseSquisqMediaPayload(squisqRaw);
|
|
191
|
+
if (payload && payload.mimeType.startsWith('image/')) {
|
|
192
|
+
event.preventDefault();
|
|
193
|
+
moveSelectionToDropPoint(view, event);
|
|
194
|
+
insertImageNode(view, payload.name, payload.alt);
|
|
195
|
+
return true;
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
const imageFiles = filesFromDataTransfer(dt);
|
|
199
|
+
if (imageFiles.length === 0 || !mediaProviderRef.current)
|
|
200
|
+
return false;
|
|
201
|
+
event.preventDefault();
|
|
202
|
+
moveSelectionToDropPoint(view, event);
|
|
203
|
+
uploadAndInsertImages(view, imageFiles, mediaProviderRef.current);
|
|
204
|
+
return true;
|
|
205
|
+
},
|
|
81
206
|
},
|
|
82
207
|
});
|
|
83
208
|
// Register / unregister the Tiptap editor instance with the shared context
|
|
@@ -87,6 +212,13 @@ export function WysiwygEditor({ placeholder = 'Start typing your markdown…', c
|
|
|
87
212
|
}
|
|
88
213
|
return () => setTiptapEditor(null);
|
|
89
214
|
}, [editor, setTiptapEditor]);
|
|
215
|
+
// Tiptap reads `editable` only at creation; mirror later changes via
|
|
216
|
+
// setEditable so flipping readOnly from the host takes effect without
|
|
217
|
+
// remounting the editor.
|
|
218
|
+
useEffect(() => {
|
|
219
|
+
if (editor)
|
|
220
|
+
editor.setEditable(!readOnly);
|
|
221
|
+
}, [editor, readOnly]);
|
|
90
222
|
// Sync external changes into Tiptap
|
|
91
223
|
useEffect(() => {
|
|
92
224
|
if (!editor)
|
|
@@ -104,6 +236,109 @@ export function WysiwygEditor({ placeholder = 'Start typing your markdown…', c
|
|
|
104
236
|
}, [markdownSource, editor]);
|
|
105
237
|
return (_jsx("div", { className: `squisq-wysiwyg-container${className ? ` ${className}` : ''}`, style: { width: '100%', height: '100%', overflow: 'auto' }, "data-testid": "wysiwyg-container", children: _jsx(EditorContent, { editor: editor, style: { height: '100%' } }) }));
|
|
106
238
|
}
|
|
239
|
+
// ── Image drop / paste helpers ─────────────────────────────────────
|
|
240
|
+
/** Extract image File objects from a DataTransfer (drop event). */
|
|
241
|
+
function filesFromDataTransfer(dt) {
|
|
242
|
+
const files = [];
|
|
243
|
+
for (let i = 0; i < dt.files.length; i++) {
|
|
244
|
+
const file = dt.files[i];
|
|
245
|
+
if (file.type.startsWith('image/'))
|
|
246
|
+
files.push(file);
|
|
247
|
+
}
|
|
248
|
+
return files;
|
|
249
|
+
}
|
|
250
|
+
/** Extract image File objects from a clipboard's items (paste event). */
|
|
251
|
+
function filesFromClipboard(clipboard) {
|
|
252
|
+
const files = [];
|
|
253
|
+
// clipboardData.items is the most reliable source for pasted images
|
|
254
|
+
if (clipboard.items) {
|
|
255
|
+
for (let i = 0; i < clipboard.items.length; i++) {
|
|
256
|
+
const item = clipboard.items[i];
|
|
257
|
+
if (item.kind === 'file' && item.type.startsWith('image/')) {
|
|
258
|
+
const file = item.getAsFile();
|
|
259
|
+
if (file)
|
|
260
|
+
files.push(file);
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
return files;
|
|
265
|
+
}
|
|
266
|
+
/**
|
|
267
|
+
* Upload image files to the MediaProvider and insert <img> nodes at the
|
|
268
|
+
* current selection. Inserts a placeholder name when files lack one
|
|
269
|
+
* (e.g., screenshots from the system clipboard).
|
|
270
|
+
*/
|
|
271
|
+
async function uploadAndInsertImages(
|
|
272
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
273
|
+
view, files, mediaProvider) {
|
|
274
|
+
for (const file of files) {
|
|
275
|
+
try {
|
|
276
|
+
const buffer = await file.arrayBuffer();
|
|
277
|
+
const mimeType = file.type || 'image/png';
|
|
278
|
+
const name = file.name && file.name !== 'image.png'
|
|
279
|
+
? file.name
|
|
280
|
+
: `pasted-${uniquePasteToken()}.${extFromMime(mimeType)}`;
|
|
281
|
+
const relativePath = await mediaProvider.addMedia(name, buffer, mimeType);
|
|
282
|
+
const altText = name.replace(/\.[^.]+$/, '').replace(/[-_]/g, ' ');
|
|
283
|
+
insertImageNode(view, relativePath, altText);
|
|
284
|
+
}
|
|
285
|
+
catch (err) {
|
|
286
|
+
console.error('Failed to upload dropped image:', err);
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
/** Insert an image node at the current selection using the schema image type. */
|
|
291
|
+
function insertImageNode(
|
|
292
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
293
|
+
view, src, alt) {
|
|
294
|
+
const { schema } = view.state;
|
|
295
|
+
const imageType = schema.nodes.image;
|
|
296
|
+
if (!imageType)
|
|
297
|
+
return;
|
|
298
|
+
const node = imageType.create({ src, alt });
|
|
299
|
+
const tr = view.state.tr.replaceSelectionWith(node);
|
|
300
|
+
view.dispatch(tr);
|
|
301
|
+
}
|
|
302
|
+
/** Move the selection to the document position under the drop event's coordinates. */
|
|
303
|
+
function moveSelectionToDropPoint(
|
|
304
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
305
|
+
view, event) {
|
|
306
|
+
const coords = view.posAtCoords({ left: event.clientX, top: event.clientY });
|
|
307
|
+
if (!coords)
|
|
308
|
+
return;
|
|
309
|
+
const tr = view.state.tr.setSelection(
|
|
310
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
311
|
+
view.state.selection.constructor.near(view.state.doc.resolve(coords.pos)));
|
|
312
|
+
view.dispatch(tr);
|
|
313
|
+
}
|
|
314
|
+
/**
|
|
315
|
+
* Produce a unique token for a pasted-file name. `Date.now()` alone can
|
|
316
|
+
* collide when a user pastes several clipboard images in the same tick
|
|
317
|
+
* (multi-image paste from a screenshot grid, for example), which would make
|
|
318
|
+
* `MediaProvider.addMedia` overwrite or reject later entries. Prefer
|
|
319
|
+
* `crypto.randomUUID()` when available and fall back to a counter so the
|
|
320
|
+
* helper stays pure-JS-everywhere.
|
|
321
|
+
*/
|
|
322
|
+
let pasteCounter = 0;
|
|
323
|
+
function uniquePasteToken() {
|
|
324
|
+
if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {
|
|
325
|
+
return crypto.randomUUID();
|
|
326
|
+
}
|
|
327
|
+
pasteCounter = (pasteCounter + 1) % 1000000;
|
|
328
|
+
return `${Date.now()}-${pasteCounter.toString(36)}`;
|
|
329
|
+
}
|
|
330
|
+
function extFromMime(mime) {
|
|
331
|
+
const map = {
|
|
332
|
+
'image/png': 'png',
|
|
333
|
+
'image/jpeg': 'jpg',
|
|
334
|
+
'image/jpg': 'jpg',
|
|
335
|
+
'image/gif': 'gif',
|
|
336
|
+
'image/webp': 'webp',
|
|
337
|
+
'image/svg+xml': 'svg',
|
|
338
|
+
'image/avif': 'avif',
|
|
339
|
+
};
|
|
340
|
+
return map[mime.toLowerCase()] ?? 'png';
|
|
341
|
+
}
|
|
107
342
|
/**
|
|
108
343
|
* Hook to access the Tiptap editor instance for toolbar commands.
|
|
109
344
|
* The WysiwygEditor must be mounted as a sibling or descendant.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"WysiwygEditor.js","sourceRoot":"","sources":["../src/WysiwygEditor.tsx"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"WysiwygEditor.js","sourceRoot":"","sources":["../src/WysiwygEditor.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detectMarkdown.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/detectMarkdown.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { looksLikeMarkdown } from '../detectMarkdown';
|
|
3
|
+
describe('looksLikeMarkdown', () => {
|
|
4
|
+
it('returns false for empty / trivial text', () => {
|
|
5
|
+
expect(looksLikeMarkdown('')).toBe(false);
|
|
6
|
+
expect(looksLikeMarkdown(' ')).toBe(false);
|
|
7
|
+
expect(looksLikeMarkdown('Hello')).toBe(false);
|
|
8
|
+
});
|
|
9
|
+
it('returns false for plain prose', () => {
|
|
10
|
+
expect(looksLikeMarkdown('Hello world. This is just a sentence.')).toBe(false);
|
|
11
|
+
expect(looksLikeMarkdown('Multiple\nlines of\nplain text without any markdown syntax.')).toBe(false);
|
|
12
|
+
});
|
|
13
|
+
it('detects ATX headings', () => {
|
|
14
|
+
expect(looksLikeMarkdown('# Heading')).toBe(true);
|
|
15
|
+
expect(looksLikeMarkdown('## Subheading')).toBe(true);
|
|
16
|
+
expect(looksLikeMarkdown('###### Tiny heading')).toBe(true);
|
|
17
|
+
expect(looksLikeMarkdown('Some intro\n\n## A heading\n\nBody text')).toBe(true);
|
|
18
|
+
});
|
|
19
|
+
it('does not treat hash in middle of line as a heading', () => {
|
|
20
|
+
expect(looksLikeMarkdown('See issue #123 for details.')).toBe(false);
|
|
21
|
+
});
|
|
22
|
+
it('detects bullet lists', () => {
|
|
23
|
+
expect(looksLikeMarkdown('- first\n- second\n- third')).toBe(true);
|
|
24
|
+
expect(looksLikeMarkdown('* item one\n* item two')).toBe(true);
|
|
25
|
+
expect(looksLikeMarkdown('+ plus item')).toBe(true);
|
|
26
|
+
});
|
|
27
|
+
it('detects ordered lists', () => {
|
|
28
|
+
expect(looksLikeMarkdown('1. step one\n2. step two')).toBe(true);
|
|
29
|
+
});
|
|
30
|
+
it('detects blockquotes', () => {
|
|
31
|
+
expect(looksLikeMarkdown('> A quoted line')).toBe(true);
|
|
32
|
+
});
|
|
33
|
+
it('detects code fences', () => {
|
|
34
|
+
expect(looksLikeMarkdown('```\nconst x = 1;\n```')).toBe(true);
|
|
35
|
+
expect(looksLikeMarkdown('```ts\nlet y = 2;\n```')).toBe(true);
|
|
36
|
+
});
|
|
37
|
+
it('detects GFM tables', () => {
|
|
38
|
+
expect(looksLikeMarkdown('| Col1 | Col2 |\n| --- | --- |\n| a | b |')).toBe(true);
|
|
39
|
+
});
|
|
40
|
+
it('detects task lists', () => {
|
|
41
|
+
expect(looksLikeMarkdown('- [ ] todo\n- [x] done')).toBe(true);
|
|
42
|
+
});
|
|
43
|
+
it('detects mixed inline patterns (2+ hits)', () => {
|
|
44
|
+
expect(looksLikeMarkdown('**Important:** see [the docs](http://example.com) for more')).toBe(true);
|
|
45
|
+
expect(looksLikeMarkdown('Use `foo()` and read [the page](http://x.com)')).toBe(true);
|
|
46
|
+
});
|
|
47
|
+
it('does not trigger on a single inline pattern in plain prose', () => {
|
|
48
|
+
expect(looksLikeMarkdown('Visit https://example.com or [the docs](http://docs.com)')).toBe(false);
|
|
49
|
+
expect(looksLikeMarkdown('She said `hello` to him')).toBe(false);
|
|
50
|
+
});
|
|
51
|
+
it('detects markdown with windows line endings', () => {
|
|
52
|
+
expect(looksLikeMarkdown('# Heading\r\n\r\nBody text')).toBe(true);
|
|
53
|
+
});
|
|
54
|
+
it('detects a full resume-style document', () => {
|
|
55
|
+
const text = `# Mike Ammerlaan
|
|
56
|
+
|
|
57
|
+
## Projects
|
|
58
|
+
|
|
59
|
+
Qualla (qualla.com) - Designed and built a map-driven storytelling platform.
|
|
60
|
+
|
|
61
|
+
## **Experience**
|
|
62
|
+
|
|
63
|
+
**Principal Product Manager, Minecraft**
|
|
64
|
+
Jan 2021 - Present
|
|
65
|
+
Driving the creator platform.`;
|
|
66
|
+
expect(looksLikeMarkdown(text)).toBe(true);
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
//# sourceMappingURL=detectMarkdown.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detectMarkdown.test.js","sourceRoot":"","sources":["../../src/__tests__/detectMarkdown.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fileKind.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/fileKind.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { detectLanguageFromFileName, resolveFileKind } from '../fileKind';
|
|
3
|
+
describe('detectLanguageFromFileName', () => {
|
|
4
|
+
it('detects common extensions', () => {
|
|
5
|
+
expect(detectLanguageFromFileName('foo.ts')).toBe('typescript');
|
|
6
|
+
expect(detectLanguageFromFileName('app.py')).toBe('python');
|
|
7
|
+
expect(detectLanguageFromFileName('data.json')).toBe('json');
|
|
8
|
+
expect(detectLanguageFromFileName('index.html')).toBe('html');
|
|
9
|
+
expect(detectLanguageFromFileName('styles.css')).toBe('css');
|
|
10
|
+
expect(detectLanguageFromFileName('README.md')).toBe('markdown');
|
|
11
|
+
});
|
|
12
|
+
it('uses the final extension for multi-dot names', () => {
|
|
13
|
+
expect(detectLanguageFromFileName('archive.tar.gz')).toBe(null);
|
|
14
|
+
expect(detectLanguageFromFileName('component.test.ts')).toBe('typescript');
|
|
15
|
+
});
|
|
16
|
+
it('accepts bare extensions with or without a leading dot', () => {
|
|
17
|
+
expect(detectLanguageFromFileName('.ts')).toBe('typescript');
|
|
18
|
+
expect(detectLanguageFromFileName('ts')).toBe('typescript');
|
|
19
|
+
});
|
|
20
|
+
it('matches extensionless names like Dockerfile', () => {
|
|
21
|
+
expect(detectLanguageFromFileName('Dockerfile')).toBe('dockerfile');
|
|
22
|
+
expect(detectLanguageFromFileName('dockerfile')).toBe('dockerfile');
|
|
23
|
+
});
|
|
24
|
+
it('is case-insensitive for extensions', () => {
|
|
25
|
+
expect(detectLanguageFromFileName('foo.TS')).toBe('typescript');
|
|
26
|
+
expect(detectLanguageFromFileName('App.JSX')).toBe('javascript');
|
|
27
|
+
});
|
|
28
|
+
it('strips leading paths', () => {
|
|
29
|
+
expect(detectLanguageFromFileName('src/lib/foo.ts')).toBe('typescript');
|
|
30
|
+
expect(detectLanguageFromFileName('C:\\Users\\me\\app.py')).toBe('python');
|
|
31
|
+
});
|
|
32
|
+
it('returns null for unknown extensions', () => {
|
|
33
|
+
expect(detectLanguageFromFileName('foo.xyz')).toBe(null);
|
|
34
|
+
expect(detectLanguageFromFileName('mystery.blob')).toBe(null);
|
|
35
|
+
});
|
|
36
|
+
it('returns null for empty or trailing-dot inputs', () => {
|
|
37
|
+
expect(detectLanguageFromFileName('')).toBe(null);
|
|
38
|
+
expect(detectLanguageFromFileName(' ')).toBe(null);
|
|
39
|
+
expect(detectLanguageFromFileName('foo.')).toBe(null);
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
describe('resolveFileKind', () => {
|
|
43
|
+
it('defaults to markdown mode when nothing is supplied', () => {
|
|
44
|
+
expect(resolveFileKind()).toEqual({ mode: 'markdown', language: 'markdown' });
|
|
45
|
+
});
|
|
46
|
+
it('returns code mode for recognized code extensions', () => {
|
|
47
|
+
expect(resolveFileKind('foo.ts')).toEqual({ mode: 'code', language: 'typescript' });
|
|
48
|
+
expect(resolveFileKind('app.py')).toEqual({ mode: 'code', language: 'python' });
|
|
49
|
+
expect(resolveFileKind('data.json')).toEqual({ mode: 'code', language: 'json' });
|
|
50
|
+
});
|
|
51
|
+
it('keeps markdown extensions in markdown mode', () => {
|
|
52
|
+
expect(resolveFileKind('README.md')).toEqual({ mode: 'markdown', language: 'markdown' });
|
|
53
|
+
expect(resolveFileKind('doc.markdown')).toEqual({ mode: 'markdown', language: 'markdown' });
|
|
54
|
+
});
|
|
55
|
+
it('keeps .txt in markdown mode with plaintext language', () => {
|
|
56
|
+
expect(resolveFileKind('notes.txt')).toEqual({ mode: 'markdown', language: 'plaintext' });
|
|
57
|
+
});
|
|
58
|
+
it('falls back to markdown mode for unknown extensions', () => {
|
|
59
|
+
expect(resolveFileKind('foo.xyz')).toEqual({ mode: 'markdown', language: 'markdown' });
|
|
60
|
+
});
|
|
61
|
+
it('lets the explicit language override fileName detection', () => {
|
|
62
|
+
expect(resolveFileKind('foo.md', 'typescript')).toEqual({
|
|
63
|
+
mode: 'code',
|
|
64
|
+
language: 'typescript',
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
it('stays in markdown mode when language override is markdown or plaintext', () => {
|
|
68
|
+
expect(resolveFileKind('foo.ts', 'markdown')).toEqual({
|
|
69
|
+
mode: 'markdown',
|
|
70
|
+
language: 'markdown',
|
|
71
|
+
});
|
|
72
|
+
expect(resolveFileKind(undefined, 'plaintext')).toEqual({
|
|
73
|
+
mode: 'markdown',
|
|
74
|
+
language: 'plaintext',
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
it('accepts a language with no fileName', () => {
|
|
78
|
+
expect(resolveFileKind(undefined, 'rust')).toEqual({ mode: 'code', language: 'rust' });
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
//# sourceMappingURL=fileKind.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fileKind.test.js","sourceRoot":"","sources":["../../src/__tests__/fileKind.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mediaAttachmentFlow.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/mediaAttachmentFlow.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { markdownToTiptap, tiptapToMarkdown } from '../tiptapBridge';
|
|
3
|
+
/**
|
|
4
|
+
* Attachment-flow regression: earlier versions of MediaBin dropped
|
|
5
|
+
* uploaded files into the bin without inserting a markdown ref into
|
|
6
|
+
* the editor body. A user would upload an image, hit Send in the
|
|
7
|
+
* downstream chat composer, and the outgoing markdown would have no
|
|
8
|
+
* image reference — the gezel would reply "nothing came through."
|
|
9
|
+
*
|
|
10
|
+
* The fix: after `mediaProvider.addMedia(...)` succeeds, MediaBin
|
|
11
|
+
* fires `onMediaUploaded(relativePath, name, mimeType)`. The
|
|
12
|
+
* EditorShell wires this to an `insertAtCursor` that emits
|
|
13
|
+
* `` so the file actually participates
|
|
14
|
+
* in the outgoing markdown.
|
|
15
|
+
*
|
|
16
|
+
* These tests exercise the contract directly: the markdown snippet
|
|
17
|
+
* produced by the upload callback, once round-tripped through the
|
|
18
|
+
* editor's markdown↔HTML bridge, must round-trip back to a form
|
|
19
|
+
* the gezel service's image-extraction regex can see.
|
|
20
|
+
*/
|
|
21
|
+
function fakeMediaProvider(records) {
|
|
22
|
+
let counter = 0;
|
|
23
|
+
return {
|
|
24
|
+
async addMedia(name, _data, _mime) {
|
|
25
|
+
counter += 1;
|
|
26
|
+
const relative = `attachments/${counter}-${name}`;
|
|
27
|
+
records.push(relative);
|
|
28
|
+
return relative;
|
|
29
|
+
},
|
|
30
|
+
async resolveUrl(relPath) {
|
|
31
|
+
return relPath;
|
|
32
|
+
},
|
|
33
|
+
async listMedia() {
|
|
34
|
+
return [];
|
|
35
|
+
},
|
|
36
|
+
async removeMedia(_relPath) {
|
|
37
|
+
/* no-op */
|
|
38
|
+
},
|
|
39
|
+
dispose() {
|
|
40
|
+
/* no-op */
|
|
41
|
+
},
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Re-implements the exact snippet EditorShell's `insertMediaRef`
|
|
46
|
+
* builds. Keeping this aligned with the real impl would ordinarily
|
|
47
|
+
* rely on directly importing the helper; since it's currently inline
|
|
48
|
+
* in EditorShell, mirror the logic here and lean on the test to
|
|
49
|
+
* alert us if we drift apart.
|
|
50
|
+
*/
|
|
51
|
+
function buildAttachmentSnippet(relativePath, name, mimeType) {
|
|
52
|
+
const alt = name.replace(/\.[^.]+$/, '').replace(/[-_]/g, ' ');
|
|
53
|
+
return mimeType.startsWith('image/') ? `` : `[${alt}](${relativePath})`;
|
|
54
|
+
}
|
|
55
|
+
describe('media attachment flow', () => {
|
|
56
|
+
it('addMedia → buildAttachmentSnippet → markdown round-trip keeps the ref', async () => {
|
|
57
|
+
const records = [];
|
|
58
|
+
const provider = fakeMediaProvider(records);
|
|
59
|
+
// Simulate MediaBin.handleFileChange for a single PNG drop.
|
|
60
|
+
const pngBytes = new Uint8Array([0x89, 0x50, 0x4e, 0x47]);
|
|
61
|
+
const relative = await provider.addMedia('my_screenshot.png', pngBytes, 'image/png');
|
|
62
|
+
expect(records).toEqual([relative]);
|
|
63
|
+
const snippet = buildAttachmentSnippet(relative, 'my_screenshot.png', 'image/png');
|
|
64
|
+
expect(snippet).toBe(``);
|
|
65
|
+
// Insert snippet into the editor: markdown → HTML → markdown.
|
|
66
|
+
// This is the path a real insertAtCursor + tiptap onUpdate goes
|
|
67
|
+
// through. The outbound markdown must still contain the ref.
|
|
68
|
+
const html = markdownToTiptap(snippet);
|
|
69
|
+
expect(html).toMatch(/<img\b/);
|
|
70
|
+
expect(html).toContain(`src="${relative}"`);
|
|
71
|
+
const back = tiptapToMarkdown(html);
|
|
72
|
+
expect(back).toContain(``);
|
|
73
|
+
});
|
|
74
|
+
it('handles empty-alt (most common pasted-image shape)', async () => {
|
|
75
|
+
const records = [];
|
|
76
|
+
const provider = fakeMediaProvider(records);
|
|
77
|
+
const relative = await provider.addMedia('pasted.png', new Uint8Array([0]), 'image/png');
|
|
78
|
+
// Simulate what happens when alt is empty — common for bare pastes
|
|
79
|
+
// where the user hasn't typed a caption.
|
|
80
|
+
const snippet = ``;
|
|
81
|
+
const html = markdownToTiptap(snippet);
|
|
82
|
+
expect(html).toMatch(/<img\b/);
|
|
83
|
+
expect(html).toContain(`src="${relative}"`);
|
|
84
|
+
const back = tiptapToMarkdown(html);
|
|
85
|
+
expect(back).toContain(``);
|
|
86
|
+
});
|
|
87
|
+
it('non-image files fall back to a plain link, still preserving the ref', async () => {
|
|
88
|
+
const records = [];
|
|
89
|
+
const provider = fakeMediaProvider(records);
|
|
90
|
+
const relative = await provider.addMedia('design.pdf', new Uint8Array([0]), 'application/pdf');
|
|
91
|
+
const snippet = buildAttachmentSnippet(relative, 'design.pdf', 'application/pdf');
|
|
92
|
+
expect(snippet).toBe(`[design](${relative})`);
|
|
93
|
+
// Non-images don't go through the `<img>` regex — they stay as
|
|
94
|
+
// plain markdown links, which the service-side extractor ignores
|
|
95
|
+
// but the UI renders as normal hyperlinks.
|
|
96
|
+
expect(snippet).not.toContain('!');
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
//# sourceMappingURL=mediaAttachmentFlow.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mediaAttachmentFlow.test.js","sourceRoot":"","sources":["../../src/__tests__/mediaAttachmentFlow.test.ts"],"names":[],"mappings":"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"}
|