@sigil-dev/compiler 0.7.5 → 0.7.6

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.
@@ -1,51 +1,57 @@
1
1
  import { types as t } from "@babel/core";
2
2
 
3
3
  export const ATTR_MAP: Record<string, string> = {
4
- class: "className",
5
- for: "htmlFor",
4
+ class: "className",
5
+ for: "htmlFor",
6
6
  };
7
7
 
8
8
  export function containsSignal(
9
- expr: t.Expression,
10
- signals: Set<string>,
9
+ expr: t.Expression,
10
+ signals: Set<string>,
11
11
  ): boolean {
12
- if (t.isIdentifier(expr)) return signals.has(expr.name);
13
- if (t.isCallExpression(expr)) {
14
- return containsSignal(expr.callee as t.Expression, signals);
15
- }
16
- if (t.isBinaryExpression(expr)) {
17
- return (
18
- containsSignal(expr.left as t.Expression, signals) ||
19
- containsSignal(expr.right, signals)
20
- );
21
- }
22
- if (t.isMemberExpression(expr)) {
23
- return containsSignal(expr.object as t.Expression, signals);
24
- }
25
- if (t.isLogicalExpression(expr)) {
26
- return (
27
- containsSignal(expr.left as t.Expression, signals) ||
28
- containsSignal(expr.right as t.Expression, signals)
29
- );
30
- }
31
- if (t.isConditionalExpression(expr)) {
32
- return (
33
- containsSignal(expr.test as t.Expression, signals) ||
34
- containsSignal(expr.consequent as t.Expression, signals) ||
35
- containsSignal(expr.alternate as t.Expression, signals)
36
- );
37
- }
38
- return false;
12
+ if (t.isIdentifier(expr)) return signals.has(expr.name);
13
+ if (t.isCallExpression(expr)) {
14
+ return containsSignal(expr.callee as t.Expression, signals);
15
+ }
16
+ if (t.isBinaryExpression(expr)) {
17
+ return (
18
+ containsSignal(expr.left as t.Expression, signals) ||
19
+ containsSignal(expr.right, signals)
20
+ );
21
+ }
22
+ if (t.isMemberExpression(expr)) {
23
+ return containsSignal(expr.object as t.Expression, signals);
24
+ }
25
+ if (t.isLogicalExpression(expr)) {
26
+ return (
27
+ containsSignal(expr.left as t.Expression, signals) ||
28
+ containsSignal(expr.right as t.Expression, signals)
29
+ );
30
+ }
31
+ if (t.isConditionalExpression(expr)) {
32
+ return (
33
+ containsSignal(expr.test as t.Expression, signals) ||
34
+ containsSignal(expr.consequent as t.Expression, signals) ||
35
+ containsSignal(expr.alternate as t.Expression, signals)
36
+ );
37
+ }
38
+ if (t.isUnaryExpression(expr)) {
39
+ return containsSignal(expr.argument as t.Expression, signals);
40
+ }
41
+ if (t.isTemplateLiteral(expr)) {
42
+ return expr.expressions.some(e => containsSignal(e as t.Expression, signals));
43
+ }
44
+ return false;
39
45
  }
40
46
 
41
47
  export function isPrimitive(expr: t.Expression): boolean {
42
- if (t.isCallExpression(expr) && t.isIdentifier(expr.callee)) return true;
43
- if (t.isBinaryExpression(expr)) return true;
44
- if (t.isTemplateLiteral(expr)) return true;
45
- if (t.isIdentifier(expr)) return true;
46
- // user().name is a MemberExpression after state handler rewrites user → user()
47
- if (t.isMemberExpression(expr)) return true;
48
- return false;
48
+ if (t.isCallExpression(expr) && t.isIdentifier(expr.callee)) return true;
49
+ if (t.isBinaryExpression(expr)) return true;
50
+ if (t.isTemplateLiteral(expr)) return true;
51
+ if (t.isIdentifier(expr)) return true;
52
+ // user().name is a MemberExpression after state handler rewrites user → user()
53
+ if (t.isMemberExpression(expr)) return true;
54
+ return false;
49
55
  }
50
56
 
