1 | /**
|
2 | * @license
|
3 | * Copyright Google LLC All Rights Reserved.
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style license that can be
|
6 | * found in the LICENSE file at https://angular.io/license
|
7 | */
|
8 | (function (factory) {
|
9 | if (typeof module === "object" && typeof module.exports === "object") {
|
10 | var v = factory(require, exports);
|
11 | if (v !== undefined) module.exports = v;
|
12 | }
|
13 | else if (typeof define === "function" && define.amd) {
|
14 | define("@angular/language-service/src/ts_utils", ["require", "exports", "tslib", "typescript/lib/tsserverlibrary"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.getClassDeclFromDecoratorProp = exports.getClassDeclOfInlineTemplateNode = exports.getPropertyAssignmentFromValue = exports.findTightestNode = exports.findPropertyValueOfType = exports.getDirectiveClassLike = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var ts = require("typescript/lib/tsserverlibrary");
|
22 | /**
|
23 | * Return metadata about `node` if it looks like an Angular directive class.
|
24 | * In this case, potential matches are `@NgModule`, `@Component`, `@Directive`,
|
25 | * `@Pipe`, etc.
|
26 | * These class declarations all share some common attributes, namely their
|
27 | * decorator takes exactly one parameter and the parameter must be an object
|
28 | * literal.
|
29 | *
|
30 | * For example,
|
31 | * v---------- `decoratorId`
|
32 | * @NgModule({ <
|
33 | * declarations: [], < classDecln-al
|
34 | * }) <
|
35 | * class AppModule {} <
|
36 | * ^----- `classId`
|
37 | *
|
38 | * @param node Potential node that represents an Angular directive.
|
39 | */
|
40 | function getDirectiveClassLike(node) {
|
41 | var e_1, _a;
|
42 | if (!ts.isClassDeclaration(node) || !node.name || !node.decorators) {
|
43 | return;
|
44 | }
|
45 | try {
|
46 | for (var _b = tslib_1.__values(node.decorators), _c = _b.next(); !_c.done; _c = _b.next()) {
|
47 | var d = _c.value;
|
48 | var expr = d.expression;
|
49 | if (!ts.isCallExpression(expr) || expr.arguments.length !== 1 ||
|
50 | !ts.isIdentifier(expr.expression)) {
|
51 | continue;
|
52 | }
|
53 | var arg = expr.arguments[0];
|
54 | if (ts.isObjectLiteralExpression(arg)) {
|
55 | return {
|
56 | decoratorId: expr.expression,
|
57 | classId: node.name,
|
58 | };
|
59 | }
|
60 | }
|
61 | }
|
62 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
63 | finally {
|
64 | try {
|
65 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
66 | }
|
67 | finally { if (e_1) throw e_1.error; }
|
68 | }
|
69 | }
|
70 | exports.getDirectiveClassLike = getDirectiveClassLike;
|
71 | /**
|
72 | * Finds the value of a property assignment that is nested in a TypeScript node and is of a certain
|
73 | * type T.
|
74 | *
|
75 | * @param startNode node to start searching for nested property assignment from
|
76 | * @param propName property assignment name
|
77 | * @param predicate function to verify that a node is of type T.
|
78 | * @return node property assignment value of type T, or undefined if none is found
|
79 | */
|
80 | function findPropertyValueOfType(startNode, propName, predicate) {
|
81 | if (ts.isPropertyAssignment(startNode) && startNode.name.getText() === propName) {
|
82 | var initializer = startNode.initializer;
|
83 | if (predicate(initializer))
|
84 | return initializer;
|
85 | }
|
86 | return startNode.forEachChild(function (c) { return findPropertyValueOfType(c, propName, predicate); });
|
87 | }
|
88 | exports.findPropertyValueOfType = findPropertyValueOfType;
|
89 | /**
|
90 | * Return the node that most tightly encompass the specified `position`.
|
91 | * @param node
|
92 | * @param position
|
93 | */
|
94 | function findTightestNode(node, position) {
|
95 | if (node.getStart() <= position && position < node.getEnd()) {
|
96 | return node.forEachChild(function (c) { return findTightestNode(c, position); }) || node;
|
97 | }
|
98 | }
|
99 | exports.findTightestNode = findTightestNode;
|
100 | /**
|
101 | * Returns a property assignment from the assignment value if the property name
|
102 | * matches the specified `key`, or `undefined` if there is no match.
|
103 | */
|
104 | function getPropertyAssignmentFromValue(value, key) {
|
105 | var propAssignment = value.parent;
|
106 | if (!propAssignment || !ts.isPropertyAssignment(propAssignment) ||
|
107 | propAssignment.name.getText() !== key) {
|
108 | return;
|
109 | }
|
110 | return propAssignment;
|
111 | }
|
112 | exports.getPropertyAssignmentFromValue = getPropertyAssignmentFromValue;
|
113 | /**
|
114 | * Given the node which is the string of the inline template for a component, returns the
|
115 | * `ts.ClassDeclaration` for the component.
|
116 | */
|
117 | function getClassDeclOfInlineTemplateNode(templateStringNode) {
|
118 | if (!ts.isStringLiteralLike(templateStringNode)) {
|
119 | return;
|
120 | }
|
121 | var tmplAsgn = getPropertyAssignmentFromValue(templateStringNode, 'template');
|
122 | if (!tmplAsgn) {
|
123 | return;
|
124 | }
|
125 | return getClassDeclFromDecoratorProp(tmplAsgn);
|
126 | }
|
127 | exports.getClassDeclOfInlineTemplateNode = getClassDeclOfInlineTemplateNode;
|
128 | /**
|
129 | * Given a decorator property assignment, return the ClassDeclaration node that corresponds to the
|
130 | * directive class the property applies to.
|
131 | * If the property assignment is not on a class decorator, no declaration is returned.
|
132 | *
|
133 | * For example,
|
134 | *
|
135 | * @Component({
|
136 | * template: '<div></div>'
|
137 | * ^^^^^^^^^^^^^^^^^^^^^^^---- property assignment
|
138 | * })
|
139 | * class AppComponent {}
|
140 | * ^---- class declaration node
|
141 | *
|
142 | * @param propAsgnNode property assignment
|
143 | */
|
144 | function getClassDeclFromDecoratorProp(propAsgnNode) {
|
145 | if (!propAsgnNode.parent || !ts.isObjectLiteralExpression(propAsgnNode.parent)) {
|
146 | return;
|
147 | }
|
148 | var objLitExprNode = propAsgnNode.parent;
|
149 | if (!objLitExprNode.parent || !ts.isCallExpression(objLitExprNode.parent)) {
|
150 | return;
|
151 | }
|
152 | var callExprNode = objLitExprNode.parent;
|
153 | if (!callExprNode.parent || !ts.isDecorator(callExprNode.parent)) {
|
154 | return;
|
155 | }
|
156 | var decorator = callExprNode.parent;
|
157 | if (!decorator.parent || !ts.isClassDeclaration(decorator.parent)) {
|
158 | return;
|
159 | }
|
160 | var classDeclNode = decorator.parent;
|
161 | return classDeclNode;
|
162 | }
|
163 | exports.getClassDeclFromDecoratorProp = getClassDeclFromDecoratorProp;
|
164 | });
|
165 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ts_utils.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/ts_utils.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as ts from 'typescript/lib/tsserverlibrary';\n\ninterface DirectiveClassLike {\n  decoratorId: ts.Identifier;  // decorator identifier, like @Component\n  classId: ts.Identifier;\n}\n\n/**\n * Return metadata about `node` if it looks like an Angular directive class.\n * In this case, potential matches are `@NgModule`, `@Component`, `@Directive`,\n * `@Pipe`, etc.\n * These class declarations all share some common attributes, namely their\n * decorator takes exactly one parameter and the parameter must be an object\n * literal.\n *\n * For example,\n *     v---------- `decoratorId`\n * @NgModule({           <\n *   declarations: [],   < classDecln-al\n * })                    <\n * class AppModule {}    <\n *          ^----- `classId`\n *\n * @param node Potential node that represents an Angular directive.\n */\nexport function getDirectiveClassLike(node: ts.Node): DirectiveClassLike|undefined {\n  if (!ts.isClassDeclaration(node) || !node.name || !node.decorators) {\n    return;\n  }\n  for (const d of node.decorators) {\n    const expr = d.expression;\n    if (!ts.isCallExpression(expr) || expr.arguments.length !== 1 ||\n        !ts.isIdentifier(expr.expression)) {\n      continue;\n    }\n    const arg = expr.arguments[0];\n    if (ts.isObjectLiteralExpression(arg)) {\n      return {\n        decoratorId: expr.expression,\n        classId: node.name,\n      };\n    }\n  }\n}\n\n/**\n * Finds the value of a property assignment that is nested in a TypeScript node and is of a certain\n * type T.\n *\n * @param startNode node to start searching for nested property assignment from\n * @param propName property assignment name\n * @param predicate function to verify that a node is of type T.\n * @return node property assignment value of type T, or undefined if none is found\n */\nexport function findPropertyValueOfType<T extends ts.Node>(\n    startNode: ts.Node, propName: string, predicate: (node: ts.Node) => node is T): T|undefined {\n  if (ts.isPropertyAssignment(startNode) && startNode.name.getText() === propName) {\n    const {initializer} = startNode;\n    if (predicate(initializer)) return initializer;\n  }\n  return startNode.forEachChild(c => findPropertyValueOfType(c, propName, predicate));\n}\n\n/**\n * Return the node that most tightly encompass the specified `position`.\n * @param node\n * @param position\n */\nexport function findTightestNode(node: ts.Node, position: number): ts.Node|undefined {\n  if (node.getStart() <= position && position < node.getEnd()) {\n    return node.forEachChild(c => findTightestNode(c, position)) || node;\n  }\n}\n\n/**\n * Returns a property assignment from the assignment value if the property name\n * matches the specified `key`, or `undefined` if there is no match.\n */\nexport function getPropertyAssignmentFromValue(value: ts.Node, key: string): ts.PropertyAssignment|\n    undefined {\n  const propAssignment = value.parent;\n  if (!propAssignment || !ts.isPropertyAssignment(propAssignment) ||\n      propAssignment.name.getText() !== key) {\n    return;\n  }\n  return propAssignment;\n}\n\n/**\n * Given the node which is the string of the inline template for a component, returns the\n * `ts.ClassDeclaration` for the component.\n */\nexport function getClassDeclOfInlineTemplateNode(templateStringNode: ts.Node): ts.ClassDeclaration|\n    undefined {\n  if (!ts.isStringLiteralLike(templateStringNode)) {\n    return;\n  }\n  const tmplAsgn = getPropertyAssignmentFromValue(templateStringNode, 'template');\n  if (!tmplAsgn) {\n    return;\n  }\n  return getClassDeclFromDecoratorProp(tmplAsgn);\n}\n\n/**\n * Given a decorator property assignment, return the ClassDeclaration node that corresponds to the\n * directive class the property applies to.\n * If the property assignment is not on a class decorator, no declaration is returned.\n *\n * For example,\n *\n * @Component({\n *   template: '<div></div>'\n *   ^^^^^^^^^^^^^^^^^^^^^^^---- property assignment\n * })\n * class AppComponent {}\n *           ^---- class declaration node\n *\n * @param propAsgnNode property assignment\n */\nexport function getClassDeclFromDecoratorProp(propAsgnNode: ts.PropertyAssignment):\n    ts.ClassDeclaration|undefined {\n  if (!propAsgnNode.parent || !ts.isObjectLiteralExpression(propAsgnNode.parent)) {\n    return;\n  }\n  const objLitExprNode = propAsgnNode.parent;\n  if (!objLitExprNode.parent || !ts.isCallExpression(objLitExprNode.parent)) {\n    return;\n  }\n  const callExprNode = objLitExprNode.parent;\n  if (!callExprNode.parent || !ts.isDecorator(callExprNode.parent)) {\n    return;\n  }\n  const decorator = callExprNode.parent;\n  if (!decorator.parent || !ts.isClassDeclaration(decorator.parent)) {\n    return;\n  }\n  const classDeclNode = decorator.parent;\n  return classDeclNode;\n}\n"]} |
\ | No newline at end of file |