@echozedlabs/react 0.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/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 Eric Zimmerman (echozed)
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,70 @@
1
+ # @echozedlabs/react
2
+
3
+ [![npm](https://img.shields.io/npm/v/@echozedlabs/react.svg)](https://www.npmjs.com/package/@echozedlabs/react)
4
+ [![license: MIT](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/echozulucode/markdown-editor/blob/main/LICENSE)
5
+
6
+ A modular, embeddable **Markdown editor for React** — Markdown-first with a byte-stable round-trip, and four modes behind one component: **source / hybrid / preview / rich text (WYSIWYG)**.
7
+
8
+ Raw Markdown bytes are the source of truth, so switching modes never silently rewrites your document; frontmatter is preserved verbatim.
9
+
10
+ ## Install
11
+
12
+ ```bash
13
+ npm i @echozedlabs/react react react-dom
14
+ # or: pnpm add @echozedlabs/react
15
+ ```
16
+
17
+ `react` and `react-dom` (>=18.2) are peer dependencies.
18
+
19
+ ## Usage
20
+
21
+ ```tsx
22
+ import { useState } from 'react';
23
+ import { MarkdownEditor } from '@echozedlabs/react';
24
+ import '@echozedlabs/react/styles.css';
25
+
26
+ export function Editor() {
27
+ const [value, setValue] = useState('# Hello\n\nStart writing…\n');
28
+ return (
29
+ <MarkdownEditor
30
+ value={value}
31
+ onChange={setValue}
32
+ modes={['markdown', 'hybrid', 'preview', 'wysiwyg']}
33
+ initialMode="hybrid"
34
+ onSaveShortcut={() => save(value)} // Cmd/Ctrl+S
35
+ />
36
+ );
37
+ }
38
+ ```
39
+
40
+ Don't forget to import the stylesheet: `import '@echozedlabs/react/styles.css'`.
41
+
42
+ ## Key props
43
+
44
+ | Prop | Purpose |
45
+ |---|---|
46
+ | `value` / `defaultValue` | Controlled / uncontrolled Markdown string. |
47
+ | `mode` / `initialMode` / `modes` | Active mode, initial mode, and which of `markdown`/`hybrid`/`preview`/`wysiwyg` are available. |
48
+ | `onChange` / `onModeChange` | Fired on Markdown edits / mode switches. |
49
+ | `onSaveShortcut` / `onCancelShortcut` | Cmd/Ctrl+S and Escape handlers. |
50
+ | `readOnly` | Render without editing. |
51
+ | `hostServices` | Optional link-search and image-upload integration. |
52
+ | `renderers` | Override/extend the preview renderer registry. |
53
+ | `ariaLabel` | Accessible label for the editor. |
54
+
55
+ The component also forwards an imperative `MarkdownEditorHandle` ref (`getMarkdown`, `setMode`, `insertMarkdown`, …).
56
+
57
+ ## How it fits together
58
+
59
+ `@echozedlabs/react` ties together the focused packages:
60
+
61
+ - [`@echozedlabs/core`](https://www.npmjs.com/package/@echozedlabs/core) — the Markdown codec (parse/serialize/replaceBody) + frontmatter split.
62
+ - [`@echozedlabs/codemirror`](https://www.npmjs.com/package/@echozedlabs/codemirror) — CodeMirror 6 source/hybrid view.
63
+ - [`@echozedlabs/wysiwyg-lexical`](https://www.npmjs.com/package/@echozedlabs/wysiwyg-lexical) — Lexical rich-text surface.
64
+ - [`@echozedlabs/renderers`](https://www.npmjs.com/package/@echozedlabs/renderers) — markdown, Shiki, Mermaid, PlantUML renderers.
65
+
66
+ Installing `@echozedlabs/react` pulls these in automatically.
67
+
68
+ ## License
69
+
70
+ [MIT](https://github.com/echozulucode/markdown-editor/blob/main/LICENSE) © Eric Zimmerman (echozed)
@@ -0,0 +1,15 @@
1
+ import React from 'react';
2
+ import type { EditorMode, MarkdownEditorHandle, MarkdownEditorProps } from '@echozedlabs/core';
3
+ import { type RendererRegistry } from '@echozedlabs/renderers';
4
+ import type { WysiwygToolbarIcons } from '@echozedlabs/wysiwyg-lexical';
5
+ export interface MarkdownEditorComponentProps extends Omit<MarkdownEditorProps, 'extensions' | 'rendererRegistry' | 'renderers'> {
6
+ renderers?: RendererRegistry;
7
+ /** Host-supplied icons for the top-level mode switcher. Keep icon packs out of the reusable package. */
8
+ modeIcons?: Partial<Record<EditorMode, React.ReactNode>>;
9
+ /** Whether host service controls such as the explicit link-search box should render in the toolbar. */
10
+ hostServiceToolbar?: boolean;
11
+ wysiwygToolbarIcons?: WysiwygToolbarIcons;
12
+ className?: string;
13
+ }
14
+ export declare const MarkdownEditor: React.ForwardRefExoticComponent<MarkdownEditorComponentProps & React.RefAttributes<MarkdownEditorHandle>>;
15
+ //# sourceMappingURL=MarkdownEditor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MarkdownEditor.d.ts","sourceRoot":"","sources":["../src/MarkdownEditor.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,EAEV,UAAU,EAGV,oBAAoB,EACpB,mBAAmB,EAGpB,MAAM,mBAAmB,CAAC;AAM3B,OAAO,EAIL,KAAK,gBAAgB,EACtB,MAAM,wBAAwB,CAAC;AAChC,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AAUxE,MAAM,WAAW,4BACf,SAAQ,IAAI,CAAC,mBAAmB,EAAE,YAAY,GAAG,kBAAkB,GAAG,WAAW,CAAC;IAClF,SAAS,CAAC,EAAE,gBAAgB,CAAC;IAC7B,wGAAwG;IACxG,SAAS,CAAC,EAAE,OAAO,CAAC,MAAM,CAAC,UAAU,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;IACzD,uGAAuG;IACvG,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B,mBAAmB,CAAC,EAAE,mBAAmB,CAAC;IAC1C,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,eAAO,MAAM,cAAc,2GAkXzB,CAAC"}
@@ -0,0 +1,407 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import React from 'react';
3
+ import { createMarkdownEditorView, } from '@echozedlabs/codemirror';
4
+ import { createDefaultRendererRegistry, renderMarkdownToHtml, } from '@echozedlabs/renderers';
5
+ import { sanitizePreviewHtml } from './sanitizeHtml.js';
6
+ const CODEMIRROR_MODES = new Set(['markdown', 'hybrid']);
7
+ const DEFAULT_MODES = ['hybrid', 'markdown', 'preview'];
8
+ const LazyWysiwygLexicalEditor = React.lazy(async () => {
9
+ const module = await import('@echozedlabs/wysiwyg-lexical');
10
+ return { default: module.WysiwygLexicalEditor };
11
+ });
12
+ export const MarkdownEditor = React.forwardRef(function MarkdownEditor({ value, defaultValue = '', mode, defaultMode, initialMode, modes = DEFAULT_MODES, readOnly = false, className, ariaLabel = 'Markdown editor', onChange, onModeChange, onSaveShortcut, onCancelShortcut, onDiagnostics, renderers, hostServices, propertySchema, frontmatterDisplay, modeIcons, hostServiceToolbar = true, wysiwygToolbarIcons, }, forwardedRef) {
13
+ const allowedModes = React.useMemo(() => normalizeModes(modes), [modes]);
14
+ const firstMode = initialMode ?? defaultMode ?? preferredInitialMode(allowedModes);
15
+ const [internalMarkdown, setInternalMarkdown] = React.useState(defaultValue);
16
+ const [internalMode, setInternalMode] = React.useState(allowedModes.includes(firstMode) ? firstMode : preferredInitialMode(allowedModes));
17
+ const normalizedFrontmatterDisplay = frontmatterDisplay ?? 'expanded';
18
+ const [showProperties, setShowProperties] = React.useState(normalizedFrontmatterDisplay !== 'hidden');
19
+ const markdown = value ?? internalMarkdown;
20
+ const activeMode = mode ?? internalMode;
21
+ const hasFrontmatter = /^---\r?\n/.test(markdown);
22
+ const hasHostServiceControls = !readOnly && hostServiceToolbar && activeMode !== 'preview' && (hostServices?.searchLinks !== undefined || hostServices?.uploadAsset !== undefined);
23
+ const showToolbar = allowedModes.length > 1 || (activeMode === 'hybrid' && hasFrontmatter) || hasHostServiceControls;
24
+ const hostRef = React.useRef(null);
25
+ const cmRef = React.useRef(null);
26
+ const markdownRef = React.useRef(markdown);
27
+ const modeRef = React.useRef(activeMode);
28
+ const onChangeRef = React.useRef(onChange);
29
+ const onSaveShortcutRef = React.useRef(onSaveShortcut);
30
+ const onCancelShortcutRef = React.useRef(onCancelShortcut);
31
+ const onDiagnosticsRef = React.useRef(onDiagnostics);
32
+ markdownRef.current = markdown;
33
+ modeRef.current = activeMode;
34
+ onChangeRef.current = onChange;
35
+ onSaveShortcutRef.current = onSaveShortcut;
36
+ onCancelShortcutRef.current = onCancelShortcut;
37
+ onDiagnosticsRef.current = onDiagnostics;
38
+ const isCodeMirrorMode = CODEMIRROR_MODES.has(activeMode);
39
+ const hybridRenderMarkdown = React.useCallback(async (blockMarkdown, context) => {
40
+ const activeRegistry = renderers ?? createDefaultRendererRegistry();
41
+ const result = await renderMarkdownToHtml(blockMarkdown, {
42
+ registry: activeRegistry,
43
+ signal: context.signal,
44
+ });
45
+ onDiagnosticsRef.current?.(result.diagnostics);
46
+ return { html: result.html };
47
+ }, [renderers]);
48
+ const wysiwygRenderServices = React.useMemo(() => ({
49
+ renderPlantUml: async (source, context) => {
50
+ if (hostServices?.renderPlantUml) {
51
+ const result = await hostServices.renderPlantUml(source, { signal: context.signal });
52
+ return { html: result.html, diagnostics: result.diagnostics };
53
+ }
54
+ const activeRegistry = renderers ?? createDefaultRendererRegistry();
55
+ const result = await renderMarkdownToHtml(`\`\`\`plantuml\n${source}\n\`\`\``, {
56
+ registry: activeRegistry,
57
+ signal: context.signal,
58
+ });
59
+ return { html: result.html, diagnostics: result.diagnostics };
60
+ },
61
+ }), [hostServices, renderers]);
62
+ const emitDiagnostics = React.useCallback((diagnostics) => {
63
+ onDiagnosticsRef.current?.(diagnostics);
64
+ }, []);
65
+ React.useEffect(() => {
66
+ if (!isCodeMirrorMode || !hostRef.current) {
67
+ cmRef.current?.destroy();
68
+ cmRef.current = null;
69
+ return;
70
+ }
71
+ const cmMode = activeMode;
72
+ const handle = createMarkdownEditorView({
73
+ parent: hostRef.current,
74
+ markdown,
75
+ mode: cmMode,
76
+ readOnly,
77
+ attributes: {
78
+ 'aria-label': ariaLabel,
79
+ class: 'me-codemirror',
80
+ },
81
+ onChange: (nextMarkdown, meta) => {
82
+ markdownRef.current = nextMarkdown;
83
+ if (value === undefined) {
84
+ setInternalMarkdown(nextMarkdown);
85
+ }
86
+ onChangeRef.current?.(nextMarkdown, {
87
+ ...meta,
88
+ mode: modeRef.current,
89
+ });
90
+ },
91
+ hybridFrontmatterMode: showProperties
92
+ ? normalizedFrontmatterDisplay === 'collapsed'
93
+ ? 'collapsed'
94
+ : 'table'
95
+ : 'hidden',
96
+ frontmatterSchema: propertySchema,
97
+ hybridRenderMarkdown,
98
+ });
99
+ cmRef.current = handle;
100
+ return () => {
101
+ handle.destroy();
102
+ if (cmRef.current === handle) {
103
+ cmRef.current = null;
104
+ }
105
+ };
106
+ }, [ariaLabel, activeMode, hybridRenderMarkdown, isCodeMirrorMode, normalizedFrontmatterDisplay, propertySchema, readOnly, showProperties]);
107
+ React.useEffect(() => {
108
+ setShowProperties(normalizedFrontmatterDisplay !== 'hidden');
109
+ }, [normalizedFrontmatterDisplay]);
110
+ React.useEffect(() => {
111
+ const handle = cmRef.current;
112
+ if (!handle) {
113
+ return;
114
+ }
115
+ if (handle.getMarkdown() !== markdown) {
116
+ handle.setMarkdown(markdown);
117
+ }
118
+ }, [markdown]);
119
+ React.useEffect(() => {
120
+ cmRef.current?.setReadOnly(readOnly || activeMode === 'preview');
121
+ }, [activeMode, readOnly]);
122
+ React.useEffect(() => {
123
+ const onKeyDown = (event) => {
124
+ const save = (event.metaKey || event.ctrlKey) && event.key.toLowerCase() === 's';
125
+ if (save) {
126
+ event.preventDefault();
127
+ onSaveShortcutRef.current?.();
128
+ return;
129
+ }
130
+ if (event.key === 'Escape') {
131
+ onCancelShortcutRef.current?.();
132
+ }
133
+ };
134
+ const current = hostRef.current;
135
+ current?.addEventListener('keydown', onKeyDown);
136
+ return () => current?.removeEventListener('keydown', onKeyDown);
137
+ }, []);
138
+ React.useImperativeHandle(forwardedRef, () => ({
139
+ focus() {
140
+ cmRef.current?.focus();
141
+ },
142
+ getMarkdown() {
143
+ return markdownRef.current;
144
+ },
145
+ setMarkdown(nextMarkdown) {
146
+ updateMarkdown(nextMarkdown, 'programmatic');
147
+ },
148
+ getMode() {
149
+ return modeRef.current;
150
+ },
151
+ setMode(nextMode) {
152
+ selectMode(nextMode, 'programmatic');
153
+ },
154
+ getSelection() {
155
+ const selection = cmRef.current?.getSelection();
156
+ return selection ? cmSelectionToTextSelection(selection) : null;
157
+ },
158
+ setSelection(selection) {
159
+ cmRef.current?.setSelection({
160
+ anchor: selection.from,
161
+ head: selection.to,
162
+ });
163
+ },
164
+ insertMarkdown(insertedMarkdown) {
165
+ if (cmRef.current) {
166
+ cmRef.current.insertMarkdown(insertedMarkdown);
167
+ return;
168
+ }
169
+ updateMarkdown(`${markdownRef.current}${insertedMarkdown}`, 'programmatic');
170
+ },
171
+ clearHistory() {
172
+ // Native CodeMirror history is intentionally preserved for now.
173
+ },
174
+ getSnapshot() {
175
+ return {
176
+ markdown: markdownRef.current,
177
+ version: 1,
178
+ mode: modeRef.current,
179
+ selection: undefined,
180
+ };
181
+ },
182
+ replaceMarkdown(nextMarkdown, meta) {
183
+ updateMarkdown(nextMarkdown, meta?.source ?? 'programmatic');
184
+ onChangeRef.current?.(nextMarkdown, {
185
+ source: 'programmatic',
186
+ timestamp: Date.now(),
187
+ mode: modeRef.current,
188
+ ...meta,
189
+ });
190
+ },
191
+ }), [value]);
192
+ function selectMode(nextMode, source = 'user') {
193
+ if (!allowedModes.includes(nextMode) || nextMode === activeMode) {
194
+ return;
195
+ }
196
+ const previousMode = activeMode;
197
+ if (mode === undefined) {
198
+ setInternalMode(nextMode);
199
+ }
200
+ onModeChange?.(nextMode, {
201
+ previousMode,
202
+ nextMode,
203
+ source,
204
+ timestamp: Date.now(),
205
+ });
206
+ }
207
+ function updateMarkdown(nextMarkdown, source) {
208
+ markdownRef.current = nextMarkdown;
209
+ cmRef.current?.setMarkdown(nextMarkdown, { emitChange: true });
210
+ if (value === undefined) {
211
+ setInternalMarkdown(nextMarkdown);
212
+ }
213
+ onChangeRef.current?.(nextMarkdown, {
214
+ source,
215
+ mode: modeRef.current,
216
+ timestamp: Date.now(),
217
+ });
218
+ }
219
+ function insertHostMarkdown(insertedMarkdown) {
220
+ if (!insertedMarkdown || readOnly || activeMode === 'preview') {
221
+ return;
222
+ }
223
+ if (cmRef.current) {
224
+ cmRef.current.insertMarkdown(insertedMarkdown);
225
+ return;
226
+ }
227
+ const separator = markdownRef.current.length > 0 && !markdownRef.current.endsWith('\n') ? '\n' : '';
228
+ updateMarkdown(`${markdownRef.current}${separator}${insertedMarkdown}`, 'host');
229
+ }
230
+ return (_jsxs("section", { className: ['me-editor', className].filter(Boolean).join(' '), "data-mode": activeMode, "data-readonly": readOnly ? 'true' : 'false', "aria-label": ariaLabel, children: [showToolbar ? (_jsxs("div", { className: "me-toolbar", role: "toolbar", "aria-label": "Editor controls", children: [allowedModes.length > 1
231
+ ? allowedModes.map((allowedMode) => {
232
+ const label = modeLabel(allowedMode);
233
+ const icon = modeIcons?.[allowedMode];
234
+ return (_jsxs("button", { type: "button", className: "me-mode-button", "data-active": allowedMode === activeMode ? 'true' : 'false', "data-icon-only": icon ? 'true' : 'false', "aria-label": label, title: label, "aria-pressed": allowedMode === activeMode, onClick: () => selectMode(allowedMode), children: [icon ? _jsx("span", { className: "me-mode-button-icon", "aria-hidden": "true", children: icon }) : null, _jsx("span", { className: icon ? 'me-sr-only' : undefined, children: label })] }, allowedMode));
235
+ })
236
+ : null, activeMode === 'hybrid' && hasFrontmatter ? (_jsxs("button", { type: "button", className: "me-mode-button me-properties-toggle", "data-active": showProperties ? 'true' : 'false', "data-icon-only": "true", "aria-label": showProperties ? 'Hide properties' : 'Show properties', title: showProperties ? 'Hide properties' : 'Show properties', "aria-pressed": showProperties, onClick: () => setShowProperties((current) => !current), children: [_jsx("span", { className: "me-mode-button-icon", "aria-hidden": "true", children: showProperties ? _jsx(IconSvg, { path: SLIDERS_ICON_PATH, dataIcon: "sliders" }) : _jsx(IconSvg, { path: LIST_ICON_PATH, dataIcon: "list" }) }), _jsx("span", { className: "me-sr-only", children: showProperties ? 'Hide properties' : 'Show properties' })] })) : null, hasHostServiceControls ? (_jsx(HostServiceToolbar, { services: hostServices, onInsertMarkdown: insertHostMarkdown, onDiagnostics: emitDiagnostics })) : null] })) : null, isCodeMirrorMode ? (_jsx("div", { ref: hostRef, className: "me-editor-surface" })) : activeMode === 'preview' ? (_jsx(PreviewSurface, { markdown: markdown, registry: renderers, onDiagnostics: (diagnostics) => onDiagnosticsRef.current?.(diagnostics) })) : (_jsx(React.Suspense, { fallback: _jsx("div", { className: "me-wysiwyg-loading", role: "status", children: "Loading rich text editor..." }), children: _jsx(LazyWysiwygLexicalEditor, { markdown: markdown, readOnly: readOnly, ariaLabel: ariaLabel, renderServices: wysiwygRenderServices, toolbarIcons: wysiwygToolbarIcons, onDiagnostics: (diagnostics) => onDiagnosticsRef.current?.(diagnostics), onChange: (nextMarkdown, meta) => {
237
+ markdownRef.current = nextMarkdown;
238
+ if (value === undefined) {
239
+ setInternalMarkdown(nextMarkdown);
240
+ }
241
+ onChange?.(nextMarkdown, {
242
+ ...meta,
243
+ mode: activeMode,
244
+ });
245
+ } }) }))] }));
246
+ });
247
+ function IconSvg({ path, dataIcon }) {
248
+ return (_jsx("svg", { className: "me-inline-icon", "data-icon": dataIcon, viewBox: "0 0 576 512", focusable: "false", "aria-hidden": "true", children: _jsx("path", { fill: "currentColor", d: path }) }));
249
+ }
250
+ const LIST_ICON_PATH = 'M40 48C26.7 48 16 58.7 16 72v48c0 13.3 10.7 24 24 24h48c13.3 0 24-10.7 24-24V72c0-13.3-10.7-24-24-24H40zm144 16c-17.7 0-32 14.3-32 32s14.3 32 32 32h352c17.7 0 32-14.3 32-32s-14.3-32-32-32H184zM40 208c-13.3 0-24 10.7-24 24v48c0 13.3 10.7 24 24 24h48c13.3 0 24-10.7 24-24v-48c0-13.3-10.7-24-24-24H40zm144 16c-17.7 0-32 14.3-32 32s14.3 32 32 32h352c17.7 0 32-14.3 32-32s-14.3-32-32-32H184zM40 368c-13.3 0-24 10.7-24 24v48c0 13.3 10.7 24 24 24h48c13.3 0 24-10.7 24-24v-48c0-13.3-10.7-24-24-24H40zm144 16c-17.7 0-32 14.3-32 32s14.3 32 32 32h352c17.7 0 32-14.3 32-32s-14.3-32-32-32H184z';
251
+ const SLIDERS_ICON_PATH = 'M0 416c0 17.7 14.3 32 32 32h54.7c13.2 37.3 48.7 64 90.5 64s77.4-26.7 90.5-64H544c17.7 0 32-14.3 32-32s-14.3-32-32-32H267.8c-13.2-37.3-48.7-64-90.5-64s-77.4 26.7-90.5 64H32c-17.7 0-32 14.3-32 32zm128 0a49.3 49.3 0 1 1 98.7 0 49.3 49.3 0 1 1-98.7 0zM0 256c0 17.7 14.3 32 32 32h246.7c13.2 37.3 48.7 64 90.5 64s77.4-26.7 90.5-64H544c17.7 0 32-14.3 32-32s-14.3-32-32-32h-84.3c-13.2-37.3-48.7-64-90.5-64s-77.4 26.7-90.5 64H32c-17.7 0-32 14.3-32 32zm320 0a49.3 49.3 0 1 1 98.7 0 49.3 49.3 0 1 1-98.7 0zM0 96c0 17.7 14.3 32 32 32h54.7c13.2 37.3 48.7 64 90.5 64s77.4-26.7 90.5-64H544c17.7 0 32-14.3 32-32s-14.3-32-32-32H267.8C254.6 26.7 219.1 0 177.3 0S99.9 26.7 86.7 64H32C14.3 64 0 78.3 0 96zm128 0a49.3 49.3 0 1 1 98.7 0 49.3 49.3 0 1 1-98.7 0z';
252
+ function HostServiceToolbar({ services, onInsertMarkdown, onDiagnostics, }) {
253
+ const [query, setQuery] = React.useState('');
254
+ const [suggestions, setSuggestions] = React.useState([]);
255
+ const [searching, setSearching] = React.useState(false);
256
+ const [uploading, setUploading] = React.useState(false);
257
+ const fileInputId = React.useId();
258
+ React.useEffect(() => {
259
+ if (!services?.searchLinks || query.trim().length === 0) {
260
+ setSuggestions([]);
261
+ setSearching(false);
262
+ return;
263
+ }
264
+ const controller = new AbortController();
265
+ setSearching(true);
266
+ services.searchLinks(query.trim(), controller.signal)
267
+ .then((nextSuggestions) => {
268
+ if (!controller.signal.aborted) {
269
+ setSuggestions(nextSuggestions);
270
+ }
271
+ })
272
+ .catch((error) => {
273
+ if (controller.signal.aborted) {
274
+ return;
275
+ }
276
+ const message = error instanceof Error ? error.message : String(error);
277
+ onDiagnostics([{
278
+ code: 'host.searchLinks.failed',
279
+ message,
280
+ severity: 'error',
281
+ source: 'host-service',
282
+ }]);
283
+ setSuggestions([]);
284
+ })
285
+ .finally(() => {
286
+ if (!controller.signal.aborted) {
287
+ setSearching(false);
288
+ }
289
+ });
290
+ return () => controller.abort();
291
+ }, [onDiagnostics, query, services]);
292
+ async function uploadFile(file) {
293
+ if (!services?.uploadAsset) {
294
+ return;
295
+ }
296
+ const controller = new AbortController();
297
+ setUploading(true);
298
+ try {
299
+ const asset = await services.uploadAsset(file, controller.signal);
300
+ const alt = (asset.alt ?? file.name.replace(/\.[^.]+$/, '')) || 'Uploaded image';
301
+ onInsertMarkdown(`\n\n![${escapeMarkdownLabel(alt)}](${asset.url})\n`);
302
+ }
303
+ catch (error) {
304
+ const message = error instanceof Error ? error.message : String(error);
305
+ onDiagnostics([{
306
+ code: 'host.uploadAsset.failed',
307
+ message,
308
+ severity: 'error',
309
+ source: 'host-service',
310
+ }]);
311
+ }
312
+ finally {
313
+ setUploading(false);
314
+ }
315
+ }
316
+ return (_jsxs("div", { className: "me-host-tools", "aria-label": "Host services", children: [services?.searchLinks ? (_jsxs("div", { className: "me-link-search", children: [_jsx("label", { className: "me-visually-hidden", htmlFor: `${fileInputId}-link-search`, children: "Search pages" }), _jsx("input", { id: `${fileInputId}-link-search`, type: "search", value: query, placeholder: "Search pages", "aria-label": "Search pages", "aria-controls": `${fileInputId}-link-suggestions`, "aria-expanded": suggestions.length > 0, onChange: (event) => setQuery(event.target.value) }), query.trim().length > 0 ? (_jsxs("div", { id: `${fileInputId}-link-suggestions`, className: "me-link-suggestions", role: "listbox", "aria-label": "Page suggestions", children: [searching ? _jsx("div", { className: "me-link-suggestion-status", children: "Searching..." }) : null, !searching && suggestions.length === 0 ? (_jsx("div", { className: "me-link-suggestion-status", children: "No matches" })) : null, suggestions.map((suggestion) => (_jsxs("button", { type: "button", role: "option", className: "me-link-suggestion", onClick: () => {
317
+ onInsertMarkdown(suggestion.insertText);
318
+ setQuery('');
319
+ setSuggestions([]);
320
+ }, children: [_jsx("span", { children: suggestion.label }), suggestion.description ? _jsx("small", { children: suggestion.description }) : null] }, suggestion.id)))] })) : null] })) : null, services?.uploadAsset ? (_jsxs("div", { className: "me-upload-control", children: [_jsx("input", { id: `${fileInputId}-asset-upload`, type: "file", accept: "image/*", "aria-label": "Upload image", onChange: (event) => {
321
+ const file = event.currentTarget.files?.[0];
322
+ event.currentTarget.value = '';
323
+ if (file) {
324
+ void uploadFile(file);
325
+ }
326
+ } }), _jsx("label", { className: "me-upload-button", htmlFor: `${fileInputId}-asset-upload`, children: uploading ? 'Uploading...' : 'Upload image' })] })) : null] }));
327
+ }
328
+ function PreviewSurface({ markdown, registry, onDiagnostics, }) {
329
+ const [result, setResult] = React.useState(null);
330
+ const registryRef = React.useRef(registry);
331
+ registryRef.current = registry;
332
+ React.useEffect(() => {
333
+ const controller = new AbortController();
334
+ const activeRegistry = registryRef.current ?? createDefaultRendererRegistry();
335
+ renderMarkdownToHtml(markdown, {
336
+ registry: activeRegistry,
337
+ signal: controller.signal,
338
+ })
339
+ .then((nextResult) => {
340
+ if (controller.signal.aborted) {
341
+ return;
342
+ }
343
+ setResult(nextResult);
344
+ onDiagnostics(nextResult.diagnostics);
345
+ })
346
+ .catch((error) => {
347
+ if (controller.signal.aborted) {
348
+ return;
349
+ }
350
+ const message = error instanceof Error ? error.message : String(error);
351
+ const diagnostics = [
352
+ {
353
+ code: 'preview.render.failed',
354
+ message,
355
+ severity: 'error',
356
+ source: 'renderer',
357
+ },
358
+ ];
359
+ setResult({
360
+ html: `<pre class="me-preview-error">${escapeHtml(markdown)}</pre>`,
361
+ blocks: [],
362
+ diagnostics,
363
+ });
364
+ onDiagnostics(diagnostics);
365
+ });
366
+ return () => controller.abort();
367
+ }, [markdown, onDiagnostics]);
368
+ return (_jsx("div", { className: "me-preview", "aria-label": "Markdown preview", dangerouslySetInnerHTML: { __html: result?.html ? sanitizePreviewHtml(result.html) : '<p>Rendering preview...</p>' } }));
369
+ }
370
+ function normalizeModes(modes) {
371
+ const normalized = modes.filter((modeName, index) => modes.indexOf(modeName) === index);
372
+ return normalized.length > 0 ? normalized : DEFAULT_MODES;
373
+ }
374
+ function preferredInitialMode(modes) {
375
+ return modes.includes('hybrid') ? 'hybrid' : modes[0] ?? 'markdown';
376
+ }
377
+ function modeLabel(mode) {
378
+ switch (mode) {
379
+ case 'markdown':
380
+ return 'Markdown';
381
+ case 'hybrid':
382
+ return 'Hybrid';
383
+ case 'preview':
384
+ return 'Preview';
385
+ case 'wysiwyg':
386
+ // Modern, user-facing name for the WYSIWYG mode. The internal mode key
387
+ // stays 'wysiwyg' for API stability.
388
+ return 'Rich Text';
389
+ }
390
+ }
391
+ function cmSelectionToTextSelection(selection) {
392
+ return {
393
+ from: Math.min(selection.anchor, selection.head),
394
+ to: Math.max(selection.anchor, selection.head),
395
+ };
396
+ }
397
+ function escapeHtml(value) {
398
+ return value
399
+ .replace(/&/g, '&amp;')
400
+ .replace(/</g, '&lt;')
401
+ .replace(/>/g, '&gt;')
402
+ .replace(/"/g, '&quot;');
403
+ }
404
+ function escapeMarkdownLabel(value) {
405
+ return value.replace(/\\/g, '\\\\').replace(/]/g, '\\]');
406
+ }
407
+ //# sourceMappingURL=MarkdownEditor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MarkdownEditor.js","sourceRoot":"","sources":["../src/MarkdownEditor.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export { MarkdownEditor } from './MarkdownEditor.js';
2
+ export type { MarkdownEditorComponentProps } from './MarkdownEditor.js';
3
+ export type { WysiwygToolbarIconKey, WysiwygToolbarIcons } from '@echozedlabs/wysiwyg-lexical';
4
+ export type { ChangeMeta, EditorDiagnostic, EditorMode, FrontmatterPropertySchema, HostServices, LinkSuggestion, MarkdownEditorHandle, MarkdownEditorProps, ModeChangeMeta, TextSelection, } from '@echozedlabs/core';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACrD,YAAY,EAAE,4BAA4B,EAAE,MAAM,qBAAqB,CAAC;AACxE,YAAY,EAAE,qBAAqB,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AAC/F,YAAY,EACV,UAAU,EACV,gBAAgB,EAChB,UAAU,EACV,yBAAyB,EACzB,YAAY,EACZ,cAAc,EACd,oBAAoB,EACpB,mBAAmB,EACnB,cAAc,EACd,aAAa,GACd,MAAM,mBAAmB,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ export { MarkdownEditor } from './MarkdownEditor.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Sanitize preview HTML (concatenated registry-renderer output, which may
3
+ * include host/diagram HTML) before it is injected via
4
+ * dangerouslySetInnerHTML. Fails closed in a non-DOM environment.
5
+ */
6
+ export declare function sanitizePreviewHtml(html: string): string;
7
+ //# sourceMappingURL=sanitizeHtml.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sanitizeHtml.d.ts","sourceRoot":"","sources":["../src/sanitizeHtml.ts"],"names":[],"mappings":"AAEA;;;;GAIG;AACH,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAaxD"}
@@ -0,0 +1,21 @@
1
+ import DOMPurify from 'dompurify';
2
+ /**
3
+ * Sanitize preview HTML (concatenated registry-renderer output, which may
4
+ * include host/diagram HTML) before it is injected via
5
+ * dangerouslySetInnerHTML. Fails closed in a non-DOM environment.
6
+ */
7
+ export function sanitizePreviewHtml(html) {
8
+ if (typeof DOMPurify.sanitize !== 'function') {
9
+ return '';
10
+ }
11
+ // Use DOMPurify's default config (which already allows HTML + SVG + MathML and
12
+ // strips scripts/event handlers) rather than USE_PROFILES. USE_PROFILES
13
+ // restricts namespaces and DROPS the HTML inside SVG <foreignObject>, which is
14
+ // how Mermaid renders flowchart node labels — that made diagram text vanish in
15
+ // the preview. The default config keeps foreignObject's HTML labels while still
16
+ // removing <script>/on*-handlers/javascript: URLs.
17
+ return DOMPurify.sanitize(html, {
18
+ ADD_TAGS: ['foreignObject', 'style'],
19
+ });
20
+ }
21
+ //# sourceMappingURL=sanitizeHtml.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sanitizeHtml.js","sourceRoot":"","sources":["../src/sanitizeHtml.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,WAAW,CAAC;AAElC;;;;GAIG;AACH,MAAM,UAAU,mBAAmB,CAAC,IAAY;IAC9C,IAAI,OAAQ,SAAoC,CAAC,QAAQ,KAAK,UAAU,EAAE,CAAC;QACzE,OAAO,EAAE,CAAC;IACZ,CAAC;IACD,+EAA+E;IAC/E,wEAAwE;IACxE,+EAA+E;IAC/E,+EAA+E;IAC/E,gFAAgF;IAChF,mDAAmD;IACnD,OAAO,SAAS,CAAC,QAAQ,CAAC,IAAI,EAAE;QAC9B,QAAQ,EAAE,CAAC,eAAe,EAAE,OAAO,CAAC;KACrC,CAAC,CAAC;AACL,CAAC"}