51
57
  /**
@@ -53,22 +59,22 @@ export function isPrimitive(expr: t.Expression): boolean {
53
59
  * When parentVar is provided, passes it to claim() for SPA fallback (append on create).
54
60
  */
55
61
  export function getCreateElement(
56
- tag: string,
57
- hydrate: boolean,
58
- nodesVar = "__nodes",
59
- parentVar?: string,
62
+ tag: string,
63
+ hydrate: boolean,
64
+ nodesVar = "__nodes",
65
+ parentVar?: string,
60
66
  ): t.Expression {
61
- if (hydrate) {
62
- const args: t.Expression[] = [t.identifier(nodesVar), t.stringLiteral(tag)];
63
- if (parentVar) {
64
- args.push(t.identifier(parentVar));
65
- }
66
- return t.callExpression(t.identifier("claim"), args);
67
- }
68
- return t.callExpression(
69
- t.memberExpression(t.identifier("document"), t.identifier("createElement")),
70
- [t.stringLiteral(tag)],
71
- );
67
+ if (hydrate) {
68
+ const args: t.Expression[] = [t.identifier(nodesVar), t.stringLiteral(tag)];
69
+ if (parentVar) {
70
+ args.push(t.identifier(parentVar));
71
+ }
72
+ return t.callExpression(t.identifier("claim"), args);
73
+ }
74
+ return t.callExpression(
75
+ t.memberExpression(t.identifier("document"), t.identifier("createElement")),
76
+ [t.stringLiteral(tag)],
77
+ );
72
78
  }
73
79
 
74
80
  /**
@@ -76,24 +82,24 @@ export function getCreateElement(
76
82
  * When parentVar is provided, passes it to claimText() for SPA fallback.
77
83
  */
78
84
  export function getCreateText(
79
- hydrate: boolean,
80
- nodesVar = "__nodes",
81
- parentVar?: string,
85
+ hydrate: boolean,
86
+ nodesVar = "__nodes",
87
+ parentVar?: string,
82
88
  ): t.Expression {
83
- if (hydrate) {
84
- const args: t.Expression[] = [t.identifier(nodesVar)];
85
- if (parentVar) {
86
- args.push(t.identifier(parentVar));
87
- }
88
- return t.callExpression(t.identifier("claimText"), args);
89
- }
90
- return t.callExpression(
91
- t.memberExpression(
92
- t.identifier("document"),
93
- t.identifier("createTextNode"),
94
- ),
95
- [t.stringLiteral("")],
96
- );
89
+ if (hydrate) {
90
+ const args: t.Expression[] = [t.identifier(nodesVar)];
91
+ if (parentVar) {
92
+ args.push(t.identifier(parentVar));
93
+ }
94
+ return t.callExpression(t.identifier("claimText"), args);
95
+ }
96
+ return t.callExpression(
97
+ t.memberExpression(
98
+ t.identifier("document"),
99
+ t.identifier("createTextNode"),
100
+ ),
101
+ [t.stringLiteral("")],
102
+ );
97
103
  }
98
104
 
99
105
  /**
@@ -101,24 +107,24 @@ export function getCreateText(
101
107
  * Returns the generated variable name for use by children.
102
108
  */
103
109
  export function buildHydrationScope(
104
- parentVar: string,
105
- statements: t.Statement[],
106
- nodesVar: string,
110
+ parentVar: string,
111
+ statements: t.Statement[],
112
+ nodesVar: string,
107
113
  ): void {
108
- statements.push(
109
- t.variableDeclaration("const", [
110
- t.variableDeclarator(
111
- t.identifier(nodesVar),
112
- t.callExpression(
113
- t.memberExpression(t.identifier("Array"), t.identifier("from")),
114
- [
115
- t.memberExpression(
116
- t.identifier(parentVar),
117
- t.identifier("childNodes"),
118
- ),
119
- ],
120
- ),
121
- ),
122
- ]),
123
- );
114
+ statements.push(
115
+ t.variableDeclaration("const", [
116
+ t.variableDeclarator(
117
+ t.identifier(nodesVar),
118
+ t.callExpression(
119
+ t.memberExpression(t.identifier("Array"), t.identifier("from")),
120
+ [
121
+ t.memberExpression(
122
+ t.identifier(parentVar),
123
+ t.identifier("childNodes"),
124
+ ),
125
+ ],
126
+ ),
127
+ ),
128
+ ]),
129
+ );
124
130
  }
