solid-mds 0.3.2
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 +21 -0
- package/README.md +476 -0
- package/dist/index.d.ts +50 -0
- package/dist/index.jsx +402 -0
- package/package.json +66 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 Matthias Reis
|
|
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,476 @@
|
|
|
1
|
+
# solid-mds
|
|
2
|
+
|
|
3
|
+
A SolidJS library for parsing and rendering MDS (Markdown Steps) format — an
|
|
4
|
+
extended Markdown syntax designed for step-based content like slide decks,
|
|
5
|
+
tutorials, and wizards.
|
|
6
|
+
|
|
7
|
+
## Installation
|
|
8
|
+
|
|
9
|
+
```bash
|
|
10
|
+
npm install solid-mds
|
|
11
|
+
# or
|
|
12
|
+
pnpm add solid-mds
|
|
13
|
+
```
|
|
14
|
+
|
|
15
|
+
**Peer dependency:** This library requires SolidJS ^1.9.0
|
|
16
|
+
|
|
17
|
+
## Quick Start
|
|
18
|
+
|
|
19
|
+
```tsx
|
|
20
|
+
import { parse } from "solid-mds";
|
|
21
|
+
|
|
22
|
+
const markdown = `
|
|
23
|
+
# Hello World
|
|
24
|
+
|
|
25
|
+
This is **bold** and *italic* text.
|
|
26
|
+
`;
|
|
27
|
+
|
|
28
|
+
function App() {
|
|
29
|
+
const result = parse(markdown);
|
|
30
|
+
return <div>{result.steps.default.body}</div>;
|
|
31
|
+
}
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
## Opinionated Plugin Configuration
|
|
35
|
+
|
|
36
|
+
solid-mds comes pre-configured with an opinionated set of remark/rehype plugins:
|
|
37
|
+
|
|
38
|
+
- **[remark-gfm](https://github.com/remarkjs/remark-gfm)** — GitHub Flavored
|
|
39
|
+
Markdown (tables, strikethrough, autolinks, task lists)
|
|
40
|
+
- **[remark-math](https://github.com/remarkjs/remark-math)** +
|
|
41
|
+
**[rehype-katex](https://github.com/remarkjs/rehype-katex)** — LaTeX math
|
|
42
|
+
rendering
|
|
43
|
+
|
|
44
|
+
These plugins are always enabled and cannot be configured. This ensures
|
|
45
|
+
consistent rendering across all use cases.
|
|
46
|
+
|
|
47
|
+
**Note:** To display math properly, include the KaTeX CSS in your app:
|
|
48
|
+
|
|
49
|
+
```html
|
|
50
|
+
<link
|
|
51
|
+
rel="stylesheet"
|
|
52
|
+
href="https://cdn.jsdelivr.net/npm/katex@0.16.0/dist/katex.min.css"
|
|
53
|
+
/>
|
|
54
|
+
```
|
|
55
|
+
|
|
56
|
+
## Supported Markdown Features
|
|
57
|
+
|
|
58
|
+
### Standard Markdown
|
|
59
|
+
|
|
60
|
+
```markdown
|
|
61
|
+
# Heading 1
|
|
62
|
+
|
|
63
|
+
## Heading 2
|
|
64
|
+
|
|
65
|
+
### Heading 3
|
|
66
|
+
|
|
67
|
+
**bold** and _italic_ and ~~strikethrough~~
|
|
68
|
+
|
|
69
|
+
- Unordered list
|
|
70
|
+
- Another item
|
|
71
|
+
|
|
72
|
+
1. Ordered list
|
|
73
|
+
2. Second item
|
|
74
|
+
|
|
75
|
+
> Blockquotes
|
|
76
|
+
|
|
77
|
+
`inline code`
|
|
78
|
+
|
|
79
|
+
[Links](https://example.com)
|
|
80
|
+
|
|
81
|
+

|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
### GFM Extensions
|
|
85
|
+
|
|
86
|
+
#### Tables
|
|
87
|
+
|
|
88
|
+
```markdown
|
|
89
|
+
| Header 1 | Header 2 |
|
|
90
|
+
| -------- | -------- |
|
|
91
|
+
| Cell 1 | Cell 2 |
|
|
92
|
+
| Cell 3 | Cell 4 |
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
#### Task Lists
|
|
96
|
+
|
|
97
|
+
```markdown
|
|
98
|
+
- [x] Completed task
|
|
99
|
+
- [ ] Incomplete task
|
|
100
|
+
```
|
|
101
|
+
|
|
102
|
+
#### Autolinks
|
|
103
|
+
|
|
104
|
+
```markdown
|
|
105
|
+
Visit https://example.com automatically linked.
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
### Math (LaTeX)
|
|
109
|
+
|
|
110
|
+
#### Inline Math
|
|
111
|
+
|
|
112
|
+
```markdown
|
|
113
|
+
The equation $E = mc^2$ is famous.
|
|
114
|
+
```
|
|
115
|
+
|
|
116
|
+
#### Block Math
|
|
117
|
+
|
|
118
|
+
```markdown
|
|
119
|
+
$$
|
|
120
|
+
\int_{-\infty}^{\infty} e^{-x^2} dx = \sqrt{\pi}
|
|
121
|
+
$$
|
|
122
|
+
```
|
|
123
|
+
|
|
124
|
+
## MDS Format: Steps
|
|
125
|
+
|
|
126
|
+
MDS extends Markdown with a step-based structure using `+++stepId` separators:
|
|
127
|
+
|
|
128
|
+
```markdown
|
|
129
|
+
+++intro
|
|
130
|
+
# Welcome
|
|
131
|
+
|
|
132
|
+
This is the introduction step.
|
|
133
|
+
|
|
134
|
+
+++
|
|
135
|
+
|
|
136
|
+
main
|
|
137
|
+
|
|
138
|
+
# Main Content
|
|
139
|
+
|
|
140
|
+
This is the main step.
|
|
141
|
+
|
|
142
|
+
+++conclusion
|
|
143
|
+
|
|
144
|
+
# Summary
|
|
145
|
+
|
|
146
|
+
This is the conclusion.
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
### Parsing Steps
|
|
150
|
+
|
|
151
|
+
```tsx
|
|
152
|
+
import { parse } from "solid-mds";
|
|
153
|
+
|
|
154
|
+
const result = parse(markdown);
|
|
155
|
+
|
|
156
|
+
// result.first → "intro" (first step ID)
|
|
157
|
+
// result.count → 3 (number of steps)
|
|
158
|
+
// result.steps.intro.body → JSX.Element
|
|
159
|
+
// result.steps.intro.next → "main"
|
|
160
|
+
// result.steps.main.prev → "intro"
|
|
161
|
+
// result.steps.main.next → "conclusion"
|
|
162
|
+
// result.steps.conclusion.prev → "main"
|
|
163
|
+
// result.steps.conclusion.next → null
|
|
164
|
+
```
|
|
165
|
+
|
|
166
|
+
### Step Object Structure
|
|
167
|
+
|
|
168
|
+
Each step contains:
|
|
169
|
+
|
|
170
|
+
```ts
|
|
171
|
+
interface Step {
|
|
172
|
+
id: string; // Step identifier
|
|
173
|
+
body: JSX.Element; // Rendered markdown content
|
|
174
|
+
local: object; // Local metadata (see below)
|
|
175
|
+
prev: string | null; // Previous step ID
|
|
176
|
+
next: string | null; // Next step ID
|
|
177
|
+
current: number; // 1-based position
|
|
178
|
+
}
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
## Metadata Blocks
|
|
182
|
+
|
|
183
|
+
### Local Metadata (per step)
|
|
184
|
+
|
|
185
|
+
Use ` ```@| ` blocks to define key-value metadata for a step:
|
|
186
|
+
|
|
187
|
+
````markdown
|
|
188
|
+
+++slide1
|
|
189
|
+
|
|
190
|
+
```@|
|
|
191
|
+
title: Introduction
|
|
192
|
+
speaker: John Doe
|
|
193
|
+
```
|
|
194
|
+
|
|
195
|
+
# Welcome to the Presentation
|
|
196
|
+
````
|
|
197
|
+
|
|
198
|
+
Access via `result.steps.slide1.local.title`.
|
|
199
|
+
|
|
200
|
+
### Local Markdown Blocks
|
|
201
|
+
|
|
202
|
+
Use ` ```@/name ` to define named markdown content within a step:
|
|
203
|
+
|
|
204
|
+
````markdown
|
|
205
|
+
+++slide1
|
|
206
|
+
|
|
207
|
+
```@/notes
|
|
208
|
+
These are speaker notes with **formatting**.
|
|
209
|
+
```
|
|
210
|
+
|
|
211
|
+
# Main Slide Content
|
|
212
|
+
````
|
|
213
|
+
|
|
214
|
+
Access via `result.steps.slide1.local.notes` (returns `JSX.Element`).
|
|
215
|
+
|
|
216
|
+
### Global Metadata
|
|
217
|
+
|
|
218
|
+
Use ` ```@@| ` for document-wide metadata:
|
|
219
|
+
|
|
220
|
+
````markdown
|
|
221
|
+
```@@|
|
|
222
|
+
author: Jane Smith
|
|
223
|
+
version: 1.0
|
|
224
|
+
```
|
|
225
|
+
|
|
226
|
+
+++first
|
|
227
|
+
|
|
228
|
+
# First Step
|
|
229
|
+
````
|
|
230
|
+
|
|
231
|
+
Access via `result.global.author`.
|
|
232
|
+
|
|
233
|
+
### Global Markdown Blocks
|
|
234
|
+
|
|
235
|
+
Use ` ```@@/name ` for global named markdown content:
|
|
236
|
+
|
|
237
|
+
````markdown
|
|
238
|
+
```@@/footer
|
|
239
|
+
Made with ❤️ by **Our Team**
|
|
240
|
+
```
|
|
241
|
+
|
|
242
|
+
+++first
|
|
243
|
+
|
|
244
|
+
# Content
|
|
245
|
+
````
|
|
246
|
+
|
|
247
|
+
Access via `result.global.footer` (returns `JSX.Element`).
|
|
248
|
+
|
|
249
|
+
## Custom Components
|
|
250
|
+
|
|
251
|
+
Override default HTML elements or create custom block components:
|
|
252
|
+
|
|
253
|
+
### Override Standard Elements
|
|
254
|
+
|
|
255
|
+
```tsx
|
|
256
|
+
import {
|
|
257
|
+
parse,
|
|
258
|
+
ComponentMap,
|
|
259
|
+
StandardComponentProps,
|
|
260
|
+
} from "solid-mds";
|
|
261
|
+
|
|
262
|
+
const CustomHeading = (props: StandardComponentProps) => (
|
|
263
|
+
<h1 class="text-4xl font-bold text-blue-600">{props.children}</h1>
|
|
264
|
+
);
|
|
265
|
+
|
|
266
|
+
const components: ComponentMap = {
|
|
267
|
+
h1: CustomHeading,
|
|
268
|
+
};
|
|
269
|
+
|
|
270
|
+
const result = parse(markdown, components);
|
|
271
|
+
```
|
|
272
|
+
|
|
273
|
+
### Custom Blocks
|
|
274
|
+
|
|
275
|
+
Create custom components that receive special props from code blocks:
|
|
276
|
+
|
|
277
|
+
```tsx
|
|
278
|
+
import { parse, ComponentMap, CustomBlockProps } from "solid-mds";
|
|
279
|
+
|
|
280
|
+
// Define a custom component
|
|
281
|
+
const Alert = (props: CustomBlockProps) => (
|
|
282
|
+
<div class={`alert alert-${props.payload[0] || "info"}`}>
|
|
283
|
+
{props.children}
|
|
284
|
+
</div>
|
|
285
|
+
);
|
|
286
|
+
|
|
287
|
+
const components: ComponentMap = {
|
|
288
|
+
alert: Alert,
|
|
289
|
+
};
|
|
290
|
+
|
|
291
|
+
const markdown = `
|
|
292
|
+
\`\`\`alert/warning
|
|
293
|
+
This is a **warning** message!
|
|
294
|
+
\`\`\`
|
|
295
|
+
`;
|
|
296
|
+
|
|
297
|
+
const result = parse(markdown, components);
|
|
298
|
+
```
|
|
299
|
+
|
|
300
|
+
The code block ` ```slert/warning ` will render using your `Alert` component
|
|
301
|
+
with:
|
|
302
|
+
|
|
303
|
+
- `props.payload` → `["warning"]` (path segments after component name)
|
|
304
|
+
- `props.raw` → `"This is a **warning** message!"` (raw content as string if you
|
|
305
|
+
need it)
|
|
306
|
+
- `props.children` → Rendered markdown as JSX
|
|
307
|
+
|
|
308
|
+
### Custom Blocks with Data
|
|
309
|
+
|
|
310
|
+
Use the `|` suffix for YAML data instead of markdown content:
|
|
311
|
+
|
|
312
|
+
```tsx
|
|
313
|
+
const Card = (props: CustomBlockProps) => (
|
|
314
|
+
<div class="card">
|
|
315
|
+
<h2>{props.data?.title}</h2>
|
|
316
|
+
<p>{props.data?.description}</p>
|
|
317
|
+
</div>
|
|
318
|
+
);
|
|
319
|
+
|
|
320
|
+
const components: ComponentMap = { Card };
|
|
321
|
+
|
|
322
|
+
const markdown = `
|
|
323
|
+
\`\`\`Card|
|
|
324
|
+
title: My Card Title
|
|
325
|
+
description: This is the card description
|
|
326
|
+
\`\`\`
|
|
327
|
+
`;
|
|
328
|
+
|
|
329
|
+
const result = parse(markdown, components);
|
|
330
|
+
```
|
|
331
|
+
|
|
332
|
+
With ` ```Card| `, the content is parsed as YAML and available via `props.data`.
|
|
333
|
+
|
|
334
|
+
## Complete Example
|
|
335
|
+
|
|
336
|
+
```tsx
|
|
337
|
+
import { createSignal, Show, For } from "solid-js";
|
|
338
|
+
import {
|
|
339
|
+
parse,
|
|
340
|
+
ComponentMap,
|
|
341
|
+
CustomBlockProps,
|
|
342
|
+
StandardComponentProps,
|
|
343
|
+
} from "solid-mds";
|
|
344
|
+
|
|
345
|
+
// Custom components
|
|
346
|
+
const Slide = (props: CustomBlockProps) => (
|
|
347
|
+
<div class="slide" data-layout={props.payload[0]}>
|
|
348
|
+
{props.children}
|
|
349
|
+
</div>
|
|
350
|
+
);
|
|
351
|
+
|
|
352
|
+
const Code = (props: StandardComponentProps) => (
|
|
353
|
+
<pre class="bg-gray-800 text-white p-4 rounded">
|
|
354
|
+
<code>{props.children}</code>
|
|
355
|
+
</pre>
|
|
356
|
+
);
|
|
357
|
+
|
|
358
|
+
const components: ComponentMap = {
|
|
359
|
+
Slide,
|
|
360
|
+
code: Code,
|
|
361
|
+
};
|
|
362
|
+
|
|
363
|
+
// Markdown content
|
|
364
|
+
const content = `
|
|
365
|
+
\`\`\`@@|
|
|
366
|
+
title: My Presentation
|
|
367
|
+
author: Developer
|
|
368
|
+
\`\`\`
|
|
369
|
+
|
|
370
|
+
+++intro
|
|
371
|
+
\`\`\`@|
|
|
372
|
+
transition: fade
|
|
373
|
+
\`\`\`
|
|
374
|
+
|
|
375
|
+
# Welcome
|
|
376
|
+
|
|
377
|
+
Introduction slide content.
|
|
378
|
+
|
|
379
|
+
+++demo
|
|
380
|
+
\`\`\`Slide/centered
|
|
381
|
+
# Demo Time
|
|
382
|
+
|
|
383
|
+
Check out this code:
|
|
384
|
+
|
|
385
|
+
\`\`\`js
|
|
386
|
+
console.log("Hello!");
|
|
387
|
+
\`\`\`
|
|
388
|
+
\`\`\`
|
|
389
|
+
|
|
390
|
+
+++end
|
|
391
|
+
# Thank You!
|
|
392
|
+
|
|
393
|
+
Questions?
|
|
394
|
+
`;
|
|
395
|
+
|
|
396
|
+
function Presentation() {
|
|
397
|
+
const result = parse(content, components);
|
|
398
|
+
const [currentId, setCurrentId] = createSignal(result.first);
|
|
399
|
+
|
|
400
|
+
const currentStep = () => result.steps[currentId()!];
|
|
401
|
+
|
|
402
|
+
const goNext = () => {
|
|
403
|
+
const next = currentStep()?.next;
|
|
404
|
+
if (next) setCurrentId(next);
|
|
405
|
+
};
|
|
406
|
+
|
|
407
|
+
const goPrev = () => {
|
|
408
|
+
const prev = currentStep()?.prev;
|
|
409
|
+
if (prev) setCurrentId(prev);
|
|
410
|
+
};
|
|
411
|
+
|
|
412
|
+
return (
|
|
413
|
+
<div class="presentation">
|
|
414
|
+
<header>
|
|
415
|
+
<h1>{result.global?.title}</h1>
|
|
416
|
+
<span>by {result.global?.author}</span>
|
|
417
|
+
</header>
|
|
418
|
+
|
|
419
|
+
<main>
|
|
420
|
+
<Show when={currentStep()}>{currentStep().body}</Show>
|
|
421
|
+
</main>
|
|
422
|
+
|
|
423
|
+
<footer>
|
|
424
|
+
<button onClick={goPrev} disabled={!currentStep()?.prev}>
|
|
425
|
+
Previous
|
|
426
|
+
</button>
|
|
427
|
+
<span>
|
|
428
|
+
{currentStep()?.current} / {result.count}
|
|
429
|
+
</span>
|
|
430
|
+
<button onClick={goNext} disabled={!currentStep()?.next}>
|
|
431
|
+
Next
|
|
432
|
+
</button>
|
|
433
|
+
</footer>
|
|
434
|
+
</div>
|
|
435
|
+
);
|
|
436
|
+
}
|
|
437
|
+
```
|
|
438
|
+
|
|
439
|
+
## API Reference
|
|
440
|
+
|
|
441
|
+
### `parse<TGlobal, TLocal>(input: string, components?: ComponentMap): ParseResult`
|
|
442
|
+
|
|
443
|
+
Parses MDS content and returns a structured result. Please provide proper types
|
|
444
|
+
for your global and local scope to receive a properly typed result.
|
|
445
|
+
|
|
446
|
+
**Parameters:**
|
|
447
|
+
|
|
448
|
+
- `input` — The MDS/Markdown string to parse
|
|
449
|
+
- `components` — Optional map of custom components
|
|
450
|
+
|
|
451
|
+
**Returns:**
|
|
452
|
+
|
|
453
|
+
```ts
|
|
454
|
+
interface ParseResult<TGlobal, TLocal> {
|
|
455
|
+
first: string | null; // First step ID
|
|
456
|
+
steps: Record<string, Step<TLocal>>; // All steps by ID
|
|
457
|
+
count: number; // Total number of steps
|
|
458
|
+
global: TGlobal | null; // Global metadata
|
|
459
|
+
}
|
|
460
|
+
```
|
|
461
|
+
|
|
462
|
+
### Type Exports
|
|
463
|
+
|
|
464
|
+
```ts
|
|
465
|
+
import type {
|
|
466
|
+
ComponentMap,
|
|
467
|
+
CustomBlockProps,
|
|
468
|
+
StandardComponentProps,
|
|
469
|
+
ParseResult,
|
|
470
|
+
Step,
|
|
471
|
+
} from "solid-mds";
|
|
472
|
+
```
|
|
473
|
+
|
|
474
|
+
## License
|
|
475
|
+
|
|
476
|
+
MIT
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { Component, JSX } from 'solid-js';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Props for custom block components (code blocks with name/sub/path syntax)
|
|
5
|
+
*/
|
|
6
|
+
interface CustomBlockProps {
|
|
7
|
+
/** The path segments after the component name, e.g. ['substructure', 'subsub'] */
|
|
8
|
+
payload: string[];
|
|
9
|
+
/** The raw content of the code block as plain text */
|
|
10
|
+
raw: string;
|
|
11
|
+
/** Parsed YAML data when using componentName| syntax */
|
|
12
|
+
data?: Record<string, string | string[]>;
|
|
13
|
+
/** The markdown-rendered content as children */
|
|
14
|
+
children?: JSX.Element;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Standard component props for regular HTML elements
|
|
18
|
+
*/
|
|
19
|
+
interface StandardComponentProps {
|
|
20
|
+
children?: JSX.Element;
|
|
21
|
+
[key: string]: unknown;
|
|
22
|
+
}
|
|
23
|
+
type ComponentMap = {
|
|
24
|
+
[tagName: string]: Component<CustomBlockProps> | Component<StandardComponentProps>;
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
interface StepLocalBase {
|
|
28
|
+
[key: string]: unknown;
|
|
29
|
+
}
|
|
30
|
+
interface GlobalMetaBase {
|
|
31
|
+
[key: string]: unknown;
|
|
32
|
+
}
|
|
33
|
+
interface Step<TLocal = StepLocalBase> {
|
|
34
|
+
id: string;
|
|
35
|
+
local: TLocal;
|
|
36
|
+
/** Component function that renders the step body. Call it to get JSX: {step.Body()} */
|
|
37
|
+
Body: Component;
|
|
38
|
+
prev: string | null;
|
|
39
|
+
next: string | null;
|
|
40
|
+
current: number;
|
|
41
|
+
}
|
|
42
|
+
interface ParseResult<TGlobal = GlobalMetaBase, TLocal = StepLocalBase> {
|
|
43
|
+
first: string | null;
|
|
44
|
+
steps: Record<string, Step<TLocal>>;
|
|
45
|
+
count: number;
|
|
46
|
+
global: TGlobal | null;
|
|
47
|
+
}
|
|
48
|
+
declare function parse<TGlobal = GlobalMetaBase, TLocal = StepLocalBase>(input: string, components?: ComponentMap): ParseResult<TGlobal, TLocal>;
|
|
49
|
+
|
|
50
|
+
export { type ComponentMap, type CustomBlockProps, type GlobalMetaBase, type ParseResult, type StandardComponentProps, type Step, type StepLocalBase, parse };
|
package/dist/index.jsx
ADDED
|
@@ -0,0 +1,402 @@
|
|
|
1
|
+
// src/index.tsx
|
|
2
|
+
import { unified } from "unified";
|
|
3
|
+
import remarkParse from "remark-parse";
|
|
4
|
+
import remarkGfm from "remark-gfm";
|
|
5
|
+
import remarkMath from "remark-math";
|
|
6
|
+
import remarkRehype from "remark-rehype";
|
|
7
|
+
import rehypeKatex from "rehype-katex";
|
|
8
|
+
import { parse as parseYaml2 } from "yaml";
|
|
9
|
+
|
|
10
|
+
// src/hast-to-solid.tsx
|
|
11
|
+
function convertProperties(properties) {
|
|
12
|
+
if (!properties) return {};
|
|
13
|
+
const result = {};
|
|
14
|
+
for (const [key, value] of Object.entries(properties)) {
|
|
15
|
+
if (key === "className") {
|
|
16
|
+
result.class = value;
|
|
17
|
+
} else if (key === "class") {
|
|
18
|
+
result.class = Array.isArray(value) ? value.join(" ") : value;
|
|
19
|
+
} else {
|
|
20
|
+
result[key] = value;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return result;
|
|
24
|
+
}
|
|
25
|
+
function renderElement(tagName, props, children) {
|
|
26
|
+
switch (tagName) {
|
|
27
|
+
case "p":
|
|
28
|
+
return <p {...props}>{children}</p>;
|
|
29
|
+
case "h1":
|
|
30
|
+
return <h1 {...props}>{children}</h1>;
|
|
31
|
+
case "h2":
|
|
32
|
+
return <h2 {...props}>{children}</h2>;
|
|
33
|
+
case "h3":
|
|
34
|
+
return <h3 {...props}>{children}</h3>;
|
|
35
|
+
case "h4":
|
|
36
|
+
return <h4 {...props}>{children}</h4>;
|
|
37
|
+
case "h5":
|
|
38
|
+
return <h5 {...props}>{children}</h5>;
|
|
39
|
+
case "h6":
|
|
40
|
+
return <h6 {...props}>{children}</h6>;
|
|
41
|
+
case "div":
|
|
42
|
+
return <div {...props}>{children}</div>;
|
|
43
|
+
case "span":
|
|
44
|
+
return <span {...props}>{children}</span>;
|
|
45
|
+
case "a":
|
|
46
|
+
return <a {...props}>{children}</a>;
|
|
47
|
+
case "strong":
|
|
48
|
+
return <strong {...props}>{children}</strong>;
|
|
49
|
+
case "b":
|
|
50
|
+
return <b {...props}>{children}</b>;
|
|
51
|
+
case "em":
|
|
52
|
+
return <em {...props}>{children}</em>;
|
|
53
|
+
case "i":
|
|
54
|
+
return <i {...props}>{children}</i>;
|
|
55
|
+
case "code":
|
|
56
|
+
return <code {...props}>{children}</code>;
|
|
57
|
+
case "pre":
|
|
58
|
+
return <pre {...props}>{children}</pre>;
|
|
59
|
+
case "blockquote":
|
|
60
|
+
return <blockquote {...props}>{children}</blockquote>;
|
|
61
|
+
case "ul":
|
|
62
|
+
return <ul {...props}>{children}</ul>;
|
|
63
|
+
case "ol":
|
|
64
|
+
return <ol {...props}>{children}</ol>;
|
|
65
|
+
case "li":
|
|
66
|
+
return <li {...props}>{children}</li>;
|
|
67
|
+
case "hr":
|
|
68
|
+
return <hr {...props} />;
|
|
69
|
+
case "br":
|
|
70
|
+
return <br {...props} />;
|
|
71
|
+
case "img":
|
|
72
|
+
return <img {...props} />;
|
|
73
|
+
case "table":
|
|
74
|
+
return <table {...props}>{children}</table>;
|
|
75
|
+
case "thead":
|
|
76
|
+
return <thead {...props}>{children}</thead>;
|
|
77
|
+
case "tbody":
|
|
78
|
+
return <tbody {...props}>{children}</tbody>;
|
|
79
|
+
case "tr":
|
|
80
|
+
return <tr {...props}>{children}</tr>;
|
|
81
|
+
case "th":
|
|
82
|
+
return <th {...props}>{children}</th>;
|
|
83
|
+
case "td":
|
|
84
|
+
return <td {...props}>{children}</td>;
|
|
85
|
+
case "del":
|
|
86
|
+
return <del {...props}>{children}</del>;
|
|
87
|
+
case "sup":
|
|
88
|
+
return <sup {...props}>{children}</sup>;
|
|
89
|
+
case "sub":
|
|
90
|
+
return <sub {...props}>{children}</sub>;
|
|
91
|
+
// GFM elements
|
|
92
|
+
case "input":
|
|
93
|
+
return <input {...props} />;
|
|
94
|
+
case "section":
|
|
95
|
+
return <section {...props}>{children}</section>;
|
|
96
|
+
// KaTeX math elements
|
|
97
|
+
case "math":
|
|
98
|
+
return <math {...props}>{children}</math>;
|
|
99
|
+
case "semantics":
|
|
100
|
+
return <semantics {...props}>{children}</semantics>;
|
|
101
|
+
case "mrow":
|
|
102
|
+
return <mrow {...props}>{children}</mrow>;
|
|
103
|
+
case "mi":
|
|
104
|
+
return <mi {...props}>{children}</mi>;
|
|
105
|
+
case "mo":
|
|
106
|
+
return <mo {...props}>{children}</mo>;
|
|
107
|
+
case "mn":
|
|
108
|
+
return <mn {...props}>{children}</mn>;
|
|
109
|
+
case "mfrac":
|
|
110
|
+
return <mfrac {...props}>{children}</mfrac>;
|
|
111
|
+
case "msup":
|
|
112
|
+
return <msup {...props}>{children}</msup>;
|
|
113
|
+
case "msub":
|
|
114
|
+
return <msub {...props}>{children}</msub>;
|
|
115
|
+
case "msubsup":
|
|
116
|
+
return <msubsup {...props}>{children}</msubsup>;
|
|
117
|
+
case "msqrt":
|
|
118
|
+
return <msqrt {...props}>{children}</msqrt>;
|
|
119
|
+
case "mroot":
|
|
120
|
+
return <mroot {...props}>{children}</mroot>;
|
|
121
|
+
case "munder":
|
|
122
|
+
return <munder {...props}>{children}</munder>;
|
|
123
|
+
case "mover":
|
|
124
|
+
return <mover {...props}>{children}</mover>;
|
|
125
|
+
case "munderover":
|
|
126
|
+
return <munderover {...props}>{children}</munderover>;
|
|
127
|
+
case "mtable":
|
|
128
|
+
return <mtable {...props}>{children}</mtable>;
|
|
129
|
+
case "mtr":
|
|
130
|
+
return <mtr {...props}>{children}</mtr>;
|
|
131
|
+
case "mtd":
|
|
132
|
+
return <mtd {...props}>{children}</mtd>;
|
|
133
|
+
case "mtext":
|
|
134
|
+
return <mtext {...props}>{children}</mtext>;
|
|
135
|
+
case "mspace":
|
|
136
|
+
return <mspace {...props}>{children}</mspace>;
|
|
137
|
+
case "annotation":
|
|
138
|
+
return <annotation {...props}>{children}</annotation>;
|
|
139
|
+
case "svg":
|
|
140
|
+
return <svg {...props}>{children}</svg>;
|
|
141
|
+
case "path":
|
|
142
|
+
return <path {...props} />;
|
|
143
|
+
case "line":
|
|
144
|
+
return <line {...props} />;
|
|
145
|
+
case "rect":
|
|
146
|
+
return <rect {...props} />;
|
|
147
|
+
case "circle":
|
|
148
|
+
return <circle {...props} />;
|
|
149
|
+
case "g":
|
|
150
|
+
return <g {...props}>{children}</g>;
|
|
151
|
+
case "defs":
|
|
152
|
+
return <defs {...props}>{children}</defs>;
|
|
153
|
+
case "clipPath":
|
|
154
|
+
return <clipPath {...props}>{children}</clipPath>;
|
|
155
|
+
case "use":
|
|
156
|
+
return <use {...props} />;
|
|
157
|
+
default:
|
|
158
|
+
return <div {...props}>{children}</div>;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
function renderNode(node, components) {
|
|
162
|
+
if (node.type === "text") {
|
|
163
|
+
return node.value;
|
|
164
|
+
}
|
|
165
|
+
if (node.type === "element") {
|
|
166
|
+
const element = node;
|
|
167
|
+
const tagName = element.tagName;
|
|
168
|
+
const rawProps = convertProperties(element.properties);
|
|
169
|
+
const CustomComponent = components[tagName];
|
|
170
|
+
const childElements = element.children.length > 0 ? element.children.map((child) => renderNode(child, components)) : null;
|
|
171
|
+
if (CustomComponent) {
|
|
172
|
+
if (typeof rawProps.payload === "string" && typeof rawProps.raw === "string") {
|
|
173
|
+
const customBlockProps = {
|
|
174
|
+
payload: JSON.parse(rawProps.payload),
|
|
175
|
+
raw: rawProps.raw
|
|
176
|
+
};
|
|
177
|
+
if (typeof rawProps.data === "string") {
|
|
178
|
+
customBlockProps.data = JSON.parse(rawProps.data);
|
|
179
|
+
}
|
|
180
|
+
const Comp2 = CustomComponent;
|
|
181
|
+
return <Comp2 {...customBlockProps}>{childElements}</Comp2>;
|
|
182
|
+
}
|
|
183
|
+
const Comp = CustomComponent;
|
|
184
|
+
return <Comp {...rawProps}>{childElements}</Comp>;
|
|
185
|
+
}
|
|
186
|
+
return renderElement(tagName, rawProps, childElements);
|
|
187
|
+
}
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
function hastToSolidComponent(hastTree, components = {}) {
|
|
191
|
+
const treeData = hastTree;
|
|
192
|
+
const componentMap = components;
|
|
193
|
+
return function HastBody() {
|
|
194
|
+
const elements = treeData.children.map((node) => renderNode(node, componentMap));
|
|
195
|
+
return <>{elements}</>;
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
// src/remark-custom-blocks.ts
|
|
200
|
+
import { visit } from "unist-util-visit";
|
|
201
|
+
import { parse as parseYaml } from "yaml";
|
|
202
|
+
function remarkCustomBlocks(options) {
|
|
203
|
+
const { components, processor } = options;
|
|
204
|
+
return (tree) => {
|
|
205
|
+
visit(tree, "code", (node, index, parent) => {
|
|
206
|
+
if (!parent || index === void 0) return;
|
|
207
|
+
const lang = node.lang;
|
|
208
|
+
if (!lang) return;
|
|
209
|
+
const isDataBlock = lang.endsWith("|");
|
|
210
|
+
const cleanLang = isDataBlock ? lang.slice(0, -1) : lang;
|
|
211
|
+
const segments = cleanLang.split("/");
|
|
212
|
+
const componentName = segments[0];
|
|
213
|
+
const raw = node.value;
|
|
214
|
+
if (!components.has(componentName)) return;
|
|
215
|
+
const payload = segments.slice(1);
|
|
216
|
+
if (isDataBlock) {
|
|
217
|
+
const data = parseYaml(raw);
|
|
218
|
+
const customNode2 = {
|
|
219
|
+
type: "customBlock",
|
|
220
|
+
data: {
|
|
221
|
+
hName: componentName,
|
|
222
|
+
hProperties: {
|
|
223
|
+
payload: JSON.stringify(payload),
|
|
224
|
+
raw,
|
|
225
|
+
data: JSON.stringify(data)
|
|
226
|
+
}
|
|
227
|
+
},
|
|
228
|
+
children: []
|
|
229
|
+
};
|
|
230
|
+
parent.children[index] = customNode2;
|
|
231
|
+
return;
|
|
232
|
+
}
|
|
233
|
+
const contentMdast = processor.parse(raw);
|
|
234
|
+
const customNode = {
|
|
235
|
+
type: "customBlock",
|
|
236
|
+
data: {
|
|
237
|
+
hName: componentName,
|
|
238
|
+
hProperties: {
|
|
239
|
+
payload: JSON.stringify(payload),
|
|
240
|
+
raw
|
|
241
|
+
}
|
|
242
|
+
},
|
|
243
|
+
children: contentMdast.children
|
|
244
|
+
};
|
|
245
|
+
parent.children[index] = customNode;
|
|
246
|
+
});
|
|
247
|
+
};
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// src/index.tsx
|
|
251
|
+
function parseLocalMeta(content, processor, components) {
|
|
252
|
+
const metaPattern = /```@\|\s*\n([\s\S]*?)```\s*\n?/g;
|
|
253
|
+
let match;
|
|
254
|
+
let local = {};
|
|
255
|
+
let body = content;
|
|
256
|
+
while ((match = metaPattern.exec(content)) !== null) {
|
|
257
|
+
const parsed = parseYaml2(match[1]);
|
|
258
|
+
local = { ...local, ...parsed };
|
|
259
|
+
}
|
|
260
|
+
body = content.replace(metaPattern, "").trim();
|
|
261
|
+
const localMdPattern = /```@\/(\w+)\s*\n([\s\S]*?)```\s*\n?/g;
|
|
262
|
+
const localMdBlocks = {};
|
|
263
|
+
while ((match = localMdPattern.exec(body)) !== null) {
|
|
264
|
+
const name = match[1];
|
|
265
|
+
const content2 = match[2].trim();
|
|
266
|
+
if (localMdBlocks[name]) {
|
|
267
|
+
localMdBlocks[name] += "\n\n" + content2;
|
|
268
|
+
} else {
|
|
269
|
+
localMdBlocks[name] = content2;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
body = body.replace(localMdPattern, "").trim();
|
|
273
|
+
for (const [name, mdContent] of Object.entries(localMdBlocks)) {
|
|
274
|
+
const mdast = processor.parse(mdContent);
|
|
275
|
+
const hastTree = processor.runSync(mdast);
|
|
276
|
+
local[name] = hastToSolidComponent(hastTree, components);
|
|
277
|
+
}
|
|
278
|
+
return { local, body };
|
|
279
|
+
}
|
|
280
|
+
function parseGlobalMeta(input, processor, components) {
|
|
281
|
+
const globalDataPattern = /```@@\|\s*\n([\s\S]*?)```/g;
|
|
282
|
+
let match;
|
|
283
|
+
let global = {};
|
|
284
|
+
while ((match = globalDataPattern.exec(input)) !== null) {
|
|
285
|
+
const parsed = parseYaml2(match[1]);
|
|
286
|
+
global = { ...global, ...parsed };
|
|
287
|
+
}
|
|
288
|
+
const globalMdPattern = /```@@\/(\w+)\s*\n([\s\S]*?)```/g;
|
|
289
|
+
const globalMdBlocks = {};
|
|
290
|
+
while ((match = globalMdPattern.exec(input)) !== null) {
|
|
291
|
+
const name = match[1];
|
|
292
|
+
const content = match[2].trim();
|
|
293
|
+
if (globalMdBlocks[name]) {
|
|
294
|
+
globalMdBlocks[name] += "\n\n" + content;
|
|
295
|
+
} else {
|
|
296
|
+
globalMdBlocks[name] = content;
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
for (const [name, mdContent] of Object.entries(globalMdBlocks)) {
|
|
300
|
+
const mdast = processor.parse(mdContent);
|
|
301
|
+
const hastTree = processor.runSync(mdast);
|
|
302
|
+
global[name] = hastToSolidComponent(hastTree, components);
|
|
303
|
+
}
|
|
304
|
+
return Object.keys(global).length > 0 ? global : null;
|
|
305
|
+
}
|
|
306
|
+
function isInsideCodeBlock(input, position) {
|
|
307
|
+
const beforeText = input.slice(0, position);
|
|
308
|
+
const fencePattern = /^```/gm;
|
|
309
|
+
let fenceCount = 0;
|
|
310
|
+
let match;
|
|
311
|
+
while ((match = fencePattern.exec(beforeText)) !== null) {
|
|
312
|
+
fenceCount++;
|
|
313
|
+
}
|
|
314
|
+
return fenceCount % 2 === 1;
|
|
315
|
+
}
|
|
316
|
+
var VALID_ID_PATTERN = /^[a-z0-9-]+$/;
|
|
317
|
+
function validateStepId(id, lineNumber) {
|
|
318
|
+
if (!VALID_ID_PATTERN.test(id)) {
|
|
319
|
+
throw new Error(
|
|
320
|
+
`Invalid step ID "${id}" at line ${lineNumber}. Step IDs must only contain lowercase letters (a-z), numbers (0-9), and hyphens (-).`
|
|
321
|
+
);
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
function parse(input, components = {}) {
|
|
325
|
+
const stepPattern = /^\+\+\+(.+)$/gm;
|
|
326
|
+
const steps = {};
|
|
327
|
+
const componentNames = new Set(Object.keys(components));
|
|
328
|
+
const baseProcessor = unified().use(remarkParse).use(remarkGfm).use(remarkMath);
|
|
329
|
+
const processor = unified().use(remarkParse).use(remarkGfm).use(remarkMath).use(remarkCustomBlocks, {
|
|
330
|
+
components: componentNames,
|
|
331
|
+
processor: baseProcessor
|
|
332
|
+
}).use(remarkRehype, { allowDangerousHtml: true }).use(rehypeKatex);
|
|
333
|
+
const global = parseGlobalMeta(input, processor, components);
|
|
334
|
+
let match;
|
|
335
|
+
const matches = [];
|
|
336
|
+
while ((match = stepPattern.exec(input)) !== null) {
|
|
337
|
+
if (!isInsideCodeBlock(input, match.index)) {
|
|
338
|
+
const id = match[1].trim();
|
|
339
|
+
const lineNumber = input.slice(0, match.index).split("\n").length;
|
|
340
|
+
validateStepId(id, lineNumber);
|
|
341
|
+
matches.push({ id, index: match.index });
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
if (matches.length === 0) {
|
|
345
|
+
const { local, body: markdown } = parseLocalMeta(
|
|
346
|
+
input.trim(),
|
|
347
|
+
processor,
|
|
348
|
+
components
|
|
349
|
+
);
|
|
350
|
+
const mdast = processor.parse(markdown);
|
|
351
|
+
const hastTree = processor.runSync(mdast);
|
|
352
|
+
const Body = hastToSolidComponent(hastTree, components);
|
|
353
|
+
steps["default"] = {
|
|
354
|
+
id: "default",
|
|
355
|
+
local,
|
|
356
|
+
Body,
|
|
357
|
+
prev: null,
|
|
358
|
+
next: null,
|
|
359
|
+
current: 1
|
|
360
|
+
};
|
|
361
|
+
return {
|
|
362
|
+
first: "default",
|
|
363
|
+
steps,
|
|
364
|
+
count: 1,
|
|
365
|
+
global
|
|
366
|
+
};
|
|
367
|
+
}
|
|
368
|
+
for (let i = 0; i < matches.length; i++) {
|
|
369
|
+
const current = matches[i];
|
|
370
|
+
const nextMatch = matches[i + 1];
|
|
371
|
+
const prevMatch = matches[i - 1];
|
|
372
|
+
const startIndex = current.index + `+++${current.id}`.length;
|
|
373
|
+
const endIndex = nextMatch ? nextMatch.index : input.length;
|
|
374
|
+
const rawContent = input.slice(startIndex, endIndex).trim();
|
|
375
|
+
const { local, body: markdown } = parseLocalMeta(
|
|
376
|
+
rawContent,
|
|
377
|
+
processor,
|
|
378
|
+
components
|
|
379
|
+
);
|
|
380
|
+
const mdast = processor.parse(markdown);
|
|
381
|
+
const hastTree = processor.runSync(mdast);
|
|
382
|
+
const Body = hastToSolidComponent(hastTree, components);
|
|
383
|
+
steps[current.id] = {
|
|
384
|
+
id: current.id,
|
|
385
|
+
local,
|
|
386
|
+
Body,
|
|
387
|
+
prev: prevMatch ? prevMatch.id : null,
|
|
388
|
+
next: nextMatch ? nextMatch.id : null,
|
|
389
|
+
current: i + 1
|
|
390
|
+
};
|
|
391
|
+
}
|
|
392
|
+
return {
|
|
393
|
+
first: matches.length > 0 ? matches[0].id : null,
|
|
394
|
+
steps,
|
|
395
|
+
count: matches.length,
|
|
396
|
+
global
|
|
397
|
+
};
|
|
398
|
+
}
|
|
399
|
+
export {
|
|
400
|
+
parse
|
|
401
|
+
};
|
|
402
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.tsx", "../src/hast-to-solid.tsx", "../src/remark-custom-blocks.ts"],
  "sourcesContent": ["import { Component, JSX } from \"solid-js\";\nimport { unified, type Processor } from \"unified\";\nimport remarkParse from \"remark-parse\";\nimport remarkGfm from \"remark-gfm\";\nimport remarkMath from \"remark-math\";\nimport remarkRehype from \"remark-rehype\";\nimport rehypeKatex from \"rehype-katex\";\nimport { parse as parseYaml } from \"yaml\";\nimport type { Root } from \"hast\";\nimport {\n  hastToSolidJsx,\n  hastToSolidComponent,\n  ComponentMap,\n} from \"./hast-to-solid\";\nimport { remarkCustomBlocks } from \"./remark-custom-blocks\";\n\nexport type {\n  ComponentMap,\n  CustomBlockProps,\n  StandardComponentProps,\n} from \"./hast-to-solid\";\n\nexport interface StepLocalBase {\n  [key: string]: unknown;\n}\n\nexport interface GlobalMetaBase {\n  [key: string]: unknown;\n}\n\nexport interface Step<TLocal = StepLocalBase> {\n  id: string;\n  local: TLocal;\n  /** Component function that renders the step body. Call it to get JSX: {step.Body()} */\n  Body: Component;\n  prev: string | null;\n  next: string | null;\n  current: number;\n}\n\nexport interface ParseResult<TGlobal = GlobalMetaBase, TLocal = StepLocalBase> {\n  first: string | null;\n  steps: Record<string, Step<TLocal>>;\n  count: number;\n  global: TGlobal | null;\n}\n\n\nfunction parseLocalMeta(\n  content: string,\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  processor: Processor<any, any, any, any, any>,\n  components: ComponentMap,\n): { local: StepLocalBase; body: string } {\n  // Match all local metadata blocks: ```@| ... ```\n  // Multiple blocks are merged, later values override earlier ones\n  const metaPattern = /```@\\|\\s*\\n([\\s\\S]*?)```\\s*\\n?/g;\n  let match: RegExpExecArray | null;\n  let local: StepLocalBase = {};\n  let body = content;\n\n  while ((match = metaPattern.exec(content)) !== null) {\n    const parsed = parseYaml(match[1]);\n    local = { ...local, ...parsed };\n  }\n\n  // Remove all metadata blocks from body\n  body = content.replace(metaPattern, \"\").trim();\n\n  // Match local md blocks: ```@/name ... ```\n  // These are markdown blocks that get rendered and stored in local[name]\n  const localMdPattern = /```@\\/(\\w+)\\s*\\n([\\s\\S]*?)```\\s*\\n?/g;\n  const localMdBlocks: Record<string, string> = {};\n\n  while ((match = localMdPattern.exec(body)) !== null) {\n    const name = match[1];\n    const content = match[2].trim();\n    // Concatenate if multiple blocks with same name\n    if (localMdBlocks[name]) {\n      localMdBlocks[name] += \"\\n\\n\" + content;\n    } else {\n      localMdBlocks[name] = content;\n    }\n  }\n\n  // Remove all local md blocks from body\n  body = body.replace(localMdPattern, \"\").trim();\n\n  // Convert all local md blocks to JSX\n  for (const [name, mdContent] of Object.entries(localMdBlocks)) {\n    const mdast = processor.parse(mdContent);\n    const hastTree = processor.runSync(mdast) as Root;\n    local[name] = hastToSolidComponent(hastTree, components);\n  }\n\n  return { local, body };\n}\n\nfunction parseGlobalMeta(\n  input: string,\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  processor: Processor<any, any, any, any, any>,\n  components: ComponentMap,\n): GlobalMetaBase | null {\n  // Match global data blocks: ```@@| ... ```\n  // Can appear anywhere in the document, may have multiple blocks\n  const globalDataPattern = /```@@\\|\\s*\\n([\\s\\S]*?)```/g;\n  let match: RegExpExecArray | null;\n  let global: GlobalMetaBase = {};\n\n  while ((match = globalDataPattern.exec(input)) !== null) {\n    const parsed = parseYaml(match[1]);\n    global = { ...global, ...parsed };\n  }\n\n  // Match global md blocks: ```@@/name ... ```\n  // These are markdown blocks that get rendered and stored in global[name]\n  const globalMdPattern = /```@@\\/(\\w+)\\s*\\n([\\s\\S]*?)```/g;\n  const globalMdBlocks: Record<string, string> = {};\n\n  while ((match = globalMdPattern.exec(input)) !== null) {\n    const name = match[1];\n    const content = match[2].trim();\n    // Concatenate if multiple blocks with same name\n    if (globalMdBlocks[name]) {\n      globalMdBlocks[name] += \"\\n\\n\" + content;\n    } else {\n      globalMdBlocks[name] = content;\n    }\n  }\n\n  // Convert all global md blocks to Components\n  for (const [name, mdContent] of Object.entries(globalMdBlocks)) {\n    const mdast = processor.parse(mdContent);\n    const hastTree = processor.runSync(mdast) as Root;\n    global[name] = hastToSolidComponent(hastTree, components);\n  }\n\n  return Object.keys(global).length > 0 ? global : null;\n}\n\nfunction isInsideCodeBlock(input: string, position: number): boolean {\n  // Find all code block fences before this position\n  const beforeText = input.slice(0, position);\n  const fencePattern = /^```/gm;\n  let fenceCount = 0;\n  let match: RegExpExecArray | null;\n\n  while ((match = fencePattern.exec(beforeText)) !== null) {\n    fenceCount++;\n  }\n\n  // If odd number of fences, we're inside a code block\n  return fenceCount % 2 === 1;\n}\n\nconst VALID_ID_PATTERN = /^[a-z0-9-]+$/;\n\nfunction validateStepId(id: string, lineNumber: number): void {\n  if (!VALID_ID_PATTERN.test(id)) {\n    throw new Error(\n      `Invalid step ID \"${id}\" at line ${lineNumber}. ` +\n        `Step IDs must only contain lowercase letters (a-z), numbers (0-9), and hyphens (-).`,\n    );\n  }\n}\n\nexport function parse<TGlobal = GlobalMetaBase, TLocal = StepLocalBase>(\n  input: string,\n  components: ComponentMap = {},\n): ParseResult<TGlobal, TLocal> {\n  // Capture everything after +++ until end of line\n  const stepPattern = /^\\+\\+\\+(.+)$/gm;\n  const steps: Record<string, Step<TLocal>> = {};\n  const componentNames = new Set(Object.keys(components));\n  const baseProcessor = unified()\n    .use(remarkParse)\n    .use(remarkGfm)\n    .use(remarkMath);\n  const processor = unified()\n    .use(remarkParse)\n    .use(remarkGfm)\n    .use(remarkMath)\n    .use(remarkCustomBlocks, {\n      components: componentNames,\n      processor: baseProcessor,\n    })\n    .use(remarkRehype, { allowDangerousHtml: true })\n    .use(rehypeKatex);\n  const global = parseGlobalMeta(input, processor, components);\n\n  let match: RegExpExecArray | null;\n  const matches: { id: string; index: number }[] = [];\n\n  while ((match = stepPattern.exec(input)) !== null) {\n    // Skip this match if it's inside a code block\n    if (!isInsideCodeBlock(input, match.index)) {\n      const id = match[1].trim();\n      // Calculate line number for error messages\n      const lineNumber = input.slice(0, match.index).split(\"\\n\").length;\n      validateStepId(id, lineNumber);\n      matches.push({ id, index: match.index });\n    }\n  }\n\n  // If no separators found, treat the entire input as a single \"default\" step\n  if (matches.length === 0) {\n    const { local, body: markdown } = parseLocalMeta(\n      input.trim(),\n      processor,\n      components,\n    );\n\n    const mdast = processor.parse(markdown);\n    const hastTree = processor.runSync(mdast) as Root;\n    const Body = hastToSolidComponent(hastTree, components);\n\n    steps[\"default\"] = {\n      id: \"default\",\n      local: local as TLocal,\n      Body,\n      prev: null,\n      next: null,\n      current: 1,\n    };\n\n    return {\n      first: \"default\",\n      steps,\n      count: 1,\n      global: global as TGlobal | null,\n    };\n  }\n\n  for (let i = 0; i < matches.length; i++) {\n    const current = matches[i];\n    const nextMatch = matches[i + 1];\n    const prevMatch = matches[i - 1];\n\n    const startIndex = current.index + `+++${current.id}`.length;\n    const endIndex = nextMatch ? nextMatch.index : input.length;\n\n    const rawContent = input.slice(startIndex, endIndex).trim();\n    const { local, body: markdown } = parseLocalMeta(\n      rawContent,\n      processor,\n      components,\n    );\n\n    const mdast = processor.parse(markdown);\n    const hastTree = processor.runSync(mdast) as Root;\n    const Body = hastToSolidComponent(hastTree, components);\n\n    steps[current.id] = {\n      id: current.id,\n      local: local as TLocal,\n      Body,\n      prev: prevMatch ? prevMatch.id : null,\n      next: nextMatch ? nextMatch.id : null,\n      current: i + 1,\n    };\n  }\n\n  return {\n    first: matches.length > 0 ? matches[0].id : null,\n    steps,\n    count: matches.length,\n    global: global as TGlobal | null,\n  };\n}\n", "import { JSX, Component, children as resolveChildren } from \"solid-js\";\nimport type { Root, Element, Text, RootContent } from \"hast\";\n\n/**\n * Props for custom block components (code blocks with name/sub/path syntax)\n */\nexport interface CustomBlockProps {\n  /** The path segments after the component name, e.g. ['substructure', 'subsub'] */\n  payload: string[];\n  /** The raw content of the code block as plain text */\n  raw: string;\n  /** Parsed YAML data when using componentName| syntax */\n  data?: Record<string, string | string[]>;\n  /** The markdown-rendered content as children */\n  children?: JSX.Element;\n}\n\n/**\n * Standard component props for regular HTML elements\n */\nexport interface StandardComponentProps {\n  children?: JSX.Element;\n  [key: string]: unknown;\n}\n\nexport type ComponentMap = {\n  [tagName: string]: Component<CustomBlockProps> | Component<StandardComponentProps>;\n};\n\nfunction convertProperties(\n  properties: Record<string, unknown> | undefined\n): Record<string, unknown> {\n  if (!properties) return {};\n\n  const result: Record<string, unknown> = {};\n\n  for (const [key, value] of Object.entries(properties)) {\n    if (key === \"className\") {\n      result.class = value;\n    } else if (key === \"class\") {\n      result.class = Array.isArray(value) ? value.join(\" \") : value;\n    } else {\n      result[key] = value;\n    }\n  }\n\n  return result;\n}\n\nfunction renderElement(\n  tagName: string,\n  props: Record<string, unknown>,\n  children: JSX.Element\n): JSX.Element {\n  switch (tagName) {\n    case \"p\":\n      return <p {...props}>{children}</p>;\n    case \"h1\":\n      return <h1 {...props}>{children}</h1>;\n    case \"h2\":\n      return <h2 {...props}>{children}</h2>;\n    case \"h3\":\n      return <h3 {...props}>{children}</h3>;\n    case \"h4\":\n      return <h4 {...props}>{children}</h4>;\n    case \"h5\":\n      return <h5 {...props}>{children}</h5>;\n    case \"h6\":\n      return <h6 {...props}>{children}</h6>;\n    case \"div\":\n      return <div {...props}>{children}</div>;\n    case \"span\":\n      return <span {...props}>{children}</span>;\n    case \"a\":\n      return <a {...props}>{children}</a>;\n    case \"strong\":\n      return <strong {...props}>{children}</strong>;\n    case \"b\":\n      return <b {...props}>{children}</b>;\n    case \"em\":\n      return <em {...props}>{children}</em>;\n    case \"i\":\n      return <i {...props}>{children}</i>;\n    case \"code\":\n      return <code {...props}>{children}</code>;\n    case \"pre\":\n      return <pre {...props}>{children}</pre>;\n    case \"blockquote\":\n      return <blockquote {...props}>{children}</blockquote>;\n    case \"ul\":\n      return <ul {...props}>{children}</ul>;\n    case \"ol\":\n      return <ol {...props}>{children}</ol>;\n    case \"li\":\n      return <li {...props}>{children}</li>;\n    case \"hr\":\n      return <hr {...props} />;\n    case \"br\":\n      return <br {...props} />;\n    case \"img\":\n      return <img {...props} />;\n    case \"table\":\n      return <table {...props}>{children}</table>;\n    case \"thead\":\n      return <thead {...props}>{children}</thead>;\n    case \"tbody\":\n      return <tbody {...props}>{children}</tbody>;\n    case \"tr\":\n      return <tr {...props}>{children}</tr>;\n    case \"th\":\n      return <th {...props}>{children}</th>;\n    case \"td\":\n      return <td {...props}>{children}</td>;\n    case \"del\":\n      return <del {...props}>{children}</del>;\n    case \"sup\":\n      return <sup {...props}>{children}</sup>;\n    case \"sub\":\n      return <sub {...props}>{children}</sub>;\n    // GFM elements\n    case \"input\":\n      return <input {...props} />;\n    case \"section\":\n      return <section {...props}>{children}</section>;\n    // KaTeX math elements\n    case \"math\":\n      return <math {...props}>{children}</math>;\n    case \"semantics\":\n      return <semantics {...props}>{children}</semantics>;\n    case \"mrow\":\n      return <mrow {...props}>{children}</mrow>;\n    case \"mi\":\n      return <mi {...props}>{children}</mi>;\n    case \"mo\":\n      return <mo {...props}>{children}</mo>;\n    case \"mn\":\n      return <mn {...props}>{children}</mn>;\n    case \"mfrac\":\n      return <mfrac {...props}>{children}</mfrac>;\n    case \"msup\":\n      return <msup {...props}>{children}</msup>;\n    case \"msub\":\n      return <msub {...props}>{children}</msub>;\n    case \"msubsup\":\n      return <msubsup {...props}>{children}</msubsup>;\n    case \"msqrt\":\n      return <msqrt {...props}>{children}</msqrt>;\n    case \"mroot\":\n      return <mroot {...props}>{children}</mroot>;\n    case \"munder\":\n      return <munder {...props}>{children}</munder>;\n    case \"mover\":\n      return <mover {...props}>{children}</mover>;\n    case \"munderover\":\n      return <munderover {...props}>{children}</munderover>;\n    case \"mtable\":\n      return <mtable {...props}>{children}</mtable>;\n    case \"mtr\":\n      return <mtr {...props}>{children}</mtr>;\n    case \"mtd\":\n      return <mtd {...props}>{children}</mtd>;\n    case \"mtext\":\n      return <mtext {...props}>{children}</mtext>;\n    case \"mspace\":\n      return <mspace {...props}>{children}</mspace>;\n    case \"annotation\":\n      return <annotation {...props}>{children}</annotation>;\n    case \"svg\":\n      return <svg {...props}>{children}</svg>;\n    case \"path\":\n      return <path {...props} />;\n    case \"line\":\n      return <line {...props} />;\n    case \"rect\":\n      return <rect {...props} />;\n    case \"circle\":\n      return <circle {...props} />;\n    case \"g\":\n      return <g {...props}>{children}</g>;\n    case \"defs\":\n      return <defs {...props}>{children}</defs>;\n    case \"clipPath\":\n      return <clipPath {...props}>{children}</clipPath>;\n    case \"use\":\n      return <use {...props} />;\n    default:\n      return <div {...props}>{children}</div>;\n  }\n}\n\nfunction renderNode(\n  node: RootContent,\n  components: ComponentMap\n): JSX.Element | string | null {\n  if (node.type === \"text\") {\n    return (node as Text).value;\n  }\n\n  if (node.type === \"element\") {\n    const element = node as Element;\n    const tagName = element.tagName;\n    const rawProps = convertProperties(element.properties);\n    const CustomComponent = components[tagName];\n\n    const childElements =\n      element.children.length > 0\n        ? element.children.map((child) => renderNode(child, components))\n        : null;\n\n    if (CustomComponent) {\n      // Check if this is a custom block component (has payload and raw props)\n      if (typeof rawProps.payload === \"string\" && typeof rawProps.raw === \"string\") {\n        // Deserialize payload from JSON string\n        const customBlockProps: CustomBlockProps = {\n          payload: JSON.parse(rawProps.payload as string),\n          raw: rawProps.raw as string,\n        };\n        // Add data if present (for componentName| syntax)\n        if (typeof rawProps.data === \"string\") {\n          customBlockProps.data = JSON.parse(rawProps.data as string);\n        }\n        const Comp = CustomComponent as Component<CustomBlockProps>;\n        return <Comp {...customBlockProps}>{childElements}</Comp>;\n      }\n      const Comp = CustomComponent as Component<StandardComponentProps>;\n      return <Comp {...rawProps}>{childElements}</Comp>;\n    }\n\n    return renderElement(tagName, rawProps, childElements as JSX.Element);\n  }\n\n  return null;\n}\n\nexport function hastToSolidJsx(\n  hastTree: Root,\n  components: ComponentMap = {}\n): JSX.Element {\n  const elements = hastTree.children.map((node) => renderNode(node, components));\n  return <>{elements}</>;\n}\n\n/**\n * Returns a component function that renders the HAST tree when called.\n * This defers JSX element creation until render time, which is important\n * for SSR hydration to work correctly.\n *\n * We serialize the HAST tree and components info so that the exact same\n * rendering logic runs on both server and client.\n */\nexport function hastToSolidComponent(\n  hastTree: Root,\n  components: ComponentMap = {}\n): () => JSX.Element {\n  // Store the tree data, not pre-rendered elements\n  const treeData = hastTree;\n  const componentMap = components;\n\n  return function HastBody() {\n    const elements = treeData.children.map((node) => renderNode(node, componentMap));\n    return <>{elements}</>;\n  };\n}\n", "import type { Root, Code, Parent, RootContent } from \"mdast\";\nimport type { Processor } from \"unified\";\nimport { visit } from \"unist-util-visit\";\nimport { parse as parseYaml } from \"yaml\";\n\n/**\n * Component that receives custom block props\n */\nexport interface CustomBlockComponentProps {\n  /** The path segments after the component name, e.g. ['substructure', 'subsub'] */\n  payload: string[];\n  /** The raw content of the code block as plain text */\n  raw: string;\n  /** Parsed YAML data when using componentName| syntax */\n  data?: Record<string, unknown>;\n  /** The markdown-rendered content as children */\n  children?: unknown;\n}\n\n/**\n * Set of component names that are available\n */\nexport type CustomBlockComponents = Set<string>;\n\nexport interface RemarkCustomBlocksOptions {\n  /** Set of component names that are available */\n  components: CustomBlockComponents;\n  /** The processor to use for parsing nested markdown */\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  processor: Processor<any, any, any, any, any>;\n}\n\n\n/**\n * A remark plugin that transforms code blocks with `name/path/segments` syntax\n * into custom block nodes that render with the specified component.\n *\n * Code blocks like:\n * ```componentName/sub/path\n * content here\n * ```\n *\n * Will be transformed to use the component if it exists in the components set,\n * otherwise they remain as regular code blocks.\n */\nexport function remarkCustomBlocks(options: RemarkCustomBlocksOptions) {\n  const { components, processor } = options;\n\n  return (tree: Root) => {\n    visit(tree, \"code\", (node: Code, index, parent: Parent | undefined) => {\n      if (!parent || index === undefined) return;\n\n      const lang = node.lang;\n      if (!lang) return;\n\n      // Check if this is a data block (lang ends with |)\n      const isDataBlock = lang.endsWith(\"|\");\n      const cleanLang = isDataBlock ? lang.slice(0, -1) : lang;\n\n      const segments = cleanLang.split(\"/\");\n      const componentName = segments[0];\n      const raw = node.value;\n\n      // If the component doesn't exist, leave it as a regular code block\n      if (!components.has(componentName)) return;\n\n      const payload = segments.slice(1);\n\n      // For data blocks, parse YAML instead of markdown\n      if (isDataBlock) {\n        const data = parseYaml(raw) as Record<string, unknown>;\n\n        const customNode = {\n          type: \"customBlock\",\n          data: {\n            hName: componentName,\n            hProperties: {\n              payload: JSON.stringify(payload),\n              raw,\n              data: JSON.stringify(data),\n            },\n          },\n          children: [],\n        };\n\n        parent.children[index] = customNode as unknown as RootContent;\n        return;\n      }\n\n      // Parse the content as markdown to get its AST\n      const contentMdast = processor.parse(raw) as Root;\n\n      // Create a custom element that will be converted to hast\n      // We use a custom node type that remark-rehype will handle\n      const customNode = {\n        type: \"customBlock\",\n        data: {\n          hName: componentName,\n          hProperties: {\n            payload: JSON.stringify(payload),\n            raw,\n          },\n        },\n        children: contentMdast.children,\n      };\n\n      // Replace the code node with our custom block\n      parent.children[index] = customNode as unknown as RootContent;\n    });\n  };\n}\n"],
  "mappings": ";AACA,SAAS,eAA+B;AACxC,OAAO,iBAAiB;AACxB,OAAO,eAAe;AACtB,OAAO,gBAAgB;AACvB,OAAO,kBAAkB;AACzB,OAAO,iBAAiB;AACxB,SAAS,SAASA,kBAAiB;;;ACsBnC,SAAS,kBACP,YACyB;AACzB,MAAI,CAAC,WAAY,QAAO,CAAC;AAEzB,QAAM,SAAkC,CAAC;AAEzC,aAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,UAAU,GAAG;AACrD,QAAI,QAAQ,aAAa;AACvB,aAAO,QAAQ;AAAA,IACjB,WAAW,QAAQ,SAAS;AAC1B,aAAO,QAAQ,MAAM,QAAQ,KAAK,IAAI,MAAM,KAAK,GAAG,IAAI;AAAA,IAC1D,OAAO;AACL,aAAO,GAAG,IAAI;AAAA,IAChB;AAAA,EACF;AAEA,SAAO;AACT;AAEA,SAAS,cACP,SACA,OACA,UACa;AACb,UAAQ,SAAS;AAAA,IACf,KAAK;AACH,aAAO,CAAC,MAAM,QAAQ,SAAS,EAAvB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,MAAM,QAAQ,SAAS,EAAvB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,WAAW,QAAQ,SAAS,EAA5B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,MAAM,QAAQ,SAAS,EAAvB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,MAAM,QAAQ,SAAS,EAAvB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,eAAe,QAAQ,SAAS,EAAhC;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,OAAO;AAAA,IACxB,KAAK;AACH,aAAO,CAAC,OAAO,OAAO;AAAA,IACxB,KAAK;AACH,aAAO,CAAC,QAAQ,OAAO;AAAA,IACzB,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA;AAAA,IAEV,KAAK;AACH,aAAO,CAAC,UAAU,OAAO;AAAA,IAC3B,KAAK;AACH,aAAO,CAAC,YAAY,QAAQ,SAAS,EAA7B;AAAA;AAAA,IAEV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,cAAc,QAAQ,SAAS,EAA/B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,OAAO,QAAQ,SAAS,EAAxB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,YAAY,QAAQ,SAAS,EAA7B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,WAAW,QAAQ,SAAS,EAA5B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,eAAe,QAAQ,SAAS,EAAhC;AAAA,IACV,KAAK;AACH,aAAO,CAAC,WAAW,QAAQ,SAAS,EAA5B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,UAAU,QAAQ,SAAS,EAA3B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,WAAW,QAAQ,SAAS,EAA5B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,eAAe,QAAQ,SAAS,EAAhC;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,OAAO;AAAA,IAC1B,KAAK;AACH,aAAO,CAAC,SAAS,OAAO;AAAA,IAC1B,KAAK;AACH,aAAO,CAAC,SAAS,OAAO;AAAA,IAC1B,KAAK;AACH,aAAO,CAAC,WAAW,OAAO;AAAA,IAC5B,KAAK;AACH,aAAO,CAAC,MAAM,QAAQ,SAAS,EAAvB;AAAA,IACV,KAAK;AACH,aAAO,CAAC,SAAS,QAAQ,SAAS,EAA1B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,aAAa,QAAQ,SAAS,EAA9B;AAAA,IACV,KAAK;AACH,aAAO,CAAC,QAAQ,OAAO;AAAA,IACzB;AACE,aAAO,CAAC,QAAQ,QAAQ,SAAS,EAAzB;AAAA,EACZ;AACF;AAEA,SAAS,WACP,MACA,YAC6B;AAC7B,MAAI,KAAK,SAAS,QAAQ;AACxB,WAAQ,KAAc;AAAA,EACxB;AAEA,MAAI,KAAK,SAAS,WAAW;AAC3B,UAAM,UAAU;AAChB,UAAM,UAAU,QAAQ;AACxB,UAAM,WAAW,kBAAkB,QAAQ,UAAU;AACrD,UAAM,kBAAkB,WAAW,OAAO;AAE1C,UAAM,gBACJ,QAAQ,SAAS,SAAS,IACtB,QAAQ,SAAS,IAAI,CAAC,UAAU,WAAW,OAAO,UAAU,CAAC,IAC7D;AAEN,QAAI,iBAAiB;AAEnB,UAAI,OAAO,SAAS,YAAY,YAAY,OAAO,SAAS,QAAQ,UAAU;AAE5E,cAAM,mBAAqC;AAAA,UACzC,SAAS,KAAK,MAAM,SAAS,OAAiB;AAAA,UAC9C,KAAK,SAAS;AAAA,QAChB;AAEA,YAAI,OAAO,SAAS,SAAS,UAAU;AACrC,2BAAiB,OAAO,KAAK,MAAM,SAAS,IAAc;AAAA,QAC5D;AACA,cAAMC,QAAO;AACb,eAAO,CAACA,UAAS,mBAAmB,cAAc,EAA1CA;AAAA,MACV;AACA,YAAM,OAAO;AACb,aAAO,CAAC,SAAS,WAAW,cAAc,EAAlC;AAAA,IACV;AAEA,WAAO,cAAc,SAAS,UAAU,aAA4B;AAAA,EACtE;AAEA,SAAO;AACT;AAkBO,SAAS,qBACd,UACA,aAA2B,CAAC,GACT;AAEnB,QAAM,WAAW;AACjB,QAAM,eAAe;AAErB,SAAO,SAAS,WAAW;AACzB,UAAM,WAAW,SAAS,SAAS,IAAI,CAAC,SAAS,WAAW,MAAM,YAAY,CAAC;AAC/E,WAAO,GAAG,SAAS;AAAA,EACrB;AACF;;;ACpQA,SAAS,aAAa;AACtB,SAAS,SAAS,iBAAiB;AA0C5B,SAAS,mBAAmB,SAAoC;AACrE,QAAM,EAAE,YAAY,UAAU,IAAI;AAElC,SAAO,CAAC,SAAe;AACrB,UAAM,MAAM,QAAQ,CAAC,MAAY,OAAO,WAA+B;AACrE,UAAI,CAAC,UAAU,UAAU,OAAW;AAEpC,YAAM,OAAO,KAAK;AAClB,UAAI,CAAC,KAAM;AAGX,YAAM,cAAc,KAAK,SAAS,GAAG;AACrC,YAAM,YAAY,cAAc,KAAK,MAAM,GAAG,EAAE,IAAI;AAEpD,YAAM,WAAW,UAAU,MAAM,GAAG;AACpC,YAAM,gBAAgB,SAAS,CAAC;AAChC,YAAM,MAAM,KAAK;AAGjB,UAAI,CAAC,WAAW,IAAI,aAAa,EAAG;AAEpC,YAAM,UAAU,SAAS,MAAM,CAAC;AAGhC,UAAI,aAAa;AACf,cAAM,OAAO,UAAU,GAAG;AAE1B,cAAMC,cAAa;AAAA,UACjB,MAAM;AAAA,UACN,MAAM;AAAA,YACJ,OAAO;AAAA,YACP,aAAa;AAAA,cACX,SAAS,KAAK,UAAU,OAAO;AAAA,cAC/B;AAAA,cACA,MAAM,KAAK,UAAU,IAAI;AAAA,YAC3B;AAAA,UACF;AAAA,UACA,UAAU,CAAC;AAAA,QACb;AAEA,eAAO,SAAS,KAAK,IAAIA;AACzB;AAAA,MACF;AAGA,YAAM,eAAe,UAAU,MAAM,GAAG;AAIxC,YAAM,aAAa;AAAA,QACjB,MAAM;AAAA,QACN,MAAM;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,YACX,SAAS,KAAK,UAAU,OAAO;AAAA,YAC/B;AAAA,UACF;AAAA,QACF;AAAA,QACA,UAAU,aAAa;AAAA,MACzB;AAGA,aAAO,SAAS,KAAK,IAAI;AAAA,IAC3B,CAAC;AAAA,EACH;AACF;;;AF9DA,SAAS,eACP,SAEA,WACA,YACwC;AAGxC,QAAM,cAAc;AACpB,MAAI;AACJ,MAAI,QAAuB,CAAC;AAC5B,MAAI,OAAO;AAEX,UAAQ,QAAQ,YAAY,KAAK,OAAO,OAAO,MAAM;AACnD,UAAM,SAASC,WAAU,MAAM,CAAC,CAAC;AACjC,YAAQ,EAAE,GAAG,OAAO,GAAG,OAAO;AAAA,EAChC;AAGA,SAAO,QAAQ,QAAQ,aAAa,EAAE,EAAE,KAAK;AAI7C,QAAM,iBAAiB;AACvB,QAAM,gBAAwC,CAAC;AAE/C,UAAQ,QAAQ,eAAe,KAAK,IAAI,OAAO,MAAM;AACnD,UAAM,OAAO,MAAM,CAAC;AACpB,UAAMC,WAAU,MAAM,CAAC,EAAE,KAAK;AAE9B,QAAI,cAAc,IAAI,GAAG;AACvB,oBAAc,IAAI,KAAK,SAASA;AAAA,IAClC,OAAO;AACL,oBAAc,IAAI,IAAIA;AAAA,IACxB;AAAA,EACF;AAGA,SAAO,KAAK,QAAQ,gBAAgB,EAAE,EAAE,KAAK;AAG7C,aAAW,CAAC,MAAM,SAAS,KAAK,OAAO,QAAQ,aAAa,GAAG;AAC7D,UAAM,QAAQ,UAAU,MAAM,SAAS;AACvC,UAAM,WAAW,UAAU,QAAQ,KAAK;AACxC,UAAM,IAAI,IAAI,qBAAqB,UAAU,UAAU;AAAA,EACzD;AAEA,SAAO,EAAE,OAAO,KAAK;AACvB;AAEA,SAAS,gBACP,OAEA,WACA,YACuB;AAGvB,QAAM,oBAAoB;AAC1B,MAAI;AACJ,MAAI,SAAyB,CAAC;AAE9B,UAAQ,QAAQ,kBAAkB,KAAK,KAAK,OAAO,MAAM;AACvD,UAAM,SAASD,WAAU,MAAM,CAAC,CAAC;AACjC,aAAS,EAAE,GAAG,QAAQ,GAAG,OAAO;AAAA,EAClC;AAIA,QAAM,kBAAkB;AACxB,QAAM,iBAAyC,CAAC;AAEhD,UAAQ,QAAQ,gBAAgB,KAAK,KAAK,OAAO,MAAM;AACrD,UAAM,OAAO,MAAM,CAAC;AACpB,UAAM,UAAU,MAAM,CAAC,EAAE,KAAK;AAE9B,QAAI,eAAe,IAAI,GAAG;AACxB,qBAAe,IAAI,KAAK,SAAS;AAAA,IACnC,OAAO;AACL,qBAAe,IAAI,IAAI;AAAA,IACzB;AAAA,EACF;AAGA,aAAW,CAAC,MAAM,SAAS,KAAK,OAAO,QAAQ,cAAc,GAAG;AAC9D,UAAM,QAAQ,UAAU,MAAM,SAAS;AACvC,UAAM,WAAW,UAAU,QAAQ,KAAK;AACxC,WAAO,IAAI,IAAI,qBAAqB,UAAU,UAAU;AAAA,EAC1D;AAEA,SAAO,OAAO,KAAK,MAAM,EAAE,SAAS,IAAI,SAAS;AACnD;AAEA,SAAS,kBAAkB,OAAe,UAA2B;AAEnE,QAAM,aAAa,MAAM,MAAM,GAAG,QAAQ;AAC1C,QAAM,eAAe;AACrB,MAAI,aAAa;AACjB,MAAI;AAEJ,UAAQ,QAAQ,aAAa,KAAK,UAAU,OAAO,MAAM;AACvD;AAAA,EACF;AAGA,SAAO,aAAa,MAAM;AAC5B;AAEA,IAAM,mBAAmB;AAEzB,SAAS,eAAe,IAAY,YAA0B;AAC5D,MAAI,CAAC,iBAAiB,KAAK,EAAE,GAAG;AAC9B,UAAM,IAAI;AAAA,MACR,oBAAoB,EAAE,aAAa,UAAU;AAAA,IAE/C;AAAA,EACF;AACF;AAEO,SAAS,MACd,OACA,aAA2B,CAAC,GACE;AAE9B,QAAM,cAAc;AACpB,QAAM,QAAsC,CAAC;AAC7C,QAAM,iBAAiB,IAAI,IAAI,OAAO,KAAK,UAAU,CAAC;AACtD,QAAM,gBAAgB,QAAQ,EAC3B,IAAI,WAAW,EACf,IAAI,SAAS,EACb,IAAI,UAAU;AACjB,QAAM,YAAY,QAAQ,EACvB,IAAI,WAAW,EACf,IAAI,SAAS,EACb,IAAI,UAAU,EACd,IAAI,oBAAoB;AAAA,IACvB,YAAY;AAAA,IACZ,WAAW;AAAA,EACb,CAAC,EACA,IAAI,cAAc,EAAE,oBAAoB,KAAK,CAAC,EAC9C,IAAI,WAAW;AAClB,QAAM,SAAS,gBAAgB,OAAO,WAAW,UAAU;AAE3D,MAAI;AACJ,QAAM,UAA2C,CAAC;AAElD,UAAQ,QAAQ,YAAY,KAAK,KAAK,OAAO,MAAM;AAEjD,QAAI,CAAC,kBAAkB,OAAO,MAAM,KAAK,GAAG;AAC1C,YAAM,KAAK,MAAM,CAAC,EAAE,KAAK;AAEzB,YAAM,aAAa,MAAM,MAAM,GAAG,MAAM,KAAK,EAAE,MAAM,IAAI,EAAE;AAC3D,qBAAe,IAAI,UAAU;AAC7B,cAAQ,KAAK,EAAE,IAAI,OAAO,MAAM,MAAM,CAAC;AAAA,IACzC;AAAA,EACF;AAGA,MAAI,QAAQ,WAAW,GAAG;AACxB,UAAM,EAAE,OAAO,MAAM,SAAS,IAAI;AAAA,MAChC,MAAM,KAAK;AAAA,MACX;AAAA,MACA;AAAA,IACF;AAEA,UAAM,QAAQ,UAAU,MAAM,QAAQ;AACtC,UAAM,WAAW,UAAU,QAAQ,KAAK;AACxC,UAAM,OAAO,qBAAqB,UAAU,UAAU;AAEtD,UAAM,SAAS,IAAI;AAAA,MACjB,IAAI;AAAA,MACJ;AAAA,MACA;AAAA,MACA,MAAM;AAAA,MACN,MAAM;AAAA,MACN,SAAS;AAAA,IACX;AAEA,WAAO;AAAA,MACL,OAAO;AAAA,MACP;AAAA,MACA,OAAO;AAAA,MACP;AAAA,IACF;AAAA,EACF;AAEA,WAAS,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAK;AACvC,UAAM,UAAU,QAAQ,CAAC;AACzB,UAAM,YAAY,QAAQ,IAAI,CAAC;AAC/B,UAAM,YAAY,QAAQ,IAAI,CAAC;AAE/B,UAAM,aAAa,QAAQ,QAAQ,MAAM,QAAQ,EAAE,GAAG;AACtD,UAAM,WAAW,YAAY,UAAU,QAAQ,MAAM;AAErD,UAAM,aAAa,MAAM,MAAM,YAAY,QAAQ,EAAE,KAAK;AAC1D,UAAM,EAAE,OAAO,MAAM,SAAS,IAAI;AAAA,MAChC;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,UAAM,QAAQ,UAAU,MAAM,QAAQ;AACtC,UAAM,WAAW,UAAU,QAAQ,KAAK;AACxC,UAAM,OAAO,qBAAqB,UAAU,UAAU;AAEtD,UAAM,QAAQ,EAAE,IAAI;AAAA,MAClB,IAAI,QAAQ;AAAA,MACZ;AAAA,MACA;AAAA,MACA,MAAM,YAAY,UAAU,KAAK;AAAA,MACjC,MAAM,YAAY,UAAU,KAAK;AAAA,MACjC,SAAS,IAAI;AAAA,IACf;AAAA,EACF;AAEA,SAAO;AAAA,IACL,OAAO,QAAQ,SAAS,IAAI,QAAQ,CAAC,EAAE,KAAK;AAAA,IAC5C;AAAA,IACA,OAAO,QAAQ;AAAA,IACf;AAAA,EACF;AACF;",
  "names": ["parseYaml", "Comp", "customNode", "parseYaml", "content"]
}

|
package/package.json
ADDED
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "solid-mds",
|
|
3
|
+
"version": "0.3.2",
|
|
4
|
+
"description": "A SolidJS library for parsing and rendering MDS (Markdown Steps) format",
|
|
5
|
+
"license": "MIT",
|
|
6
|
+
"author": "Matthias Reis",
|
|
7
|
+
"keywords": [
|
|
8
|
+
"solid",
|
|
9
|
+
"solidjs",
|
|
10
|
+
"markdown",
|
|
11
|
+
"parser",
|
|
12
|
+
"slides",
|
|
13
|
+
"presentation",
|
|
14
|
+
"remark",
|
|
15
|
+
"rehype"
|
|
16
|
+
],
|
|
17
|
+
"type": "module",
|
|
18
|
+
"main": "./dist/index.jsx",
|
|
19
|
+
"module": "./dist/index.jsx",
|
|
20
|
+
"types": "./dist/index.d.ts",
|
|
21
|
+
"exports": {
|
|
22
|
+
".": {
|
|
23
|
+
"types": "./dist/index.d.ts",
|
|
24
|
+
"default": "./dist/index.jsx"
|
|
25
|
+
}
|
|
26
|
+
},
|
|
27
|
+
"files": [
|
|
28
|
+
"dist",
|
|
29
|
+
"README.md",
|
|
30
|
+
"LICENSE"
|
|
31
|
+
],
|
|
32
|
+
"scripts": {
|
|
33
|
+
"build": "tsup",
|
|
34
|
+
"dev": "tsup --watch",
|
|
35
|
+
"test": "vitest run",
|
|
36
|
+
"test:watch": "vitest",
|
|
37
|
+
"prepublishOnly": "pnpm run build",
|
|
38
|
+
"release": "bash ../../scripts/release.sh"
|
|
39
|
+
},
|
|
40
|
+
"dependencies": {
|
|
41
|
+
"yaml": "^2.7.0",
|
|
42
|
+
"rehype-katex": "^7.0.0",
|
|
43
|
+
"rehype-stringify": "^10.0.1",
|
|
44
|
+
"remark-gfm": "^4.0.0",
|
|
45
|
+
"remark-math": "^6.0.0",
|
|
46
|
+
"remark-parse": "^11.0.0",
|
|
47
|
+
"remark-rehype": "^11.1.1",
|
|
48
|
+
"unified": "^11.0.5",
|
|
49
|
+
"unist-util-visit": "^5.0.0"
|
|
50
|
+
},
|
|
51
|
+
"devDependencies": {
|
|
52
|
+
"@solidjs/testing-library": "^0.8.0",
|
|
53
|
+
"@types/hast": "^3.0.4",
|
|
54
|
+
"@types/mdast": "^4.0.4",
|
|
55
|
+
"jsdom": "^26.0.0",
|
|
56
|
+
"tsup": "^8.3.5",
|
|
57
|
+
"tsup-preset-solid": "^2.2.0",
|
|
58
|
+
"typescript": "^5.7.3",
|
|
59
|
+
"vite-plugin-solid": "^2.10.0",
|
|
60
|
+
"vitest": "^3.0.0"
|
|
61
|
+
},
|
|
62
|
+
"peerDependencies": {
|
|
63
|
+
"solid-js": "^1.9.0"
|
|
64
|
+
},
|
|
65
|
+
"typesVersions": {}
|
|
66
|
+
}
|