-
-
Notifications
You must be signed in to change notification settings - Fork 1.4k
/
xast.test.js
120 lines (113 loc) · 2.61 KB
/
xast.test.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/**
* @typedef {import('./types.js').XastElement} XastElement
* @typedef {import('./types.js').XastRoot} XastRoot
*/
import { visit, visitSkip, detachNodeFromParent } from './xast.js';
/**
* @type {(children: XastElement[]) => XastRoot}
*/
const root = (children) => {
return { type: 'root', children };
};
/**
* @type {(
* name: string,
* attrs?: ?Record<string, string>,
* children?: XastElement[]
* ) => XastElement}
*/
const x = (name, attrs = null, children = []) => {
return { type: 'element', name, attributes: attrs || {}, children };
};
test('visit enters into nodes', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
root: {
enter: (node) => {
entered.push(node.type);
},
},
element: {
enter: (node) => {
entered.push(`${node.type}:${node.name}`);
},
},
});
expect(entered).toStrictEqual([
'root',
'element:g',
'element:rect',
'element:circle',
'element:ellipse',
]);
});
test('visit exits from nodes', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const exited = [];
visit(ast, {
root: {
exit: (node) => {
exited.push(node.type);
},
},
element: {
exit: (node) => {
exited.push(`${node.type}:${node.name}`);
},
},
});
expect(exited).toStrictEqual([
'element:rect',
'element:circle',
'element:g',
'element:ellipse',
'root',
]);
});
test('visit skips entering children if node is detached', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
element: {
enter: (node, parentNode) => {
entered.push(node.name);
if (node.name === 'g') {
detachNodeFromParent(node, parentNode);
}
},
},
});
expect(entered).toStrictEqual(['g', 'ellipse']);
expect(ast).toStrictEqual(root([x('ellipse')]));
});
test('visit skips entering children when symbol is passed', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
element: {
enter: (node) => {
entered.push(node.name);
if (node.name === 'g') {
return visitSkip;
}
},
},
});
expect(entered).toStrictEqual(['g', 'ellipse']);
expect(ast).toStrictEqual(
root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]),
);
});