@@ -1,135 +1,126 @@
1
- import { types as t } from "@babel/core";
2
-
3
- const BIND_EVENT: Record<string, string> = {
4
- value: "input",
5
- checked: "change",
6
- };
7
-
8
- export function buildBind(
9
- varName: string,
10
- attrName: string,
11
- attr: t.JSXAttribute,
12
- statements: t.Statement[],
13
- ): void {
14
- const signal = (attr.value as t.JSXExpressionContainer)
15
- .expression as t.CallExpression; // already text()
16
-
17
- const signalId = signal.callee as t.Identifier; // text — for .set()
18
-
19
- if (attrName === "bindThis") {
20
- statements.push(
21
- t.expressionStatement(
22
- t.callExpression(
23
- t.memberExpression(signalId, t.identifier("set")),
24
- [t.identifier(varName)],
25
- ),
26
- ),
27
- );
28
- return;
29
- }
30
-
31
- const prop = attrName.slice(4).toLowerCase();
32
- const event = BIND_EVENT[prop] ?? "input";
33
-
34
- // createEffect(() => _el.prop = text())
35
- statements.push(
36
- t.expressionStatement(
37
- t.callExpression(t.identifier("createEffect"), [
38
- t.arrowFunctionExpression(
39
- [],
40
- t.blockStatement([
41
- t.expressionStatement(
42
- t.assignmentExpression(
43
- "=",
44
- t.memberExpression(t.identifier(varName), t.identifier(prop)),
45
- signal, // text() — already correct
46
- ),
47
- ),
48
- ]),
49
- ),
50
- ]),
51
- ),
52
- );
53
-
54
- // addEventListener(event, e => text.set(e.target.prop))
55
- statements.push(
56
- t.expressionStatement(
57
- t.callExpression(
58
- t.memberExpression(t.identifier(varName), t.identifier("addEventListener")),
59
- [
60
- t.stringLiteral(event),
61
- t.arrowFunctionExpression(
62
- [t.identifier("e")],
63
- t.callExpression(
64
- t.memberExpression(signalId, t.identifier("set")),
65
- [
66
- t.memberExpression(
67
- t.memberExpression(t.identifier("e"), t.identifier("target")),
68
- t.identifier(prop),
69
- ),
70
- ],
71
- ),
72
- ),
73
- ],
74
- ),
75
- ),
76
- );
77
- }
78
-
79
- export function buildProxyBind(
80
- varName: string,
81
- attrName: string,
82
- attr: t.JSXAttribute,
83
- statements: t.Statement[],
84
- signal: t.MemberExpression, // already form().name
85
- ): void {
86
- const prop = attrName.slice(4).toLowerCase();
87
- const event = BIND_EVENT[prop] ?? "input";
88
-
89
- // createEffect(() => el.prop = form().name)
90
- statements.push(
91
- t.expressionStatement(
92
- t.callExpression(t.identifier("createEffect"), [
93
- t.arrowFunctionExpression(
94
- [],
95
- t.blockStatement([
96
- t.expressionStatement(
97
- t.assignmentExpression(
98
- "=",
99
- t.memberExpression(t.identifier(varName), t.identifier(prop)),
100
- signal,
101
- ),
102
- ),
103
- ]),
104
- ),
105
- ]),
106
- ),
107
- );
108
-
109
- // addEventListener(event, e => { form().name = e.target.prop })
110
- statements.push(
111
- t.expressionStatement(
112
- t.callExpression(
113
- t.memberExpression(t.identifier(varName), t.identifier("addEventListener")),
114
- [
115
- t.stringLiteral(event),
116
- t.arrowFunctionExpression(
117
- [t.identifier("e")],
118
- t.blockStatement([
119
- t.expressionStatement(
120
- t.assignmentExpression(
121
- "=",
122
- signal as unknown as t.LVal, // form().name — valid LVal
123
- t.memberExpression(
124
- t.memberExpression(t.identifier("e"), t.identifier("target")),
125
- t.identifier(prop),
126
- ),
127
- ),
128
- ),
129
- ]),
130
- ),
131
- ],
132
- ),
133
- ),
134
- );
135
- }
1
+ import { types as t } from "@babel/core";
2
+
3
+ const BIND_EVENT: Record<string, string> = {
4
+ value: "input",
5
+ checked: "change",
6
+ innerHTML: "input",
7
+ textContent: "input",
8
+ group: "change",
9
+ };
10
+
11
+ export function buildBind(
12
+ varName: string,
13
+ bindProp: string,
14
+ attr: t.JSXAttribute,
15
+ statements: t.Statement[],
16
+ ): void {
17
+ const signal = (attr.value as t.JSXExpressionContainer)
18
+ .expression as t.CallExpression; // already text()
19
+
20
+ const signalId = signal.callee as t.Identifier; // text — for .set()
21
+
22
+ const event = BIND_EVENT[bindProp] ?? "input";
23
+
24
+ // createEffect(() => _el.prop = text())
25
+ statements.push(
26
+ t.expressionStatement(
27
+ t.callExpression(t.identifier("createEffect"), [
28
+ t.arrowFunctionExpression(
29
+ [],
30
+ t.blockStatement([
31
+ t.expressionStatement(
32
+ t.assignmentExpression(
33
+ "=",
34
+ t.memberExpression(t.identifier(varName), t.identifier(bindProp)),
35
+ signal,
36
+ ),
37
+ ),
38
+ ]),
39
+ ),
40
+ ]),
41
+ ),
42
+ );
43
+
44
+ // addEventListener(event, e => text.set(e.target.prop))
45
+ statements.push(
46
+ t.expressionStatement(
47
+ t.callExpression(
48
+ t.memberExpression(t.identifier(varName), t.identifier("addEventListener")),
49
+ [
50
+ t.stringLiteral(event),
51
+ t.arrowFunctionExpression(
52
+ [t.identifier("e")],
53
+ t.callExpression(
54
+ t.memberExpression(signalId, t.identifier("set")),
55
+ [
56
+ t.memberExpression(
57
+ t.memberExpression(t.identifier("e"), t.identifier("target")),
58
+ t.identifier(bindProp),
59
+ ),
60
+ ],
61
+ ),
62
+ ),
63
+ ],
64
+ ),
65
+ ),
66
+ );
67
+ }
68
+
69
+ export function buildProxyBind(
70
+ varName: string,
71
+ bindProp: string,
72
+ attr: t.JSXAttribute,
73
+ statements: t.Statement[],
74
+ signal: t.MemberExpression, // already form().name
75
+ ): void {
76
+ const event = BIND_EVENT[bindProp] ?? "input";
77
+
78
+ // createEffect(() => el.prop = form().name)
79
+ statements.push(
80
+ t.expressionStatement(
81
+ t.callExpression(t.identifier("createEffect"), [
82
+ t.arrowFunctionExpression(
83
+ [],
84
+ t.blockStatement([
85
+ t.expressionStatement(
86
+ t.assignmentExpression(
87
+ "=",
88
+ t.memberExpression(t.identifier(varName), t.identifier(bindProp)),
89
+ signal,
90
+ ),
91
+ ),
92
+ ]),
93
+ ),
94
+ ]),
95
+ ),
96
+ );
97
+
98
+ // addEventListener(event, e => { form().name = e.target.prop })
99
+ statements.push(
100
+ t.expressionStatement(
101
+ t.callExpression(
102
+ t.memberExpression(t.identifier(varName), t.identifier("addEventListener")),
103
+ [
104
+ t.stringLiteral(event),
105
+ t.arrowFunctionExpression(
106
+ [t.identifier("e")],
107
+ t.blockStatement([
108
+ t.expressionStatement(
109
+ t.assignmentExpression(
110
+ "=",
111
+ signal as unknown as t.LVal, // form().name — valid LVal
112
+ t.memberExpression(
113
+ t.memberExpression(t.identifier("e"), t.identifier("target")),
114
+ t.identifier(bindProp),
115
+ ),
116
+ ),
117
+ ),
118
+ ]),
119
+ ),
120
+ ],
121
+ ),
122
+ ),
123
+ );
124
+ }
125
+
126
+