|
1 |
| -import type { PresentArray } from '@glimmer/interfaces'; |
2 |
| -import type { SourceSlice } from '@glimmer/syntax'; |
3 |
| -import { ASTv2, generateSyntaxError } from '@glimmer/syntax'; |
4 |
| -import { unreachable } from '@glimmer/util'; |
5 |
| - |
6 |
| -export type HasPath<Node extends ASTv2.CallNode = ASTv2.CallNode> = Node & { |
7 |
| - head: ASTv2.PathExpression; |
8 |
| -}; |
9 |
| - |
10 |
| -export type HasArguments = |
11 |
| - | { |
12 |
| - params: PresentArray<ASTv2.ExpressionNode>; |
13 |
| - } |
14 |
| - | { |
15 |
| - hash: { |
16 |
| - pairs: PresentArray<ASTv2.NamedArgument>; |
17 |
| - }; |
18 |
| - }; |
19 |
| - |
20 |
| -export type HelperInvocation<Node extends ASTv2.CallNode = ASTv2.CallNode> = HasPath<Node> & |
21 |
| - HasArguments; |
22 |
| - |
23 |
| -export function hasPath<N extends ASTv2.CallNode>(node: N): node is HasPath<N> { |
24 |
| - return node.callee.type === 'Path'; |
25 |
| -} |
26 |
| - |
27 |
| -export function isHelperInvocation<N extends ASTv2.CallNode>( |
28 |
| - node: ASTv2.CallNode |
29 |
| -): node is HelperInvocation<N> { |
30 |
| - if (!hasPath(node)) { |
31 |
| - return false; |
32 |
| - } |
33 |
| - |
34 |
| - return !node.args.isEmpty(); |
35 |
| -} |
36 |
| - |
37 |
| -export interface SimplePath extends ASTv2.PathExpression { |
38 |
| - tail: [SourceSlice]; |
39 |
| - data: false; |
40 |
| - this: false; |
41 |
| -} |
42 |
| - |
43 |
| -export type SimpleHelper<N extends HasPath> = N & { |
44 |
| - path: SimplePath; |
45 |
| -}; |
46 |
| - |
47 |
| -export function isSimplePath(path: ASTv2.ExpressionNode): path is SimplePath { |
48 |
| - if (path.type === 'Path') { |
49 |
| - let { ref: head, tail: parts } = path; |
50 |
| - |
51 |
| - return head.type === 'Free' && !ASTv2.isStrictResolution(head.resolution) && parts.length === 0; |
52 |
| - } else { |
53 |
| - return false; |
54 |
| - } |
55 |
| -} |
56 |
| - |
57 |
| -export function isStrictHelper(expr: HasPath): boolean { |
58 |
| - if (expr.callee.type !== 'Path') { |
59 |
| - return true; |
60 |
| - } |
61 |
| - |
62 |
| - if (expr.callee.ref.type !== 'Free') { |
63 |
| - return true; |
64 |
| - } |
65 |
| - |
66 |
| - return ASTv2.isStrictResolution(expr.callee.ref.resolution); |
67 |
| -} |
68 |
| - |
69 |
| -export function assertIsValidModifier<N extends HasPath>( |
70 |
| - helper: N |
71 |
| -): asserts helper is SimpleHelper<N> { |
72 |
| - if (isStrictHelper(helper) || isSimplePath(helper.callee)) { |
73 |
| - return; |
74 |
| - } |
75 |
| - |
76 |
| - throw generateSyntaxError( |
77 |
| - `\`${printPath(helper.callee)}\` is not a valid name for a modifier`, |
78 |
| - helper.loc |
79 |
| - ); |
80 |
| -} |
81 |
| - |
82 |
| -function printPath(path: ASTv2.ExpressionNode): string { |
83 |
| - switch (path.type) { |
84 |
| - case 'Literal': |
85 |
| - return JSON.stringify(path.value); |
86 |
| - case 'Path': { |
87 |
| - let printedPath = [printPathHead(path.ref)]; |
88 |
| - printedPath.push(...path.tail.map((t) => t.chars)); |
89 |
| - return printedPath.join('.'); |
90 |
| - } |
91 |
| - case 'Call': |
92 |
| - return `(${printPath(path.callee)} ...)`; |
93 |
| - case 'Interpolate': |
94 |
| - throw unreachable('a concat statement cannot appear as the head of an expression'); |
95 |
| - } |
96 |
| -} |
97 |
| - |
98 |
| -function printPathHead(head: ASTv2.VariableReference): string { |
99 |
| - switch (head.type) { |
100 |
| - case 'Arg': |
101 |
| - return head.name.chars; |
102 |
| - case 'Free': |
103 |
| - case 'Local': |
104 |
| - return head.name; |
105 |
| - case 'This': |
106 |
| - return 'this'; |
107 |
| - } |
108 |
| -} |
| 1 | +import type { ASTv2 } from '@glimmer/syntax'; |
109 | 2 |
|
110 | 3 | /**
|
111 | 4 | * This function is checking whether an AST node is a triple-curly, which means that it's
|
|
0 commit comments