solid-mds 0.3.2 → 0.4.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/README.md +167 -245
- package/dist/index.d.ts +50 -2
- package/dist/index.jsx +29 -197
- package/package.json +7 -19
package/README.md
CHANGED
|
@@ -1,23 +1,26 @@
|
|
|
1
1
|
# solid-mds
|
|
2
2
|
|
|
3
|
-
A SolidJS library for
|
|
4
|
-
|
|
5
|
-
|
|
3
|
+
A SolidJS library for transforming MDS (Markdown Steps) HAST to Solid components.
|
|
4
|
+
|
|
5
|
+
This package works with [hast-mds](https://www.npmjs.com/package/hast-mds) to render step-based Markdown content in SolidJS applications.
|
|
6
6
|
|
|
7
7
|
## Installation
|
|
8
8
|
|
|
9
9
|
```bash
|
|
10
|
-
npm install solid-mds
|
|
10
|
+
npm install solid-mds hast-mds
|
|
11
11
|
# or
|
|
12
|
-
pnpm add solid-mds
|
|
12
|
+
pnpm add solid-mds hast-mds
|
|
13
13
|
```
|
|
14
14
|
|
|
15
|
-
**Peer
|
|
15
|
+
**Peer dependencies:**
|
|
16
|
+
- `solid-js` ^1.9.0
|
|
17
|
+
- `hast-mds` ^0.1.0
|
|
16
18
|
|
|
17
19
|
## Quick Start
|
|
18
20
|
|
|
19
21
|
```tsx
|
|
20
|
-
import { parse } from "
|
|
22
|
+
import { parse } from "hast-mds";
|
|
23
|
+
import { transform } from "solid-mds";
|
|
21
24
|
|
|
22
25
|
const markdown = `
|
|
23
26
|
# Hello World
|
|
@@ -26,238 +29,106 @@ This is **bold** and *italic* text.
|
|
|
26
29
|
`;
|
|
27
30
|
|
|
28
31
|
function App() {
|
|
29
|
-
const
|
|
30
|
-
|
|
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
|
-
```
|
|
32
|
+
const parsed = parse(markdown);
|
|
33
|
+
const result = transform(parsed);
|
|
94
34
|
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
```markdown
|
|
98
|
-
- [x] Completed task
|
|
99
|
-
- [ ] Incomplete task
|
|
35
|
+
return <div>{result.steps.default.Body()}</div>;
|
|
36
|
+
}
|
|
100
37
|
```
|
|
101
38
|
|
|
102
|
-
|
|
39
|
+
## Architecture
|
|
103
40
|
|
|
104
|
-
|
|
105
|
-
Visit https://example.com automatically linked.
|
|
106
|
-
```
|
|
41
|
+
solid-mds is designed as a thin transformation layer:
|
|
107
42
|
|
|
108
|
-
|
|
43
|
+
1. **hast-mds** parses your MDS content into HAST (Hypertext Abstract Syntax Tree)
|
|
44
|
+
2. **solid-mds** transforms HAST into renderable Solid components
|
|
109
45
|
|
|
110
|
-
|
|
46
|
+
This separation allows:
|
|
47
|
+
- Framework-agnostic parsing (use hast-mds with React, Vue, etc.)
|
|
48
|
+
- Optimized Solid rendering with SSR support
|
|
111
49
|
|
|
112
|
-
|
|
113
|
-
The equation $E = mc^2$ is famous.
|
|
114
|
-
```
|
|
50
|
+
## MDS Format
|
|
115
51
|
|
|
116
|
-
|
|
52
|
+
MDS (Markdown Steps) extends Markdown with step-based structure.
|
|
117
53
|
|
|
118
|
-
|
|
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:
|
|
54
|
+
### Step Separators
|
|
127
55
|
|
|
128
56
|
```markdown
|
|
129
57
|
+++intro
|
|
130
58
|
# Welcome
|
|
131
59
|
|
|
132
|
-
This is the introduction
|
|
133
|
-
|
|
134
|
-
+++
|
|
135
|
-
|
|
136
|
-
main
|
|
60
|
+
This is the introduction.
|
|
137
61
|
|
|
62
|
+
+++main
|
|
138
63
|
# Main Content
|
|
139
64
|
|
|
140
|
-
This is the main step.
|
|
141
|
-
|
|
142
65
|
+++conclusion
|
|
143
|
-
|
|
144
66
|
# Summary
|
|
145
|
-
|
|
146
|
-
This is the conclusion.
|
|
147
67
|
```
|
|
148
68
|
|
|
149
|
-
###
|
|
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
|
-
```
|
|
69
|
+
### Metadata Blocks
|
|
165
70
|
|
|
166
|
-
|
|
71
|
+
#### Global Metadata
|
|
167
72
|
|
|
168
|
-
|
|
73
|
+
```markdown
|
|
74
|
+
\`\`\`yaml @@
|
|
75
|
+
title: My Presentation
|
|
76
|
+
author: Jane Doe
|
|
77
|
+
\`\`\`
|
|
169
78
|
|
|
170
|
-
|
|
171
|
-
|
|
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
|
-
}
|
|
79
|
+
+++intro
|
|
80
|
+
# Content
|
|
179
81
|
```
|
|
180
82
|
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
### Local Metadata (per step)
|
|
83
|
+
Access via `result.global.title`.
|
|
184
84
|
|
|
185
|
-
|
|
85
|
+
#### Global Markdown Blocks
|
|
186
86
|
|
|
187
|
-
|
|
188
|
-
|
|
87
|
+
```markdown
|
|
88
|
+
\`\`\`md @@/footer
|
|
89
|
+
Made with **love**
|
|
90
|
+
\`\`\`
|
|
189
91
|
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
speaker: John Doe
|
|
92
|
+
+++intro
|
|
93
|
+
# Content
|
|
193
94
|
```
|
|
194
95
|
|
|
195
|
-
|
|
196
|
-
````
|
|
197
|
-
|
|
198
|
-
Access via `result.steps.slide1.local.title`.
|
|
199
|
-
|
|
200
|
-
### Local Markdown Blocks
|
|
96
|
+
Access via `result.global.footer` (returns Solid component).
|
|
201
97
|
|
|
202
|
-
|
|
98
|
+
#### Local Metadata (per step)
|
|
203
99
|
|
|
204
|
-
|
|
100
|
+
```markdown
|
|
205
101
|
+++slide1
|
|
102
|
+
\`\`\`yaml @
|
|
103
|
+
layout: centered
|
|
104
|
+
transition: fade
|
|
105
|
+
\`\`\`
|
|
206
106
|
|
|
207
|
-
|
|
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
|
|
107
|
+
# Slide Content
|
|
224
108
|
```
|
|
225
109
|
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
# First Step
|
|
229
|
-
````
|
|
230
|
-
|
|
231
|
-
Access via `result.global.author`.
|
|
110
|
+
Access via `result.steps.slide1.local.layout`.
|
|
232
111
|
|
|
233
|
-
|
|
112
|
+
#### Local Markdown Blocks
|
|
234
113
|
|
|
235
|
-
|
|
114
|
+
```markdown
|
|
115
|
+
+++slide1
|
|
116
|
+
\`\`\`md @/notes
|
|
117
|
+
Speaker **notes** here.
|
|
118
|
+
\`\`\`
|
|
236
119
|
|
|
237
|
-
|
|
238
|
-
```@@/footer
|
|
239
|
-
Made with ❤️ by **Our Team**
|
|
120
|
+
# Main Content
|
|
240
121
|
```
|
|
241
122
|
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
# Content
|
|
245
|
-
````
|
|
246
|
-
|
|
247
|
-
Access via `result.global.footer` (returns `JSX.Element`).
|
|
123
|
+
Access via `result.steps.slide1.local.notes` (returns Solid component).
|
|
248
124
|
|
|
249
125
|
## Custom Components
|
|
250
126
|
|
|
251
|
-
Override default HTML elements or create custom block components:
|
|
252
|
-
|
|
253
127
|
### Override Standard Elements
|
|
254
128
|
|
|
255
129
|
```tsx
|
|
256
|
-
import {
|
|
257
|
-
|
|
258
|
-
ComponentMap,
|
|
259
|
-
StandardComponentProps,
|
|
260
|
-
} from "solid-mds";
|
|
130
|
+
import { parse } from "hast-mds";
|
|
131
|
+
import { transform, ComponentMap, StandardComponentProps } from "solid-mds";
|
|
261
132
|
|
|
262
133
|
const CustomHeading = (props: StandardComponentProps) => (
|
|
263
134
|
<h1 class="text-4xl font-bold text-blue-600">{props.children}</h1>
|
|
@@ -267,17 +138,18 @@ const components: ComponentMap = {
|
|
|
267
138
|
h1: CustomHeading,
|
|
268
139
|
};
|
|
269
140
|
|
|
270
|
-
const
|
|
141
|
+
const parsed = parse(markdown);
|
|
142
|
+
const result = transform(parsed, components);
|
|
271
143
|
```
|
|
272
144
|
|
|
273
|
-
### Custom
|
|
145
|
+
### Custom Block Components
|
|
274
146
|
|
|
275
|
-
Create custom components
|
|
147
|
+
Create custom components using `yaml componentName` or `md componentName/path` syntax:
|
|
276
148
|
|
|
277
149
|
```tsx
|
|
278
|
-
import { parse
|
|
150
|
+
import { parse } from "hast-mds";
|
|
151
|
+
import { transform, ComponentMap, CustomBlockProps } from "solid-mds";
|
|
279
152
|
|
|
280
|
-
// Define a custom component
|
|
281
153
|
const Alert = (props: CustomBlockProps) => (
|
|
282
154
|
<div class={`alert alert-${props.payload[0] || "info"}`}>
|
|
283
155
|
{props.children}
|
|
@@ -289,25 +161,17 @@ const components: ComponentMap = {
|
|
|
289
161
|
};
|
|
290
162
|
|
|
291
163
|
const markdown = `
|
|
292
|
-
\`\`\`alert/warning
|
|
164
|
+
\`\`\`md alert/warning
|
|
293
165
|
This is a **warning** message!
|
|
294
166
|
\`\`\`
|
|
295
167
|
`;
|
|
296
168
|
|
|
297
|
-
|
|
169
|
+
// Register component names when parsing
|
|
170
|
+
const parsed = parse(markdown, new Set(["alert"]));
|
|
171
|
+
const result = transform(parsed, components);
|
|
298
172
|
```
|
|
299
173
|
|
|
300
|
-
|
|
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:
|
|
174
|
+
### Custom Blocks with YAML Data
|
|
311
175
|
|
|
312
176
|
```tsx
|
|
313
177
|
const Card = (props: CustomBlockProps) => (
|
|
@@ -317,30 +181,25 @@ const Card = (props: CustomBlockProps) => (
|
|
|
317
181
|
</div>
|
|
318
182
|
);
|
|
319
183
|
|
|
320
|
-
const components: ComponentMap = { Card };
|
|
184
|
+
const components: ComponentMap = { card: Card };
|
|
321
185
|
|
|
322
186
|
const markdown = `
|
|
323
|
-
\`\`\`
|
|
324
|
-
title: My Card
|
|
325
|
-
description:
|
|
187
|
+
\`\`\`yaml card
|
|
188
|
+
title: My Card
|
|
189
|
+
description: Card content here
|
|
326
190
|
\`\`\`
|
|
327
191
|
`;
|
|
328
192
|
|
|
329
|
-
const
|
|
193
|
+
const parsed = parse(markdown, new Set(["card"]));
|
|
194
|
+
const result = transform(parsed, components);
|
|
330
195
|
```
|
|
331
196
|
|
|
332
|
-
With ` ```Card| `, the content is parsed as YAML and available via `props.data`.
|
|
333
|
-
|
|
334
197
|
## Complete Example
|
|
335
198
|
|
|
336
199
|
```tsx
|
|
337
|
-
import { createSignal, Show
|
|
338
|
-
import {
|
|
339
|
-
|
|
340
|
-
ComponentMap,
|
|
341
|
-
CustomBlockProps,
|
|
342
|
-
StandardComponentProps,
|
|
343
|
-
} from "solid-mds";
|
|
200
|
+
import { createSignal, Show } from "solid-js";
|
|
201
|
+
import { parse } from "hast-mds";
|
|
202
|
+
import { transform, ComponentMap, CustomBlockProps, StandardComponentProps } from "solid-mds";
|
|
344
203
|
|
|
345
204
|
// Custom components
|
|
346
205
|
const Slide = (props: CustomBlockProps) => (
|
|
@@ -356,19 +215,18 @@ const Code = (props: StandardComponentProps) => (
|
|
|
356
215
|
);
|
|
357
216
|
|
|
358
217
|
const components: ComponentMap = {
|
|
359
|
-
Slide,
|
|
218
|
+
slide: Slide,
|
|
360
219
|
code: Code,
|
|
361
220
|
};
|
|
362
221
|
|
|
363
|
-
// Markdown content
|
|
364
222
|
const content = `
|
|
365
|
-
|
|
223
|
+
\`\`\`yaml @@
|
|
366
224
|
title: My Presentation
|
|
367
225
|
author: Developer
|
|
368
226
|
\`\`\`
|
|
369
227
|
|
|
370
228
|
+++intro
|
|
371
|
-
|
|
229
|
+
\`\`\`yaml @
|
|
372
230
|
transition: fade
|
|
373
231
|
\`\`\`
|
|
374
232
|
|
|
@@ -377,24 +235,19 @@ transition: fade
|
|
|
377
235
|
Introduction slide content.
|
|
378
236
|
|
|
379
237
|
+++demo
|
|
380
|
-
\`\`\`
|
|
238
|
+
\`\`\`md slide/centered
|
|
381
239
|
# Demo Time
|
|
382
240
|
|
|
383
|
-
Check out this code
|
|
384
|
-
|
|
385
|
-
\`\`\`js
|
|
386
|
-
console.log("Hello!");
|
|
387
|
-
\`\`\`
|
|
241
|
+
Check out this code!
|
|
388
242
|
\`\`\`
|
|
389
243
|
|
|
390
244
|
+++end
|
|
391
245
|
# Thank You!
|
|
392
|
-
|
|
393
|
-
Questions?
|
|
394
246
|
`;
|
|
395
247
|
|
|
396
248
|
function Presentation() {
|
|
397
|
-
const
|
|
249
|
+
const parsed = parse(content, new Set(["slide"]));
|
|
250
|
+
const result = transform(parsed, components);
|
|
398
251
|
const [currentId, setCurrentId] = createSignal(result.first);
|
|
399
252
|
|
|
400
253
|
const currentStep = () => result.steps[currentId()!];
|
|
@@ -417,7 +270,9 @@ function Presentation() {
|
|
|
417
270
|
</header>
|
|
418
271
|
|
|
419
272
|
<main>
|
|
420
|
-
<Show when={currentStep()}>
|
|
273
|
+
<Show when={currentStep()}>
|
|
274
|
+
{currentStep().Body()}
|
|
275
|
+
</Show>
|
|
421
276
|
</main>
|
|
422
277
|
|
|
423
278
|
<footer>
|
|
@@ -438,24 +293,32 @@ function Presentation() {
|
|
|
438
293
|
|
|
439
294
|
## API Reference
|
|
440
295
|
|
|
441
|
-
### `
|
|
296
|
+
### `transform<TGlobal, TLocal>(parsed, components?): ParseResult`
|
|
442
297
|
|
|
443
|
-
|
|
444
|
-
for your global and local scope to receive a properly typed result.
|
|
298
|
+
Transforms parsed MDS content (HAST) into Solid components.
|
|
445
299
|
|
|
446
300
|
**Parameters:**
|
|
447
301
|
|
|
448
|
-
- `
|
|
449
|
-
- `components` — Optional map of custom components
|
|
302
|
+
- `parsed` — Result from `hast-mds` parse() function
|
|
303
|
+
- `components` — Optional map of custom Solid components
|
|
450
304
|
|
|
451
305
|
**Returns:**
|
|
452
306
|
|
|
453
307
|
```ts
|
|
454
308
|
interface ParseResult<TGlobal, TLocal> {
|
|
455
|
-
first: string | null;
|
|
309
|
+
first: string | null; // First step ID
|
|
456
310
|
steps: Record<string, Step<TLocal>>; // All steps by ID
|
|
457
|
-
count: number;
|
|
458
|
-
global: TGlobal | null;
|
|
311
|
+
count: number; // Total number of steps
|
|
312
|
+
global: TGlobal | null; // Global metadata
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
interface Step<TLocal> {
|
|
316
|
+
id: string;
|
|
317
|
+
local: TLocal;
|
|
318
|
+
Body: Component; // Call as Body() to render
|
|
319
|
+
prev: string | null;
|
|
320
|
+
next: string | null;
|
|
321
|
+
current: number;
|
|
459
322
|
}
|
|
460
323
|
```
|
|
461
324
|
|
|
@@ -468,9 +331,68 @@ import type {
|
|
|
468
331
|
StandardComponentProps,
|
|
469
332
|
ParseResult,
|
|
470
333
|
Step,
|
|
334
|
+
// Re-exported from hast-mds
|
|
335
|
+
HastParseResult,
|
|
336
|
+
HastStep,
|
|
337
|
+
HastBody,
|
|
338
|
+
CustomComponents,
|
|
471
339
|
} from "solid-mds";
|
|
472
340
|
```
|
|
473
341
|
|
|
342
|
+
## Migration from v0.3.x
|
|
343
|
+
|
|
344
|
+
### Breaking Changes
|
|
345
|
+
|
|
346
|
+
1. **New API**: `parse()` is replaced by `transform()`
|
|
347
|
+
2. **Requires hast-mds**: Install and use `hast-mds` for parsing
|
|
348
|
+
3. **Syntax changes**: Old block syntax is no longer supported
|
|
349
|
+
|
|
350
|
+
### Before (v0.3.x)
|
|
351
|
+
|
|
352
|
+
```tsx
|
|
353
|
+
import { parse } from "solid-mds";
|
|
354
|
+
|
|
355
|
+
const result = parse(markdown, components);
|
|
356
|
+
```
|
|
357
|
+
|
|
358
|
+
### After (v0.4.0)
|
|
359
|
+
|
|
360
|
+
```tsx
|
|
361
|
+
import { parse } from "hast-mds";
|
|
362
|
+
import { transform } from "solid-mds";
|
|
363
|
+
|
|
364
|
+
const parsed = parse(markdown, new Set(Object.keys(components)));
|
|
365
|
+
const result = transform(parsed, components);
|
|
366
|
+
```
|
|
367
|
+
|
|
368
|
+
### Syntax Changes
|
|
369
|
+
|
|
370
|
+
| Old Syntax | New Syntax |
|
|
371
|
+
|------------|------------|
|
|
372
|
+
| ` ```@@\| ` | ` ```yaml @@ ` |
|
|
373
|
+
| ` ```@\| ` | ` ```yaml @ ` |
|
|
374
|
+
| ` ```@@/name ` | ` ```md @@/name ` |
|
|
375
|
+
| ` ```@/name ` | ` ```md @/name ` |
|
|
376
|
+
| ` ```componentName\| ` | ` ```yaml componentName ` |
|
|
377
|
+
| ` ```componentName/path ` | ` ```md componentName/path ` |
|
|
378
|
+
|
|
379
|
+
## Markdown Features
|
|
380
|
+
|
|
381
|
+
solid-mds (via hast-mds) supports:
|
|
382
|
+
|
|
383
|
+
- **Standard Markdown**: Headings, bold, italic, links, images, lists, blockquotes, code
|
|
384
|
+
- **GFM**: Tables, task lists, strikethrough, autolinks
|
|
385
|
+
- **Math**: LaTeX via KaTeX (`$inline$` and `$$block$$`)
|
|
386
|
+
|
|
387
|
+
For math rendering, include KaTeX CSS:
|
|
388
|
+
|
|
389
|
+
```html
|
|
390
|
+
<link
|
|
391
|
+
rel="stylesheet"
|
|
392
|
+
href="https://cdn.jsdelivr.net/npm/katex@0.16.0/dist/katex.min.css"
|
|
393
|
+
/>
|
|
394
|
+
```
|
|
395
|
+
|
|
474
396
|
## License
|
|
475
397
|
|
|
476
398
|
MIT
|
package/dist/index.d.ts
CHANGED
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import { HastParseResult } from 'hast-mds';
|
|
2
|
+
export { CustomComponents, HastBody, HastParseResult, HastStep } from 'hast-mds';
|
|
1
3
|
import { Component, JSX } from 'solid-js';
|
|
2
4
|
|
|
3
5
|
/**
|
|
@@ -24,27 +26,73 @@ type ComponentMap = {
|
|
|
24
26
|
[tagName: string]: Component<CustomBlockProps> | Component<StandardComponentProps>;
|
|
25
27
|
};
|
|
26
28
|
|
|
29
|
+
/**
|
|
30
|
+
* Base interface for step-local metadata.
|
|
31
|
+
* Users can extend this with their own types.
|
|
32
|
+
*/
|
|
27
33
|
interface StepLocalBase {
|
|
28
34
|
[key: string]: unknown;
|
|
29
35
|
}
|
|
36
|
+
/**
|
|
37
|
+
* Base interface for global metadata.
|
|
38
|
+
* Users can extend this with their own types.
|
|
39
|
+
*/
|
|
30
40
|
interface GlobalMetaBase {
|
|
31
41
|
[key: string]: unknown;
|
|
32
42
|
}
|
|
43
|
+
/**
|
|
44
|
+
* A single step in the MDS document with Solid components.
|
|
45
|
+
*/
|
|
33
46
|
interface Step<TLocal = StepLocalBase> {
|
|
47
|
+
/** Unique identifier for the step */
|
|
34
48
|
id: string;
|
|
49
|
+
/** Step-specific metadata (YAML values and markdown blocks as Components) */
|
|
35
50
|
local: TLocal;
|
|
36
51
|
/** Component function that renders the step body. Call it to get JSX: {step.Body()} */
|
|
37
52
|
Body: Component;
|
|
53
|
+
/** ID of the previous step, or null if first */
|
|
38
54
|
prev: string | null;
|
|
55
|
+
/** ID of the next step, or null if last */
|
|
39
56
|
next: string | null;
|
|
57
|
+
/** 1-based position of this step */
|
|
40
58
|
current: number;
|
|
41
59
|
}
|
|
60
|
+
/**
|
|
61
|
+
* Result of transforming an MDS document to Solid components.
|
|
62
|
+
*/
|
|
42
63
|
interface ParseResult<TGlobal = GlobalMetaBase, TLocal = StepLocalBase> {
|
|
64
|
+
/** ID of the first step, or null if no steps */
|
|
43
65
|
first: string | null;
|
|
66
|
+
/** Map of step IDs to step objects */
|
|
44
67
|
steps: Record<string, Step<TLocal>>;
|
|
68
|
+
/** Total number of steps */
|
|
45
69
|
count: number;
|
|
70
|
+
/** Global metadata (YAML values and markdown blocks as Components) */
|
|
46
71
|
global: TGlobal | null;
|
|
47
72
|
}
|
|
48
|
-
declare function parse<TGlobal = GlobalMetaBase, TLocal = StepLocalBase>(input: string, components?: ComponentMap): ParseResult<TGlobal, TLocal>;
|
|
49
73
|
|
|
50
|
-
|
|
74
|
+
/**
|
|
75
|
+
* Transform a parsed MDS document (HAST) into Solid components.
|
|
76
|
+
*
|
|
77
|
+
* This function takes the output from `hast-mds` parse() and converts all
|
|
78
|
+
* HAST trees into renderable Solid components.
|
|
79
|
+
*
|
|
80
|
+
* @example
|
|
81
|
+
* ```typescript
|
|
82
|
+
* import { parse } from "hast-mds";
|
|
83
|
+
* import { transform } from "solid-mds";
|
|
84
|
+
*
|
|
85
|
+
* const parsed = parse(mdsString, new Set(["quiz", "card"]));
|
|
86
|
+
* const result = transform(parsed, { quiz: QuizComponent, card: CardComponent });
|
|
87
|
+
*
|
|
88
|
+
* // Use in Solid component:
|
|
89
|
+
* <div>{result.steps.intro.Body()}</div>
|
|
90
|
+
* ```
|
|
91
|
+
*
|
|
92
|
+
* @param parsed - The result from hast-mds parse() function
|
|
93
|
+
* @param components - Optional map of custom Solid components
|
|
94
|
+
* @returns ParseResult with Solid components
|
|
95
|
+
*/
|
|
96
|
+
declare function transform<TGlobal = GlobalMetaBase, TLocal = StepLocalBase>(parsed: HastParseResult<TGlobal, TLocal>, components?: ComponentMap): ParseResult<TGlobal, TLocal>;
|
|
97
|
+
|
|
98
|
+
export { type ComponentMap, type CustomBlockProps, type GlobalMetaBase, type ParseResult, type StandardComponentProps, type Step, type StepLocalBase, transform };
|
package/dist/index.jsx
CHANGED
|
@@ -1,12 +1,3 @@
|
|
|
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
1
|
// src/hast-to-solid.tsx
|
|
11
2
|
function convertProperties(properties) {
|
|
12
3
|
if (!properties) return {};
|
|
@@ -196,207 +187,48 @@ function hastToSolidComponent(hastTree, components = {}) {
|
|
|
196
187
|
};
|
|
197
188
|
}
|
|
198
189
|
|
|
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
190
|
// src/index.tsx
|
|
251
|
-
function
|
|
252
|
-
|
|
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 };
|
|
191
|
+
function isHastBody(value) {
|
|
192
|
+
return typeof value === "object" && value !== null && "type" in value && value.type === "hast" && "node" in value;
|
|
279
193
|
}
|
|
280
|
-
function
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
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;
|
|
194
|
+
function transformMetadata(metadata, components) {
|
|
195
|
+
if (metadata === null) return null;
|
|
196
|
+
const result = {};
|
|
197
|
+
for (const [key, value] of Object.entries(
|
|
198
|
+
metadata
|
|
199
|
+
)) {
|
|
200
|
+
if (isHastBody(value)) {
|
|
201
|
+
result[key] = hastToSolidComponent(value.node, components);
|
|
295
202
|
} else {
|
|
296
|
-
|
|
203
|
+
result[key] = value;
|
|
297
204
|
}
|
|
298
205
|
}
|
|
299
|
-
|
|
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
|
-
}
|
|
206
|
+
return result;
|
|
323
207
|
}
|
|
324
|
-
function
|
|
325
|
-
const stepPattern = /^\+\+\+(.+)$/gm;
|
|
208
|
+
function transform(parsed, components = {}) {
|
|
326
209
|
const steps = {};
|
|
327
|
-
const
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
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,
|
|
210
|
+
for (const id of Object.keys(parsed.steps)) {
|
|
211
|
+
const hastStep = parsed.steps[id];
|
|
212
|
+
const Body = hastToSolidComponent(hastStep.body.node, components);
|
|
213
|
+
const transformedLocal = transformMetadata(hastStep.local, components);
|
|
214
|
+
steps[id] = {
|
|
215
|
+
id: hastStep.id,
|
|
216
|
+
local: transformedLocal,
|
|
386
217
|
Body,
|
|
387
|
-
prev:
|
|
388
|
-
next:
|
|
389
|
-
current:
|
|
218
|
+
prev: hastStep.prev,
|
|
219
|
+
next: hastStep.next,
|
|
220
|
+
current: hastStep.current
|
|
390
221
|
};
|
|
391
222
|
}
|
|
223
|
+
const transformedGlobal = transformMetadata(parsed.global, components);
|
|
392
224
|
return {
|
|
393
|
-
first:
|
|
225
|
+
first: parsed.first,
|
|
394
226
|
steps,
|
|
395
|
-
count:
|
|
396
|
-
global
|
|
227
|
+
count: parsed.count,
|
|
228
|
+
global: transformedGlobal
|
|
397
229
|
};
|
|
398
230
|
}
|
|
399
231
|
export {
|
|
400
|
-
|
|
232
|
+
transform
|
|
401
233
|
};
|
|
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": "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",
  "names": ["parseYaml", "Comp", "customNode", "parseYaml", "content"]
}

|
|
234
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/hast-to-solid.tsx", "../src/index.tsx"],
  "sourcesContent": ["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 { HastParseResult, HastBody, HastStep } from \"hast-mds\";\nimport { hastToSolidComponent, ComponentMap } from \"./hast-to-solid\";\nimport type { Step, ParseResult, StepLocalBase, GlobalMetaBase } from \"./types\";\n\n// Re-export types from hast-to-solid\nexport type {\n  ComponentMap,\n  CustomBlockProps,\n  StandardComponentProps,\n} from \"./hast-to-solid\";\n\n// Re-export types from types.ts\nexport type { Step, ParseResult, StepLocalBase, GlobalMetaBase } from \"./types\";\n\n// Re-export hast-mds types for convenience\nexport type {\n  HastParseResult,\n  HastStep,\n  HastBody,\n  CustomComponents,\n} from \"hast-mds\";\n\n/**\n * Type guard to check if a value is a HastBody\n */\nfunction isHastBody(value: unknown): value is HastBody {\n  return (\n    typeof value === \"object\" &&\n    value !== null &&\n    \"type\" in value &&\n    (value as HastBody).type === \"hast\" &&\n    \"node\" in value\n  );\n}\n\n/**\n * Recursively transform metadata objects, converting HastBody to Solid components\n */\nfunction transformMetadata<T>(\n  metadata: T | null,\n  components: ComponentMap,\n): T | null {\n  if (metadata === null) return null;\n\n  const result: Record<string, unknown> = {};\n\n  for (const [key, value] of Object.entries(\n    metadata as Record<string, unknown>,\n  )) {\n    if (isHastBody(value)) {\n      // Convert HastBody to Solid component\n      result[key] = hastToSolidComponent(value.node, components);\n    } else {\n      result[key] = value;\n    }\n  }\n\n  return result as T;\n}\n\n/**\n * Transform a parsed MDS document (HAST) into Solid components.\n *\n * This function takes the output from `hast-mds` parse() and converts all\n * HAST trees into renderable Solid components.\n *\n * @example\n * ```typescript\n * import { parse } from \"hast-mds\";\n * import { transform } from \"solid-mds\";\n *\n * const parsed = parse(mdsString, new Set([\"quiz\", \"card\"]));\n * const result = transform(parsed, { quiz: QuizComponent, card: CardComponent });\n *\n * // Use in Solid component:\n * <div>{result.steps.intro.Body()}</div>\n * ```\n *\n * @param parsed - The result from hast-mds parse() function\n * @param components - Optional map of custom Solid components\n * @returns ParseResult with Solid components\n */\nexport function transform<TGlobal = GlobalMetaBase, TLocal = StepLocalBase>(\n  parsed: HastParseResult<TGlobal, TLocal>,\n  components: ComponentMap = {},\n): ParseResult<TGlobal, TLocal> {\n  const steps: Record<string, Step<TLocal>> = {};\n\n  for (const id of Object.keys(parsed.steps)) {\n    const hastStep: HastStep<TLocal> = parsed.steps[id];\n    const Body = hastToSolidComponent(hastStep.body.node, components);\n    const transformedLocal = transformMetadata(hastStep.local, components);\n\n    steps[id] = {\n      id: hastStep.id,\n      local: transformedLocal as TLocal,\n      Body,\n      prev: hastStep.prev,\n      next: hastStep.next,\n      current: hastStep.current,\n    };\n  }\n\n  const transformedGlobal = transformMetadata(parsed.global, components);\n\n  return {\n    first: parsed.first,\n    steps,\n    count: parsed.count,\n    global: transformedGlobal as TGlobal | null,\n  };\n}\n"],
  "mappings": "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",
  "names": ["Comp"]
}

|
package/package.json
CHANGED
|
@@ -1,18 +1,16 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "solid-mds",
|
|
3
|
-
"version": "0.
|
|
4
|
-
"description": "A SolidJS library for
|
|
3
|
+
"version": "0.4.0",
|
|
4
|
+
"description": "A SolidJS library for transforming MDS (Markdown Steps) HAST to Solid components",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"author": "Matthias Reis",
|
|
7
7
|
"keywords": [
|
|
8
8
|
"solid",
|
|
9
9
|
"solidjs",
|
|
10
10
|
"markdown",
|
|
11
|
-
"
|
|
11
|
+
"hast",
|
|
12
12
|
"slides",
|
|
13
|
-
"presentation"
|
|
14
|
-
"remark",
|
|
15
|
-
"rehype"
|
|
13
|
+
"presentation"
|
|
16
14
|
],
|
|
17
15
|
"type": "module",
|
|
18
16
|
"main": "./dist/index.jsx",
|
|
@@ -37,21 +35,10 @@
|
|
|
37
35
|
"prepublishOnly": "pnpm run build",
|
|
38
36
|
"release": "bash ../../scripts/release.sh"
|
|
39
37
|
},
|
|
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
38
|
"devDependencies": {
|
|
52
39
|
"@solidjs/testing-library": "^0.8.0",
|
|
53
40
|
"@types/hast": "^3.0.4",
|
|
54
|
-
"
|
|
41
|
+
"hast-mds": "workspace:*",
|
|
55
42
|
"jsdom": "^26.0.0",
|
|
56
43
|
"tsup": "^8.3.5",
|
|
57
44
|
"tsup-preset-solid": "^2.2.0",
|
|
@@ -60,7 +47,8 @@
|
|
|
60
47
|
"vitest": "^3.0.0"
|
|
61
48
|
},
|
|
62
49
|
"peerDependencies": {
|
|
63
|
-
"solid-js": "^1.9.0"
|
|
50
|
+
"solid-js": "^1.9.0",
|
|
51
|
+
"hast-mds": "^0.1.0"
|
|
64
52
|
},
|
|
65
53
|
"typesVersions": {}
|
|
66
54
|
}
|