UNPKG

17.2 kBJavaScriptView Raw
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 "use strict";
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