-
Notifications
You must be signed in to change notification settings - Fork 4
/
index.js
85 lines (75 loc) · 3.01 KB
/
index.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
"use strict"
const privateClassElements = require("acorn-private-class-elements")
module.exports = function(Parser) {
const ExtendedParser = privateClassElements(Parser)
const acorn = Parser.acorn || require("acorn")
const tt = acorn.tokTypes
return class extends ExtendedParser {
_maybeParseFieldValue(field) {
if (this.eat(tt.eq)) {
const oldInFieldValue = this._inStaticFieldScope
this._inStaticFieldScope = this.currentThisScope()
field.value = this.parseExpression()
this._inStaticFieldScope = oldInFieldValue
} else field.value = null
}
// Parse fields
parseClassElement(_constructorAllowsSuper) {
if (this.options.ecmaVersion < 8 || !this.isContextual("static")) {
return super.parseClassElement.apply(this, arguments)
}
const branch = this._branch()
branch.next()
if ([tt.name, tt.bracketL, tt.string, tt.num, this.privateIdentifierToken].indexOf(branch.type) == -1 && !branch.type.keyword) {
return super.parseClassElement.apply(this, arguments)
}
if (branch.type == tt.bracketL) {
let count = 0
do {
if (branch.eat(tt.bracketL)) ++count
else if (branch.eat(tt.bracketR)) --count
else branch.next()
} while (count > 0)
} else branch.next()
if (branch.type != tt.eq && !branch.canInsertSemicolon() && branch.type != tt.semi) {
return super.parseClassElement.apply(this, arguments)
}
const node = this.startNode()
node.static = this.eatContextual("static")
if (this.type == this.privateIdentifierToken) {
this.parsePrivateClassElementName(node)
} else {
this.parsePropertyName(node)
}
if ((node.key.type === "Identifier" && node.key.name === "constructor") ||
(node.key.type === "Literal" && !node.computed && node.key.value === "constructor")) {
this.raise(node.key.start, "Classes may not have a field called constructor")
}
if ((node.key.name || node.key.value) === "prototype" && !node.computed) {
this.raise(node.key.start, "Classes may not have a static property named prototype")
}
this.enterScope(64 | 2 | 1) // See acorn's scopeflags.js
this._maybeParseFieldValue(node)
this.exitScope()
this.finishNode(node, "PropertyDefinition")
this.semicolon()
return node
}
// Parse private static methods
parsePropertyName(prop) {
if (prop.static && this.type == this.privateIdentifierToken) {
this.parsePrivateClassElementName(prop)
} else {
super.parsePropertyName(prop)
}
}
// Prohibit arguments in class field initializers
parseIdent(liberal, isBinding) {
const ident = super.parseIdent(liberal, isBinding)
if (this._inStaticFieldScope && this.currentThisScope() === this._inStaticFieldScope && ident.name == "arguments") {
this.raise(ident.start, "A static class field initializer may not contain arguments")
}
return ident
}
}
}