@bendyline/squisq-editor-react 1.0.0 → 1.1.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/DropZoneOverlay.d.ts +24 -0
- package/dist/DropZoneOverlay.d.ts.map +1 -0
- package/dist/DropZoneOverlay.js +53 -0
- package/dist/DropZoneOverlay.js.map +1 -0
- package/dist/EditorContext.d.ts +79 -0
- package/dist/EditorContext.d.ts.map +1 -0
- package/dist/EditorContext.js +204 -0
- package/dist/EditorContext.js.map +1 -0
- package/dist/EditorShell.d.ts +39 -0
- package/dist/EditorShell.d.ts.map +1 -0
- package/dist/EditorShell.js +104 -0
- package/dist/EditorShell.js.map +1 -0
- package/dist/MediaBin.d.ts +18 -0
- package/dist/MediaBin.d.ts.map +1 -0
- package/dist/MediaBin.js +141 -0
- package/dist/MediaBin.js.map +1 -0
- package/dist/PreviewPanel.d.ts +33 -0
- package/dist/PreviewPanel.d.ts.map +1 -0
- package/dist/PreviewPanel.js +385 -0
- package/dist/PreviewPanel.js.map +1 -0
- package/dist/RawEditor.d.ts +25 -0
- package/dist/RawEditor.d.ts.map +1 -0
- package/dist/RawEditor.js +100 -0
- package/dist/RawEditor.js.map +1 -0
- package/dist/StatusBar.d.ts +15 -0
- package/dist/StatusBar.d.ts.map +1 -0
- package/dist/StatusBar.js +24 -0
- package/dist/StatusBar.js.map +1 -0
- package/dist/TemplateAnnotation.d.ts +20 -0
- package/dist/TemplateAnnotation.d.ts.map +1 -0
- package/dist/TemplateAnnotation.js +69 -0
- package/dist/TemplateAnnotation.js.map +1 -0
- package/dist/Toolbar.d.ts +23 -0
- package/dist/Toolbar.d.ts.map +1 -0
- package/dist/Toolbar.js +350 -0
- package/dist/Toolbar.js.map +1 -0
- package/dist/ViewSwitcher.d.ts +14 -0
- package/dist/ViewSwitcher.d.ts.map +1 -0
- package/dist/ViewSwitcher.js +20 -0
- package/dist/ViewSwitcher.js.map +1 -0
- package/dist/WysiwygEditor.d.ts +28 -0
- package/dist/WysiwygEditor.d.ts.map +1 -0
- package/dist/WysiwygEditor.js +111 -0
- package/dist/WysiwygEditor.js.map +1 -0
- package/dist/hooks/useFileDrop.d.ts +41 -0
- package/dist/hooks/useFileDrop.d.ts.map +1 -0
- package/dist/hooks/useFileDrop.js +167 -0
- package/dist/hooks/useFileDrop.js.map +1 -0
- package/dist/index.d.ts +43 -268
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +39 -3832
- package/dist/index.js.map +1 -1
- package/dist/tiptapBridge.d.ts +24 -0
- package/dist/tiptapBridge.d.ts.map +1 -0
- package/dist/tiptapBridge.js +358 -0
- package/dist/tiptapBridge.js.map +1 -0
- package/dist/utils/dropUtils.d.ts +36 -0
- package/dist/utils/dropUtils.d.ts.map +1 -0
- package/dist/utils/dropUtils.js +71 -0
- package/dist/utils/dropUtils.js.map +1 -0
- package/package.json +5 -5
- package/src/DropZoneOverlay.tsx +137 -0
- package/src/EditorContext.tsx +56 -0
- package/src/EditorShell.tsx +102 -8
- package/src/MediaBin.tsx +223 -0
- package/src/Toolbar.tsx +21 -1
- package/src/hooks/useFileDrop.ts +226 -0
- package/src/index.ts +23 -0
- package/src/styles/editor.css +318 -0
- package/src/utils/dropUtils.ts +88 -0
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* DropZoneOverlay
|
|
3
|
+
*
|
|
4
|
+
* Full-editor overlay that appears when files are dragged over the editor.
|
|
5
|
+
* Shows contextual drop zones depending on the type of files being dragged:
|
|
6
|
+
* - Media files → single "Media" drop zone
|
|
7
|
+
* - Text files → two zones: "Insert" (at cursor) and "Replace" (all content)
|
|
8
|
+
* - Mixed files → all three zones
|
|
9
|
+
*/
|
|
10
|
+
import type { DragContentType, UseFileDropResult } from './hooks/useFileDrop';
|
|
11
|
+
export interface DropZoneOverlayProps {
|
|
12
|
+
/** What kind of content is being dragged */
|
|
13
|
+
dragContentType: DragContentType;
|
|
14
|
+
/** Factory that creates event props for a specific drop target */
|
|
15
|
+
zoneProps: UseFileDropResult['zoneProps'];
|
|
16
|
+
/** Whether a MediaProvider is available (disables media zone when false) */
|
|
17
|
+
hasMediaProvider: boolean;
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Full-size overlay with contextual drop targets for file uploads.
|
|
21
|
+
* Rendered conditionally by EditorShell when files are dragged over the editor.
|
|
22
|
+
*/
|
|
23
|
+
export declare function DropZoneOverlay({ dragContentType, zoneProps, hasMediaProvider, }: DropZoneOverlayProps): import("react/jsx-runtime").JSX.Element;
|
|
24
|
+
//# sourceMappingURL=DropZoneOverlay.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DropZoneOverlay.d.ts","sourceRoot":"","sources":["../src/DropZoneOverlay.tsx"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,KAAK,EAAE,eAAe,EAAc,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAE1F,MAAM,WAAW,oBAAoB;IACnC,4CAA4C;IAC5C,eAAe,EAAE,eAAe,CAAC;IACjC,kEAAkE;IAClE,SAAS,EAAE,iBAAiB,CAAC,WAAW,CAAC,CAAC;IAC1C,4EAA4E;IAC5E,gBAAgB,EAAE,OAAO,CAAC;CAC3B;AAED;;;GAGG;AACH,wBAAgB,eAAe,CAAC,EAC9B,eAAe,EACf,SAAS,EACT,gBAAgB,GACjB,EAAE,oBAAoB,2CAyCtB"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* DropZoneOverlay
|
|
4
|
+
*
|
|
5
|
+
* Full-editor overlay that appears when files are dragged over the editor.
|
|
6
|
+
* Shows contextual drop zones depending on the type of files being dragged:
|
|
7
|
+
* - Media files → single "Media" drop zone
|
|
8
|
+
* - Text files → two zones: "Insert" (at cursor) and "Replace" (all content)
|
|
9
|
+
* - Mixed files → all three zones
|
|
10
|
+
*/
|
|
11
|
+
import { useState } from 'react';
|
|
12
|
+
/**
|
|
13
|
+
* Full-size overlay with contextual drop targets for file uploads.
|
|
14
|
+
* Rendered conditionally by EditorShell when files are dragged over the editor.
|
|
15
|
+
*/
|
|
16
|
+
export function DropZoneOverlay({ dragContentType, zoneProps, hasMediaProvider, }) {
|
|
17
|
+
const showMedia = dragContentType === 'media' || dragContentType === 'mixed';
|
|
18
|
+
const showText = dragContentType === 'text' || dragContentType === 'mixed';
|
|
19
|
+
return (_jsx("div", { className: "squisq-drop-overlay", children: _jsxs("div", { className: "squisq-drop-overlay-inner", children: [showMedia && (_jsx(DropZone, { target: "media", zoneProps: zoneProps, icon: "\uD83D\uDCF7", label: "Media", description: hasMediaProvider ? 'Add to file bin' : 'No file storage configured', disabled: !hasMediaProvider, variant: "media" })), showText && (_jsxs(_Fragment, { children: [_jsx(DropZone, { target: "insert", zoneProps: zoneProps, icon: "\uD83D\uDCCB", label: "Insert", description: "Insert content at cursor", variant: "insert" }), _jsx(DropZone, { target: "replace", zoneProps: zoneProps, icon: "\uD83D\uDD04", label: "Replace", description: "Replace all editor content", variant: "replace" })] }))] }) }));
|
|
20
|
+
}
|
|
21
|
+
function DropZone({ target, zoneProps, icon, label, description, disabled, variant, }) {
|
|
22
|
+
const [isHovering, setIsHovering] = useState(false);
|
|
23
|
+
const props = zoneProps(target);
|
|
24
|
+
return (_jsxs("div", { className: [
|
|
25
|
+
'squisq-drop-zone',
|
|
26
|
+
`squisq-drop-zone--${variant}`,
|
|
27
|
+
isHovering && !disabled ? 'squisq-drop-zone--active' : '',
|
|
28
|
+
disabled ? 'squisq-drop-zone--disabled' : '',
|
|
29
|
+
]
|
|
30
|
+
.filter(Boolean)
|
|
31
|
+
.join(' '), onDragOver: (e) => {
|
|
32
|
+
if (disabled) {
|
|
33
|
+
e.preventDefault();
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
props.onDragOver(e);
|
|
37
|
+
}, onDragEnter: (e) => {
|
|
38
|
+
e.preventDefault();
|
|
39
|
+
if (!disabled)
|
|
40
|
+
setIsHovering(true);
|
|
41
|
+
}, onDragLeave: (e) => {
|
|
42
|
+
e.preventDefault();
|
|
43
|
+
setIsHovering(false);
|
|
44
|
+
}, onDrop: (e) => {
|
|
45
|
+
setIsHovering(false);
|
|
46
|
+
if (disabled) {
|
|
47
|
+
e.preventDefault();
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
props.onDrop(e);
|
|
51
|
+
}, children: [_jsx("span", { className: "squisq-drop-zone-icon", children: icon }), _jsx("span", { className: "squisq-drop-zone-label", children: label }), _jsx("span", { className: "squisq-drop-zone-desc", children: description })] }));
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=DropZoneOverlay.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DropZoneOverlay.js","sourceRoot":"","sources":["../src/DropZoneOverlay.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* EditorContext
|
|
3
|
+
*
|
|
4
|
+
* Shared React context that synchronizes state across all three editor views
|
|
5
|
+
* (Raw/Monaco, WYSIWYG/Tiptap, Preview/DocPlayer). When any view modifies the
|
|
6
|
+
* markdown source, the context re-parses and regenerates the MarkdownDocument
|
|
7
|
+
* and Doc so all views stay in sync.
|
|
8
|
+
*/
|
|
9
|
+
import { type ReactNode } from 'react';
|
|
10
|
+
import type { Doc } from '@bendyline/squisq/schemas';
|
|
11
|
+
import type { MarkdownDocument } from '@bendyline/squisq/markdown';
|
|
12
|
+
import type { Editor as TiptapEditor } from '@tiptap/core';
|
|
13
|
+
import type { editor as MonacoEditorNs } from 'monaco-editor';
|
|
14
|
+
/** Monaco standalone code editor instance type */
|
|
15
|
+
type MonacoEditor = MonacoEditorNs.IStandaloneCodeEditor;
|
|
16
|
+
export type EditorView = 'raw' | 'wysiwyg' | 'preview';
|
|
17
|
+
export type EditorTheme = 'light' | 'dark';
|
|
18
|
+
export interface EditorState {
|
|
19
|
+
/** Raw markdown source string */
|
|
20
|
+
markdownSource: string;
|
|
21
|
+
/** Parsed markdown document (JSON DOM) */
|
|
22
|
+
markdownDoc: MarkdownDocument | null;
|
|
23
|
+
/** Generated Doc (block hierarchy) */
|
|
24
|
+
doc: Doc | null;
|
|
25
|
+
/** Currently active editor view */
|
|
26
|
+
activeView: EditorView;
|
|
27
|
+
/** Parse error, if any */
|
|
28
|
+
parseError: string | null;
|
|
29
|
+
/** Whether a parse is pending */
|
|
30
|
+
isParsing: boolean;
|
|
31
|
+
/** Current color theme */
|
|
32
|
+
theme: EditorTheme;
|
|
33
|
+
}
|
|
34
|
+
export interface EditorActions {
|
|
35
|
+
/** Set markdown source and trigger re-parse */
|
|
36
|
+
setMarkdownSource: (source: string) => void;
|
|
37
|
+
/** Set markdown from a MarkdownDocument (e.g. from WYSIWYG) */
|
|
38
|
+
setMarkdownDoc: (doc: MarkdownDocument) => void;
|
|
39
|
+
/** Switch the active view */
|
|
40
|
+
setActiveView: (view: EditorView) => void;
|
|
41
|
+
/** Register / unregister the Tiptap editor instance (called by WysiwygEditor) */
|
|
42
|
+
setTiptapEditor: (editor: TiptapEditor | null) => void;
|
|
43
|
+
/** Register / unregister the Monaco editor instance (called by RawEditor) */
|
|
44
|
+
setMonacoEditor: (editor: MonacoEditor | null) => void;
|
|
45
|
+
/** Set the color theme */
|
|
46
|
+
setTheme: (theme: EditorTheme) => void;
|
|
47
|
+
/** Insert text at the current cursor position in the active editor */
|
|
48
|
+
insertAtCursor: (text: string) => void;
|
|
49
|
+
/** Replace all editor content with the given text */
|
|
50
|
+
replaceAll: (text: string) => void;
|
|
51
|
+
}
|
|
52
|
+
export interface EditorContextValue extends EditorState, EditorActions {
|
|
53
|
+
/** The live Tiptap editor instance (null when WYSIWYG is not mounted) */
|
|
54
|
+
tiptapEditor: TiptapEditor | null;
|
|
55
|
+
/** The live Monaco editor instance (null when Raw is not mounted) */
|
|
56
|
+
monacoEditor: MonacoEditor | null;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Hook to access the editor context. Must be used within an EditorProvider.
|
|
60
|
+
*/
|
|
61
|
+
export declare function useEditorContext(): EditorContextValue;
|
|
62
|
+
export interface EditorProviderProps {
|
|
63
|
+
/** Initial markdown content */
|
|
64
|
+
initialMarkdown?: string;
|
|
65
|
+
/** Initial active view */
|
|
66
|
+
initialView?: EditorView;
|
|
67
|
+
/** Article ID used when generating the Doc */
|
|
68
|
+
articleId?: string;
|
|
69
|
+
/** Color theme */
|
|
70
|
+
theme?: EditorTheme;
|
|
71
|
+
children: ReactNode;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Provides shared editor state to all child components.
|
|
75
|
+
* Automatically parses markdown and generates a Doc whenever the source changes.
|
|
76
|
+
*/
|
|
77
|
+
export declare function EditorProvider({ initialMarkdown, initialView, articleId, theme: initialTheme, children, }: EditorProviderProps): import("react/jsx-runtime").JSX.Element;
|
|
78
|
+
export {};
|
|
79
|
+
//# sourceMappingURL=EditorContext.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EditorContext.d.ts","sourceRoot":"","sources":["../src/EditorContext.tsx"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAQL,KAAK,SAAS,EACf,MAAM,OAAO,CAAC;AACf,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,2BAA2B,CAAC;AACrD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,4BAA4B,CAAC;AAGnE,OAAO,KAAK,EAAE,MAAM,IAAI,YAAY,EAAE,MAAM,cAAc,CAAC;AAC3D,OAAO,KAAK,EAAE,MAAM,IAAI,cAAc,EAAE,MAAM,eAAe,CAAC;AAG9D,kDAAkD;AAClD,KAAK,YAAY,GAAG,cAAc,CAAC,qBAAqB,CAAC;AAIzD,MAAM,MAAM,UAAU,GAAG,KAAK,GAAG,SAAS,GAAG,SAAS,CAAC;AACvD,MAAM,MAAM,WAAW,GAAG,OAAO,GAAG,MAAM,CAAC;AAE3C,MAAM,WAAW,WAAW;IAC1B,iCAAiC;IACjC,cAAc,EAAE,MAAM,CAAC;IACvB,0CAA0C;IAC1C,WAAW,EAAE,gBAAgB,GAAG,IAAI,CAAC;IACrC,sCAAsC;IACtC,GAAG,EAAE,GAAG,GAAG,IAAI,CAAC;IAChB,mCAAmC;IACnC,UAAU,EAAE,UAAU,CAAC;IACvB,0BAA0B;IAC1B,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,iCAAiC;IACjC,SAAS,EAAE,OAAO,CAAC;IACnB,0BAA0B;IAC1B,KAAK,EAAE,WAAW,CAAC;CACpB;AAED,MAAM,WAAW,aAAa;IAC5B,+CAA+C;IAC/C,iBAAiB,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI,CAAC;IAC5C,+DAA+D;IAC/D,cAAc,EAAE,CAAC,GAAG,EAAE,gBAAgB,KAAK,IAAI,CAAC;IAChD,6BAA6B;IAC7B,aAAa,EAAE,CAAC,IAAI,EAAE,UAAU,KAAK,IAAI,CAAC;IAC1C,iFAAiF;IACjF,eAAe,EAAE,CAAC,MAAM,EAAE,YAAY,GAAG,IAAI,KAAK,IAAI,CAAC;IACvD,6EAA6E;IAC7E,eAAe,EAAE,CAAC,MAAM,EAAE,YAAY,GAAG,IAAI,KAAK,IAAI,CAAC;IACvD,0BAA0B;IAC1B,QAAQ,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;IACvC,sEAAsE;IACtE,cAAc,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IACvC,qDAAqD;IACrD,UAAU,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;CACpC;AAED,MAAM,WAAW,kBAAmB,SAAQ,WAAW,EAAE,aAAa;IACpE,yEAAyE;IACzE,YAAY,EAAE,YAAY,GAAG,IAAI,CAAC;IAClC,qEAAqE;IACrE,YAAY,EAAE,YAAY,GAAG,IAAI,CAAC;CACnC;AAMD;;GAEG;AAEH,wBAAgB,gBAAgB,IAAI,kBAAkB,CAMrD;AAID,MAAM,WAAW,mBAAmB;IAClC,+BAA+B;IAC/B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,0BAA0B;IAC1B,WAAW,CAAC,EAAE,UAAU,CAAC;IACzB,8CAA8C;IAC9C,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,kBAAkB;IAClB,KAAK,CAAC,EAAE,WAAW,CAAC;IACpB,QAAQ,EAAE,SAAS,CAAC;CACrB;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,EAC7B,eAAoB,EACpB,WAAmB,EACnB,SAAsB,EACtB,KAAK,EAAE,YAAsB,EAC7B,QAAQ,GACT,EAAE,mBAAmB,2CA4LrB"}
|
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* EditorContext
|
|
4
|
+
*
|
|
5
|
+
* Shared React context that synchronizes state across all three editor views
|
|
6
|
+
* (Raw/Monaco, WYSIWYG/Tiptap, Preview/DocPlayer). When any view modifies the
|
|
7
|
+
* markdown source, the context re-parses and regenerates the MarkdownDocument
|
|
8
|
+
* and Doc so all views stay in sync.
|
|
9
|
+
*/
|
|
10
|
+
import { createContext, useContext, useState, useCallback, useMemo, useRef, useEffect, } from 'react';
|
|
11
|
+
import { parseMarkdown, stringifyMarkdown } from '@bendyline/squisq/markdown';
|
|
12
|
+
import { markdownToDoc } from '@bendyline/squisq/doc';
|
|
13
|
+
import { markdownToTiptap } from './tiptapBridge';
|
|
14
|
+
// ─── Context ─────────────────────────────────────────────
|
|
15
|
+
const EditorContext = createContext(null);
|
|
16
|
+
/**
|
|
17
|
+
* Hook to access the editor context. Must be used within an EditorProvider.
|
|
18
|
+
*/
|
|
19
|
+
// eslint-disable-next-line react-refresh/only-export-components
|
|
20
|
+
export function useEditorContext() {
|
|
21
|
+
const ctx = useContext(EditorContext);
|
|
22
|
+
if (!ctx) {
|
|
23
|
+
throw new Error('useEditorContext must be used within an <EditorProvider>');
|
|
24
|
+
}
|
|
25
|
+
return ctx;
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Provides shared editor state to all child components.
|
|
29
|
+
* Automatically parses markdown and generates a Doc whenever the source changes.
|
|
30
|
+
*/
|
|
31
|
+
export function EditorProvider({ initialMarkdown = '', initialView = 'raw', articleId = 'untitled', theme: initialTheme = 'light', children, }) {
|
|
32
|
+
const [markdownSource, setMarkdownSourceRaw] = useState(initialMarkdown);
|
|
33
|
+
const [markdownDoc, setMarkdownDocState] = useState(null);
|
|
34
|
+
const [doc, setDoc] = useState(null);
|
|
35
|
+
const [activeView, setActiveView] = useState(initialView);
|
|
36
|
+
const [parseError, setParseError] = useState(null);
|
|
37
|
+
const [isParsing, setIsParsing] = useState(false);
|
|
38
|
+
const [theme, setTheme] = useState(initialTheme);
|
|
39
|
+
const [tiptapEditor, setTiptapEditor] = useState(null);
|
|
40
|
+
const [monacoEditor, setMonacoEditor] = useState(null);
|
|
41
|
+
const articleIdRef = useRef(articleId);
|
|
42
|
+
articleIdRef.current = articleId;
|
|
43
|
+
// Sync theme when prop changes
|
|
44
|
+
useEffect(() => {
|
|
45
|
+
setTheme(initialTheme);
|
|
46
|
+
}, [initialTheme]);
|
|
47
|
+
// Debounced parse on markdown source change
|
|
48
|
+
const parseTimeoutRef = useRef(null);
|
|
49
|
+
const doParse = useCallback((source) => {
|
|
50
|
+
setIsParsing(true);
|
|
51
|
+
try {
|
|
52
|
+
const parsed = parseMarkdown(source);
|
|
53
|
+
setMarkdownDocState(parsed);
|
|
54
|
+
setParseError(null);
|
|
55
|
+
// Generate Doc from parsed markdown
|
|
56
|
+
try {
|
|
57
|
+
const generatedDoc = markdownToDoc(parsed, {
|
|
58
|
+
articleId: articleIdRef.current,
|
|
59
|
+
});
|
|
60
|
+
setDoc(generatedDoc);
|
|
61
|
+
}
|
|
62
|
+
catch (docErr) {
|
|
63
|
+
// Doc generation can fail but markdown parse succeeded
|
|
64
|
+
setDoc(null);
|
|
65
|
+
console.warn('Doc generation failed:', docErr instanceof Error ? docErr.message : docErr);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
catch (err) {
|
|
69
|
+
setParseError(err instanceof Error ? err.message : 'Parse error');
|
|
70
|
+
setMarkdownDocState(null);
|
|
71
|
+
setDoc(null);
|
|
72
|
+
}
|
|
73
|
+
finally {
|
|
74
|
+
setIsParsing(false);
|
|
75
|
+
}
|
|
76
|
+
}, []);
|
|
77
|
+
// Parse on source changes with debounce
|
|
78
|
+
useEffect(() => {
|
|
79
|
+
if (parseTimeoutRef.current) {
|
|
80
|
+
clearTimeout(parseTimeoutRef.current);
|
|
81
|
+
}
|
|
82
|
+
parseTimeoutRef.current = setTimeout(() => {
|
|
83
|
+
doParse(markdownSource);
|
|
84
|
+
}, 150);
|
|
85
|
+
return () => {
|
|
86
|
+
if (parseTimeoutRef.current) {
|
|
87
|
+
clearTimeout(parseTimeoutRef.current);
|
|
88
|
+
}
|
|
89
|
+
};
|
|
90
|
+
}, [markdownSource, doParse]);
|
|
91
|
+
// Initial parse
|
|
92
|
+
useEffect(() => {
|
|
93
|
+
if (initialMarkdown) {
|
|
94
|
+
doParse(initialMarkdown);
|
|
95
|
+
}
|
|
96
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
97
|
+
}, []);
|
|
98
|
+
const setMarkdownSource = useCallback((source) => {
|
|
99
|
+
setMarkdownSourceRaw(source);
|
|
100
|
+
}, []);
|
|
101
|
+
const insertAtCursor = useCallback((text) => {
|
|
102
|
+
if (activeView === 'wysiwyg' && tiptapEditor) {
|
|
103
|
+
// Insert as HTML so formatting is preserved
|
|
104
|
+
const html = markdownToTiptap(text);
|
|
105
|
+
tiptapEditor.chain().focus().insertContent(html).run();
|
|
106
|
+
}
|
|
107
|
+
else if (activeView === 'raw' && monacoEditor) {
|
|
108
|
+
const position = monacoEditor.getPosition();
|
|
109
|
+
if (position) {
|
|
110
|
+
const model = monacoEditor.getModel();
|
|
111
|
+
if (model) {
|
|
112
|
+
const range = {
|
|
113
|
+
startLineNumber: position.lineNumber,
|
|
114
|
+
startColumn: position.column,
|
|
115
|
+
endLineNumber: position.lineNumber,
|
|
116
|
+
endColumn: position.column,
|
|
117
|
+
};
|
|
118
|
+
monacoEditor.executeEdits('drop', [{ range, text }]);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
// No cursor — append
|
|
123
|
+
setMarkdownSourceRaw((prev) => prev + '\n\n' + text);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
else {
|
|
127
|
+
// Preview or no editor — append to end
|
|
128
|
+
setMarkdownSourceRaw((prev) => prev + '\n\n' + text);
|
|
129
|
+
}
|
|
130
|
+
}, [activeView, tiptapEditor, monacoEditor]);
|
|
131
|
+
const replaceAll = useCallback((text) => {
|
|
132
|
+
setMarkdownSourceRaw(text);
|
|
133
|
+
// Push to editors if mounted
|
|
134
|
+
if (tiptapEditor) {
|
|
135
|
+
const html = markdownToTiptap(text);
|
|
136
|
+
tiptapEditor.commands.setContent(html);
|
|
137
|
+
}
|
|
138
|
+
if (monacoEditor) {
|
|
139
|
+
monacoEditor.setValue(text);
|
|
140
|
+
}
|
|
141
|
+
}, [tiptapEditor, monacoEditor]);
|
|
142
|
+
const setMarkdownDoc = useCallback((newDoc) => {
|
|
143
|
+
setMarkdownDocState(newDoc);
|
|
144
|
+
// Stringify to update the raw source
|
|
145
|
+
try {
|
|
146
|
+
const newSource = stringifyMarkdown(newDoc);
|
|
147
|
+
setMarkdownSourceRaw(newSource);
|
|
148
|
+
setParseError(null);
|
|
149
|
+
// Generate Doc
|
|
150
|
+
try {
|
|
151
|
+
const generatedDoc = markdownToDoc(newDoc, {
|
|
152
|
+
articleId: articleIdRef.current,
|
|
153
|
+
});
|
|
154
|
+
setDoc(generatedDoc);
|
|
155
|
+
}
|
|
156
|
+
catch (docErr) {
|
|
157
|
+
setDoc(null);
|
|
158
|
+
console.warn('Doc generation failed:', docErr instanceof Error ? docErr.message : docErr);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
catch (err) {
|
|
162
|
+
setParseError(err instanceof Error ? err.message : 'Stringify error');
|
|
163
|
+
}
|
|
164
|
+
}, []);
|
|
165
|
+
const value = useMemo(() => ({
|
|
166
|
+
markdownSource,
|
|
167
|
+
markdownDoc,
|
|
168
|
+
doc,
|
|
169
|
+
activeView,
|
|
170
|
+
parseError,
|
|
171
|
+
isParsing,
|
|
172
|
+
theme,
|
|
173
|
+
tiptapEditor,
|
|
174
|
+
monacoEditor,
|
|
175
|
+
setMarkdownSource,
|
|
176
|
+
setMarkdownDoc,
|
|
177
|
+
setActiveView,
|
|
178
|
+
setTiptapEditor,
|
|
179
|
+
setMonacoEditor,
|
|
180
|
+
setTheme,
|
|
181
|
+
insertAtCursor,
|
|
182
|
+
replaceAll,
|
|
183
|
+
}), [
|
|
184
|
+
markdownSource,
|
|
185
|
+
markdownDoc,
|
|
186
|
+
doc,
|
|
187
|
+
activeView,
|
|
188
|
+
parseError,
|
|
189
|
+
isParsing,
|
|
190
|
+
theme,
|
|
191
|
+
tiptapEditor,
|
|
192
|
+
monacoEditor,
|
|
193
|
+
setMarkdownSource,
|
|
194
|
+
setMarkdownDoc,
|
|
195
|
+
setActiveView,
|
|
196
|
+
setTiptapEditor,
|
|
197
|
+
setMonacoEditor,
|
|
198
|
+
setTheme,
|
|
199
|
+
insertAtCursor,
|
|
200
|
+
replaceAll,
|
|
201
|
+
]);
|
|
202
|
+
return _jsx(EditorContext.Provider, { value: value, children: children });
|
|
203
|
+
}
|
|
204
|
+
//# sourceMappingURL=EditorContext.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EditorContext.js","sourceRoot":"","sources":["../src/EditorContext.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* EditorShell
|
|
3
|
+
*
|
|
4
|
+
* Top-level shell component that composes the Toolbar, ViewSwitcher, editor
|
|
5
|
+
* views, and StatusBar into a complete editing experience. Wraps everything
|
|
6
|
+
* in an EditorProvider for shared state.
|
|
7
|
+
*/
|
|
8
|
+
import { type EditorView } from './EditorContext';
|
|
9
|
+
import type { MediaProvider } from '@bendyline/squisq/schemas';
|
|
10
|
+
export type { EditorTheme } from './EditorContext';
|
|
11
|
+
export interface EditorShellProps {
|
|
12
|
+
/** Initial markdown content */
|
|
13
|
+
initialMarkdown?: string;
|
|
14
|
+
/** Initial active view */
|
|
15
|
+
/** Initial active view (default: 'wysiwyg') */
|
|
16
|
+
initialView?: EditorView;
|
|
17
|
+
/** Article ID for Doc generation */
|
|
18
|
+
articleId?: string;
|
|
19
|
+
/** Base path for media URLs in preview */
|
|
20
|
+
basePath?: string;
|
|
21
|
+
/** Called when markdown source changes */
|
|
22
|
+
onChange?: (source: string) => void;
|
|
23
|
+
/** Color theme: 'light' or 'dark' (default: 'light') */
|
|
24
|
+
theme?: 'light' | 'dark';
|
|
25
|
+
/** Additional class name */
|
|
26
|
+
className?: string;
|
|
27
|
+
/** CSS height for the shell container (default: '100vh') */
|
|
28
|
+
height?: string;
|
|
29
|
+
/** Optional MediaProvider for the Files panel. When set (even to null), a Files toggle appears in the toolbar. */
|
|
30
|
+
mediaProvider?: MediaProvider | null;
|
|
31
|
+
/** Show the Files toggle in the toolbar. Defaults to true when mediaProvider is passed. */
|
|
32
|
+
showFilesToggle?: boolean;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Complete markdown editor shell with toolbar, view switcher, and three
|
|
36
|
+
* editing modes: Raw (Monaco), WYSIWYG (Tiptap), and Preview.
|
|
37
|
+
*/
|
|
38
|
+
export declare function EditorShell({ initialMarkdown, initialView, articleId, basePath, onChange, theme, className, height, mediaProvider, showFilesToggle, }: EditorShellProps): import("react/jsx-runtime").JSX.Element;
|
|
39
|
+
//# sourceMappingURL=EditorShell.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EditorShell.d.ts","sourceRoot":"","sources":["../src/EditorShell.tsx"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAGH,OAAO,EAAoC,KAAK,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAepF,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAE/D,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAEnD,MAAM,WAAW,gBAAgB;IAC/B,+BAA+B;IAC/B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,0BAA0B;IAC1B,+CAA+C;IAC/C,WAAW,CAAC,EAAE,UAAU,CAAC;IACzB,oCAAoC;IACpC,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,0CAA0C;IAC1C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,0CAA0C;IAC1C,QAAQ,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI,CAAC;IACpC,wDAAwD;IACxD,KAAK,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IACzB,4BAA4B;IAC5B,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,4DAA4D;IAC5D,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,kHAAkH;IAClH,aAAa,CAAC,EAAE,aAAa,GAAG,IAAI,CAAC;IACrC,2FAA2F;IAC3F,eAAe,CAAC,EAAE,OAAO,CAAC;CAC3B;AAED;;;GAGG;AACH,wBAAgB,WAAW,CAAC,EAC1B,eAAoB,EACpB,WAAuB,EACvB,SAAsB,EACtB,QAAc,EACd,QAAQ,EACR,KAAe,EACf,SAAS,EACT,MAAgB,EAChB,aAAa,EACb,eAAe,GAChB,EAAE,gBAAgB,2CAqBlB"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* EditorShell
|
|
4
|
+
*
|
|
5
|
+
* Top-level shell component that composes the Toolbar, ViewSwitcher, editor
|
|
6
|
+
* views, and StatusBar into a complete editing experience. Wraps everything
|
|
7
|
+
* in an EditorProvider for shared state.
|
|
8
|
+
*/
|
|
9
|
+
import { useEffect, useState, useCallback } from 'react';
|
|
10
|
+
import { EditorProvider, useEditorContext } from './EditorContext';
|
|
11
|
+
import { Toolbar } from './Toolbar';
|
|
12
|
+
import { StatusBar } from './StatusBar';
|
|
13
|
+
import { RawEditor } from './RawEditor';
|
|
14
|
+
import { WysiwygEditor } from './WysiwygEditor';
|
|
15
|
+
import { PreviewPanel } from './PreviewPanel';
|
|
16
|
+
import { MediaBin } from './MediaBin';
|
|
17
|
+
import { DropZoneOverlay } from './DropZoneOverlay';
|
|
18
|
+
import { useFileDrop } from './hooks/useFileDrop';
|
|
19
|
+
import { partitionFiles, processMediaFiles, processTextFile, processTextFiles, } from './utils/dropUtils';
|
|
20
|
+
/**
|
|
21
|
+
* Complete markdown editor shell with toolbar, view switcher, and three
|
|
22
|
+
* editing modes: Raw (Monaco), WYSIWYG (Tiptap), and Preview.
|
|
23
|
+
*/
|
|
24
|
+
export function EditorShell({ initialMarkdown = '', initialView = 'wysiwyg', articleId = 'untitled', basePath = '/', onChange, theme = 'light', className, height = '100vh', mediaProvider, showFilesToggle, }) {
|
|
25
|
+
// Show the toggle when explicitly opted in, or when mediaProvider prop was passed at all
|
|
26
|
+
const filesToggleEnabled = showFilesToggle ?? mediaProvider !== undefined;
|
|
27
|
+
return (_jsx(EditorProvider, { initialMarkdown: initialMarkdown, initialView: initialView, articleId: articleId, theme: theme, children: _jsx(EditorShellInner, { basePath: basePath, onChange: onChange, className: className, height: height, mediaProvider: mediaProvider ?? null, filesToggleEnabled: filesToggleEnabled }) }));
|
|
28
|
+
}
|
|
29
|
+
function EditorShellInner({ basePath, onChange, className, height, mediaProvider, filesToggleEnabled, }) {
|
|
30
|
+
const { activeView, markdownSource, theme, insertAtCursor, replaceAll } = useEditorContext();
|
|
31
|
+
const [showFiles, setShowFiles] = useState(false);
|
|
32
|
+
const [mediaRefreshKey, setMediaRefreshKey] = useState(0);
|
|
33
|
+
const isDark = theme === 'dark';
|
|
34
|
+
const handleToggleFiles = useCallback(() => {
|
|
35
|
+
setShowFiles((prev) => !prev);
|
|
36
|
+
}, []);
|
|
37
|
+
// ── Drag-and-drop file handling ──
|
|
38
|
+
const handleFileDrop = useCallback(async (files, target) => {
|
|
39
|
+
try {
|
|
40
|
+
const { media, text } = partitionFiles(files);
|
|
41
|
+
// Process media files
|
|
42
|
+
if (media.length > 0 && mediaProvider) {
|
|
43
|
+
await processMediaFiles(media, mediaProvider);
|
|
44
|
+
setMediaRefreshKey((k) => k + 1);
|
|
45
|
+
// Auto-open the media bin so the user sees the new files
|
|
46
|
+
if (!showFiles)
|
|
47
|
+
setShowFiles(true);
|
|
48
|
+
}
|
|
49
|
+
// Process text files
|
|
50
|
+
if (text.length > 0) {
|
|
51
|
+
if (target === 'replace') {
|
|
52
|
+
// Replace with first text file
|
|
53
|
+
const content = await processTextFile(text[0]);
|
|
54
|
+
replaceAll(content);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
// Insert all text files concatenated
|
|
58
|
+
const content = await processTextFiles(text);
|
|
59
|
+
insertAtCursor(content);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
catch (err) {
|
|
64
|
+
console.error('Failed to process dropped files:', err instanceof Error ? err.message : err);
|
|
65
|
+
}
|
|
66
|
+
}, [mediaProvider, showFiles, replaceAll, insertAtCursor]);
|
|
67
|
+
const { isDragging, dragContentType, containerProps, zoneProps } = useFileDrop({
|
|
68
|
+
onDrop: handleFileDrop,
|
|
69
|
+
});
|
|
70
|
+
// Notify parent of changes
|
|
71
|
+
useEffect(() => {
|
|
72
|
+
onChange?.(markdownSource);
|
|
73
|
+
}, [markdownSource, onChange]);
|
|
74
|
+
// Keyboard shortcuts for view switching
|
|
75
|
+
useEffect(() => {
|
|
76
|
+
const handler = (e) => {
|
|
77
|
+
if (e.ctrlKey || e.metaKey) {
|
|
78
|
+
switch (e.key) {
|
|
79
|
+
case '1':
|
|
80
|
+
e.preventDefault();
|
|
81
|
+
document.querySelector('[data-view="wysiwyg"]')?.click();
|
|
82
|
+
break;
|
|
83
|
+
case '2':
|
|
84
|
+
e.preventDefault();
|
|
85
|
+
document.querySelector('[data-view="raw"]')?.click();
|
|
86
|
+
break;
|
|
87
|
+
case '3':
|
|
88
|
+
e.preventDefault();
|
|
89
|
+
document.querySelector('[data-view="preview"]')?.click();
|
|
90
|
+
break;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
window.addEventListener('keydown', handler);
|
|
95
|
+
return () => window.removeEventListener('keydown', handler);
|
|
96
|
+
}, []);
|
|
97
|
+
return (_jsxs("div", { className: `squisq-editor-shell ${className || ''}`, "data-theme": theme, style: {
|
|
98
|
+
display: 'flex',
|
|
99
|
+
flexDirection: 'column',
|
|
100
|
+
height,
|
|
101
|
+
overflow: 'hidden',
|
|
102
|
+
}, ...containerProps, children: [_jsx("div", { className: "squisq-editor-header", children: _jsx(Toolbar, { showFiles: showFiles, onToggleFiles: filesToggleEnabled ? handleToggleFiles : undefined }) }), _jsxs("div", { className: "squisq-editor-content", style: { flex: 1, overflow: 'hidden', position: 'relative', display: 'flex' }, children: [_jsxs("div", { style: { flex: 1, overflow: 'hidden', position: 'relative' }, children: [activeView === 'raw' && _jsx(RawEditor, { theme: theme === 'dark' ? 'vs-dark' : 'vs' }), activeView === 'wysiwyg' && _jsx(WysiwygEditor, {}), activeView === 'preview' && _jsx(PreviewPanel, { basePath: basePath })] }), showFiles && (_jsx(MediaBin, { mediaProvider: mediaProvider, isDark: isDark, refreshKey: mediaRefreshKey })), isDragging && (_jsx(DropZoneOverlay, { dragContentType: dragContentType, zoneProps: zoneProps, hasMediaProvider: mediaProvider !== null }))] }), _jsx(StatusBar, {})] }));
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=EditorShell.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EditorShell.js","sourceRoot":"","sources":["../src/EditorShell.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MediaBin
|
|
3
|
+
*
|
|
4
|
+
* Toggleable side panel that displays files associated with the current
|
|
5
|
+
* content. Shows image thumbnails, icons for other types, file sizes,
|
|
6
|
+
* and provides an upload button to add new media.
|
|
7
|
+
*/
|
|
8
|
+
import type { MediaProvider } from '@bendyline/squisq/schemas';
|
|
9
|
+
export interface MediaBinProps {
|
|
10
|
+
/** The active MediaProvider (null when no media context is available) */
|
|
11
|
+
mediaProvider: MediaProvider | null;
|
|
12
|
+
/** Whether the editor is in dark mode */
|
|
13
|
+
isDark: boolean;
|
|
14
|
+
/** Incremented externally to signal a re-scan of the media list */
|
|
15
|
+
refreshKey?: number;
|
|
16
|
+
}
|
|
17
|
+
export declare function MediaBin({ mediaProvider, isDark, refreshKey }: MediaBinProps): import("react/jsx-runtime").JSX.Element;
|
|
18
|
+
//# sourceMappingURL=MediaBin.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MediaBin.d.ts","sourceRoot":"","sources":["../src/MediaBin.tsx"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAGH,OAAO,KAAK,EAAE,aAAa,EAAc,MAAM,2BAA2B,CAAC;AAM3E,MAAM,WAAW,aAAa;IAC5B,yEAAyE;IACzE,aAAa,EAAE,aAAa,GAAG,IAAI,CAAC;IACpC,yCAAyC;IACzC,MAAM,EAAE,OAAO,CAAC;IAChB,mEAAmE;IACnE,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AA6BD,wBAAgB,QAAQ,CAAC,EAAE,aAAa,EAAE,MAAM,EAAE,UAAU,EAAE,EAAE,aAAa,2CA2K5E"}
|