UNPKG

15.5 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.findPropertyValueOfType = exports.getDirectiveClassLike = exports.getClassDeclFromDecoratorProp = exports.getPropertyAssignmentFromValue = exports.findTightestNode = void 0;
20 var tslib_1 = require("tslib");
21 var ts = require("typescript/lib/tsserverlibrary");
22 /**
23 * Return the node that most tightly encompass the specified `position`.
24 * @param node
25 * @param position
26 */
27 function findTightestNode(node, position) {
28 if (node.getStart() <= position && position < node.getEnd()) {
29 return node.forEachChild(function (c) { return findTightestNode(c, position); }) || node;
30 }
31 }
32 exports.findTightestNode = findTightestNode;
33 /**
34 * Returns a property assignment from the assignment value if the property name
35 * matches the specified `key`, or `undefined` if there is no match.
36 */
37 function getPropertyAssignmentFromValue(value, key) {
38 var propAssignment = value.parent;
39 if (!propAssignment || !ts.isPropertyAssignment(propAssignment) ||
40 propAssignment.name.getText() !== key) {
41 return;
42 }
43 return propAssignment;
44 }
45 exports.getPropertyAssignmentFromValue = getPropertyAssignmentFromValue;
46 /**
47 * Given a decorator property assignment, return the ClassDeclaration node that corresponds to the
48 * directive class the property applies to.
49 * If the property assignment is not on a class decorator, no declaration is returned.
50 *
51 * For example,
52 *
53 * @Component({
54 * template: '<div></div>'
55 * ^^^^^^^^^^^^^^^^^^^^^^^---- property assignment
56 * })
57 * class AppComponent {}
58 * ^---- class declaration node
59 *
60 * @param propAsgn property assignment
61 */
62 function getClassDeclFromDecoratorProp(propAsgnNode) {
63 if (!propAsgnNode.parent || !ts.isObjectLiteralExpression(propAsgnNode.parent)) {
64 return;
65 }
66 var objLitExprNode = propAsgnNode.parent;
67 if (!objLitExprNode.parent || !ts.isCallExpression(objLitExprNode.parent)) {
68 return;
69 }
70 var callExprNode = objLitExprNode.parent;
71 if (!callExprNode.parent || !ts.isDecorator(callExprNode.parent)) {
72 return;
73 }
74 var decorator = callExprNode.parent;
75 if (!decorator.parent || !ts.isClassDeclaration(decorator.parent)) {
76 return;
77 }
78 var classDeclNode = decorator.parent;
79 return classDeclNode;
80 }
81 exports.getClassDeclFromDecoratorProp = getClassDeclFromDecoratorProp;
82 /**
83 * Return metadata about `node` if it looks like an Angular directive class.
84 * In this case, potential matches are `@NgModule`, `@Component`, `@Directive`,
85 * `@Pipe`, etc.
86 * These class declarations all share some common attributes, namely their
87 * decorator takes exactly one parameter and the parameter must be an object
88 * literal.
89 *
90 * For example,
91 * v---------- `decoratorId`
92 * @NgModule({ <
93 * declarations: [], < classDecln-al
94 * }) <
95 * class AppModule {} <
96 * ^----- `classId`
97 *
98 * @param node Potential node that represents an Angular directive.
99 */
100 function getDirectiveClassLike(node) {
101 var e_1, _a;
102 if (!ts.isClassDeclaration(node) || !node.name || !node.decorators) {
103 return;
104 }
105 try {
106 for (var _b = tslib_1.__values(node.decorators), _c = _b.next(); !_c.done; _c = _b.next()) {
107 var d = _c.value;
108 var expr = d.expression;
109 if (!ts.isCallExpression(expr) || expr.arguments.length !== 1 ||
110 !ts.isIdentifier(expr.expression)) {
111 continue;
112 }
113 var arg = expr.arguments[0];
114 if (ts.isObjectLiteralExpression(arg)) {
115 return {
116 decoratorId: expr.expression,
117 classId: node.name,
118 };
119 }
120 }
121 }
122 catch (e_1_1) { e_1 = { error: e_1_1 }; }
123 finally {
124 try {
125 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
126 }
127 finally { if (e_1) throw e_1.error; }
128 }
129 }
130 exports.getDirectiveClassLike = getDirectiveClassLike;
131 /**
132 * Finds the value of a property assignment that is nested in a TypeScript node and is of a certain
133 * type T.
134 *
135 * @param startNode node to start searching for nested property assignment from
136 * @param propName property assignment name
137 * @param predicate function to verify that a node is of type T.
138 * @return node property assignment value of type T, or undefined if none is found
139 */
140 function findPropertyValueOfType(startNode, propName, predicate) {
141 if (ts.isPropertyAssignment(startNode) && startNode.name.getText() === propName) {
142 var initializer = startNode.initializer;
143 if (predicate(initializer))
144 return initializer;
145 }
146 return startNode.forEachChild(function (c) { return findPropertyValueOfType(c, propName, predicate); });
147 }
148 exports.findPropertyValueOfType = findPropertyValueOfType;
149});
150//# 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\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 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 propAsgn 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\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"]}
\No newline at end of file