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 CHANGED
@@ -1,23 +1,26 @@
1
1
  # solid-mds
2
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.
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 dependency:** This library requires SolidJS ^1.9.0
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 "solid-mds";
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 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
- ![Images](image.png)
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
- #### Task Lists
96
-
97
- ```markdown
98
- - [x] Completed task
99
- - [ ] Incomplete task
35
+ return <div>{result.steps.default.Body()}</div>;
36
+ }
100
37
  ```
101
38
 
102
- #### Autolinks
39
+ ## Architecture
103
40
 
104
- ```markdown
105
- Visit https://example.com automatically linked.
106
- ```
41
+ solid-mds is designed as a thin transformation layer:
107
42
 
108
- ### Math (LaTeX)
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
- #### Inline Math
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
- ```markdown
113
- The equation $E = mc^2$ is famous.
114
- ```
50
+ ## MDS Format
115
51
 
116
- #### Block Math
52
+ MDS (Markdown Steps) extends Markdown with step-based structure.
117
53
 
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:
54
+ ### Step Separators
127
55
 
128
56
  ```markdown
129
57
  +++intro
130
58
  # Welcome
131
59
 
132
- This is the introduction step.
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
- ### 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
- ```
69
+ ### Metadata Blocks
165
70
 
166
- ### Step Object Structure
71
+ #### Global Metadata
167
72
 
168
- Each step contains:
73
+ ```markdown
74
+ \`\`\`yaml @@
75
+ title: My Presentation
76
+ author: Jane Doe
77
+ \`\`\`
169
78
 
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
- }
79
+ +++intro
80
+ # Content
179
81
  ```
180
82
 
181
- ## Metadata Blocks
182
-
183
- ### Local Metadata (per step)
83
+ Access via `result.global.title`.
184
84
 
185
- Use ` ```@| ` blocks to define key-value metadata for a step:
85
+ #### Global Markdown Blocks
186
86
 
187
- ````markdown
188
- +++slide1
87
+ ```markdown
88
+ \`\`\`md @@/footer
89
+ Made with **love**
90
+ \`\`\`
189
91
 
190
- ```@|
191
- title: Introduction
192
- speaker: John Doe
92
+ +++intro
93
+ # Content
193
94
  ```
194
95
 
195
- # Welcome to the Presentation
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
- Use ` ```@/name ` to define named markdown content within a step:
98
+ #### Local Metadata (per step)
203
99
 
204
- ````markdown
100
+ ```markdown
205
101
  +++slide1
102
+ \`\`\`yaml @
103
+ layout: centered
104
+ transition: fade
105
+ \`\`\`
206
106
 
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
107
+ # Slide Content
224
108
  ```
225
109
 
226
- +++first
227
-
228
- # First Step
229
- ````
230
-
231
- Access via `result.global.author`.
110
+ Access via `result.steps.slide1.local.layout`.
232
111
 
233
- ### Global Markdown Blocks
112
+ #### Local Markdown Blocks
234
113
 
235
- Use ` ```@@/name ` for global named markdown content:
114
+ ```markdown
115
+ +++slide1
116
+ \`\`\`md @/notes
117
+ Speaker **notes** here.
118
+ \`\`\`
236
119
 
237
- ````markdown
238
- ```@@/footer
239
- Made with ❤️ by **Our Team**
120
+ # Main Content
240
121
  ```
241
122
 
242
- +++first
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
- parse,
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 result = parse(markdown, components);
141
+ const parsed = parse(markdown);
142
+ const result = transform(parsed, components);
271
143
  ```
272
144
 
273
- ### Custom Blocks
145
+ ### Custom Block Components
274
146
 
275
- Create custom components that receive special props from code blocks:
147
+ Create custom components using `yaml componentName` or `md componentName/path` syntax:
276
148
 
277
149
  ```tsx
278
- import { parse, ComponentMap, CustomBlockProps } from "solid-mds";
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
- const result = parse(markdown, components);
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
- 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:
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
- \`\`\`Card|
324
- title: My Card Title
325
- description: This is the card description
187
+ \`\`\`yaml card
188
+ title: My Card
189
+ description: Card content here
326
190
  \`\`\`
327
191
  `;
328
192
 
329
- const result = parse(markdown, components);
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, For } from "solid-js";
338
- import {
339
- parse,
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
- \`\`\`Slide/centered
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 result = parse(content, components);
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()}>{currentStep().body}</Show>
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
- ### `parse<TGlobal, TLocal>(input: string, components?: ComponentMap): ParseResult`
296
+ ### `transform<TGlobal, TLocal>(parsed, components?): ParseResult`
442
297
 
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.
298
+ Transforms parsed MDS content (HAST) into Solid components.
445
299
 
446
300
  **Parameters:**
447
301
 
448
- - `input` — The MDS/Markdown string to parse
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; // First step ID
309
+ first: string | null; // First step ID
456
310
  steps: Record<string, Step<TLocal>>; // All steps by ID
457
- count: number; // Total number of steps
458
- global: TGlobal | null; // Global metadata
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
- export { type ComponentMap, type CustomBlockProps, type GlobalMetaBase, type ParseResult, type StandardComponentProps, type Step, type StepLocalBase, parse };
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 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 };
191
+ function isHastBody(value) {
192
+ return typeof value === "object" && value !== null && "type" in value && value.type === "hast" && "node" in value;
279
193
  }
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;
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
- globalMdBlocks[name] = content;
203
+ result[key] = value;
297
204
  }
298
205
  }
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
- }
206
+ return result;
323
207
  }
324
- function parse(input, components = {}) {
325
- const stepPattern = /^\+\+\+(.+)$/gm;
208
+ function transform(parsed, components = {}) {
326
209
  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,
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: prevMatch ? prevMatch.id : null,
388
- next: nextMatch ? nextMatch.id : null,
389
- current: i + 1
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: matches.length > 0 ? matches[0].id : null,
225
+ first: parsed.first,
394
226
  steps,
395
- count: matches.length,
396
- global
227
+ count: parsed.count,
228
+ global: transformedGlobal
397
229
  };
398
230
  }
399
231
  export {
400
- parse
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.3.2",
4
- "description": "A SolidJS library for parsing and rendering MDS (Markdown Steps) format",
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
- "parser",
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
- "@types/mdast": "^4.0.4",
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
  }