@furystack/shades-common-components 12.1.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 +42 -0
- package/esm/components/avatar.d.ts.map +1 -1
- package/esm/components/avatar.js +3 -1
- package/esm/components/avatar.js.map +1 -1
- package/esm/components/avatar.spec.js +4 -4
- package/esm/components/avatar.spec.js.map +1 -1
- package/esm/components/icons/icon-definitions.d.ts +82 -0
- package/esm/components/icons/icon-definitions.d.ts.map +1 -1
- package/esm/components/icons/icon-definitions.js +717 -0
- package/esm/components/icons/icon-definitions.js.map +1 -1
- package/esm/components/icons/icon-definitions.spec.js +22 -2
- package/esm/components/icons/icon-definitions.spec.js.map +1 -1
- package/esm/components/icons/icon-types.d.ts +10 -0
- package/esm/components/icons/icon-types.d.ts.map +1 -1
- package/esm/components/icons/index.d.ts +1 -1
- package/esm/components/icons/index.d.ts.map +1 -1
- 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/page-container/index.d.ts +1 -1
- package/esm/components/page-container/index.js +1 -1
- package/esm/components/page-container/page-header.d.ts +5 -5
- package/esm/components/page-container/page-header.d.ts.map +1 -1
- package/esm/components/page-container/page-header.js +3 -3
- 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/suggest/index.d.ts +1 -1
- package/esm/components/suggest/index.d.ts.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 +3 -3
- package/src/components/avatar.spec.tsx +4 -4
- package/src/components/avatar.tsx +3 -1
- package/src/components/icons/icon-definitions.spec.ts +28 -2
- package/src/components/icons/icon-definitions.ts +759 -0
- package/src/components/icons/icon-types.ts +12 -0
- package/src/components/icons/index.ts +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/page-container/index.tsx +1 -1
- package/src/components/page-container/page-header.tsx +5 -5
- package/src/components/styles.tsx +1 -0
- package/src/components/suggest/index.tsx +1 -1
- 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
|
@@ -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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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-editor.spec.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-editor.spec.tsx"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,111 @@
|
|
|
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 { MarkdownEditor } from './markdown-editor.js';
|
|
6
|
+
describe('MarkdownEditor', () => {
|
|
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(MarkdownEditor, { value: "# Hello" }),
|
|
21
|
+
});
|
|
22
|
+
await sleepAsync(50);
|
|
23
|
+
const el = document.querySelector('shade-markdown-editor');
|
|
24
|
+
expect(el).not.toBeNull();
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
describe('side-by-side layout (default)', () => {
|
|
28
|
+
it('should render input and preview panes side by side', async () => {
|
|
29
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
30
|
+
const rootElement = document.getElementById('root');
|
|
31
|
+
initializeShadeRoot({
|
|
32
|
+
injector,
|
|
33
|
+
rootElement,
|
|
34
|
+
jsxElement: createComponent(MarkdownEditor, { value: "# Hello" }),
|
|
35
|
+
});
|
|
36
|
+
await sleepAsync(50);
|
|
37
|
+
const split = document.querySelector('.md-editor-split');
|
|
38
|
+
expect(split).not.toBeNull();
|
|
39
|
+
expect(split.dataset.layout).toBe('side-by-side');
|
|
40
|
+
const input = document.querySelector('shade-markdown-editor shade-markdown-input');
|
|
41
|
+
expect(input).not.toBeNull();
|
|
42
|
+
const display = document.querySelector('shade-markdown-editor shade-markdown-display');
|
|
43
|
+
expect(display).not.toBeNull();
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
});
|
|
47
|
+
describe('above-below layout', () => {
|
|
48
|
+
it('should render with above-below layout', async () => {
|
|
49
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
50
|
+
const rootElement = document.getElementById('root');
|
|
51
|
+
initializeShadeRoot({
|
|
52
|
+
injector,
|
|
53
|
+
rootElement,
|
|
54
|
+
jsxElement: createComponent(MarkdownEditor, { value: "# Hello", layout: "above-below" }),
|
|
55
|
+
});
|
|
56
|
+
await sleepAsync(50);
|
|
57
|
+
const split = document.querySelector('.md-editor-split');
|
|
58
|
+
expect(split).not.toBeNull();
|
|
59
|
+
expect(split.dataset.layout).toBe('above-below');
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
describe('tabs layout', () => {
|
|
64
|
+
it('should render with tabs layout', async () => {
|
|
65
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
66
|
+
const rootElement = document.getElementById('root');
|
|
67
|
+
initializeShadeRoot({
|
|
68
|
+
injector,
|
|
69
|
+
rootElement,
|
|
70
|
+
jsxElement: createComponent(MarkdownEditor, { value: "# Hello", layout: "tabs" }),
|
|
71
|
+
});
|
|
72
|
+
await sleepAsync(50);
|
|
73
|
+
const tabs = document.querySelector('shade-markdown-editor shade-tabs');
|
|
74
|
+
expect(tabs).not.toBeNull();
|
|
75
|
+
const tabButtons = document.querySelectorAll('shade-markdown-editor .shade-tab-btn');
|
|
76
|
+
expect(tabButtons.length).toBe(2);
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
it('should show the edit tab by default', async () => {
|
|
80
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
81
|
+
const rootElement = document.getElementById('root');
|
|
82
|
+
initializeShadeRoot({
|
|
83
|
+
injector,
|
|
84
|
+
rootElement,
|
|
85
|
+
jsxElement: createComponent(MarkdownEditor, { value: "# Hello", layout: "tabs" }),
|
|
86
|
+
});
|
|
87
|
+
await sleepAsync(50);
|
|
88
|
+
const input = document.querySelector('shade-markdown-editor shade-markdown-input');
|
|
89
|
+
expect(input).not.toBeNull();
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
});
|
|
93
|
+
it('should pass value to both input and display', async () => {
|
|
94
|
+
await usingAsync(new Injector(), async (injector) => {
|
|
95
|
+
const rootElement = document.getElementById('root');
|
|
96
|
+
const mdContent = '# Test Content';
|
|
97
|
+
initializeShadeRoot({
|
|
98
|
+
injector,
|
|
99
|
+
rootElement,
|
|
100
|
+
jsxElement: createComponent(MarkdownEditor, { value: mdContent }),
|
|
101
|
+
});
|
|
102
|
+
await sleepAsync(50);
|
|
103
|
+
const textarea = document.querySelector('shade-markdown-editor textarea');
|
|
104
|
+
expect(textarea.value).toBe(mdContent);
|
|
105
|
+
const heading = document.querySelector('shade-markdown-editor shade-markdown-display shade-typography[data-variant="h1"]');
|
|
106
|
+
expect(heading).not.toBeNull();
|
|
107
|
+
expect(heading?.textContent).toContain('Test Content');
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
});
|
|
111
|
+
//# sourceMappingURL=markdown-editor.spec.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-editor.spec.js","sourceRoot":"","sources":["../../../src/components/markdown/markdown-editor.spec.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
export type MarkdownInputProps = {
|
|
2
|
+
/** The current Markdown string */
|
|
3
|
+
value: string;
|
|
4
|
+
/** Called when the value changes */
|
|
5
|
+
onValueChange?: (newValue: string) => void;
|
|
6
|
+
/** Maximum image file size in bytes for base64 paste. Defaults to 256KB. */
|
|
7
|
+
maxImageSizeBytes?: number;
|
|
8
|
+
/** Whether the textarea is read-only */
|
|
9
|
+
readOnly?: boolean;
|
|
10
|
+
/** Whether the textarea is disabled */
|
|
11
|
+
disabled?: boolean;
|
|
12
|
+
/** Placeholder text */
|
|
13
|
+
placeholder?: string;
|
|
14
|
+
/** Label shown above the textarea */
|
|
15
|
+
labelTitle?: string;
|
|
16
|
+
/** Number of visible text rows */
|
|
17
|
+
rows?: number;
|
|
18
|
+
};
|
|
19
|
+
/**
|
|
20
|
+
* Markdown text input with base64 image paste support.
|
|
21
|
+
* When the user pastes an image below the configured size limit,
|
|
22
|
+
* it is inlined as a `` Markdown image.
|
|
23
|
+
*/
|
|
24
|
+
export declare const MarkdownInput: (props: MarkdownInputProps & Omit<Partial<HTMLElement>, "style"> & {
|
|
25
|
+
style?: Partial<CSSStyleDeclaration>;
|
|
26
|
+
} & {
|
|
27
|
+
ref?: import("@furystack/shades").RefObject<Element>;
|
|
28
|
+
}, children?: import("@furystack/shades").ChildrenList) => JSX.Element;
|
|
29
|
+
//# sourceMappingURL=markdown-input.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-input.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-input.tsx"],"names":[],"mappings":"AAKA,MAAM,MAAM,kBAAkB,GAAG;IAC/B,kCAAkC;IAClC,KAAK,EAAE,MAAM,CAAA;IACb,oCAAoC;IACpC,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,KAAK,IAAI,CAAA;IAC1C,4EAA4E;IAC5E,iBAAiB,CAAC,EAAE,MAAM,CAAA;IAC1B,wCAAwC;IACxC,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,uCAAuC;IACvC,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,uBAAuB;IACvB,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB,qCAAqC;IACrC,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,kCAAkC;IAClC,IAAI,CAAC,EAAE,MAAM,CAAA;CACd,CAAA;AAED;;;;GAIG;AACH,eAAO,MAAM,aAAa;;;;sEAiHxB,CAAA"}
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import { Shade, createComponent } from '@furystack/shades';
|
|
2
|
+
import { cssVariableTheme } from '../../services/css-variable-theme.js';
|
|
3
|
+
const DEFAULT_MAX_IMAGE_SIZE = 256 * 1024;
|
|
4
|
+
/**
|
|
5
|
+
* Markdown text input with base64 image paste support.
|
|
6
|
+
* When the user pastes an image below the configured size limit,
|
|
7
|
+
* it is inlined as a `` Markdown image.
|
|
8
|
+
*/
|
|
9
|
+
export const MarkdownInput = Shade({
|
|
10
|
+
shadowDomName: 'shade-markdown-input',
|
|
11
|
+
css: {
|
|
12
|
+
display: 'block',
|
|
13
|
+
marginBottom: '1em',
|
|
14
|
+
'& label': {
|
|
15
|
+
display: 'flex',
|
|
16
|
+
flexDirection: 'column',
|
|
17
|
+
alignItems: 'flex-start',
|
|
18
|
+
fontSize: cssVariableTheme.typography.fontSize.xs,
|
|
19
|
+
color: cssVariableTheme.text.secondary,
|
|
20
|
+
padding: '1em',
|
|
21
|
+
borderRadius: cssVariableTheme.shape.borderRadius.md,
|
|
22
|
+
border: `1px solid ${cssVariableTheme.action.subtleBorder}`,
|
|
23
|
+
transition: `color ${cssVariableTheme.transitions.duration.slow} ${cssVariableTheme.transitions.easing.default}`,
|
|
24
|
+
},
|
|
25
|
+
'&[data-disabled] label': {
|
|
26
|
+
color: cssVariableTheme.text.disabled,
|
|
27
|
+
},
|
|
28
|
+
'&:focus-within label': {
|
|
29
|
+
color: cssVariableTheme.palette.primary.main,
|
|
30
|
+
},
|
|
31
|
+
'& textarea': {
|
|
32
|
+
border: 'none',
|
|
33
|
+
backgroundColor: 'transparent',
|
|
34
|
+
outline: 'none',
|
|
35
|
+
fontSize: cssVariableTheme.typography.fontSize.sm,
|
|
36
|
+
fontFamily: 'monospace',
|
|
37
|
+
width: '100%',
|
|
38
|
+
resize: 'vertical',
|
|
39
|
+
color: cssVariableTheme.text.primary,
|
|
40
|
+
boxShadow: '0px 0px 0px rgba(128,128,128,0.1)',
|
|
41
|
+
transition: `box-shadow ${cssVariableTheme.transitions.duration.normal} ease`,
|
|
42
|
+
lineHeight: cssVariableTheme.typography.lineHeight.relaxed,
|
|
43
|
+
padding: `${cssVariableTheme.spacing.sm} 0`,
|
|
44
|
+
},
|
|
45
|
+
'&:focus-within textarea': {
|
|
46
|
+
boxShadow: `0px 3px 0px ${cssVariableTheme.palette.primary.main}`,
|
|
47
|
+
},
|
|
48
|
+
},
|
|
49
|
+
render: ({ props, useHostProps, useRef }) => {
|
|
50
|
+
const maxSize = props.maxImageSizeBytes ?? DEFAULT_MAX_IMAGE_SIZE;
|
|
51
|
+
const textareaRef = useRef('textarea');
|
|
52
|
+
useHostProps({
|
|
53
|
+
'data-disabled': props.disabled ? '' : undefined,
|
|
54
|
+
});
|
|
55
|
+
const handleInput = (ev) => {
|
|
56
|
+
const target = ev.target;
|
|
57
|
+
props.onValueChange?.(target.value);
|
|
58
|
+
};
|
|
59
|
+
const handlePaste = (ev) => {
|
|
60
|
+
const items = ev.clipboardData?.items;
|
|
61
|
+
if (!items)
|
|
62
|
+
return;
|
|
63
|
+
for (let i = 0; i < items.length; i++) {
|
|
64
|
+
const item = items[i];
|
|
65
|
+
if (item.type.startsWith('image/')) {
|
|
66
|
+
const file = item.getAsFile();
|
|
67
|
+
if (!file || file.size > maxSize)
|
|
68
|
+
continue;
|
|
69
|
+
ev.preventDefault();
|
|
70
|
+
const reader = new FileReader();
|
|
71
|
+
reader.onload = () => {
|
|
72
|
+
const base64 = reader.result;
|
|
73
|
+
const textarea = textareaRef.current;
|
|
74
|
+
if (!textarea)
|
|
75
|
+
return;
|
|
76
|
+
const start = textarea.selectionStart;
|
|
77
|
+
const end = textarea.selectionEnd;
|
|
78
|
+
const before = textarea.value.slice(0, start);
|
|
79
|
+
const after = textarea.value.slice(end);
|
|
80
|
+
const imageMarkdown = ``;
|
|
81
|
+
const newValue = before + imageMarkdown + after;
|
|
82
|
+
textarea.value = newValue;
|
|
83
|
+
const cursorPos = start + imageMarkdown.length;
|
|
84
|
+
textarea.setSelectionRange(cursorPos, cursorPos);
|
|
85
|
+
props.onValueChange?.(newValue);
|
|
86
|
+
};
|
|
87
|
+
reader.onerror = () => {
|
|
88
|
+
console.warn('Failed to read pasted image file');
|
|
89
|
+
};
|
|
90
|
+
reader.readAsDataURL(file);
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
return (createComponent("label", null,
|
|
96
|
+
props.labelTitle ? createComponent("span", null, props.labelTitle) : null,
|
|
97
|
+
createComponent("textarea", { ref: textareaRef, value: props.value, oninput: handleInput, onpaste: handlePaste, readOnly: props.readOnly, disabled: props.disabled, placeholder: props.placeholder, rows: props.rows ?? 10 })));
|
|
98
|
+
},
|
|
99
|
+
});
|
|
100
|
+
//# sourceMappingURL=markdown-input.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-input.js","sourceRoot":"","sources":["../../../src/components/markdown/markdown-input.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAA;AAC1D,OAAO,EAAE,gBAAgB,EAAE,MAAM,sCAAsC,CAAA;AAEvE,MAAM,sBAAsB,GAAG,GAAG,GAAG,IAAI,CAAA;AAqBzC;;;;GAIG;AACH,MAAM,CAAC,MAAM,aAAa,GAAG,KAAK,CAAqB;IACrD,aAAa,EAAE,sBAAsB;IACrC,GAAG,EAAE;QACH,OAAO,EAAE,OAAO;QAChB,YAAY,EAAE,KAAK;QAEnB,SAAS,EAAE;YACT,OAAO,EAAE,MAAM;YACf,aAAa,EAAE,QAAQ;YACvB,UAAU,EAAE,YAAY;YACxB,QAAQ,EAAE,gBAAgB,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;YACjD,KAAK,EAAE,gBAAgB,CAAC,IAAI,CAAC,SAAS;YACtC,OAAO,EAAE,KAAK;YACd,YAAY,EAAE,gBAAgB,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE;YACpD,MAAM,EAAE,aAAa,gBAAgB,CAAC,MAAM,CAAC,YAAY,EAAE;YAC3D,UAAU,EAAE,SAAS,gBAAgB,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,IAAI,gBAAgB,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,EAAE;SACjH;QAED,wBAAwB,EAAE;YACxB,KAAK,EAAE,gBAAgB,CAAC,IAAI,CAAC,QAAQ;SACtC;QAED,sBAAsB,EAAE;YACtB,KAAK,EAAE,gBAAgB,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI;SAC7C;QAED,YAAY,EAAE;YACZ,MAAM,EAAE,MAAM;YACd,eAAe,EAAE,aAAa;YAC9B,OAAO,EAAE,MAAM;YACf,QAAQ,EAAE,gBAAgB,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;YACjD,UAAU,EAAE,WAAW;YACvB,KAAK,EAAE,MAAM;YACb,MAAM,EAAE,UAAU;YAClB,KAAK,EAAE,gBAAgB,CAAC,IAAI,CAAC,OAAO;YACpC,SAAS,EAAE,mCAAmC;YAC9C,UAAU,EAAE,cAAc,gBAAgB,CAAC,WAAW,CAAC,QAAQ,CAAC,MAAM,OAAO;YAC7E,UAAU,EAAE,gBAAgB,CAAC,UAAU,CAAC,UAAU,CAAC,OAAO;YAC1D,OAAO,EAAE,GAAG,gBAAgB,CAAC,OAAO,CAAC,EAAE,IAAI;SAC5C;QAED,yBAAyB,EAAE;YACzB,SAAS,EAAE,eAAe,gBAAgB,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE;SAClE;KACF;IACD,MAAM,EAAE,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,MAAM,EAAE,EAAE,EAAE;QAC1C,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,IAAI,sBAAsB,CAAA;QACjE,MAAM,WAAW,GAAG,MAAM,CAAsB,UAAU,CAAC,CAAA;QAE3D,YAAY,CAAC;YACX,eAAe,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS;SACjD,CAAC,CAAA;QAEF,MAAM,WAAW,GAAG,CAAC,EAAS,EAAE,EAAE;YAChC,MAAM,MAAM,GAAG,EAAE,CAAC,MAA6B,CAAA;YAC/C,KAAK,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;QACrC,CAAC,CAAA;QAED,MAAM,WAAW,GAAG,CAAC,EAAkB,EAAE,EAAE;YACzC,MAAM,KAAK,GAAG,EAAE,CAAC,aAAa,EAAE,KAAK,CAAA;YACrC,IAAI,CAAC,KAAK;gBAAE,OAAM;YAElB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACtC,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAA;gBACrB,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;oBACnC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,EAAE,CAAA;oBAC7B,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,GAAG,OAAO;wBAAE,SAAQ;oBAE1C,EAAE,CAAC,cAAc,EAAE,CAAA;oBAEnB,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAA;oBAC/B,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;wBACnB,MAAM,MAAM,GAAG,MAAM,CAAC,MAAgB,CAAA;wBACtC,MAAM,QAAQ,GAAG,WAAW,CAAC,OAAO,CAAA;wBACpC,IAAI,CAAC,QAAQ;4BAAE,OAAM;wBAErB,MAAM,KAAK,GAAG,QAAQ,CAAC,cAAc,CAAA;wBACrC,MAAM,GAAG,GAAG,QAAQ,CAAC,YAAY,CAAA;wBACjC,MAAM,MAAM,GAAG,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAA;wBAC7C,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;wBACvC,MAAM,aAAa,GAAG,mBAAmB,MAAM,GAAG,CAAA;wBAClD,MAAM,QAAQ,GAAG,MAAM,GAAG,aAAa,GAAG,KAAK,CAAA;wBAE/C,QAAQ,CAAC,KAAK,GAAG,QAAQ,CAAA;wBACzB,MAAM,SAAS,GAAG,KAAK,GAAG,aAAa,CAAC,MAAM,CAAA;wBAC9C,QAAQ,CAAC,iBAAiB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAA;wBAChD,KAAK,CAAC,aAAa,EAAE,CAAC,QAAQ,CAAC,CAAA;oBACjC,CAAC,CAAA;oBACD,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE;wBACpB,OAAO,CAAC,IAAI,CAAC,kCAAkC,CAAC,CAAA;oBAClD,CAAC,CAAA;oBACD,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAA;oBAC1B,OAAM;gBACR,CAAC;YACH,CAAC;QACH,CAAC,CAAA;QAED,OAAO,CACL;YACG,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,8BAAO,KAAK,CAAC,UAAU,CAAQ,CAAC,CAAC,CAAC,IAAI;YAC1D,8BACE,GAAG,EAAE,WAAW,EAChB,KAAK,EAAE,KAAK,CAAC,KAAK,EAClB,OAAO,EAAE,WAAW,EACpB,OAAO,EAAE,WAAW,EACpB,QAAQ,EAAE,KAAK,CAAC,QAAQ,EACxB,QAAQ,EAAE,KAAK,CAAC,QAAQ,EACxB,WAAW,EAAE,KAAK,CAAC,WAAW,EAC9B,IAAI,EAAE,KAAK,CAAC,IAAI,IAAI,EAAE,GACtB,CACI,CACT,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"markdown-input.spec.d.ts","sourceRoot":"","sources":["../../../src/components/markdown/markdown-input.spec.tsx"],"names":[],"mappings":""}
|