@pyreon/compiler 0.1.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/lib/index.js ADDED
@@ -0,0 +1,547 @@
1
+ import ts from "typescript";
2
+
3
+ //#region src/jsx.ts
4
+ /**
5
+ * JSX transform — wraps dynamic JSX expressions in `() =>` so the Pyreon runtime
6
+ * receives reactive getters instead of eagerly-evaluated snapshot values.
7
+ *
8
+ * Rules:
9
+ * - `<div>{expr}</div>` → `<div>{() => expr}</div>` (child)
10
+ * - `<div class={expr}>` → `<div class={() => expr}>` (prop)
11
+ * - `<button onClick={fn}>` → unchanged (event handler)
12
+ * - `<div>{() => expr}</div>` → unchanged (already wrapped)
13
+ * - `<div>{"literal"}</div>` → unchanged (static)
14
+ *
15
+ * Static VNode hoisting:
16
+ * - Fully static JSX in expression containers is hoisted to module scope:
17
+ * `{<span>Hello</span>}` → `const _$h0 = <span>Hello</span>` + `{_$h0}`
18
+ * - Hoisted nodes are created ONCE at module initialisation, not per-instance.
19
+ * - A JSX node is static if: all props are string literals / booleans / static
20
+ * values, and all children are text nodes or other static JSX nodes.
21
+ *
22
+ * Template emission:
23
+ * - JSX element trees with ≥ 2 DOM elements (no components, no spread attrs)
24
+ * are compiled to `_tpl(html, bindFn)` calls instead of nested `h()` calls.
25
+ * - The HTML string is parsed once via <template>.innerHTML, then cloneNode(true)
26
+ * for each instance (~5-10x faster than sequential createElement calls).
27
+ * - Static attributes are baked into the HTML string; dynamic attributes and
28
+ * text content use renderEffect in the bind function.
29
+ *
30
+ * Implementation: TypeScript parser for positions + magic-string replacements.
31
+ * No extra runtime dependencies — `typescript` is already in devDependencies.
32
+ *
33
+ * Known limitation (v0): expressions inside *nested* JSX within a child
34
+ * expression container are not individually wrapped. They are still reactive
35
+ * because the outer wrapper re-evaluates the whole subtree, just at a coarser
36
+ * granularity. Fine-grained nested wrapping is planned for a future pass.
37
+ */
38
+ const SKIP_PROPS = new Set(["key", "ref"]);
39
+ const EVENT_RE = /^on[A-Z]/;
40
+ function transformJSX(code, filename = "input.tsx") {
41
+ const scriptKind = filename.endsWith(".tsx") || filename.endsWith(".jsx") ? ts.ScriptKind.TSX : ts.ScriptKind.TSX;
42
+ const sf = ts.createSourceFile(filename, code, ts.ScriptTarget.ESNext, true, scriptKind);
43
+ const replacements = [];
44
+ const warnings = [];
45
+ function warn(node, message, warnCode) {
46
+ const { line, character } = sf.getLineAndCharacterOfPosition(node.getStart(sf));
47
+ warnings.push({
48
+ message,
49
+ line: line + 1,
50
+ column: character,
51
+ code: warnCode
52
+ });
53
+ }
54
+ const hoists = [];
55
+ let hoistIdx = 0;
56
+ let needsTplImport = false;
57
+ /**
58
+ * If `node` is a fully-static JSX element/fragment, register a module-scope
59
+ * hoist for it and return the generated variable name. Otherwise return null.
60
+ */
61
+ function maybeHoist(node) {
62
+ if ((ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node) || ts.isJsxFragment(node)) && isStaticJSXNode(node)) {
63
+ const name = `_$h${hoistIdx++}`;
64
+ const text = code.slice(node.getStart(sf), node.getEnd());
65
+ hoists.push({
66
+ name,
67
+ text
68
+ });
69
+ return name;
70
+ }
71
+ return null;
72
+ }
73
+ function wrap(expr) {
74
+ const start = expr.getStart(sf);
75
+ const end = expr.getEnd();
76
+ replacements.push({
77
+ start,
78
+ end,
79
+ text: `() => ${code.slice(start, end)}`
80
+ });
81
+ }
82
+ /** Try to hoist or wrap an expression, pushing a replacement if needed. */
83
+ function hoistOrWrap(expr) {
84
+ const hoistName = maybeHoist(expr);
85
+ if (hoistName) replacements.push({
86
+ start: expr.getStart(sf),
87
+ end: expr.getEnd(),
88
+ text: hoistName
89
+ });
90
+ else if (shouldWrap(expr)) wrap(expr);
91
+ }
92
+ /** Try to emit a template for a JsxElement. Returns true if handled. */
93
+ function tryTemplateEmit(node) {
94
+ if (templateElementCount(node) < 1) return false;
95
+ const tplCall = buildTemplateCall(node);
96
+ if (!tplCall) return false;
97
+ const start = node.getStart(sf);
98
+ const end = node.getEnd();
99
+ const parent = node.parent;
100
+ const needsBraces = parent && (ts.isJsxElement(parent) || ts.isJsxFragment(parent));
101
+ replacements.push({
102
+ start,
103
+ end,
104
+ text: needsBraces ? `{${tplCall}}` : tplCall
105
+ });
106
+ needsTplImport = true;
107
+ return true;
108
+ }
109
+ /** Emit warnings for common JSX mistakes (e.g. <For> without by). */
110
+ function checkForWarnings(node) {
111
+ const opening = ts.isJsxElement(node) ? node.openingElement : node;
112
+ if ((ts.isIdentifier(opening.tagName) ? opening.tagName.text : "") !== "For") return;
113
+ if (!opening.attributes.properties.some((p) => ts.isJsxAttribute(p) && ts.isIdentifier(p.name) && p.name.text === "by")) warn(opening.tagName, `<For> without a "by" prop will use index-based diffing, which is slower and may cause bugs with stateful children. Add by={(item) => item.id} for efficient keyed reconciliation.`, "missing-key-on-for");
114
+ }
115
+ /** Handle a JSX attribute node — wrap or hoist its value if needed. */
116
+ function handleJsxAttribute(node) {
117
+ const name = ts.isIdentifier(node.name) ? node.name.text : "";
118
+ const openingEl = node.parent.parent;
119
+ const tagName = ts.isIdentifier(openingEl.tagName) ? openingEl.tagName.text : "";
120
+ if (tagName.length > 0 && tagName.charAt(0) !== tagName.charAt(0).toLowerCase()) return;
121
+ if (SKIP_PROPS.has(name) || EVENT_RE.test(name)) return;
122
+ if (!node.initializer || !ts.isJsxExpression(node.initializer)) return;
123
+ const expr = node.initializer.expression;
124
+ if (expr) hoistOrWrap(expr);
125
+ }
126
+ /** Handle a JSX expression in child position — wrap or hoist. */
127
+ function handleJsxExpression(node) {
128
+ const expr = node.expression;
129
+ if (expr) hoistOrWrap(expr);
130
+ }
131
+ function walk(node) {
132
+ if (ts.isJsxElement(node) && tryTemplateEmit(node)) return;
133
+ if (ts.isJsxSelfClosingElement(node) || ts.isJsxElement(node)) checkForWarnings(node);
134
+ if (ts.isJsxAttribute(node)) {
135
+ handleJsxAttribute(node);
136
+ return;
137
+ }
138
+ if (ts.isJsxExpression(node)) {
139
+ handleJsxExpression(node);
140
+ return;
141
+ }
142
+ ts.forEachChild(node, walk);
143
+ }
144
+ walk(sf);
145
+ if (replacements.length === 0 && hoists.length === 0) return {
146
+ code,
147
+ warnings
148
+ };
149
+ replacements.sort((a, b) => b.start - a.start);
150
+ let result = code;
151
+ for (const r of replacements) result = result.slice(0, r.start) + r.text + result.slice(r.end);
152
+ if (hoists.length > 0) result = hoists.map((h) => `const ${h.name} = /*@__PURE__*/ ${h.text}\n`).join("") + result;
153
+ if (needsTplImport) result = `import { _tpl } from "@pyreon/runtime-dom";\nimport { _bind } from "@pyreon/reactivity";\n` + result;
154
+ return {
155
+ code: result,
156
+ usesTemplates: needsTplImport,
157
+ warnings
158
+ };
159
+ /** Check if a single attribute would prevent template emission. */
160
+ function hasBailAttr(node) {
161
+ for (const attr of jsxAttrs(node)) {
162
+ if (ts.isJsxSpreadAttribute(attr)) return true;
163
+ if (ts.isJsxAttribute(attr) && ts.isIdentifier(attr.name) && attr.name.text === "key") return true;
164
+ }
165
+ return false;
166
+ }
167
+ /**
168
+ * Count template-eligible elements for a single JSX child.
169
+ * Returns 0 for skippable children, -1 for bail, positive for element count.
170
+ */
171
+ function countChildForTemplate(child) {
172
+ if (ts.isJsxText(child)) return 0;
173
+ if (ts.isJsxElement(child) || ts.isJsxSelfClosingElement(child)) return templateElementCount(child);
174
+ if (ts.isJsxExpression(child)) {
175
+ if (!child.expression) return 0;
176
+ return containsJSXInExpr(child.expression) ? -1 : 0;
177
+ }
178
+ if (ts.isJsxFragment(child)) return templateFragmentCount(child);
179
+ return -1;
180
+ }
181
+ /**
182
+ * Count DOM elements in a JSX subtree. Returns -1 if the tree is not
183
+ * eligible for template emission.
184
+ */
185
+ function templateElementCount(node) {
186
+ const tag = jsxTagName(node);
187
+ if (!tag || !isLowerCase(tag)) return -1;
188
+ if (hasBailAttr(node)) return -1;
189
+ if (!ts.isJsxElement(node)) return 1;
190
+ let count = 1;
191
+ for (const child of node.children) {
192
+ const c = countChildForTemplate(child);
193
+ if (c === -1) return -1;
194
+ count += c;
195
+ }
196
+ return count;
197
+ }
198
+ /** Count template-eligible elements inside a fragment. */
199
+ function templateFragmentCount(frag) {
200
+ let count = 0;
201
+ for (const child of frag.children) {
202
+ const c = countChildForTemplate(child);
203
+ if (c === -1) return -1;
204
+ count += c;
205
+ }
206
+ return count;
207
+ }
208
+ /**
209
+ * Build the complete `_tpl("html", (__root) => { ... })` call string
210
+ * for a template-eligible JSX element tree. Returns null if codegen fails.
211
+ */
212
+ function buildTemplateCall(node) {
213
+ const bindLines = [];
214
+ const disposerNames = [];
215
+ let varIdx = 0;
216
+ let dispIdx = 0;
217
+ function nextVar() {
218
+ return `__e${varIdx++}`;
219
+ }
220
+ function nextDisp() {
221
+ const name = `__d${dispIdx++}`;
222
+ disposerNames.push(name);
223
+ return name;
224
+ }
225
+ function nextTextVar() {
226
+ return `__t${varIdx++}`;
227
+ }
228
+ /** Resolve the variable name for an element given its accessor path. */
229
+ function resolveElementVar(accessor, hasDynamic) {
230
+ if (accessor === "__root") return "__root";
231
+ if (hasDynamic) {
232
+ const v = nextVar();
233
+ bindLines.push(`const ${v} = ${accessor}`);
234
+ return v;
235
+ }
236
+ return accessor;
237
+ }
238
+ /** Emit bind line for a ref attribute. */
239
+ function emitRef(attr, varName) {
240
+ if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return;
241
+ if (!attr.initializer.expression) return;
242
+ bindLines.push(`${sliceExpr(attr.initializer.expression)}.current = ${varName}`);
243
+ }
244
+ /** Emit addEventListener bind line for an event handler attribute. */
245
+ function emitEventListener(attr, attrName, varName) {
246
+ const eventName = (attrName[2] ?? "").toLowerCase() + attrName.slice(3);
247
+ if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return;
248
+ if (!attr.initializer.expression) return;
249
+ bindLines.push(`${varName}.addEventListener("${eventName}", ${sliceExpr(attr.initializer.expression)})`);
250
+ }
251
+ /** Return HTML string for a static attribute expression, or null if not static. */
252
+ function staticAttrToHtml(exprNode, htmlAttrName) {
253
+ if (!isStatic(exprNode)) return null;
254
+ if (ts.isStringLiteral(exprNode)) return ` ${htmlAttrName}="${escapeHtmlAttr(exprNode.text)}"`;
255
+ if (ts.isNumericLiteral(exprNode)) return ` ${htmlAttrName}="${exprNode.text}"`;
256
+ if (exprNode.kind === ts.SyntaxKind.TrueKeyword) return ` ${htmlAttrName}`;
257
+ return "";
258
+ }
259
+ /** Unwrap a reactive accessor expression for use inside _bind(). */
260
+ function unwrapAccessor(exprNode) {
261
+ if (ts.isArrowFunction(exprNode) && !ts.isBlock(exprNode.body)) return {
262
+ expr: sliceExpr(exprNode.body),
263
+ isReactive: true
264
+ };
265
+ if (ts.isArrowFunction(exprNode) || ts.isFunctionExpression(exprNode)) return {
266
+ expr: `(${sliceExpr(exprNode)})()`,
267
+ isReactive: true
268
+ };
269
+ return {
270
+ expr: sliceExpr(exprNode),
271
+ isReactive: containsCall(exprNode)
272
+ };
273
+ }
274
+ /** Emit bind line for a dynamic (non-static) attribute. */
275
+ function emitDynamicAttr(_expr, exprNode, htmlAttrName, varName) {
276
+ const { expr, isReactive } = unwrapAccessor(exprNode);
277
+ const setter = htmlAttrName === "class" ? `${varName}.className = ${expr}` : `${varName}.setAttribute("${htmlAttrName}", ${expr})`;
278
+ if (isReactive) {
279
+ const d = nextDisp();
280
+ bindLines.push(`const ${d} = _bind(() => { ${setter} })`);
281
+ } else bindLines.push(setter);
282
+ }
283
+ /** Emit bind line or HTML for an expression attribute value. */
284
+ function emitAttrExpression(exprNode, htmlAttrName, varName) {
285
+ const staticHtml = staticAttrToHtml(exprNode, htmlAttrName);
286
+ if (staticHtml !== null) return staticHtml;
287
+ emitDynamicAttr(sliceExpr(exprNode), exprNode, htmlAttrName, varName);
288
+ return "";
289
+ }
290
+ /** Emit side-effects for special attrs (ref, event). Returns true if handled. */
291
+ function tryEmitSpecialAttr(attr, attrName, varName) {
292
+ if (attrName === "ref") {
293
+ emitRef(attr, varName);
294
+ return true;
295
+ }
296
+ if (EVENT_RE.test(attrName)) {
297
+ emitEventListener(attr, attrName, varName);
298
+ return true;
299
+ }
300
+ return false;
301
+ }
302
+ /** Convert an attribute initializer to HTML. Returns empty string for side-effect-only attrs. */
303
+ function attrInitializerToHtml(attr, htmlAttrName, varName) {
304
+ if (!attr.initializer) return ` ${htmlAttrName}`;
305
+ if (ts.isStringLiteral(attr.initializer)) return ` ${htmlAttrName}="${escapeHtmlAttr(attr.initializer.text)}"`;
306
+ if (ts.isJsxExpression(attr.initializer) && attr.initializer.expression) return emitAttrExpression(attr.initializer.expression, htmlAttrName, varName);
307
+ return "";
308
+ }
309
+ /** Process a single attribute, returning HTML to append. */
310
+ function processOneAttr(attr, varName) {
311
+ if (!ts.isJsxAttribute(attr)) return "";
312
+ const attrName = ts.isIdentifier(attr.name) ? attr.name.text : "";
313
+ if (attrName === "key") return "";
314
+ if (tryEmitSpecialAttr(attr, attrName, varName)) return "";
315
+ return attrInitializerToHtml(attr, JSX_TO_HTML_ATTR[attrName] ?? attrName, varName);
316
+ }
317
+ /** Process all attributes on an element, returning the HTML attribute string. */
318
+ function processAttrs(el, varName) {
319
+ let htmlAttrs = "";
320
+ for (const attr of jsxAttrs(el)) htmlAttrs += processOneAttr(attr, varName);
321
+ return htmlAttrs;
322
+ }
323
+ /** Emit bind lines for a reactive text expression child. */
324
+ function emitReactiveTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder) {
325
+ const tVar = nextTextVar();
326
+ const d = nextDisp();
327
+ bindLines.push(`const ${tVar} = document.createTextNode("")`);
328
+ if (needsPlaceholder) bindLines.push(`${parentRef}.replaceChild(${tVar}, ${parentRef}.childNodes[${childNodeIdx}])`);
329
+ else bindLines.push(`${varName}.appendChild(${tVar})`);
330
+ bindLines.push(`const ${d} = _bind(() => { ${tVar}.data = ${expr} })`);
331
+ return needsPlaceholder ? "<!>" : "";
332
+ }
333
+ /** Emit bind lines for a static text expression child. */
334
+ function emitStaticTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder) {
335
+ if (needsPlaceholder) {
336
+ const tVar = nextTextVar();
337
+ bindLines.push(`const ${tVar} = document.createTextNode(${expr})`);
338
+ bindLines.push(`${parentRef}.replaceChild(${tVar}, ${parentRef}.childNodes[${childNodeIdx}])`);
339
+ return "<!>";
340
+ }
341
+ bindLines.push(`${varName}.textContent = ${expr}`);
342
+ return "";
343
+ }
344
+ /** Process a single flat child, returning the HTML contribution or null on failure. */
345
+ function processOneChild(child, varName, parentRef, useMixed, useMultiExpr, childNodeIdx) {
346
+ if (child.kind === "text") return escapeHtmlText(child.text);
347
+ if (child.kind === "element") {
348
+ const childAccessor = useMixed ? `${parentRef}.childNodes[${childNodeIdx}]` : `${parentRef}.children[${child.elemIdx}]`;
349
+ return processElement(child.node, childAccessor);
350
+ }
351
+ const needsPlaceholder = useMixed || useMultiExpr;
352
+ const { expr, isReactive } = unwrapAccessor(child.expression);
353
+ if (isReactive) return emitReactiveTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder);
354
+ return emitStaticTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder);
355
+ }
356
+ /** Process children of a JsxElement, returning the children HTML. */
357
+ function processChildren(el, varName, accessor) {
358
+ const flatChildren = flattenChildren(el.children);
359
+ const { useMixed, useMultiExpr } = analyzeChildren(flatChildren);
360
+ const parentRef = accessor === "__root" ? "__root" : varName;
361
+ let html = "";
362
+ let childNodeIdx = 0;
363
+ for (const child of flatChildren) {
364
+ const childHtml = processOneChild(child, varName, parentRef, useMixed, useMultiExpr, childNodeIdx);
365
+ if (childHtml === null) return null;
366
+ html += childHtml;
367
+ childNodeIdx++;
368
+ }
369
+ return html;
370
+ }
371
+ /** Process a single DOM element for template emission. Returns the HTML string or null. */
372
+ function processElement(el, accessor) {
373
+ const tag = jsxTagName(el);
374
+ if (!tag) return null;
375
+ const varName = resolveElementVar(accessor, elementHasDynamic(el));
376
+ let html = `<${tag}${processAttrs(el, varName)}>`;
377
+ if (ts.isJsxElement(el)) {
378
+ const childHtml = processChildren(el, varName, accessor);
379
+ if (childHtml === null) return null;
380
+ html += childHtml;
381
+ }
382
+ if (!VOID_ELEMENTS.has(tag)) html += `</${tag}>`;
383
+ return html;
384
+ }
385
+ const html = processElement(node, "__root");
386
+ if (html === null) return null;
387
+ const escaped = html.replace(/\\/g, "\\\\").replace(/"/g, "\\\"");
388
+ if (bindLines.length === 0 && disposerNames.length === 0) return `_tpl("${escaped}", () => null)`;
389
+ let body = bindLines.map((l) => ` ${l}`).join("\n");
390
+ if (disposerNames.length > 0) body += `\n return () => { ${disposerNames.map((d) => `${d}()`).join("; ")} }`;
391
+ else body += "\n return null";
392
+ return `_tpl("${escaped}", (__root) => {\n${body}\n})`;
393
+ }
394
+ /** Classify a single JSX child into a FlatChild descriptor. */
395
+ function classifyJsxChild(child, out, elemIdxRef, recurse) {
396
+ if (ts.isJsxText(child)) {
397
+ const trimmed = child.text.replace(/\n\s*/g, "").trim();
398
+ if (trimmed) out.push({
399
+ kind: "text",
400
+ text: trimmed
401
+ });
402
+ return;
403
+ }
404
+ if (ts.isJsxElement(child) || ts.isJsxSelfClosingElement(child)) {
405
+ out.push({
406
+ kind: "element",
407
+ node: child,
408
+ elemIdx: elemIdxRef.value++
409
+ });
410
+ return;
411
+ }
412
+ if (ts.isJsxExpression(child)) {
413
+ if (child.expression) out.push({
414
+ kind: "expression",
415
+ expression: child.expression
416
+ });
417
+ return;
418
+ }
419
+ if (ts.isJsxFragment(child)) recurse(child.children);
420
+ }
421
+ /**
422
+ * Flatten JSX children, inlining fragment children and stripping whitespace-only text.
423
+ * Returns a flat array of child descriptors with element indices pre-computed.
424
+ */
425
+ function flattenChildren(children) {
426
+ const flatList = [];
427
+ const elemIdxRef = { value: 0 };
428
+ function addChildren(kids) {
429
+ for (const child of kids) classifyJsxChild(child, flatList, elemIdxRef, addChildren);
430
+ }
431
+ addChildren(children);
432
+ return flatList;
433
+ }
434
+ /** Analyze flat children to determine indexing strategy. */
435
+ function analyzeChildren(flatChildren) {
436
+ const hasElem = flatChildren.some((c) => c.kind === "element");
437
+ const hasNonElem = flatChildren.some((c) => c.kind !== "element");
438
+ const exprCount = flatChildren.filter((c) => c.kind === "expression").length;
439
+ return {
440
+ useMixed: hasElem && hasNonElem,
441
+ useMultiExpr: exprCount > 1
442
+ };
443
+ }
444
+ /** Check if a single attribute is dynamic (has ref, event, or non-static expression). */
445
+ function attrIsDynamic(attr) {
446
+ if (!ts.isJsxAttribute(attr)) return false;
447
+ const name = ts.isIdentifier(attr.name) ? attr.name.text : "";
448
+ if (name === "ref") return true;
449
+ if (EVENT_RE.test(name)) return true;
450
+ if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return false;
451
+ const expr = attr.initializer.expression;
452
+ return expr ? !isStatic(expr) : false;
453
+ }
454
+ /** Check if an element has any dynamic attributes, events, ref, or expression children */
455
+ function elementHasDynamic(node) {
456
+ if (jsxAttrs(node).some(attrIsDynamic)) return true;
457
+ if (ts.isJsxElement(node)) return node.children.some((c) => ts.isJsxExpression(c) && c.expression !== void 0);
458
+ return false;
459
+ }
460
+ /** Slice expression source from the original code */
461
+ function sliceExpr(expr) {
462
+ return code.slice(expr.getStart(sf), expr.getEnd());
463
+ }
464
+ /** Get tag name string */
465
+ function jsxTagName(node) {
466
+ const tag = ts.isJsxElement(node) ? node.openingElement.tagName : node.tagName;
467
+ return ts.isIdentifier(tag) ? tag.text : "";
468
+ }
469
+ /** Get attribute list */
470
+ function jsxAttrs(node) {
471
+ return ts.isJsxElement(node) ? node.openingElement.attributes.properties : node.attributes.properties;
472
+ }
473
+ }
474
+ const VOID_ELEMENTS = new Set([
475
+ "area",
476
+ "base",
477
+ "br",
478
+ "col",
479
+ "embed",
480
+ "hr",
481
+ "img",
482
+ "input",
483
+ "link",
484
+ "meta",
485
+ "param",
486
+ "source",
487
+ "track",
488
+ "wbr"
489
+ ]);
490
+ const JSX_TO_HTML_ATTR = {
491
+ className: "class",
492
+ htmlFor: "for"
493
+ };
494
+ function isLowerCase(s) {
495
+ return s.length > 0 && s[0] === s[0]?.toLowerCase();
496
+ }
497
+ /** Check if an expression subtree contains JSX nodes */
498
+ function containsJSXInExpr(node) {
499
+ if (ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node) || ts.isJsxFragment(node)) return true;
500
+ return ts.forEachChild(node, containsJSXInExpr) ?? false;
501
+ }
502
+ function escapeHtmlAttr(s) {
503
+ return s.replace(/&/g, "&amp;").replace(/"/g, "&quot;");
504
+ }
505
+ function escapeHtmlText(s) {
506
+ return s.replace(/&/g, "&amp;").replace(/</g, "&lt;");
507
+ }
508
+ function isStaticJSXNode(node) {
509
+ if (ts.isJsxSelfClosingElement(node)) return isStaticAttrs(node.attributes);
510
+ if (ts.isJsxFragment(node)) return node.children.every(isStaticChild);
511
+ return isStaticAttrs(node.openingElement.attributes) && node.children.every(isStaticChild);
512
+ }
513
+ function isStaticAttrs(attrs) {
514
+ return attrs.properties.every((prop) => {
515
+ if (!ts.isJsxAttribute(prop)) return false;
516
+ if (!prop.initializer) return true;
517
+ if (ts.isStringLiteral(prop.initializer)) return true;
518
+ const expr = prop.initializer.expression;
519
+ return expr ? isStatic(expr) : true;
520
+ });
521
+ }
522
+ function isStaticChild(child) {
523
+ if (ts.isJsxText(child)) return true;
524
+ if (ts.isJsxSelfClosingElement(child)) return isStaticJSXNode(child);
525
+ if (ts.isJsxElement(child)) return isStaticJSXNode(child);
526
+ if (ts.isJsxFragment(child)) return isStaticJSXNode(child);
527
+ const expr = child.expression;
528
+ return expr ? isStatic(expr) : true;
529
+ }
530
+ function isStatic(node) {
531
+ return ts.isStringLiteral(node) || ts.isNumericLiteral(node) || ts.isNoSubstitutionTemplateLiteral(node) || node.kind === ts.SyntaxKind.TrueKeyword || node.kind === ts.SyntaxKind.FalseKeyword || node.kind === ts.SyntaxKind.NullKeyword || node.kind === ts.SyntaxKind.UndefinedKeyword;
532
+ }
533
+ function shouldWrap(node) {
534
+ if (ts.isArrowFunction(node) || ts.isFunctionExpression(node)) return false;
535
+ if (isStatic(node)) return false;
536
+ return containsCall(node);
537
+ }
538
+ function containsCall(node) {
539
+ if (ts.isCallExpression(node)) return true;
540
+ if (ts.isTaggedTemplateExpression(node)) return true;
541
+ if (ts.isArrowFunction(node) || ts.isFunctionExpression(node)) return false;
542
+ return ts.forEachChild(node, containsCall) ?? false;
543
+ }
544
+
545
+ //#endregion
546
+ export { transformJSX };
547
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":[],"sources":["../src/jsx.ts"],"sourcesContent":["/**\n * JSX transform — wraps dynamic JSX expressions in `() =>` so the Pyreon runtime\n * receives reactive getters instead of eagerly-evaluated snapshot values.\n *\n * Rules:\n * - `<div>{expr}</div>` → `<div>{() => expr}</div>` (child)\n * - `<div class={expr}>` → `<div class={() => expr}>` (prop)\n * - `<button onClick={fn}>` → unchanged (event handler)\n * - `<div>{() => expr}</div>` → unchanged (already wrapped)\n * - `<div>{\"literal\"}</div>` → unchanged (static)\n *\n * Static VNode hoisting:\n * - Fully static JSX in expression containers is hoisted to module scope:\n * `{<span>Hello</span>}` → `const _$h0 = <span>Hello</span>` + `{_$h0}`\n * - Hoisted nodes are created ONCE at module initialisation, not per-instance.\n * - A JSX node is static if: all props are string literals / booleans / static\n * values, and all children are text nodes or other static JSX nodes.\n *\n * Template emission:\n * - JSX element trees with ≥ 2 DOM elements (no components, no spread attrs)\n * are compiled to `_tpl(html, bindFn)` calls instead of nested `h()` calls.\n * - The HTML string is parsed once via <template>.innerHTML, then cloneNode(true)\n * for each instance (~5-10x faster than sequential createElement calls).\n * - Static attributes are baked into the HTML string; dynamic attributes and\n * text content use renderEffect in the bind function.\n *\n * Implementation: TypeScript parser for positions + magic-string replacements.\n * No extra runtime dependencies — `typescript` is already in devDependencies.\n *\n * Known limitation (v0): expressions inside *nested* JSX within a child\n * expression container are not individually wrapped. They are still reactive\n * because the outer wrapper re-evaluates the whole subtree, just at a coarser\n * granularity. Fine-grained nested wrapping is planned for a future pass.\n */\n\nimport ts from \"typescript\"\n\nexport interface CompilerWarning {\n /** Warning message */\n message: string\n /** Source file line number (1-based) */\n line: number\n /** Source file column number (0-based) */\n column: number\n /** Warning code for filtering */\n code: \"signal-call-in-jsx\" | \"missing-key-on-for\" | \"signal-in-static-prop\"\n}\n\nexport interface TransformResult {\n /** Transformed source code (JSX preserved, only expression containers modified) */\n code: string\n /** Whether the output uses _tpl/_re template helpers (needs auto-import) */\n usesTemplates?: boolean\n /** Compiler warnings for common mistakes */\n warnings: CompilerWarning[]\n}\n\n// Props that should never be wrapped in a reactive getter\nconst SKIP_PROPS = new Set([\"key\", \"ref\"])\n// Event handler pattern: onClick, onInput, onMouseEnter, …\nconst EVENT_RE = /^on[A-Z]/\n\nexport function transformJSX(code: string, filename = \"input.tsx\"): TransformResult {\n const scriptKind =\n filename.endsWith(\".tsx\") || filename.endsWith(\".jsx\") ? ts.ScriptKind.TSX : ts.ScriptKind.TSX // default to TSX so JSX is always parsed\n\n const sf = ts.createSourceFile(\n filename,\n code,\n ts.ScriptTarget.ESNext,\n /* setParentNodes */ true,\n scriptKind,\n )\n\n type Replacement = { start: number; end: number; text: string }\n const replacements: Replacement[] = []\n const warnings: CompilerWarning[] = []\n\n function warn(node: ts.Node, message: string, warnCode: CompilerWarning[\"code\"]): void {\n const { line, character } = sf.getLineAndCharacterOfPosition(node.getStart(sf))\n warnings.push({ message, line: line + 1, column: character, code: warnCode })\n }\n\n // ── Static hoisting state ─────────────────────────────────────────────────\n type Hoist = { name: string; text: string }\n const hoists: Hoist[] = []\n let hoistIdx = 0\n let needsTplImport = false\n\n /**\n * If `node` is a fully-static JSX element/fragment, register a module-scope\n * hoist for it and return the generated variable name. Otherwise return null.\n */\n function maybeHoist(node: ts.Node): string | null {\n if (\n (ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node) || ts.isJsxFragment(node)) &&\n isStaticJSXNode(node as ts.JsxElement | ts.JsxSelfClosingElement | ts.JsxFragment)\n ) {\n const name = `_$h${hoistIdx++}`\n const text = code.slice(node.getStart(sf), node.getEnd())\n hoists.push({ name, text })\n return name\n }\n return null\n }\n\n function wrap(expr: ts.Expression): void {\n const start = expr.getStart(sf)\n const end = expr.getEnd()\n replacements.push({ start, end, text: `() => ${code.slice(start, end)}` })\n }\n\n /** Try to hoist or wrap an expression, pushing a replacement if needed. */\n function hoistOrWrap(expr: ts.Expression): void {\n const hoistName = maybeHoist(expr)\n if (hoistName) {\n replacements.push({ start: expr.getStart(sf), end: expr.getEnd(), text: hoistName })\n } else if (shouldWrap(expr)) {\n wrap(expr)\n }\n }\n\n // ── walk sub-handlers ───────────────────────────────────────────────────────\n\n /** Try to emit a template for a JsxElement. Returns true if handled. */\n function tryTemplateEmit(node: ts.JsxElement): boolean {\n const elemCount = templateElementCount(node)\n if (elemCount < 1) return false\n const tplCall = buildTemplateCall(node)\n if (!tplCall) return false\n const start = node.getStart(sf)\n const end = node.getEnd()\n const parent = node.parent\n const needsBraces = parent && (ts.isJsxElement(parent) || ts.isJsxFragment(parent))\n replacements.push({ start, end, text: needsBraces ? `{${tplCall}}` : tplCall })\n needsTplImport = true\n return true\n }\n\n /** Emit warnings for common JSX mistakes (e.g. <For> without by). */\n function checkForWarnings(node: ts.JsxElement | ts.JsxSelfClosingElement): void {\n const opening = ts.isJsxElement(node) ? node.openingElement : node\n const tagName = ts.isIdentifier(opening.tagName) ? opening.tagName.text : \"\"\n if (tagName !== \"For\") return\n const hasBy = opening.attributes.properties.some(\n (p) => ts.isJsxAttribute(p) && ts.isIdentifier(p.name) && p.name.text === \"by\",\n )\n if (!hasBy) {\n warn(\n opening.tagName,\n `<For> without a \"by\" prop will use index-based diffing, which is slower and may cause bugs with stateful children. Add by={(item) => item.id} for efficient keyed reconciliation.`,\n \"missing-key-on-for\",\n )\n }\n }\n\n /** Handle a JSX attribute node — wrap or hoist its value if needed. */\n function handleJsxAttribute(node: ts.JsxAttribute): void {\n const name = ts.isIdentifier(node.name) ? node.name.text : \"\"\n const openingEl = node.parent.parent as ts.JsxOpeningElement | ts.JsxSelfClosingElement\n const tagName = ts.isIdentifier(openingEl.tagName) ? openingEl.tagName.text : \"\"\n const isComponentElement =\n tagName.length > 0 && tagName.charAt(0) !== tagName.charAt(0).toLowerCase()\n if (isComponentElement) return\n if (SKIP_PROPS.has(name) || EVENT_RE.test(name)) return\n if (!node.initializer || !ts.isJsxExpression(node.initializer)) return\n const expr = node.initializer.expression\n if (expr) hoistOrWrap(expr)\n }\n\n /** Handle a JSX expression in child position — wrap or hoist. */\n function handleJsxExpression(node: ts.JsxExpression): void {\n const expr = node.expression\n if (expr) hoistOrWrap(expr)\n }\n\n function walk(node: ts.Node): void {\n if (ts.isJsxElement(node) && tryTemplateEmit(node)) return\n if (ts.isJsxSelfClosingElement(node) || ts.isJsxElement(node)) checkForWarnings(node)\n if (ts.isJsxAttribute(node)) {\n handleJsxAttribute(node)\n return\n }\n if (ts.isJsxExpression(node)) {\n handleJsxExpression(node)\n return\n }\n ts.forEachChild(node, walk)\n }\n\n walk(sf)\n\n if (replacements.length === 0 && hoists.length === 0) return { code, warnings }\n\n // Apply replacements from right to left so earlier positions stay valid\n replacements.sort((a, b) => b.start - a.start)\n\n let result = code\n for (const r of replacements) {\n result = result.slice(0, r.start) + r.text + result.slice(r.end)\n }\n\n // Prepend module-scope hoisted static VNode declarations\n if (hoists.length > 0) {\n const preamble = hoists.map((h) => `const ${h.name} = /*@__PURE__*/ ${h.text}\\n`).join(\"\")\n result = preamble + result\n }\n\n // Prepend template imports if _tpl() was emitted\n if (needsTplImport) {\n result =\n `import { _tpl } from \"@pyreon/runtime-dom\";\\nimport { _bind } from \"@pyreon/reactivity\";\\n` +\n result\n }\n\n return { code: result, usesTemplates: needsTplImport, warnings }\n\n // ── Template emission helpers (closures over sf, code) ──────────────────────\n\n /** Check if a single attribute would prevent template emission. */\n function hasBailAttr(node: ts.JsxElement | ts.JsxSelfClosingElement): boolean {\n for (const attr of jsxAttrs(node)) {\n if (ts.isJsxSpreadAttribute(attr)) return true\n if (ts.isJsxAttribute(attr) && ts.isIdentifier(attr.name) && attr.name.text === \"key\")\n return true\n }\n return false\n }\n\n /**\n * Count template-eligible elements for a single JSX child.\n * Returns 0 for skippable children, -1 for bail, positive for element count.\n */\n function countChildForTemplate(child: ts.JsxChild): number {\n if (ts.isJsxText(child)) return 0\n if (ts.isJsxElement(child) || ts.isJsxSelfClosingElement(child))\n return templateElementCount(child)\n if (ts.isJsxExpression(child)) {\n if (!child.expression) return 0\n return containsJSXInExpr(child.expression) ? -1 : 0\n }\n if (ts.isJsxFragment(child)) return templateFragmentCount(child)\n return -1\n }\n\n /**\n * Count DOM elements in a JSX subtree. Returns -1 if the tree is not\n * eligible for template emission.\n */\n function templateElementCount(node: ts.JsxElement | ts.JsxSelfClosingElement): number {\n const tag = jsxTagName(node)\n if (!tag || !isLowerCase(tag)) return -1\n if (hasBailAttr(node)) return -1\n if (!ts.isJsxElement(node)) return 1\n\n let count = 1\n for (const child of node.children) {\n const c = countChildForTemplate(child)\n if (c === -1) return -1\n count += c\n }\n return count\n }\n\n /** Count template-eligible elements inside a fragment. */\n function templateFragmentCount(frag: ts.JsxFragment): number {\n let count = 0\n for (const child of frag.children) {\n const c = countChildForTemplate(child)\n if (c === -1) return -1\n count += c\n }\n return count\n }\n\n /**\n * Build the complete `_tpl(\"html\", (__root) => { ... })` call string\n * for a template-eligible JSX element tree. Returns null if codegen fails.\n */\n function buildTemplateCall(node: ts.JsxElement | ts.JsxSelfClosingElement): string | null {\n const bindLines: string[] = []\n const disposerNames: string[] = []\n let varIdx = 0\n let dispIdx = 0\n\n function nextVar(): string {\n return `__e${varIdx++}`\n }\n function nextDisp(): string {\n const name = `__d${dispIdx++}`\n disposerNames.push(name)\n return name\n }\n function nextTextVar(): string {\n return `__t${varIdx++}`\n }\n\n /** Resolve the variable name for an element given its accessor path. */\n function resolveElementVar(accessor: string, hasDynamic: boolean): string {\n if (accessor === \"__root\") return \"__root\"\n if (hasDynamic) {\n const v = nextVar()\n bindLines.push(`const ${v} = ${accessor}`)\n return v\n }\n return accessor\n }\n\n /** Emit bind line for a ref attribute. */\n function emitRef(attr: ts.JsxAttribute, varName: string): void {\n if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return\n if (!attr.initializer.expression) return\n bindLines.push(`${sliceExpr(attr.initializer.expression)}.current = ${varName}`)\n }\n\n /** Emit addEventListener bind line for an event handler attribute. */\n function emitEventListener(attr: ts.JsxAttribute, attrName: string, varName: string): void {\n const eventName = (attrName[2] ?? \"\").toLowerCase() + attrName.slice(3)\n if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return\n if (!attr.initializer.expression) return\n bindLines.push(\n `${varName}.addEventListener(\"${eventName}\", ${sliceExpr(attr.initializer.expression)})`,\n )\n }\n\n /** Return HTML string for a static attribute expression, or null if not static. */\n function staticAttrToHtml(exprNode: ts.Expression, htmlAttrName: string): string | null {\n if (!isStatic(exprNode)) return null\n if (ts.isStringLiteral(exprNode)) return ` ${htmlAttrName}=\"${escapeHtmlAttr(exprNode.text)}\"`\n if (ts.isNumericLiteral(exprNode)) return ` ${htmlAttrName}=\"${exprNode.text}\"`\n if (exprNode.kind === ts.SyntaxKind.TrueKeyword) return ` ${htmlAttrName}`\n return \"\" // false/null/undefined → omit\n }\n\n /** Unwrap a reactive accessor expression for use inside _bind(). */\n function unwrapAccessor(exprNode: ts.Expression): { expr: string; isReactive: boolean } {\n // Concise arrow: () => value() → unwrap to \"value()\"\n if (ts.isArrowFunction(exprNode) && !ts.isBlock(exprNode.body)) {\n return { expr: sliceExpr(exprNode.body as ts.Expression), isReactive: true }\n }\n // Block-body arrow/function: invoke it\n if (ts.isArrowFunction(exprNode) || ts.isFunctionExpression(exprNode)) {\n return { expr: `(${sliceExpr(exprNode)})()`, isReactive: true }\n }\n return { expr: sliceExpr(exprNode), isReactive: containsCall(exprNode) }\n }\n\n /** Emit bind line for a dynamic (non-static) attribute. */\n function emitDynamicAttr(\n _expr: string,\n exprNode: ts.Expression,\n htmlAttrName: string,\n varName: string,\n ): void {\n const { expr, isReactive } = unwrapAccessor(exprNode)\n const setter =\n htmlAttrName === \"class\"\n ? `${varName}.className = ${expr}`\n : `${varName}.setAttribute(\"${htmlAttrName}\", ${expr})`\n\n if (isReactive) {\n const d = nextDisp()\n bindLines.push(`const ${d} = _bind(() => { ${setter} })`)\n } else {\n bindLines.push(setter)\n }\n }\n\n /** Emit bind line or HTML for an expression attribute value. */\n function emitAttrExpression(\n exprNode: ts.Expression,\n htmlAttrName: string,\n varName: string,\n ): string {\n const staticHtml = staticAttrToHtml(exprNode, htmlAttrName)\n if (staticHtml !== null) return staticHtml\n emitDynamicAttr(sliceExpr(exprNode), exprNode, htmlAttrName, varName)\n return \"\"\n }\n\n /** Emit side-effects for special attrs (ref, event). Returns true if handled. */\n function tryEmitSpecialAttr(attr: ts.JsxAttribute, attrName: string, varName: string): boolean {\n if (attrName === \"ref\") {\n emitRef(attr, varName)\n return true\n }\n if (EVENT_RE.test(attrName)) {\n emitEventListener(attr, attrName, varName)\n return true\n }\n return false\n }\n\n /** Convert an attribute initializer to HTML. Returns empty string for side-effect-only attrs. */\n function attrInitializerToHtml(\n attr: ts.JsxAttribute,\n htmlAttrName: string,\n varName: string,\n ): string {\n if (!attr.initializer) return ` ${htmlAttrName}`\n if (ts.isStringLiteral(attr.initializer))\n return ` ${htmlAttrName}=\"${escapeHtmlAttr(attr.initializer.text)}\"`\n if (ts.isJsxExpression(attr.initializer) && attr.initializer.expression)\n return emitAttrExpression(attr.initializer.expression, htmlAttrName, varName)\n return \"\"\n }\n\n /** Process a single attribute, returning HTML to append. */\n function processOneAttr(attr: ts.JsxAttributeLike, varName: string): string {\n if (!ts.isJsxAttribute(attr)) return \"\"\n const attrName = ts.isIdentifier(attr.name) ? attr.name.text : \"\"\n if (attrName === \"key\") return \"\"\n if (tryEmitSpecialAttr(attr, attrName, varName)) return \"\"\n return attrInitializerToHtml(attr, JSX_TO_HTML_ATTR[attrName] ?? attrName, varName)\n }\n\n /** Process all attributes on an element, returning the HTML attribute string. */\n function processAttrs(el: ts.JsxElement | ts.JsxSelfClosingElement, varName: string): string {\n let htmlAttrs = \"\"\n for (const attr of jsxAttrs(el)) htmlAttrs += processOneAttr(attr, varName)\n return htmlAttrs\n }\n\n /** Emit bind lines for a reactive text expression child. */\n function emitReactiveTextChild(\n expr: string,\n varName: string,\n parentRef: string,\n childNodeIdx: number,\n needsPlaceholder: boolean,\n ): string {\n const tVar = nextTextVar()\n const d = nextDisp()\n bindLines.push(`const ${tVar} = document.createTextNode(\"\")`)\n if (needsPlaceholder) {\n bindLines.push(\n `${parentRef}.replaceChild(${tVar}, ${parentRef}.childNodes[${childNodeIdx}])`,\n )\n } else {\n bindLines.push(`${varName}.appendChild(${tVar})`)\n }\n bindLines.push(`const ${d} = _bind(() => { ${tVar}.data = ${expr} })`)\n return needsPlaceholder ? \"<!>\" : \"\"\n }\n\n /** Emit bind lines for a static text expression child. */\n function emitStaticTextChild(\n expr: string,\n varName: string,\n parentRef: string,\n childNodeIdx: number,\n needsPlaceholder: boolean,\n ): string {\n if (needsPlaceholder) {\n const tVar = nextTextVar()\n bindLines.push(`const ${tVar} = document.createTextNode(${expr})`)\n bindLines.push(\n `${parentRef}.replaceChild(${tVar}, ${parentRef}.childNodes[${childNodeIdx}])`,\n )\n return \"<!>\"\n }\n bindLines.push(`${varName}.textContent = ${expr}`)\n return \"\"\n }\n\n /** Process a single flat child, returning the HTML contribution or null on failure. */\n function processOneChild(\n child: FlatChild,\n varName: string,\n parentRef: string,\n useMixed: boolean,\n useMultiExpr: boolean,\n childNodeIdx: number,\n ): string | null {\n if (child.kind === \"text\") return escapeHtmlText(child.text)\n if (child.kind === \"element\") {\n const childAccessor = useMixed\n ? `${parentRef}.childNodes[${childNodeIdx}]`\n : `${parentRef}.children[${child.elemIdx}]`\n return processElement(child.node, childAccessor)\n }\n // expression\n const needsPlaceholder = useMixed || useMultiExpr\n const { expr, isReactive } = unwrapAccessor(child.expression)\n if (isReactive) {\n return emitReactiveTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder)\n }\n return emitStaticTextChild(expr, varName, parentRef, childNodeIdx, needsPlaceholder)\n }\n\n /** Process children of a JsxElement, returning the children HTML. */\n function processChildren(el: ts.JsxElement, varName: string, accessor: string): string | null {\n const flatChildren = flattenChildren(el.children)\n const { useMixed, useMultiExpr } = analyzeChildren(flatChildren)\n const parentRef = accessor === \"__root\" ? \"__root\" : varName\n\n let html = \"\"\n let childNodeIdx = 0\n\n for (const child of flatChildren) {\n const childHtml = processOneChild(\n child,\n varName,\n parentRef,\n useMixed,\n useMultiExpr,\n childNodeIdx,\n )\n if (childHtml === null) return null\n html += childHtml\n childNodeIdx++\n }\n\n return html\n }\n\n /** Process a single DOM element for template emission. Returns the HTML string or null. */\n function processElement(\n el: ts.JsxElement | ts.JsxSelfClosingElement,\n accessor: string,\n ): string | null {\n const tag = jsxTagName(el)\n if (!tag) return null\n\n const varName = resolveElementVar(accessor, elementHasDynamic(el))\n const htmlAttrs = processAttrs(el, varName)\n let html = `<${tag}${htmlAttrs}>`\n\n if (ts.isJsxElement(el)) {\n const childHtml = processChildren(el, varName, accessor)\n if (childHtml === null) return null\n html += childHtml\n }\n\n if (!VOID_ELEMENTS.has(tag)) html += `</${tag}>`\n return html\n }\n\n const html = processElement(node, \"__root\")\n if (html === null) return null\n\n // Build bind function body\n const escaped = html.replace(/\\\\/g, \"\\\\\\\\\").replace(/\"/g, '\\\\\"')\n\n if (bindLines.length === 0 && disposerNames.length === 0) {\n return `_tpl(\"${escaped}\", () => null)`\n }\n\n let body = bindLines.map((l) => ` ${l}`).join(\"\\n\")\n if (disposerNames.length > 0) {\n body += `\\n return () => { ${disposerNames.map((d) => `${d}()`).join(\"; \")} }`\n } else {\n body += \"\\n return null\"\n }\n\n return `_tpl(\"${escaped}\", (__root) => {\\n${body}\\n})`\n }\n\n /** Flat child descriptor for template children processing */\n type FlatChild =\n | { kind: \"text\"; text: string }\n | { kind: \"element\"; node: ts.JsxElement | ts.JsxSelfClosingElement; elemIdx: number }\n | { kind: \"expression\"; expression: ts.Expression }\n\n /** Classify a single JSX child into a FlatChild descriptor. */\n function classifyJsxChild(\n child: ts.JsxChild,\n out: FlatChild[],\n elemIdxRef: { value: number },\n recurse: (kids: ts.NodeArray<ts.JsxChild>) => void,\n ): void {\n if (ts.isJsxText(child)) {\n const trimmed = child.text.replace(/\\n\\s*/g, \"\").trim()\n if (trimmed) out.push({ kind: \"text\", text: trimmed })\n return\n }\n if (ts.isJsxElement(child) || ts.isJsxSelfClosingElement(child)) {\n out.push({ kind: \"element\", node: child, elemIdx: elemIdxRef.value++ })\n return\n }\n if (ts.isJsxExpression(child)) {\n if (child.expression) out.push({ kind: \"expression\", expression: child.expression })\n return\n }\n if (ts.isJsxFragment(child)) recurse(child.children)\n }\n\n /**\n * Flatten JSX children, inlining fragment children and stripping whitespace-only text.\n * Returns a flat array of child descriptors with element indices pre-computed.\n */\n function flattenChildren(children: ts.NodeArray<ts.JsxChild>): FlatChild[] {\n const flatList: FlatChild[] = []\n const elemIdxRef = { value: 0 }\n\n function addChildren(kids: ts.NodeArray<ts.JsxChild>): void {\n for (const child of kids) classifyJsxChild(child, flatList, elemIdxRef, addChildren)\n }\n\n addChildren(children)\n return flatList\n }\n\n /** Analyze flat children to determine indexing strategy. */\n function analyzeChildren(flatChildren: FlatChild[]): {\n useMixed: boolean\n useMultiExpr: boolean\n } {\n const hasElem = flatChildren.some((c) => c.kind === \"element\")\n const hasNonElem = flatChildren.some((c) => c.kind !== \"element\")\n const exprCount = flatChildren.filter((c) => c.kind === \"expression\").length\n return { useMixed: hasElem && hasNonElem, useMultiExpr: exprCount > 1 }\n }\n\n /** Check if a single attribute is dynamic (has ref, event, or non-static expression). */\n function attrIsDynamic(attr: ts.JsxAttributeLike): boolean {\n if (!ts.isJsxAttribute(attr)) return false\n const name = ts.isIdentifier(attr.name) ? attr.name.text : \"\"\n if (name === \"ref\") return true\n if (EVENT_RE.test(name)) return true\n if (!attr.initializer || !ts.isJsxExpression(attr.initializer)) return false\n const expr = attr.initializer.expression\n return expr ? !isStatic(expr) : false\n }\n\n /** Check if an element has any dynamic attributes, events, ref, or expression children */\n function elementHasDynamic(node: ts.JsxElement | ts.JsxSelfClosingElement): boolean {\n if (jsxAttrs(node).some(attrIsDynamic)) return true\n if (ts.isJsxElement(node)) {\n return node.children.some((c) => ts.isJsxExpression(c) && c.expression !== undefined)\n }\n return false\n }\n\n /** Slice expression source from the original code */\n function sliceExpr(expr: ts.Expression): string {\n return code.slice(expr.getStart(sf), expr.getEnd())\n }\n\n /** Get tag name string */\n function jsxTagName(node: ts.JsxElement | ts.JsxSelfClosingElement): string {\n const tag = ts.isJsxElement(node) ? node.openingElement.tagName : node.tagName\n return ts.isIdentifier(tag) ? tag.text : \"\"\n }\n\n /** Get attribute list */\n function jsxAttrs(\n node: ts.JsxElement | ts.JsxSelfClosingElement,\n ): ts.NodeArray<ts.JsxAttributeLike> {\n return ts.isJsxElement(node)\n ? node.openingElement.attributes.properties\n : node.attributes.properties\n }\n}\n\n// ─── Template constants ──────────────────────────────────────────────────────\n\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n])\n\nconst JSX_TO_HTML_ATTR: Record<string, string> = {\n className: \"class\",\n htmlFor: \"for\",\n}\n\nfunction isLowerCase(s: string): boolean {\n return s.length > 0 && s[0] === s[0]?.toLowerCase()\n}\n\n/** Check if an expression subtree contains JSX nodes */\nfunction containsJSXInExpr(node: ts.Node): boolean {\n if (ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node) || ts.isJsxFragment(node))\n return true\n return ts.forEachChild(node, containsJSXInExpr) ?? false\n}\n\nfunction escapeHtmlAttr(s: string): string {\n return s.replace(/&/g, \"&amp;\").replace(/\"/g, \"&quot;\")\n}\n\nfunction escapeHtmlText(s: string): string {\n return s.replace(/&/g, \"&amp;\").replace(/</g, \"&lt;\")\n}\n\n// ─── Static JSX analysis ──────────────────────────────────────────────────────\n\ntype StaticJSXNode = ts.JsxElement | ts.JsxSelfClosingElement | ts.JsxFragment\n\nfunction isStaticJSXNode(node: StaticJSXNode): boolean {\n if (ts.isJsxSelfClosingElement(node)) {\n return isStaticAttrs(node.attributes)\n }\n if (ts.isJsxFragment(node)) {\n return node.children.every(isStaticChild)\n }\n // JsxElement\n return isStaticAttrs(node.openingElement.attributes) && node.children.every(isStaticChild)\n}\n\nfunction isStaticAttrs(attrs: ts.JsxAttributes): boolean {\n return attrs.properties.every((prop) => {\n // Spread attribute — always dynamic\n if (!ts.isJsxAttribute(prop)) return false\n // Boolean shorthand: <input disabled />\n if (!prop.initializer) return true\n // String literal: class=\"foo\"\n if (ts.isStringLiteral(prop.initializer)) return true\n // Must be JsxExpression — the only remaining JsxAttributeValue type\n const expr = (prop.initializer as ts.JsxExpression).expression\n return expr ? isStatic(expr) : true\n })\n}\n\nfunction isStaticChild(child: ts.JsxChild): boolean {\n // Plain text content\n if (ts.isJsxText(child)) return true\n // Nested JSX elements\n if (ts.isJsxSelfClosingElement(child)) return isStaticJSXNode(child)\n if (ts.isJsxElement(child)) return isStaticJSXNode(child)\n if (ts.isJsxFragment(child)) return isStaticJSXNode(child)\n // Must be JsxExpression — the only remaining JsxChild type\n const expr = (child as ts.JsxExpression).expression\n return expr ? isStatic(expr) : true\n}\n\n// ─── General helpers ──────────────────────────────────────────────────────────\n\nfunction isStatic(node: ts.Expression): boolean {\n return (\n ts.isStringLiteral(node) ||\n ts.isNumericLiteral(node) ||\n ts.isNoSubstitutionTemplateLiteral(node) ||\n node.kind === ts.SyntaxKind.TrueKeyword ||\n node.kind === ts.SyntaxKind.FalseKeyword ||\n node.kind === ts.SyntaxKind.NullKeyword ||\n node.kind === ts.SyntaxKind.UndefinedKeyword\n )\n}\n\nfunction shouldWrap(node: ts.Expression): boolean {\n // Already a function — user explicitly wrapped or it's a callback\n if (ts.isArrowFunction(node) || ts.isFunctionExpression(node)) return false\n // Static literal — no signals involved\n if (isStatic(node)) return false\n // Only wrap if the expression tree contains a call — signal reads are always\n // function calls (e.g. `count()`, `name()`). Plain identifiers, object literals\n // like `style={{ color: \"red\" }}`, array literals, and member accesses are\n // left as-is to avoid unnecessary reactive wrappers.\n return containsCall(node)\n}\n\nfunction containsCall(node: ts.Node): boolean {\n if (ts.isCallExpression(node)) return true\n if (ts.isTaggedTemplateExpression(node)) return true\n // Don't recurse into nested functions — they're self-contained\n if (ts.isArrowFunction(node) || ts.isFunctionExpression(node)) return false\n return ts.forEachChild(node, containsCall) ?? false\n}\n"],"mappings":"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"}