@furystack/shades-common-components 12.2.0 → 12.3.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/CHANGELOG.md +26 -0
- package/esm/components/index.d.ts +1 -0
- package/esm/components/index.d.ts.map +1 -1
- package/esm/components/index.js +1 -0
- package/esm/components/index.js.map +1 -1
- package/esm/components/markdown/index.d.ts +5 -0
- package/esm/components/markdown/index.d.ts.map +1 -0
- package/esm/components/markdown/index.js +5 -0
- package/esm/components/markdown/index.js.map +1 -0
- package/esm/components/markdown/markdown-display.d.ts +19 -0
- package/esm/components/markdown/markdown-display.d.ts.map +1 -0
- package/esm/components/markdown/markdown-display.js +149 -0
- package/esm/components/markdown/markdown-display.js.map +1 -0
- package/esm/components/markdown/markdown-display.spec.d.ts +2 -0
- package/esm/components/markdown/markdown-display.spec.d.ts.map +1 -0
- package/esm/components/markdown/markdown-display.spec.js +191 -0
- package/esm/components/markdown/markdown-display.spec.js.map +1 -0
- package/esm/components/markdown/markdown-editor.d.ts +25 -0
- package/esm/components/markdown/markdown-editor.d.ts.map +1 -0
- package/esm/components/markdown/markdown-editor.js +113 -0
- package/esm/components/markdown/markdown-editor.js.map +1 -0
- package/esm/components/markdown/markdown-editor.spec.d.ts +2 -0
- package/esm/components/markdown/markdown-editor.spec.d.ts.map +1 -0
- package/esm/components/markdown/markdown-editor.spec.js +111 -0
- package/esm/components/markdown/markdown-editor.spec.js.map +1 -0
- package/esm/components/markdown/markdown-input.d.ts +29 -0
- package/esm/components/markdown/markdown-input.d.ts.map +1 -0
- package/esm/components/markdown/markdown-input.js +100 -0
- package/esm/components/markdown/markdown-input.js.map +1 -0
- package/esm/components/markdown/markdown-input.spec.d.ts +2 -0
- package/esm/components/markdown/markdown-input.spec.d.ts.map +1 -0
- package/esm/components/markdown/markdown-input.spec.js +215 -0
- package/esm/components/markdown/markdown-input.spec.js.map +1 -0
- package/esm/components/markdown/markdown-parser.d.ts +82 -0
- package/esm/components/markdown/markdown-parser.d.ts.map +1 -0
- package/esm/components/markdown/markdown-parser.js +274 -0
- package/esm/components/markdown/markdown-parser.js.map +1 -0
- package/esm/components/markdown/markdown-parser.spec.d.ts +2 -0
- package/esm/components/markdown/markdown-parser.spec.d.ts.map +1 -0
- package/esm/components/markdown/markdown-parser.spec.js +229 -0
- package/esm/components/markdown/markdown-parser.spec.js.map +1 -0
- package/esm/components/styles.d.ts +1 -0
- package/esm/components/styles.d.ts.map +1 -1
- package/esm/components/styles.js.map +1 -1
- package/esm/components/typography.d.ts.map +1 -1
- package/esm/components/typography.js +26 -14
- package/esm/components/typography.js.map +1 -1
- package/esm/services/css-variable-theme.d.ts +3 -0
- package/esm/services/css-variable-theme.d.ts.map +1 -1
- package/esm/services/css-variable-theme.js +3 -0
- package/esm/services/css-variable-theme.js.map +1 -1
- package/esm/services/css-variable-theme.spec.js +3 -0
- package/esm/services/css-variable-theme.spec.js.map +1 -1
- package/esm/services/default-dark-palette.d.ts +8 -0
- package/esm/services/default-dark-palette.d.ts.map +1 -0
- package/esm/services/default-dark-palette.js +56 -0
- package/esm/services/default-dark-palette.js.map +1 -0
- package/esm/services/default-dark-theme.d.ts +3 -0
- package/esm/services/default-dark-theme.d.ts.map +1 -1
- package/esm/services/default-dark-theme.js +7 -4
- package/esm/services/default-dark-theme.js.map +1 -1
- package/esm/services/default-light-theme.d.ts +3 -0
- package/esm/services/default-light-theme.d.ts.map +1 -1
- package/esm/services/default-light-theme.js +3 -0
- package/esm/services/default-light-theme.js.map +1 -1
- package/esm/services/index.d.ts +1 -0
- package/esm/services/index.d.ts.map +1 -1
- package/esm/services/index.js +1 -0
- package/esm/services/index.js.map +1 -1
- package/esm/services/theme-provider-service.d.ts +10 -1
- package/esm/services/theme-provider-service.d.ts.map +1 -1
- package/esm/services/theme-provider-service.js.map +1 -1
- package/package.json +1 -1
- package/src/components/index.ts +1 -0
- package/src/components/markdown/index.ts +4 -0
- package/src/components/markdown/markdown-display.spec.tsx +243 -0
- package/src/components/markdown/markdown-display.tsx +202 -0
- package/src/components/markdown/markdown-editor.spec.tsx +142 -0
- package/src/components/markdown/markdown-editor.tsx +167 -0
- package/src/components/markdown/markdown-input.spec.tsx +274 -0
- package/src/components/markdown/markdown-input.tsx +143 -0
- package/src/components/markdown/markdown-parser.spec.ts +258 -0
- package/src/components/markdown/markdown-parser.ts +333 -0
- package/src/components/styles.tsx +1 -0
- package/src/components/typography.tsx +28 -15
- package/src/services/css-variable-theme.spec.ts +3 -0
- package/src/services/css-variable-theme.ts +3 -0
- package/src/services/default-dark-palette.ts +57 -0
- package/src/services/default-dark-theme.ts +7 -4
- package/src/services/default-light-theme.ts +3 -0
- package/src/services/index.ts +1 -0
- package/src/services/theme-provider-service.ts +7 -1
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,31 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## [12.3.0] - 2026-02-19
|
|
4
|
+
|
|
5
|
+
### ✨ Features
|
|
6
|
+
|
|
7
|
+
### Markdown Components
|
|
8
|
+
|
|
9
|
+
A new set of zero-dependency Markdown components for rendering and editing Markdown content.
|
|
10
|
+
|
|
11
|
+
**`parseMarkdown(source)`** — Converts a Markdown string into a typed AST. Supports headings, paragraphs, ordered/unordered lists, task-list checkboxes, fenced code blocks with language hints, blockquotes, horizontal rules, and inline formatting (bold, italic, inline code, links, images).
|
|
12
|
+
|
|
13
|
+
**`toggleCheckbox(source, lineIndex)`** — Toggles a checkbox at the given source line index in a raw Markdown string, returning the updated string.
|
|
14
|
+
|
|
15
|
+
**`MarkdownDisplay`** — Renders a Markdown string as styled HTML using FuryStack Shades components. When `readOnly` is set to `false`, task-list checkboxes become interactive and report changes via an `onChange` callback.
|
|
16
|
+
|
|
17
|
+
**`MarkdownInput`** — A textarea for editing raw Markdown. Supports pasting images from the clipboard, which are inlined as base64-encoded `` Markdown images (configurable size limit, defaults to 256 KB).
|
|
18
|
+
|
|
19
|
+
**`MarkdownEditor`** — A combined editor with an input pane and a live preview pane. Supports three layout modes: `side-by-side`, `tabs` (Edit / Preview), and `above-below`. Checkboxes toggled in the preview pane update the source text.
|
|
20
|
+
|
|
21
|
+
### 🧪 Tests
|
|
22
|
+
|
|
23
|
+
- Added unit tests for `parseMarkdown` and `parseInline` covering headings, paragraphs, lists, checkboxes, code blocks, blockquotes, horizontal rules, and inline formatting
|
|
24
|
+
- Added unit tests for `toggleCheckbox` verifying checked/unchecked toggling and out-of-bounds handling
|
|
25
|
+
- Added unit tests for `MarkdownDisplay` rendering and interactive checkbox toggling
|
|
26
|
+
- Added unit tests for `MarkdownEditor` layout switching between side-by-side, tabs, and above-below modes
|
|
27
|
+
- Added unit tests for `MarkdownInput` text input and image paste behavior
|
|
28
|
+
|
|
3
29
|
## [12.2.0] - 2026-02-19
|
|
4
30
|
|
|
5
31
|
### ⬆️ Dependencies
|
|
@@ -29,6 +29,7 @@ export * from './inputs/index.js';
|
|
|
29
29
|
export * from './linear-progress.js';
|
|
30
30
|
export * from './list/index.js';
|
|
31
31
|
export * from './loader.js';
|
|
32
|
+
export * from './markdown/index.js';
|
|
32
33
|
export * from './menu/index.js';
|
|
33
34
|
export * from './modal.js';
|
|
34
35
|
export * from './noty-list.js';
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/components/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,cAAc,CAAA;AAC5B,cAAc,aAAa,CAAA;AAC3B,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,WAAW,CAAA;AACzB,cAAc,wBAAwB,CAAA;AACtC,cAAc,4BAA4B,CAAA;AAC1C,cAAc,yBAAyB,CAAA;AACvC,cAAc,sBAAsB,CAAA;AACpC,cAAc,aAAa,CAAA;AAC3B,cAAc,cAAc,CAAA;AAC5B,cAAc,mBAAmB,CAAA;AACjC,cAAc,eAAe,CAAA;AAC7B,cAAc,UAAU,CAAA;AACxB,cAAc,WAAW,CAAA;AACzB,cAAc,WAAW,CAAA;AACzB,cAAc,kBAAkB,CAAA;AAChC,cAAc,YAAY,CAAA;AAC1B,cAAc,mBAAmB,CAAA;AACjC,cAAc,sBAAsB,CAAA;AACpC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,gBAAgB,CAAA;AAC9B,cAAc,2BAA2B,CAAA;AACzC,cAAc,wBAAwB,CAAA;AACtC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA;AAC3B,cAAc,eAAe,CAAA;AAC7B,cAAc,aAAa,CAAA;AAC3B,cAAc,oBAAoB,CAAA;AAClC,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,cAAc,CAAA;AAC5B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/components/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,cAAc,CAAA;AAC5B,cAAc,aAAa,CAAA;AAC3B,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,WAAW,CAAA;AACzB,cAAc,wBAAwB,CAAA;AACtC,cAAc,4BAA4B,CAAA;AAC1C,cAAc,yBAAyB,CAAA;AACvC,cAAc,sBAAsB,CAAA;AACpC,cAAc,aAAa,CAAA;AAC3B,cAAc,cAAc,CAAA;AAC5B,cAAc,mBAAmB,CAAA;AACjC,cAAc,eAAe,CAAA;AAC7B,cAAc,UAAU,CAAA;AACxB,cAAc,WAAW,CAAA;AACzB,cAAc,WAAW,CAAA;AACzB,cAAc,kBAAkB,CAAA;AAChC,cAAc,YAAY,CAAA;AAC1B,cAAc,mBAAmB,CAAA;AACjC,cAAc,sBAAsB,CAAA;AACpC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,aAAa,CAAA;AAC3B,cAAc,qBAAqB,CAAA;AACnC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,gBAAgB,CAAA;AAC9B,cAAc,2BAA2B,CAAA;AACzC,cAAc,wBAAwB,CAAA;AACtC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA;AAC3B,cAAc,eAAe,CAAA;AAC7B,cAAc,aAAa,CAAA;AAC3B,cAAc,oBAAoB,CAAA;AAClC,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,cAAc,CAAA;AAC5B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA"}
|
package/esm/components/index.js
CHANGED
|
@@ -29,6 +29,7 @@ export * from './inputs/index.js';
|
|
|
29
29
|
export * from './linear-progress.js';
|
|
30
30
|
export * from './list/index.js';
|
|
31
31
|
export * from './loader.js';
|
|
32
|
+
export * from './markdown/index.js';
|
|
32
33
|
export * from './menu/index.js';
|
|
33
34
|
export * from './modal.js';
|
|
34
35
|
export * from './noty-list.js';
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/components/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,cAAc,CAAA;AAC5B,cAAc,aAAa,CAAA;AAC3B,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,WAAW,CAAA;AACzB,cAAc,wBAAwB,CAAA;AACtC,cAAc,4BAA4B,CAAA;AAC1C,cAAc,yBAAyB,CAAA;AACvC,cAAc,sBAAsB,CAAA;AACpC,cAAc,aAAa,CAAA;AAC3B,cAAc,cAAc,CAAA;AAC5B,cAAc,mBAAmB,CAAA;AACjC,cAAc,eAAe,CAAA;AAC7B,cAAc,UAAU,CAAA;AACxB,cAAc,WAAW,CAAA;AACzB,cAAc,WAAW,CAAA;AACzB,cAAc,kBAAkB,CAAA;AAChC,cAAc,YAAY,CAAA;AAC1B,cAAc,mBAAmB,CAAA;AACjC,cAAc,sBAAsB,CAAA;AACpC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,gBAAgB,CAAA;AAC9B,cAAc,2BAA2B,CAAA;AACzC,cAAc,wBAAwB,CAAA;AACtC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA;AAC3B,cAAc,eAAe,CAAA;AAC7B,cAAc,aAAa,CAAA;AAC3B,cAAc,oBAAoB,CAAA;AAClC,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,cAAc,CAAA;AAC5B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA"}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/components/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,cAAc,CAAA;AAC5B,cAAc,aAAa,CAAA;AAC3B,cAAc,YAAY,CAAA;AAC1B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA;AACjC,cAAc,aAAa,CAAA;AAC3B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,WAAW,CAAA;AACzB,cAAc,wBAAwB,CAAA;AACtC,cAAc,4BAA4B,CAAA;AAC1C,cAAc,yBAAyB,CAAA;AACvC,cAAc,sBAAsB,CAAA;AACpC,cAAc,aAAa,CAAA;AAC3B,cAAc,cAAc,CAAA;AAC5B,cAAc,mBAAmB,CAAA;AACjC,cAAc,eAAe,CAAA;AAC7B,cAAc,UAAU,CAAA;AACxB,cAAc,WAAW,CAAA;AACzB,cAAc,WAAW,CAAA;AACzB,cAAc,kBAAkB,CAAA;AAChC,cAAc,YAAY,CAAA;AAC1B,cAAc,mBAAmB,CAAA;AACjC,cAAc,sBAAsB,CAAA;AACpC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,aAAa,CAAA;AAC3B,cAAc,qBAAqB,CAAA;AACnC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,gBAAgB,CAAA;AAC9B,cAAc,2BAA2B,CAAA;AACzC,cAAc,wBAAwB,CAAA;AACtC,cAAc,iBAAiB,CAAA;AAC/B,cAAc,YAAY,CAAA;AAC1B,cAAc,aAAa,CAAA;AAC3B,cAAc,aAAa,CAAA;AAC3B,cAAc,eAAe,CAAA;AAC7B,cAAc,aAAa,CAAA;AAC3B,cAAc,oBAAoB,CAAA;AAClC,cAAc,WAAW,CAAA;AACzB,cAAc,eAAe,CAAA;AAC7B,cAAc,cAAc,CAAA;AAC5B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,iBAAiB,CAAA;AAC/B,cAAc,mBAAmB,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,uBAAuB,CAAA;AACrC,cAAc,qBAAqB,CAAA;AACnC,cAAc,sBAAsB,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/components/markdown/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAA;AACpC,cAAc,uBAAuB,CAAA;AACrC,cAAc,qBAAqB,CAAA;AACnC,cAAc,sBAAsB,CAAA"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
export type MarkdownDisplayProps = {
|
|
2
|
+
/** The raw Markdown string to render */
|
|
3
|
+
content: string;
|
|
4
|
+
/** When false, checkboxes can be toggled. Defaults to true. */
|
|
5
|
+
readOnly?: boolean;
|
|
6
|
+
/** Called with the updated Markdown string when a checkbox is toggled */
|
|
7
|
+
onChange?: (newContent: string) => void;
|
|
8
|
+
};
|
|
9
|
+
/**
|
|
10
|
+
* Renders a Markdown string using FuryStack Shades components.
|
|
11
|
+
* Supports headings, paragraphs, lists, checkboxes, code blocks,
|
|
12
|
+
* blockquotes, images, links, and horizontal rules.
|
|
13
|
+
*/
|
|
14
|
+
export declare const MarkdownDisplay: (props: MarkdownDisplayProps & Omit<Partial<HTMLElement>, "style"> & {
|
|
15
|
+
style?: Partial<CSSStyleDeclaration>;
|
|
16
|
+
} & {
|
|
17
|
+
ref?: import("@furystack/shades").RefObject<Element>;
|
|
18
|
+
}, children?: import("@furystack/shades").ChildrenList) => JSX.Element;
|
|
19
|
+
//# sourceMappingURL=markdown-display.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-display.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-display.tsx"],"names":[],"mappings":"AAOA,MAAM,MAAM,oBAAoB,GAAG;IACjC,wCAAwC;IACxC,OAAO,EAAE,MAAM,CAAA;IACf,+DAA+D;IAC/D,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,yEAAyE;IACzE,QAAQ,CAAC,EAAE,CAAC,UAAU,EAAE,MAAM,KAAK,IAAI,CAAA;CACxC,CAAA;AAuFD;;;;GAIG;AACH,eAAO,MAAM,eAAe;;;;sEA+F1B,CAAA"}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Shade, createComponent } from '@furystack/shades';
|
|
2
|
+
import { cssVariableTheme } from '../../services/css-variable-theme.js';
|
|
3
|
+
import { Checkbox } from '../inputs/checkbox.js';
|
|
4
|
+
import { Typography } from '../typography.js';
|
|
5
|
+
import { parseMarkdown, toggleCheckbox } from './markdown-parser.js';
|
|
6
|
+
const renderInline = (nodes) => {
|
|
7
|
+
return (createComponent(createComponent, null, nodes.map((node) => {
|
|
8
|
+
switch (node.type) {
|
|
9
|
+
case 'text':
|
|
10
|
+
return createComponent(createComponent, null, node.content);
|
|
11
|
+
case 'bold':
|
|
12
|
+
return createComponent("strong", null, renderInline(node.children));
|
|
13
|
+
case 'italic':
|
|
14
|
+
return createComponent("em", null, renderInline(node.children));
|
|
15
|
+
case 'code':
|
|
16
|
+
return createComponent("code", { className: "md-inline-code" }, node.content);
|
|
17
|
+
case 'link':
|
|
18
|
+
return (createComponent("a", { className: "md-link", href: node.href, target: "_blank", rel: "noopener noreferrer" }, renderInline(node.children)));
|
|
19
|
+
case 'image':
|
|
20
|
+
return createComponent("img", { className: "md-image", src: node.src, alt: node.alt });
|
|
21
|
+
default:
|
|
22
|
+
return createComponent(createComponent, null);
|
|
23
|
+
}
|
|
24
|
+
})));
|
|
25
|
+
};
|
|
26
|
+
const variantForLevel = (level) => {
|
|
27
|
+
const map = { 1: 'h1', 2: 'h2', 3: 'h3', 4: 'h4', 5: 'h5', 6: 'h6' };
|
|
28
|
+
return map[level];
|
|
29
|
+
};
|
|
30
|
+
const renderBlock = (node, _index, options) => {
|
|
31
|
+
switch (node.type) {
|
|
32
|
+
case 'heading':
|
|
33
|
+
return createComponent(Typography, { variant: variantForLevel(node.level) }, renderInline(node.children));
|
|
34
|
+
case 'paragraph':
|
|
35
|
+
return createComponent(Typography, { variant: "body1" }, renderInline(node.children));
|
|
36
|
+
case 'codeBlock':
|
|
37
|
+
return (createComponent("pre", { className: "md-code-block", "data-language": node.language || undefined },
|
|
38
|
+
createComponent("code", null, node.content)));
|
|
39
|
+
case 'blockquote':
|
|
40
|
+
return (createComponent("blockquote", { className: "md-blockquote" }, node.children.map((child, i) => renderBlock(child, i, options))));
|
|
41
|
+
case 'horizontalRule':
|
|
42
|
+
return createComponent("hr", { className: "md-hr" });
|
|
43
|
+
case 'list': {
|
|
44
|
+
const listItems = node.items.map((item) => {
|
|
45
|
+
if (item.checkbox !== undefined) {
|
|
46
|
+
const handleChange = () => {
|
|
47
|
+
if (!options.readOnly && options.onChange) {
|
|
48
|
+
options.onChange(toggleCheckbox(options.content, item.sourceLineIndex));
|
|
49
|
+
}
|
|
50
|
+
};
|
|
51
|
+
return (createComponent("li", { className: "md-list-item md-checkbox-item", "data-source-line": String(item.sourceLineIndex) },
|
|
52
|
+
createComponent(Checkbox, { checked: item.checkbox === 'checked', disabled: options.readOnly, onchange: handleChange }),
|
|
53
|
+
createComponent("span", { className: "md-checkbox-label" }, renderInline(item.children))));
|
|
54
|
+
}
|
|
55
|
+
return createComponent("li", { className: "md-list-item" }, renderInline(item.children));
|
|
56
|
+
});
|
|
57
|
+
if (node.ordered) {
|
|
58
|
+
return createComponent("ol", { className: "md-list" }, listItems);
|
|
59
|
+
}
|
|
60
|
+
return createComponent("ul", { className: "md-list" }, listItems);
|
|
61
|
+
}
|
|
62
|
+
default:
|
|
63
|
+
return createComponent(createComponent, null);
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
/**
|
|
67
|
+
* Renders a Markdown string using FuryStack Shades components.
|
|
68
|
+
* Supports headings, paragraphs, lists, checkboxes, code blocks,
|
|
69
|
+
* blockquotes, images, links, and horizontal rules.
|
|
70
|
+
*/
|
|
71
|
+
export const MarkdownDisplay = Shade({
|
|
72
|
+
shadowDomName: 'shade-markdown-display',
|
|
73
|
+
css: {
|
|
74
|
+
display: 'block',
|
|
75
|
+
fontFamily: cssVariableTheme.typography.fontFamily,
|
|
76
|
+
color: cssVariableTheme.text.primary,
|
|
77
|
+
lineHeight: cssVariableTheme.typography.lineHeight.relaxed,
|
|
78
|
+
'& .md-inline-code': {
|
|
79
|
+
fontFamily: 'monospace',
|
|
80
|
+
backgroundColor: cssVariableTheme.action.hoverBackground,
|
|
81
|
+
padding: '2px 6px',
|
|
82
|
+
borderRadius: cssVariableTheme.shape.borderRadius.xs,
|
|
83
|
+
fontSize: '0.9em',
|
|
84
|
+
},
|
|
85
|
+
'& .md-code-block': {
|
|
86
|
+
fontFamily: 'monospace',
|
|
87
|
+
backgroundColor: cssVariableTheme.background.default,
|
|
88
|
+
border: `1px solid ${cssVariableTheme.action.subtleBorder}`,
|
|
89
|
+
borderRadius: cssVariableTheme.shape.borderRadius.md,
|
|
90
|
+
padding: cssVariableTheme.spacing.md,
|
|
91
|
+
overflow: 'auto',
|
|
92
|
+
fontSize: cssVariableTheme.typography.fontSize.sm,
|
|
93
|
+
margin: `${cssVariableTheme.spacing.sm} 0`,
|
|
94
|
+
},
|
|
95
|
+
'& .md-code-block code': {
|
|
96
|
+
font: 'inherit',
|
|
97
|
+
whiteSpace: 'pre',
|
|
98
|
+
},
|
|
99
|
+
'& .md-blockquote': {
|
|
100
|
+
borderLeft: `4px solid ${cssVariableTheme.palette.primary.main}`,
|
|
101
|
+
margin: `${cssVariableTheme.spacing.sm} 0`,
|
|
102
|
+
padding: `${cssVariableTheme.spacing.sm} ${cssVariableTheme.spacing.md}`,
|
|
103
|
+
color: cssVariableTheme.text.secondary,
|
|
104
|
+
},
|
|
105
|
+
'& .md-link': {
|
|
106
|
+
color: cssVariableTheme.palette.primary.main,
|
|
107
|
+
textDecoration: 'none',
|
|
108
|
+
},
|
|
109
|
+
'& .md-link:hover': {
|
|
110
|
+
textDecoration: 'underline',
|
|
111
|
+
},
|
|
112
|
+
'& .md-image': {
|
|
113
|
+
maxWidth: '100%',
|
|
114
|
+
borderRadius: cssVariableTheme.shape.borderRadius.md,
|
|
115
|
+
},
|
|
116
|
+
'& .md-hr': {
|
|
117
|
+
border: 'none',
|
|
118
|
+
borderTop: `1px solid ${cssVariableTheme.divider}`,
|
|
119
|
+
margin: `${cssVariableTheme.spacing.md} 0`,
|
|
120
|
+
},
|
|
121
|
+
'& .md-list': {
|
|
122
|
+
paddingLeft: cssVariableTheme.spacing.xl,
|
|
123
|
+
margin: `${cssVariableTheme.spacing.sm} 0`,
|
|
124
|
+
},
|
|
125
|
+
'& .md-list-item': {
|
|
126
|
+
marginBottom: cssVariableTheme.spacing.xs,
|
|
127
|
+
fontSize: cssVariableTheme.typography.fontSize.md,
|
|
128
|
+
},
|
|
129
|
+
'& .md-checkbox-item': {
|
|
130
|
+
listStyle: 'none',
|
|
131
|
+
display: 'flex',
|
|
132
|
+
alignItems: 'center',
|
|
133
|
+
gap: cssVariableTheme.spacing.sm,
|
|
134
|
+
},
|
|
135
|
+
'& .md-checkbox-label': {
|
|
136
|
+
fontSize: cssVariableTheme.typography.fontSize.md,
|
|
137
|
+
},
|
|
138
|
+
},
|
|
139
|
+
render: ({ props }) => {
|
|
140
|
+
const readOnly = props.readOnly !== false;
|
|
141
|
+
const ast = parseMarkdown(props.content);
|
|
142
|
+
return (createComponent("div", { className: "md-root" }, ast.map((node, i) => renderBlock(node, i, {
|
|
143
|
+
content: props.content,
|
|
144
|
+
readOnly,
|
|
145
|
+
onChange: props.onChange,
|
|
146
|
+
}))));
|
|
147
|
+
},
|
|
148
|
+
});
|
|
149
|
+
//# sourceMappingURL=markdown-display.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-display.js","sourceRoot":"","sources":["../../../src/components/markdown/markdown-display.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-display.spec.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-display.spec.tsx"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
import { Injector } from '@furystack/inject';
|
|
2
|
+
import { createComponent, initializeShadeRoot } from '@furystack/shades';
|
|
3
|
+
import { sleepAsync, usingAsync } from '@furystack/utils';
|
|
4
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
5
|
+
import { MarkdownDisplay } from './markdown-display.js';
|
|
6
|
+
describe('MarkdownDisplay', () => {
|
|
7
|
+
beforeEach(() => {
|
|
8
|
+
document.body.innerHTML = '<div id="root"></div>';
|
|
9
|
+
});
|
|
10
|
+
afterEach(() => {
|
|
11
|
+
document.body.innerHTML = '';
|
|
12
|
+
vi.restoreAllMocks();
|
|
13
|
+
});
|
|
14
|
+
it('should render with shadow DOM', async () => {
|
|
15
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
16
|
+
const rootElement = document.getElementById('root');
|
|
17
|
+
initializeShadeRoot({
|
|
18
|
+
injector,
|
|
19
|
+
rootElement,
|
|
20
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "Hello" }),
|
|
21
|
+
});
|
|
22
|
+
await sleepAsync(50);
|
|
23
|
+
const el = document.querySelector('shade-markdown-display');
|
|
24
|
+
expect(el).not.toBeNull();
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
it('should render a heading', async () => {
|
|
28
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
29
|
+
const rootElement = document.getElementById('root');
|
|
30
|
+
initializeShadeRoot({
|
|
31
|
+
injector,
|
|
32
|
+
rootElement,
|
|
33
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "# Hello World" }),
|
|
34
|
+
});
|
|
35
|
+
await sleepAsync(50);
|
|
36
|
+
const typography = document.querySelector('shade-markdown-display shade-typography');
|
|
37
|
+
expect(typography).not.toBeNull();
|
|
38
|
+
expect(typography?.getAttribute('data-variant')).toBe('h1');
|
|
39
|
+
expect(typography?.textContent).toContain('Hello World');
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
it('should render a paragraph', async () => {
|
|
43
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
44
|
+
const rootElement = document.getElementById('root');
|
|
45
|
+
initializeShadeRoot({
|
|
46
|
+
injector,
|
|
47
|
+
rootElement,
|
|
48
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "Just a paragraph." }),
|
|
49
|
+
});
|
|
50
|
+
await sleepAsync(50);
|
|
51
|
+
const typography = document.querySelector('shade-markdown-display shade-typography[data-variant="body1"]');
|
|
52
|
+
expect(typography).not.toBeNull();
|
|
53
|
+
expect(typography?.textContent).toContain('Just a paragraph.');
|
|
54
|
+
});
|
|
55
|
+
});
|
|
56
|
+
it('should render a code block', async () => {
|
|
57
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
58
|
+
const rootElement = document.getElementById('root');
|
|
59
|
+
initializeShadeRoot({
|
|
60
|
+
injector,
|
|
61
|
+
rootElement,
|
|
62
|
+
jsxElement: createComponent(MarkdownDisplay, { content: '```js\nconsole.log("hi")\n```' }),
|
|
63
|
+
});
|
|
64
|
+
await sleepAsync(50);
|
|
65
|
+
const codeBlock = document.querySelector('shade-markdown-display .md-code-block');
|
|
66
|
+
expect(codeBlock).not.toBeNull();
|
|
67
|
+
expect(codeBlock?.textContent).toContain('console.log("hi")');
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
it('should render a list', async () => {
|
|
71
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
72
|
+
const rootElement = document.getElementById('root');
|
|
73
|
+
initializeShadeRoot({
|
|
74
|
+
injector,
|
|
75
|
+
rootElement,
|
|
76
|
+
jsxElement: createComponent(MarkdownDisplay, { content: '- Item A\n- Item B' }),
|
|
77
|
+
});
|
|
78
|
+
await sleepAsync(50);
|
|
79
|
+
const list = document.querySelector('shade-markdown-display ul');
|
|
80
|
+
expect(list).not.toBeNull();
|
|
81
|
+
const items = document.querySelectorAll('shade-markdown-display .md-list-item');
|
|
82
|
+
expect(items.length).toBe(2);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
it('should render checkboxes as disabled when readOnly (default)', async () => {
|
|
86
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
87
|
+
const rootElement = document.getElementById('root');
|
|
88
|
+
initializeShadeRoot({
|
|
89
|
+
injector,
|
|
90
|
+
rootElement,
|
|
91
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "- [ ] Task" }),
|
|
92
|
+
});
|
|
93
|
+
await sleepAsync(50);
|
|
94
|
+
const checkbox = document.querySelector('shade-markdown-display shade-checkbox');
|
|
95
|
+
expect(checkbox).not.toBeNull();
|
|
96
|
+
expect(checkbox?.hasAttribute('data-disabled')).toBe(true);
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
it('should render checkboxes as enabled when readOnly is false', async () => {
|
|
100
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
101
|
+
const rootElement = document.getElementById('root');
|
|
102
|
+
const onChange = vi.fn();
|
|
103
|
+
initializeShadeRoot({
|
|
104
|
+
injector,
|
|
105
|
+
rootElement,
|
|
106
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "- [ ] Task", readOnly: false, onChange: onChange }),
|
|
107
|
+
});
|
|
108
|
+
await sleepAsync(50);
|
|
109
|
+
const checkbox = document.querySelector('shade-markdown-display shade-checkbox');
|
|
110
|
+
expect(checkbox).not.toBeNull();
|
|
111
|
+
expect(checkbox?.hasAttribute('data-disabled')).toBe(false);
|
|
112
|
+
const input = checkbox?.querySelector('input[type="checkbox"]');
|
|
113
|
+
expect(input).not.toBeNull();
|
|
114
|
+
input.click();
|
|
115
|
+
await sleepAsync(50);
|
|
116
|
+
expect(onChange).toHaveBeenCalledOnce();
|
|
117
|
+
expect(onChange).toHaveBeenCalledWith('- [x] Task');
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
it('should render a blockquote', async () => {
|
|
121
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
122
|
+
const rootElement = document.getElementById('root');
|
|
123
|
+
initializeShadeRoot({
|
|
124
|
+
injector,
|
|
125
|
+
rootElement,
|
|
126
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "> Quote text" }),
|
|
127
|
+
});
|
|
128
|
+
await sleepAsync(50);
|
|
129
|
+
const bq = document.querySelector('shade-markdown-display .md-blockquote');
|
|
130
|
+
expect(bq).not.toBeNull();
|
|
131
|
+
expect(bq?.textContent).toContain('Quote text');
|
|
132
|
+
});
|
|
133
|
+
});
|
|
134
|
+
it('should render a horizontal rule', async () => {
|
|
135
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
136
|
+
const rootElement = document.getElementById('root');
|
|
137
|
+
initializeShadeRoot({
|
|
138
|
+
injector,
|
|
139
|
+
rootElement,
|
|
140
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "---" }),
|
|
141
|
+
});
|
|
142
|
+
await sleepAsync(50);
|
|
143
|
+
const hr = document.querySelector('shade-markdown-display .md-hr');
|
|
144
|
+
expect(hr).not.toBeNull();
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
it('should render links', async () => {
|
|
148
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
149
|
+
const rootElement = document.getElementById('root');
|
|
150
|
+
initializeShadeRoot({
|
|
151
|
+
injector,
|
|
152
|
+
rootElement,
|
|
153
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "[Click here](https://example.com)" }),
|
|
154
|
+
});
|
|
155
|
+
await sleepAsync(50);
|
|
156
|
+
const link = document.querySelector('shade-markdown-display .md-link');
|
|
157
|
+
expect(link).not.toBeNull();
|
|
158
|
+
expect(link?.href).toContain('example.com');
|
|
159
|
+
expect(link?.textContent).toContain('Click here');
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
it('should render images', async () => {
|
|
163
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
164
|
+
const rootElement = document.getElementById('root');
|
|
165
|
+
initializeShadeRoot({
|
|
166
|
+
injector,
|
|
167
|
+
rootElement,
|
|
168
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "" }),
|
|
169
|
+
});
|
|
170
|
+
await sleepAsync(50);
|
|
171
|
+
const img = document.querySelector('shade-markdown-display .md-image');
|
|
172
|
+
expect(img).not.toBeNull();
|
|
173
|
+
expect(img?.alt).toBe('alt text');
|
|
174
|
+
});
|
|
175
|
+
});
|
|
176
|
+
it('should render empty for empty content', async () => {
|
|
177
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
178
|
+
const rootElement = document.getElementById('root');
|
|
179
|
+
initializeShadeRoot({
|
|
180
|
+
injector,
|
|
181
|
+
rootElement,
|
|
182
|
+
jsxElement: createComponent(MarkdownDisplay, { content: "" }),
|
|
183
|
+
});
|
|
184
|
+
await sleepAsync(50);
|
|
185
|
+
const root = document.querySelector('shade-markdown-display .md-root');
|
|
186
|
+
expect(root).not.toBeNull();
|
|
187
|
+
expect(root?.children.length).toBe(0);
|
|
188
|
+
});
|
|
189
|
+
});
|
|
190
|
+
});
|
|
191
|
+
//# sourceMappingURL=markdown-display.spec.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-display.spec.js","sourceRoot":"","sources":["../../../src/components/markdown/markdown-display.spec.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export type MarkdownEditorLayout = 'side-by-side' | 'tabs' | 'above-below';
|
|
2
|
+
export type MarkdownEditorProps = {
|
|
3
|
+
/** The current Markdown string */
|
|
4
|
+
value: string;
|
|
5
|
+
/** Called when the value changes (from either the input or checkbox toggle) */
|
|
6
|
+
onValueChange?: (newValue: string) => void;
|
|
7
|
+
/** Layout mode for the editor. Defaults to 'side-by-side'. */
|
|
8
|
+
layout?: MarkdownEditorLayout;
|
|
9
|
+
/** Maximum image file size in bytes for base64 paste */
|
|
10
|
+
maxImageSizeBytes?: number;
|
|
11
|
+
/** When true, the editor is read-only */
|
|
12
|
+
readOnly?: boolean;
|
|
13
|
+
/** Inline styles applied to the host element */
|
|
14
|
+
style?: Partial<CSSStyleDeclaration>;
|
|
15
|
+
};
|
|
16
|
+
/**
|
|
17
|
+
* Combined Markdown editor with an input pane and a live preview pane.
|
|
18
|
+
* Supports three layouts: side-by-side, tabs (Edit/Preview), or above-below.
|
|
19
|
+
*/
|
|
20
|
+
export declare const MarkdownEditor: (props: MarkdownEditorProps & Omit<Partial<HTMLElement>, "style"> & {
|
|
21
|
+
style?: Partial<CSSStyleDeclaration>;
|
|
22
|
+
} & {
|
|
23
|
+
ref?: import("@furystack/shades").RefObject<Element>;
|
|
24
|
+
}, children?: import("@furystack/shades").ChildrenList) => JSX.Element;
|
|
25
|
+
//# sourceMappingURL=markdown-editor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-editor.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-editor.tsx"],"names":[],"mappings":"AAMA,MAAM,MAAM,oBAAoB,GAAG,cAAc,GAAG,MAAM,GAAG,aAAa,CAAA;AAE1E,MAAM,MAAM,mBAAmB,GAAG;IAChC,kCAAkC;IAClC,KAAK,EAAE,MAAM,CAAA;IACb,+EAA+E;IAC/E,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,KAAK,IAAI,CAAA;IAC1C,8DAA8D;IAC9D,MAAM,CAAC,EAAE,oBAAoB,CAAA;IAC7B,wDAAwD;IACxD,iBAAiB,CAAC,EAAE,MAAM,CAAA;IAC1B,yCAAyC;IACzC,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,gDAAgD;IAChD,KAAK,CAAC,EAAE,OAAO,CAAC,mBAAmB,CAAC,CAAA;CACrC,CAAA;AAID;;;GAGG;AACH,eAAO,MAAM,cAAc;;;;sEAyIzB,CAAA"}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import { Shade, createComponent } from '@furystack/shades';
|
|
2
|
+
import { cssVariableTheme } from '../../services/css-variable-theme.js';
|
|
3
|
+
import { Tabs } from '../tabs.js';
|
|
4
|
+
import { MarkdownDisplay } from './markdown-display.js';
|
|
5
|
+
import { MarkdownInput } from './markdown-input.js';
|
|
6
|
+
/**
|
|
7
|
+
* Combined Markdown editor with an input pane and a live preview pane.
|
|
8
|
+
* Supports three layouts: side-by-side, tabs (Edit/Preview), or above-below.
|
|
9
|
+
*/
|
|
10
|
+
export const MarkdownEditor = Shade({
|
|
11
|
+
shadowDomName: 'shade-markdown-editor',
|
|
12
|
+
css: {
|
|
13
|
+
display: 'flex',
|
|
14
|
+
flexDirection: 'column',
|
|
15
|
+
border: `1px solid ${cssVariableTheme.action.subtleBorder}`,
|
|
16
|
+
borderRadius: cssVariableTheme.shape.borderRadius.md,
|
|
17
|
+
overflow: 'hidden',
|
|
18
|
+
minHeight: '0',
|
|
19
|
+
'& .md-editor-split': {
|
|
20
|
+
display: 'flex',
|
|
21
|
+
flex: '1',
|
|
22
|
+
minHeight: '0',
|
|
23
|
+
},
|
|
24
|
+
'& .md-editor-split[data-layout="side-by-side"]': {
|
|
25
|
+
flexDirection: 'row',
|
|
26
|
+
},
|
|
27
|
+
'& .md-editor-split[data-layout="above-below"]': {
|
|
28
|
+
flexDirection: 'column',
|
|
29
|
+
minHeight: 'auto',
|
|
30
|
+
},
|
|
31
|
+
'& .md-editor-pane': {
|
|
32
|
+
flex: '1',
|
|
33
|
+
minWidth: '0',
|
|
34
|
+
minHeight: '0',
|
|
35
|
+
overflow: 'auto',
|
|
36
|
+
display: 'flex',
|
|
37
|
+
flexDirection: 'column',
|
|
38
|
+
},
|
|
39
|
+
'& .md-editor-pane-input': {
|
|
40
|
+
borderRight: 'none',
|
|
41
|
+
},
|
|
42
|
+
'& .md-editor-split[data-layout="side-by-side"] .md-editor-pane-input': {
|
|
43
|
+
borderRight: `1px solid ${cssVariableTheme.action.subtleBorder}`,
|
|
44
|
+
},
|
|
45
|
+
'& .md-editor-split[data-layout="above-below"] .md-editor-pane-input': {
|
|
46
|
+
borderBottom: `1px solid ${cssVariableTheme.action.subtleBorder}`,
|
|
47
|
+
},
|
|
48
|
+
'& .md-editor-split[data-layout="above-below"] .md-editor-pane': {
|
|
49
|
+
flex: 'none',
|
|
50
|
+
overflow: 'visible',
|
|
51
|
+
minHeight: 'auto',
|
|
52
|
+
},
|
|
53
|
+
'& .md-editor-split[data-layout="above-below"] shade-markdown-input textarea': {
|
|
54
|
+
overflow: 'hidden',
|
|
55
|
+
fieldSizing: 'content',
|
|
56
|
+
},
|
|
57
|
+
'& .md-editor-pane-preview': {
|
|
58
|
+
padding: cssVariableTheme.spacing.md,
|
|
59
|
+
},
|
|
60
|
+
'& shade-markdown-input': {
|
|
61
|
+
marginBottom: '0',
|
|
62
|
+
flex: '1',
|
|
63
|
+
display: 'flex',
|
|
64
|
+
flexDirection: 'column',
|
|
65
|
+
},
|
|
66
|
+
'& shade-markdown-input label': {
|
|
67
|
+
border: 'none',
|
|
68
|
+
borderRadius: '0',
|
|
69
|
+
flex: '1',
|
|
70
|
+
display: 'flex',
|
|
71
|
+
flexDirection: 'column',
|
|
72
|
+
},
|
|
73
|
+
'& shade-markdown-input textarea': {
|
|
74
|
+
flex: '1',
|
|
75
|
+
resize: 'none',
|
|
76
|
+
},
|
|
77
|
+
'& shade-tabs': {
|
|
78
|
+
flex: '1',
|
|
79
|
+
minHeight: '0',
|
|
80
|
+
},
|
|
81
|
+
'& .md-editor-tab-content': {
|
|
82
|
+
padding: cssVariableTheme.spacing.md,
|
|
83
|
+
overflow: 'auto',
|
|
84
|
+
},
|
|
85
|
+
},
|
|
86
|
+
render: ({ props, useState, useHostProps }) => {
|
|
87
|
+
const layout = props.layout ?? 'side-by-side';
|
|
88
|
+
useHostProps({
|
|
89
|
+
...(props.style ? { style: props.style } : {}),
|
|
90
|
+
});
|
|
91
|
+
const [activeTab, setActiveTab] = useState('activeTab', 'edit');
|
|
92
|
+
const inputPane = (createComponent(MarkdownInput, { value: props.value, onValueChange: props.onValueChange, maxImageSizeBytes: props.maxImageSizeBytes, readOnly: props.readOnly }));
|
|
93
|
+
const previewPane = createComponent(MarkdownDisplay, { content: props.value, readOnly: false, onChange: props.onValueChange });
|
|
94
|
+
if (layout === 'tabs') {
|
|
95
|
+
return (createComponent(Tabs, { activeKey: activeTab, onTabChange: (key) => setActiveTab(key), tabs: [
|
|
96
|
+
{
|
|
97
|
+
header: createComponent(createComponent, null, "Edit"),
|
|
98
|
+
hash: 'edit',
|
|
99
|
+
component: createComponent("div", { className: "md-editor-tab-content" }, inputPane),
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
header: createComponent(createComponent, null, "Preview"),
|
|
103
|
+
hash: 'preview',
|
|
104
|
+
component: createComponent("div", { className: "md-editor-tab-content" }, previewPane),
|
|
105
|
+
},
|
|
106
|
+
] }));
|
|
107
|
+
}
|
|
108
|
+
return (createComponent("div", { className: "md-editor-split", "data-layout": layout },
|
|
109
|
+
createComponent("div", { className: "md-editor-pane md-editor-pane-input" }, inputPane),
|
|
110
|
+
createComponent("div", { className: "md-editor-pane md-editor-pane-preview" }, previewPane)));
|
|
111
|
+
},
|
|
112
|
+
});
|
|
113
|
+
//# sourceMappingURL=markdown-editor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-editor.js","sourceRoot":"","sources":["../../../src/components/markdown/markdown-editor.tsx"],"names":[],"mappings":"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"}
|