-
Notifications
You must be signed in to change notification settings - Fork 35
/
Copy pathArrowFunctionExpression.ts
109 lines (98 loc) · 3.27 KB
/
ArrowFunctionExpression.ts
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
import type { HasEffectsContext, InclusionContext } from '../ExecutionContext';
import type { NodeInteraction } from '../NodeInteractions';
import { INTERACTION_CALLED } from '../NodeInteractions';
import type ChildScope from '../scopes/ChildScope';
import ReturnValueScope from '../scopes/ReturnValueScope';
import { type ObjectPath, UNKNOWN_PATH } from '../utils/PathTracker';
import type BlockStatement from './BlockStatement';
import type CallExpression from './CallExpression';
import Identifier from './Identifier';
import * as NodeType from './NodeType';
import { Flag, isFlagSet, setFlag } from './shared/BitFlags';
import FunctionBase from './shared/FunctionBase';
import type { ExpressionNode, IncludeChildren } from './shared/Node';
import { ObjectEntity } from './shared/ObjectEntity';
import { OBJECT_PROTOTYPE } from './shared/ObjectPrototype';
import type { DeclarationPatternNode } from './shared/Pattern';
export default class ArrowFunctionExpression extends FunctionBase {
declare body: BlockStatement | ExpressionNode;
declare params: DeclarationPatternNode[];
declare preventChildBlockScope: true;
declare scope: ReturnValueScope;
declare type: NodeType.tArrowFunctionExpression;
protected objectEntity: ObjectEntity | null = null;
get expression(): boolean {
return isFlagSet(this.flags, Flag.expression);
}
set expression(value: boolean) {
this.flags = setFlag(this.flags, Flag.expression, value);
}
createScope(parentScope: ChildScope): void {
this.scope = new ReturnValueScope(parentScope, false);
}
hasEffects(): boolean {
return false;
}
hasEffectsOnInteractionAtPath(
path: ObjectPath,
interaction: NodeInteraction,
context: HasEffectsContext
): boolean {
if (
this.annotationNoSideEffects &&
path.length === 0 &&
interaction.type === INTERACTION_CALLED
) {
return false;
}
if (super.hasEffectsOnInteractionAtPath(path, interaction, context)) {
return true;
}
if (interaction.type === INTERACTION_CALLED) {
if (this.annotationNoSideEffects) {
return false;
}
const { ignore, brokenFlow } = context;
context.ignore = {
breaks: false,
continues: false,
labels: new Set(),
returnYield: true,
this: false
};
if (this.body.hasEffects(context)) return true;
context.ignore = ignore;
context.brokenFlow = brokenFlow;
}
return false;
}
protected onlyFunctionCallUsed(): boolean {
const isIIFE =
this.parent.type === NodeType.CallExpression &&
(this.parent as CallExpression).callee === this;
return isIIFE || super.onlyFunctionCallUsed();
}
include(context: InclusionContext, includeChildrenRecursively: IncludeChildren): void {
super.include(context, includeChildrenRecursively);
for (const parameter of this.params) {
if (!(parameter instanceof Identifier)) {
parameter.include(context, includeChildrenRecursively);
}
}
}
includeNode(context: InclusionContext) {
this.included = true;
this.body.includePath(UNKNOWN_PATH, context);
for (const parameter of this.params) {
if (!(parameter instanceof Identifier)) {
parameter.includePath(UNKNOWN_PATH, context);
}
}
}
protected getObjectEntity(): ObjectEntity {
if (this.objectEntity !== null) {
return this.objectEntity;
}
return (this.objectEntity = new ObjectEntity([], OBJECT_PROTOTYPE));
}
}