@fuzdev/fuz_ui 0.181.1 → 0.182.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/dist/DeclarationDetail.svelte +2 -2
- package/dist/Mdz.svelte +1 -1
- package/dist/MdzPrecompiled.svelte +30 -0
- package/dist/MdzPrecompiled.svelte.d.ts +11 -0
- package/dist/MdzPrecompiled.svelte.d.ts.map +1 -0
- package/dist/declaration.svelte.d.ts +1 -1
- package/dist/library.svelte.d.ts +1 -1
- package/dist/mdz_to_svelte.d.ts +38 -0
- package/dist/mdz_to_svelte.d.ts.map +1 -0
- package/dist/mdz_to_svelte.js +89 -0
- package/dist/svelte_preprocess_mdz.d.ts +65 -0
- package/dist/svelte_preprocess_mdz.d.ts.map +1 -0
- package/dist/svelte_preprocess_mdz.js +513 -0
- package/dist/tsdoc_mdz.d.ts +1 -1
- package/dist/tsdoc_mdz.js +1 -1
- package/package.json +25 -8
- package/src/lib/mdz_to_svelte.ts +127 -0
- package/src/lib/svelte_preprocess_mdz.ts +726 -0
- package/src/lib/tsdoc_mdz.ts +1 -1
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Converts parsed `MdzNode` arrays to Svelte markup strings.
|
|
3
|
+
*
|
|
4
|
+
* Used by the `svelte_preprocess_mdz` preprocessor to expand static `<Mdz content="...">` usages
|
|
5
|
+
* into pre-rendered Svelte markup at build time. The output for each node type matches what
|
|
6
|
+
* `MdzNodeView.svelte` renders at runtime, so precompiled and runtime rendering are identical.
|
|
7
|
+
*
|
|
8
|
+
* @module
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import {UnreachableError} from '@fuzdev/fuz_util/error.js';
|
|
12
|
+
import {escape_svelte_text} from '@fuzdev/fuz_util/svelte_preprocess_helpers.js';
|
|
13
|
+
import {escape_js_string} from '@fuzdev/fuz_util/string.js';
|
|
14
|
+
|
|
15
|
+
import type {MdzNode} from './mdz.js';
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Result of converting `MdzNode` arrays to Svelte markup.
|
|
19
|
+
*/
|
|
20
|
+
export interface MdzToSvelteResult {
|
|
21
|
+
/** Generated Svelte markup string. */
|
|
22
|
+
markup: string;
|
|
23
|
+
|
|
24
|
+
/** Required imports: `Map<local_name, {path, kind}>`. */
|
|
25
|
+
imports: Map<string, {path: string; kind: 'default' | 'named'}>;
|
|
26
|
+
|
|
27
|
+
/** Whether content references unconfigured Component or Element tags. */
|
|
28
|
+
has_unconfigured_tags: boolean;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Converts an array of `MdzNode` to a Svelte markup string.
|
|
33
|
+
*
|
|
34
|
+
* Each node type produces output matching what `MdzNodeView.svelte` renders at runtime.
|
|
35
|
+
* Collects required imports and flags unconfigured component/element references.
|
|
36
|
+
*
|
|
37
|
+
* @param nodes Parsed mdz nodes to render.
|
|
38
|
+
* @param components Component name to import path mapping (e.g., `{Alert: '$lib/Alert.svelte'}`).
|
|
39
|
+
* If content references a component not in this map, `has_unconfigured_tags` is set.
|
|
40
|
+
* @param elements Allowed HTML element names (e.g., `new Set(['aside', 'details'])`).
|
|
41
|
+
* If content references an element not in this set, `has_unconfigured_tags` is set.
|
|
42
|
+
*/
|
|
43
|
+
export const mdz_to_svelte = (
|
|
44
|
+
nodes: Array<MdzNode>,
|
|
45
|
+
components: Record<string, string>,
|
|
46
|
+
elements: ReadonlySet<string>,
|
|
47
|
+
): MdzToSvelteResult => {
|
|
48
|
+
const imports: Map<string, {path: string; kind: 'default' | 'named'}> = new Map();
|
|
49
|
+
let has_unconfigured_tags = false;
|
|
50
|
+
|
|
51
|
+
const render_nodes = (children: Array<MdzNode>): string => {
|
|
52
|
+
return children.map((child) => render_node(child)).join('');
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
const render_node = (node: MdzNode): string => {
|
|
56
|
+
switch (node.type) {
|
|
57
|
+
case 'Text':
|
|
58
|
+
return escape_svelte_text(node.content);
|
|
59
|
+
|
|
60
|
+
case 'Code':
|
|
61
|
+
imports.set('DocsLink', {path: '@fuzdev/fuz_ui/DocsLink.svelte', kind: 'default'});
|
|
62
|
+
return `<DocsLink reference={'${escape_js_string(node.content)}'} />`;
|
|
63
|
+
|
|
64
|
+
case 'Codeblock': {
|
|
65
|
+
imports.set('Code', {path: '@fuzdev/fuz_code/Code.svelte', kind: 'default'});
|
|
66
|
+
const lang_attr =
|
|
67
|
+
node.lang === null ? 'lang={null}' : `lang={'${escape_js_string(node.lang)}'}`;
|
|
68
|
+
return `<Code ${lang_attr} content={'${escape_js_string(node.content)}'} />`;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
case 'Bold':
|
|
72
|
+
return `<strong>${render_nodes(node.children)}</strong>`;
|
|
73
|
+
|
|
74
|
+
case 'Italic':
|
|
75
|
+
return `<em>${render_nodes(node.children)}</em>`;
|
|
76
|
+
|
|
77
|
+
case 'Strikethrough':
|
|
78
|
+
return `<s>${render_nodes(node.children)}</s>`;
|
|
79
|
+
|
|
80
|
+
case 'Link': {
|
|
81
|
+
const children_markup = render_nodes(node.children);
|
|
82
|
+
if (node.link_type === 'internal') {
|
|
83
|
+
if (node.reference.startsWith('#') || node.reference.startsWith('?')) {
|
|
84
|
+
return `<a href={'${escape_js_string(node.reference)}'}>${children_markup}</a>`;
|
|
85
|
+
}
|
|
86
|
+
imports.set('resolve', {path: '$app/paths', kind: 'named'});
|
|
87
|
+
return `<a href={resolve('${escape_js_string(node.reference)}')}>${children_markup}</a>`;
|
|
88
|
+
}
|
|
89
|
+
// External link — matches MdzNodeView: target="_blank" rel="noopener"
|
|
90
|
+
return `<a href={'${escape_js_string(node.reference)}'} target="_blank" rel="noopener">${children_markup}</a>`;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
case 'Paragraph':
|
|
94
|
+
return `<p>${render_nodes(node.children)}</p>`;
|
|
95
|
+
|
|
96
|
+
case 'Hr':
|
|
97
|
+
return '<hr />';
|
|
98
|
+
|
|
99
|
+
case 'Heading':
|
|
100
|
+
return `<h${node.level}>${render_nodes(node.children)}</h${node.level}>`;
|
|
101
|
+
|
|
102
|
+
case 'Element': {
|
|
103
|
+
if (!elements.has(node.name)) {
|
|
104
|
+
has_unconfigured_tags = true;
|
|
105
|
+
return '';
|
|
106
|
+
}
|
|
107
|
+
return `<${node.name}>${render_nodes(node.children)}</${node.name}>`;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
case 'Component': {
|
|
111
|
+
const import_path = components[node.name];
|
|
112
|
+
if (!import_path) {
|
|
113
|
+
has_unconfigured_tags = true;
|
|
114
|
+
return '';
|
|
115
|
+
}
|
|
116
|
+
imports.set(node.name, {path: import_path, kind: 'default'});
|
|
117
|
+
return `<${node.name}>${render_nodes(node.children)}</${node.name}>`;
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
default:
|
|
121
|
+
throw new UnreachableError(node);
|
|
122
|
+
}
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
const markup = render_nodes(nodes);
|
|
126
|
+
return {markup, imports, has_unconfigured_tags};
|
|
127
|
+
};
|