forked from lumeland/pug
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathwalk.js
More file actions
116 lines (109 loc) · 3.07 KB
/
walk.js
File metadata and controls
116 lines (109 loc) · 3.07 KB
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
export default function walkAST(ast, before, after, options) {
if (after && typeof after === "object" && typeof options === "undefined") {
options = after;
after = null;
}
options = options || { includeDependencies: false };
var parents = (options.parents = options.parents || []);
var replace = function replace(replacement) {
if (Array.isArray(replacement) && !replace.arrayAllowed) {
throw new Error(
"replace() can only be called with an array if the last parent is a Block or NamedBlock",
);
}
ast = replacement;
};
replace.arrayAllowed = parents[0] &&
(/^(Named)?Block$/.test(parents[0].type) ||
(parents[0].type === "RawInclude" && ast.type === "IncludeFilter"));
if (before) {
var result = before(ast, replace);
if (result === false) {
return ast;
} else if (Array.isArray(ast)) {
// return right here to skip after() call on array
return walkAndMergeNodes(ast);
}
}
parents.unshift(ast);
switch (ast.type) {
case "NamedBlock":
case "Block":
ast.nodes = walkAndMergeNodes(ast.nodes);
break;
case "Case":
case "Filter":
case "Mixin":
case "Tag":
case "InterpolatedTag":
case "When":
case "Code":
case "While":
if (ast.block) {
ast.block = walkAST(ast.block, before, after, options);
}
break;
case "Each":
if (ast.block) {
ast.block = walkAST(ast.block, before, after, options);
}
if (ast.alternate) {
ast.alternate = walkAST(ast.alternate, before, after, options);
}
break;
case "EachOf":
if (ast.block) {
ast.block = walkAST(ast.block, before, after, options);
}
break;
case "Conditional":
if (ast.consequent) {
ast.consequent = walkAST(ast.consequent, before, after, options);
}
if (ast.alternate) {
ast.alternate = walkAST(ast.alternate, before, after, options);
}
break;
case "Include":
walkAST(ast.block, before, after, options);
walkAST(ast.file, before, after, options);
break;
case "Extends":
walkAST(ast.file, before, after, options);
break;
case "RawInclude":
ast.filters = walkAndMergeNodes(ast.filters);
walkAST(ast.file, before, after, options);
break;
case "Attrs":
case "BlockComment":
case "Comment":
case "Doctype":
case "IncludeFilter":
case "MixinBlock":
case "YieldBlock":
case "Text":
break;
case "FileReference":
if (options.includeDependencies && ast.ast) {
walkAST(ast.ast, before, after, options);
}
break;
default:
throw new Error("Unexpected node type " + ast.type);
break;
}
parents.shift();
after && after(ast, replace);
return ast;
function walkAndMergeNodes(nodes) {
return nodes.reduce(function (nodes, node) {
var result = walkAST(node, before, after, options);
if (Array.isArray(result)) {
return nodes.concat(result);
} else {
return nodes.concat([result]);
}
}, []);
}
}