@angular/language-service 9.0.0-rc.9 → 9.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@angular/language-service",
3
- "version": "9.0.0-rc.9",
3
+ "version": "9.0.0",
4
4
  "description": "Angular - language services",
5
5
  "main": "./bundles/language-service.umd.js",
6
6
  "module": "./fesm5/language-service.js",
@@ -11,11 +11,12 @@
11
11
  if (v !== undefined) module.exports = v;
12
12
  }
13
13
  else if (typeof define === "function" && define.amd) {
14
- define("@angular/language-service/src/definitions", ["require", "exports", "path", "typescript", "@angular/language-service/src/locate_symbol", "@angular/language-service/src/template", "@angular/language-service/src/utils"], factory);
14
+ define("@angular/language-service/src/definitions", ["require", "exports", "tslib", "path", "typescript", "@angular/language-service/src/locate_symbol", "@angular/language-service/src/template", "@angular/language-service/src/utils"], factory);
15
15
  }
16
16
  })(function (require, exports) {
17
17
  "use strict";
18
18
  Object.defineProperty(exports, "__esModule", { value: true });
19
+ var tslib_1 = require("tslib");
19
20
  var path = require("path");
20
21
  var ts = require("typescript"); // used as value and is provided at runtime
21
22
  var locate_symbol_1 = require("@angular/language-service/src/locate_symbol");
@@ -39,32 +40,68 @@
39
40
  * @param position
40
41
  */
41
42
  function getDefinitionAndBoundSpan(info, position) {
42
- var symbolInfo = locate_symbol_1.locateSymbol(info, position);
43
- if (!symbolInfo) {
43
+ var e_1, _a, e_2, _b;
44
+ var symbols = locate_symbol_1.locateSymbols(info, position);
45
+ if (!symbols.length) {
44
46
  return;
45
47
  }
46
- var textSpan = ngSpanToTsTextSpan(symbolInfo.span);
47
- var symbol = symbolInfo.symbol;
48
- var container = symbol.container, locations = symbol.definition;
49
- if (!locations || !locations.length) {
50
- // symbol.definition is really the locations of the symbol. There could be
51
- // more than one. No meaningful info could be provided without any location.
52
- return { textSpan: textSpan };
48
+ var seen = new Set();
49
+ var definitions = [];
50
+ try {
51
+ for (var symbols_1 = tslib_1.__values(symbols), symbols_1_1 = symbols_1.next(); !symbols_1_1.done; symbols_1_1 = symbols_1.next()) {
52
+ var symbolInfo = symbols_1_1.value;
53
+ var symbol = symbolInfo.symbol;
54
+ // symbol.definition is really the locations of the symbol. There could be
55
+ // more than one. No meaningful info could be provided without any location.
56
+ var kind = symbol.kind, name_1 = symbol.name, container = symbol.container, locations = symbol.definition;
57
+ if (!locations || !locations.length) {
58
+ continue;
59
+ }
60
+ var containerKind = container ? container.kind : ts.ScriptElementKind.unknown;
61
+ var containerName = container ? container.name : '';
62
+ try {
63
+ for (var locations_1 = (e_2 = void 0, tslib_1.__values(locations)), locations_1_1 = locations_1.next(); !locations_1_1.done; locations_1_1 = locations_1.next()) {
64
+ var _c = locations_1_1.value, fileName = _c.fileName, span = _c.span;
65
+ var textSpan = ngSpanToTsTextSpan(span);
66
+ // In cases like two-way bindings, a request for the definitions of an expression may return
67
+ // two of the same definition:
68
+ // [(ngModel)]="prop"
69
+ // ^^^^ -- one definition for the property binding, one for the event binding
70
+ // To prune duplicate definitions, tag definitions with unique location signatures and ignore
71
+ // definitions whose locations have already been seen.
72
+ var signature = textSpan.start + ":" + textSpan.length + "@" + fileName;
73
+ if (seen.has(signature))
74
+ continue;
75
+ definitions.push({
76
+ kind: kind,
77
+ name: name_1,
78
+ containerKind: containerKind,
79
+ containerName: containerName,
80
+ textSpan: ngSpanToTsTextSpan(span),
81
+ fileName: fileName,
82
+ });
83
+ seen.add(signature);
84
+ }
85
+ }
86
+ catch (e_2_1) { e_2 = { error: e_2_1 }; }
87
+ finally {
88
+ try {
89
+ if (locations_1_1 && !locations_1_1.done && (_b = locations_1.return)) _b.call(locations_1);
90
+ }
91
+ finally { if (e_2) throw e_2.error; }
92
+ }
93
+ }
94
+ }
95
+ catch (e_1_1) { e_1 = { error: e_1_1 }; }
96
+ finally {
97
+ try {
98
+ if (symbols_1_1 && !symbols_1_1.done && (_a = symbols_1.return)) _a.call(symbols_1);
99
+ }
100
+ finally { if (e_1) throw e_1.error; }
53
101
  }
54
- var containerKind = container ? container.kind : ts.ScriptElementKind.unknown;
55
- var containerName = container ? container.name : '';
56
- var definitions = locations.map(function (location) {
57
- return {
58
- kind: symbol.kind,
59
- name: symbol.name,
60
- containerKind: containerKind,
61
- containerName: containerName,
62
- textSpan: ngSpanToTsTextSpan(location.span),
63
- fileName: location.fileName,
64
- };
65
- });
66
102
  return {
67
- definitions: definitions, textSpan: textSpan,
103
+ definitions: definitions,
104
+ textSpan: symbols[0].span,
68
105
  };
69
106
  }
70
107
  exports.getDefinitionAndBoundSpan = getDefinitionAndBoundSpan;
@@ -142,4 +179,4 @@
142
179
  };
143
180
  }
144
181
  });
145
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/definitions.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;;;;;;;;;;;;IAEH,2BAA6B;IAC7B,+BAAiC,CAAC,2CAA2C;IAE7E,6EAA6C;IAC7C,mEAAoF;IAEpF,6DAAyC;IAEzC;;;;OAIG;IACH,SAAS,kBAAkB,CAAC,IAAU;QACpC,OAAO;YACL,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,MAAM,EAAE,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,KAAK;SAC9B,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACH,SAAgB,yBAAyB,CACrC,IAAe,EAAE,QAAgB;QACnC,IAAM,UAAU,GAAG,4BAAY,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAChD,IAAI,CAAC,UAAU,EAAE;YACf,OAAO;SACR;QACD,IAAM,QAAQ,GAAG,kBAAkB,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QAC9C,IAAA,0BAAM,CAAe;QACrB,IAAA,4BAAS,EAAE,6BAAqB,CAAW;QAClD,IAAI,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YACnC,0EAA0E;YAC1E,4EAA4E;YAC5E,OAAO,EAAC,QAAQ,UAAA,EAAC,CAAC;SACnB;QACD,IAAM,aAAa,GAAG,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,iBAAiB,CAAC,OAAO,CAAC;QAChF,IAAM,aAAa,GAAG,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;QACtD,IAAM,WAAW,GAAG,SAAS,CAAC,GAAG,CAAC,UAAC,QAAQ;YACzC,OAAO;gBACL,IAAI,EAAE,MAAM,CAAC,IAA4B;gBACzC,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,aAAa,EAAE,aAAqC;gBACpD,aAAa,EAAE,aAAa;gBAC5B,QAAQ,EAAE,kBAAkB,CAAC,QAAQ,CAAC,IAAI,CAAC;gBAC3C,QAAQ,EAAE,QAAQ,CAAC,QAAQ;aAC5B,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,OAAO;YACH,WAAW,aAAA,EAAE,QAAQ,UAAA;SACxB,CAAC;IACJ,CAAC;IA7BD,8DA6BC;IAED;;OAEG;IACH,SAAgB,2BAA2B,CACvC,EAAiB,EAAE,QAAgB,EACnC,QAA0C;QAC5C,IAAM,IAAI,GAAG,wBAAgB,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;QAC5C,IAAI,CAAC,IAAI;YAAE,OAAO;QAClB,QAAQ,IAAI,CAAC,IAAI,EAAE;YACjB,KAAK,EAAE,CAAC,UAAU,CAAC,aAAa,CAAC;YACjC,KAAK,EAAE,CAAC,UAAU,CAAC,6BAA6B;gBAC9C,mEAAmE;gBACnE,OAAO,kBAAkB,CAAC,IAA4B,EAAE,QAAQ,CAAC,CAAC;YACpE;gBACE,OAAO,SAAS,CAAC;SACpB;IACH,CAAC;IAbD,kEAaC;IAED;;;;OAIG;IACH,SAAS,kBAAkB,CACvB,OAA6B,EAC7B,QAA0C;QAC5C,iGAAiG;QACjG,6FAA6F;QAC7F,uBAAuB;QACvB,MAAM;QACN,4CAA4C;QAC5C,yDAAyD;QACzD,MAAM;QACN,iFAAiF;QACjF,+EAA+E;QAC/E,EAAE;QACF,kCAAkC;QAClC,IAAI,IAAI,GAAG,yCAA8B,CAAC,OAAO,CAAC,CAAC;QACnD,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,aAAa,EAAE;YAClD,+EAA+E;YAC/E,IAAI,GAAG,yCAA8B,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;YACtD,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,WAAW,EAAE;gBAChD,uBAAuB;gBACvB,OAAO;aACR;SACF;QAED,gGAAgG;QAChG,UAAU;QACV,IAAI,CAAC,mCAAwB,CAAC,IAAI,CAAC;YAAE,OAAO;QAE5C,IAAM,EAAE,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC;QACnC,8FAA8F;QAC9F,8BAA8B;QAC9B,IAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,QAAQ,CAAC,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;QAE/D,6FAA6F;QAC7F,4FAA4F;QAC5F,IAAI,QAAQ,CAAC,UAAU,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC;YAAE,OAAO;QAE7D,IAAM,mBAAmB,GAAwB,CAAC;gBAChD,IAAI,EAAE,EAAE,CAAC,iBAAiB,CAAC,kBAAkB;gBAC7C,IAAI,EAAE,GAAG;gBACT,aAAa,EAAE,EAAE,CAAC,iBAAiB,CAAC,OAAO;gBAC3C,aAAa,EAAE,EAAE;gBACjB,iEAAiE;gBACjE,QAAQ,EAAE,EAAC,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAC;gBAC/B,QAAQ,EAAE,GAAG;aACd,CAAC,CAAC;QAEH,OAAO;YACL,WAAW,EAAE,mBAAmB;YAChC,QAAQ,EAAE;gBACR,sDAAsD;gBACtD,KAAK,EAAE,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC;gBAC7B,MAAM,EAAE,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC;aAC/B;SACF,CAAC;IACJ,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 path from 'path';\nimport * as ts from 'typescript'; // used as value and is provided at runtime\nimport {AstResult} from './common';\nimport {locateSymbol} from './locate_symbol';\nimport {getPropertyAssignmentFromValue, isClassDecoratorProperty} from './template';\nimport {Span, TemplateSource} from './types';\nimport {findTightestNode} from './utils';\n\n/**\n * Convert Angular Span to TypeScript TextSpan. Angular Span has 'start' and\n * 'end' whereas TS TextSpan has 'start' and 'length'.\n * @param span Angular Span\n */\nfunction ngSpanToTsTextSpan(span: Span): ts.TextSpan {\n  return {\n    start: span.start,\n    length: span.end - span.start,\n  };\n}\n\n/**\n * Traverse the template AST and look for the symbol located at `position`, then\n * return its definition and span of bound text.\n * @param info\n * @param position\n */\nexport function getDefinitionAndBoundSpan(\n    info: AstResult, position: number): ts.DefinitionInfoAndBoundSpan|undefined {\n  const symbolInfo = locateSymbol(info, position);\n  if (!symbolInfo) {\n    return;\n  }\n  const textSpan = ngSpanToTsTextSpan(symbolInfo.span);\n  const {symbol} = symbolInfo;\n  const {container, definition: locations} = symbol;\n  if (!locations || !locations.length) {\n    // symbol.definition is really the locations of the symbol. There could be\n    // more than one. No meaningful info could be provided without any location.\n    return {textSpan};\n  }\n  const containerKind = container ? container.kind : ts.ScriptElementKind.unknown;\n  const containerName = container ? container.name : '';\n  const definitions = locations.map((location) => {\n    return {\n      kind: symbol.kind as ts.ScriptElementKind,\n      name: symbol.name,\n      containerKind: containerKind as ts.ScriptElementKind,\n      containerName: containerName,\n      textSpan: ngSpanToTsTextSpan(location.span),\n      fileName: location.fileName,\n    };\n  });\n  return {\n      definitions, textSpan,\n  };\n}\n\n/**\n * Gets an Angular-specific definition in a TypeScript source file.\n */\nexport function getTsDefinitionAndBoundSpan(\n    sf: ts.SourceFile, position: number,\n    tsLsHost: Readonly<ts.LanguageServiceHost>): ts.DefinitionInfoAndBoundSpan|undefined {\n  const node = findTightestNode(sf, position);\n  if (!node) return;\n  switch (node.kind) {\n    case ts.SyntaxKind.StringLiteral:\n    case ts.SyntaxKind.NoSubstitutionTemplateLiteral:\n      // Attempt to extract definition of a URL in a property assignment.\n      return getUrlFromProperty(node as ts.StringLiteralLike, tsLsHost);\n    default:\n      return undefined;\n  }\n}\n\n/**\n * Attempts to get the definition of a file whose URL is specified in a property assignment in a\n * directive decorator.\n * Currently applies to `templateUrl` and `styleUrls` properties.\n */\nfunction getUrlFromProperty(\n    urlNode: ts.StringLiteralLike,\n    tsLsHost: Readonly<ts.LanguageServiceHost>): ts.DefinitionInfoAndBoundSpan|undefined {\n  // Get the property assignment node corresponding to the `templateUrl` or `styleUrls` assignment.\n  // These assignments are specified differently; `templateUrl` is a string, and `styleUrls` is\n  // an array of strings:\n  //   {\n  //        templateUrl: './template.ng.html',\n  //        styleUrls: ['./style.css', './other-style.css']\n  //   }\n  // `templateUrl`'s property assignment can be found from the string literal node;\n  // `styleUrls`'s property assignment can be found from the array (parent) node.\n  //\n  // First search for `templateUrl`.\n  let asgn = getPropertyAssignmentFromValue(urlNode);\n  if (!asgn || asgn.name.getText() !== 'templateUrl') {\n    // `templateUrl` assignment not found; search for `styleUrls` array assignment.\n    asgn = getPropertyAssignmentFromValue(urlNode.parent);\n    if (!asgn || asgn.name.getText() !== 'styleUrls') {\n      // Nothing found, bail.\n      return;\n    }\n  }\n\n  // If the property assignment is not a property of a class decorator, don't generate definitions\n  // for it.\n  if (!isClassDecoratorProperty(asgn)) return;\n\n  const sf = urlNode.getSourceFile();\n  // Extract url path specified by the url node, which is relative to the TypeScript source file\n  // the url node is defined in.\n  const url = path.join(path.dirname(sf.fileName), urlNode.text);\n\n  // If the file does not exist, bail. It is possible that the TypeScript language service host\n  // does not have a `fileExists` method, in which case optimistically assume the file exists.\n  if (tsLsHost.fileExists && !tsLsHost.fileExists(url)) return;\n\n  const templateDefinitions: ts.DefinitionInfo[] = [{\n    kind: ts.ScriptElementKind.externalModuleName,\n    name: url,\n    containerKind: ts.ScriptElementKind.unknown,\n    containerName: '',\n    // Reading the template is expensive, so don't provide a preview.\n    textSpan: {start: 0, length: 0},\n    fileName: url,\n  }];\n\n  return {\n    definitions: templateDefinitions,\n    textSpan: {\n      // Exclude opening and closing quotes in the url span.\n      start: urlNode.getStart() + 1,\n      length: urlNode.getWidth() - 2,\n    },\n  };\n}\n"]}
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/definitions.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 path from 'path';\nimport * as ts from 'typescript'; // used as value and is provided at runtime\nimport {AstResult} from './common';\nimport {locateSymbols} from './locate_symbol';\nimport {getPropertyAssignmentFromValue, isClassDecoratorProperty} from './template';\nimport {Span} from './types';\nimport {findTightestNode} from './utils';\n\n/**\n * Convert Angular Span to TypeScript TextSpan. Angular Span has 'start' and\n * 'end' whereas TS TextSpan has 'start' and 'length'.\n * @param span Angular Span\n */\nfunction ngSpanToTsTextSpan(span: Span): ts.TextSpan {\n  return {\n    start: span.start,\n    length: span.end - span.start,\n  };\n}\n\n/**\n * Traverse the template AST and look for the symbol located at `position`, then\n * return its definition and span of bound text.\n * @param info\n * @param position\n */\nexport function getDefinitionAndBoundSpan(\n    info: AstResult, position: number): ts.DefinitionInfoAndBoundSpan|undefined {\n  const symbols = locateSymbols(info, position);\n  if (!symbols.length) {\n    return;\n  }\n\n  const seen = new Set<string>();\n  const definitions: ts.DefinitionInfo[] = [];\n  for (const symbolInfo of symbols) {\n    const {symbol} = symbolInfo;\n\n    // symbol.definition is really the locations of the symbol. There could be\n    // more than one. No meaningful info could be provided without any location.\n    const {kind, name, container, definition: locations} = symbol;\n    if (!locations || !locations.length) {\n      continue;\n    }\n\n    const containerKind =\n        container ? container.kind as ts.ScriptElementKind : ts.ScriptElementKind.unknown;\n    const containerName = container ? container.name : '';\n\n    for (const {fileName, span} of locations) {\n      const textSpan = ngSpanToTsTextSpan(span);\n      // In cases like two-way bindings, a request for the definitions of an expression may return\n      // two of the same definition:\n      //    [(ngModel)]=\"prop\"\n      //                 ^^^^  -- one definition for the property binding, one for the event binding\n      // To prune duplicate definitions, tag definitions with unique location signatures and ignore\n      // definitions whose locations have already been seen.\n      const signature = `${textSpan.start}:${textSpan.length}@${fileName}`;\n      if (seen.has(signature)) continue;\n\n      definitions.push({\n        kind: kind as ts.ScriptElementKind,\n        name,\n        containerKind,\n        containerName,\n        textSpan: ngSpanToTsTextSpan(span),\n        fileName: fileName,\n      });\n      seen.add(signature);\n    }\n  }\n\n  return {\n    definitions,\n    textSpan: symbols[0].span,\n  };\n}\n\n/**\n * Gets an Angular-specific definition in a TypeScript source file.\n */\nexport function getTsDefinitionAndBoundSpan(\n    sf: ts.SourceFile, position: number,\n    tsLsHost: Readonly<ts.LanguageServiceHost>): ts.DefinitionInfoAndBoundSpan|undefined {\n  const node = findTightestNode(sf, position);\n  if (!node) return;\n  switch (node.kind) {\n    case ts.SyntaxKind.StringLiteral:\n    case ts.SyntaxKind.NoSubstitutionTemplateLiteral:\n      // Attempt to extract definition of a URL in a property assignment.\n      return getUrlFromProperty(node as ts.StringLiteralLike, tsLsHost);\n    default:\n      return undefined;\n  }\n}\n\n/**\n * Attempts to get the definition of a file whose URL is specified in a property assignment in a\n * directive decorator.\n * Currently applies to `templateUrl` and `styleUrls` properties.\n */\nfunction getUrlFromProperty(\n    urlNode: ts.StringLiteralLike,\n    tsLsHost: Readonly<ts.LanguageServiceHost>): ts.DefinitionInfoAndBoundSpan|undefined {\n  // Get the property assignment node corresponding to the `templateUrl` or `styleUrls` assignment.\n  // These assignments are specified differently; `templateUrl` is a string, and `styleUrls` is\n  // an array of strings:\n  //   {\n  //        templateUrl: './template.ng.html',\n  //        styleUrls: ['./style.css', './other-style.css']\n  //   }\n  // `templateUrl`'s property assignment can be found from the string literal node;\n  // `styleUrls`'s property assignment can be found from the array (parent) node.\n  //\n  // First search for `templateUrl`.\n  let asgn = getPropertyAssignmentFromValue(urlNode);\n  if (!asgn || asgn.name.getText() !== 'templateUrl') {\n    // `templateUrl` assignment not found; search for `styleUrls` array assignment.\n    asgn = getPropertyAssignmentFromValue(urlNode.parent);\n    if (!asgn || asgn.name.getText() !== 'styleUrls') {\n      // Nothing found, bail.\n      return;\n    }\n  }\n\n  // If the property assignment is not a property of a class decorator, don't generate definitions\n  // for it.\n  if (!isClassDecoratorProperty(asgn)) return;\n\n  const sf = urlNode.getSourceFile();\n  // Extract url path specified by the url node, which is relative to the TypeScript source file\n  // the url node is defined in.\n  const url = path.join(path.dirname(sf.fileName), urlNode.text);\n\n  // If the file does not exist, bail. It is possible that the TypeScript language service host\n  // does not have a `fileExists` method, in which case optimistically assume the file exists.\n  if (tsLsHost.fileExists && !tsLsHost.fileExists(url)) return;\n\n  const templateDefinitions: ts.DefinitionInfo[] = [{\n    kind: ts.ScriptElementKind.externalModuleName,\n    name: url,\n    containerKind: ts.ScriptElementKind.unknown,\n    containerName: '',\n    // Reading the template is expensive, so don't provide a preview.\n    textSpan: {start: 0, length: 0},\n    fileName: url,\n  }];\n\n  return {\n    definitions: templateDefinitions,\n    textSpan: {\n      // Exclude opening and closing quotes in the url span.\n      start: urlNode.getStart() + 1,\n      length: urlNode.getWidth() - 2,\n    },\n  };\n}\n"]}
@@ -31,10 +31,3 @@ export declare function getDeclarationDiagnostics(declarations: ng.Declaration[]
31
31
  * @param file
32
32
  */
33
33
  export declare function ngDiagnosticToTsDiagnostic(d: ng.Diagnostic, file: ts.SourceFile | undefined): ts.Diagnostic;
34
- /**
35
- * Return elements filtered by unique span.
36
- * @param elements
37
- */
38
- export declare function uniqueBySpan<T extends {
39
- span: ng.Span;
40
- }>(elements: T[]): T[];
@@ -126,9 +126,6 @@
126
126
  }
127
127
  finally { if (e_4) throw e_4.error; }
128
128
  }
129
- if (!metadata) {
130
- continue; // declaration is not an Angular directive
131
- }
132
129
  if (metadata.isComponent) {
133
130
  if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {
134
131
  results.push({
@@ -260,38 +257,5 @@
260
257
  };
261
258
  }
262
259
  exports.ngDiagnosticToTsDiagnostic = ngDiagnosticToTsDiagnostic;
263
- /**
264
- * Return elements filtered by unique span.
265
- * @param elements
266
- */
267
- function uniqueBySpan(elements) {
268
- var e_5, _a;
269
- var result = [];
270
- var map = new Map();
271
- try {
272
- for (var elements_1 = tslib_1.__values(elements), elements_1_1 = elements_1.next(); !elements_1_1.done; elements_1_1 = elements_1.next()) {
273
- var element = elements_1_1.value;
274
- var span = element.span;
275
- var set = map.get(span.start);
276
- if (!set) {
277
- set = new Set();
278
- map.set(span.start, set);
279
- }
280
- if (!set.has(span.end)) {
281
- set.add(span.end);
282
- result.push(element);
283
- }
284
- }
285
- }
286
- catch (e_5_1) { e_5 = { error: e_5_1 }; }
287
- finally {
288
- try {
289
- if (elements_1_1 && !elements_1_1.done && (_a = elements_1.return)) _a.call(elements_1);
290
- }
291
- finally { if (e_5) throw e_5.error; }
292
- }
293
- return result;
294
- }
295
- exports.uniqueBySpan = uniqueBySpan;
296
260
  });
297
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/diagnostics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 {NgAnalyzedModules} from '@angular/compiler';\nimport * as path from 'path';\nimport * as ts from 'typescript';\n\nimport {AstResult} from './common';\nimport {getTemplateExpressionDiagnostics} from './expression_diagnostics';\nimport * as ng from './types';\nimport {TypeScriptServiceHost} from './typescript_host';\nimport {findPropertyValueOfType, findTightestNode, offsetSpan, spanOf} from './utils';\n\n\n/**\n * Return diagnostic information for the parsed AST of the template.\n * @param ast contains HTML and template AST\n */\nexport function getTemplateDiagnostics(ast: AstResult): ng.Diagnostic[] {\n  const {parseErrors, templateAst, htmlAst, template} = ast;\n  if (parseErrors && parseErrors.length) {\n    return parseErrors.map(e => {\n      return {\n        kind: ts.DiagnosticCategory.Error,\n        span: offsetSpan(spanOf(e.span), template.span.start),\n        message: e.msg,\n      };\n    });\n  }\n  return getTemplateExpressionDiagnostics({\n    templateAst: templateAst,\n    htmlAst: htmlAst,\n    offset: template.span.start,\n    query: template.query,\n    members: template.members,\n  });\n}\n\n/**\n * Generate an error message that indicates a directive is not part of any\n * NgModule.\n * @param name class name\n * @param isComponent true if directive is an Angular Component\n */\nfunction missingDirective(name: string, isComponent: boolean) {\n  const type = isComponent ? 'Component' : 'Directive';\n  return `${type} '${name}' is not included in a module and will not be ` +\n      'available inside a template. Consider adding it to a NgModule declaration.';\n}\n\n/**\n * Performs a variety diagnostics on directive declarations.\n *\n * @param declarations Angular directive declarations\n * @param modules NgModules in the project\n * @param host TypeScript service host used to perform TypeScript queries\n * @return diagnosed errors, if any\n */\nexport function getDeclarationDiagnostics(\n    declarations: ng.Declaration[], modules: NgAnalyzedModules,\n    host: Readonly<TypeScriptServiceHost>): ng.Diagnostic[] {\n  const directives = new Set<ng.StaticSymbol>();\n  for (const ngModule of modules.ngModules) {\n    for (const directive of ngModule.declaredDirectives) {\n      directives.add(directive.reference);\n    }\n  }\n\n  const results: ng.Diagnostic[] = [];\n\n  for (const declaration of declarations) {\n    const {errors, metadata, type, declarationSpan} = declaration;\n\n    const sf = host.getSourceFile(type.filePath);\n    if (!sf) {\n      host.error(`directive ${type.name} exists but has no source file`);\n      return [];\n    }\n    // TypeScript identifier of the directive declaration annotation (e.g. \"Component\" or\n    // \"Directive\") on a directive class.\n    const directiveIdentifier = findTightestNode(sf, declarationSpan.start);\n    if (!directiveIdentifier) {\n      host.error(`directive ${type.name} exists but has no identifier`);\n      return [];\n    }\n\n    for (const error of errors) {\n      results.push({\n        kind: ts.DiagnosticCategory.Error,\n        message: error.message,\n        span: error.span,\n      });\n    }\n    if (!metadata) {\n      continue;  // declaration is not an Angular directive\n    }\n    if (metadata.isComponent) {\n      if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {\n        results.push({\n          kind: ts.DiagnosticCategory.Suggestion,\n          message: missingDirective(type.name, metadata.isComponent),\n          span: declarationSpan,\n        });\n      }\n      const {template, templateUrl, styleUrls} = metadata.template !;\n      if (template === null && !templateUrl) {\n        results.push({\n          kind: ts.DiagnosticCategory.Error,\n          message: `Component '${type.name}' must have a template or templateUrl`,\n          span: declarationSpan,\n        });\n      } else if (templateUrl) {\n        if (template) {\n          results.push({\n            kind: ts.DiagnosticCategory.Error,\n            message: `Component '${type.name}' must not have both template and templateUrl`,\n            span: declarationSpan,\n          });\n        }\n\n        // Find templateUrl value from the directive call expression, which is the parent of the\n        // directive identifier.\n        //\n        // TODO: We should create an enum of the various properties a directive can have to use\n        // instead of string literals. We can then perform a mass migration of all literal usages.\n        const templateUrlNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);\n        if (!templateUrlNode) {\n          host.error(`templateUrl ${templateUrl} exists but its TypeScript node doesn't`);\n          return [];\n        }\n\n        results.push(...validateUrls([templateUrlNode], host.tsLsHost));\n      }\n\n      if (styleUrls.length > 0) {\n        // Find styleUrls value from the directive call expression, which is the parent of the\n        // directive identifier.\n        const styleUrlsNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);\n        if (!styleUrlsNode) {\n          host.error(`styleUrls property exists but its TypeScript node doesn't'`);\n          return [];\n        }\n\n        results.push(...validateUrls(styleUrlsNode.elements, host.tsLsHost));\n      }\n    } else if (!directives.has(declaration.type)) {\n      results.push({\n        kind: ts.DiagnosticCategory.Suggestion,\n        message: missingDirective(type.name, metadata.isComponent),\n        span: declarationSpan,\n      });\n    }\n  }\n\n  return results;\n}\n\n/**\n * Checks that URLs on a directive point to a valid file.\n * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other\n * checks.\n *\n * @param urls urls to check for validity\n * @param tsLsHost TS LS host used for querying filesystem information\n * @return diagnosed url errors, if any\n */\nfunction validateUrls(\n    urls: ArrayLike<ts.Expression>, tsLsHost: Readonly<ts.LanguageServiceHost>): ng.Diagnostic[] {\n  if (!tsLsHost.fileExists) {\n    return [];\n  }\n\n  const allErrors: ng.Diagnostic[] = [];\n  // TODO(ayazhafiz): most of this logic can be unified with the logic in\n  // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.\n  for (let i = 0; i < urls.length; ++i) {\n    const urlNode = urls[i];\n    if (!ts.isStringLiteralLike(urlNode)) {\n      // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be\n      // picked up by the TS Language Server.\n      continue;\n    }\n    const curPath = urlNode.getSourceFile().fileName;\n    const url = path.join(path.dirname(curPath), urlNode.text);\n    if (tsLsHost.fileExists(url)) continue;\n\n    allErrors.push({\n      kind: ts.DiagnosticCategory.Error,\n      message: `URL does not point to a valid file`,\n      // Exclude opening and closing quotes in the url span.\n      span: {start: urlNode.getStart() + 1, end: urlNode.end - 1},\n    });\n  }\n  return allErrors;\n}\n\n/**\n * Return a recursive data structure that chains diagnostic messages.\n * @param chain\n */\nfunction chainDiagnostics(chain: ng.DiagnosticMessageChain): ts.DiagnosticMessageChain {\n  return {\n    messageText: chain.message,\n    category: ts.DiagnosticCategory.Error,\n    code: 0,\n    next: chain.next ? chain.next.map(chainDiagnostics) : undefined\n  };\n}\n\n/**\n * Convert ng.Diagnostic to ts.Diagnostic.\n * @param d diagnostic\n * @param file\n */\nexport function ngDiagnosticToTsDiagnostic(\n    d: ng.Diagnostic, file: ts.SourceFile | undefined): ts.Diagnostic {\n  return {\n    file,\n    start: d.span.start,\n    length: d.span.end - d.span.start,\n    messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),\n    category: d.kind,\n    code: 0,\n    source: 'ng',\n  };\n}\n\n/**\n * Return elements filtered by unique span.\n * @param elements\n */\nexport function uniqueBySpan<T extends{span: ng.Span}>(elements: T[]): T[] {\n  const result: T[] = [];\n  const map = new Map<number, Set<number>>();\n  for (const element of elements) {\n    const {span} = element;\n    let set = map.get(span.start);\n    if (!set) {\n      set = new Set();\n      map.set(span.start, set);\n    }\n    if (!set.has(span.end)) {\n      set.add(span.end);\n      result.push(element);\n    }\n  }\n  return result;\n}\n"]}
261
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/diagnostics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 {NgAnalyzedModules} from '@angular/compiler';\nimport * as path from 'path';\nimport * as ts from 'typescript';\n\nimport {AstResult} from './common';\nimport {getTemplateExpressionDiagnostics} from './expression_diagnostics';\nimport * as ng from './types';\nimport {TypeScriptServiceHost} from './typescript_host';\nimport {findPropertyValueOfType, findTightestNode, offsetSpan, spanOf} from './utils';\n\n\n/**\n * Return diagnostic information for the parsed AST of the template.\n * @param ast contains HTML and template AST\n */\nexport function getTemplateDiagnostics(ast: AstResult): ng.Diagnostic[] {\n  const {parseErrors, templateAst, htmlAst, template} = ast;\n  if (parseErrors && parseErrors.length) {\n    return parseErrors.map(e => {\n      return {\n        kind: ts.DiagnosticCategory.Error,\n        span: offsetSpan(spanOf(e.span), template.span.start),\n        message: e.msg,\n      };\n    });\n  }\n  return getTemplateExpressionDiagnostics({\n    templateAst: templateAst,\n    htmlAst: htmlAst,\n    offset: template.span.start,\n    query: template.query,\n    members: template.members,\n  });\n}\n\n/**\n * Generate an error message that indicates a directive is not part of any\n * NgModule.\n * @param name class name\n * @param isComponent true if directive is an Angular Component\n */\nfunction missingDirective(name: string, isComponent: boolean) {\n  const type = isComponent ? 'Component' : 'Directive';\n  return `${type} '${name}' is not included in a module and will not be ` +\n      'available inside a template. Consider adding it to a NgModule declaration.';\n}\n\n/**\n * Performs a variety diagnostics on directive declarations.\n *\n * @param declarations Angular directive declarations\n * @param modules NgModules in the project\n * @param host TypeScript service host used to perform TypeScript queries\n * @return diagnosed errors, if any\n */\nexport function getDeclarationDiagnostics(\n    declarations: ng.Declaration[], modules: NgAnalyzedModules,\n    host: Readonly<TypeScriptServiceHost>): ng.Diagnostic[] {\n  const directives = new Set<ng.StaticSymbol>();\n  for (const ngModule of modules.ngModules) {\n    for (const directive of ngModule.declaredDirectives) {\n      directives.add(directive.reference);\n    }\n  }\n\n  const results: ng.Diagnostic[] = [];\n\n  for (const declaration of declarations) {\n    const {errors, metadata, type, declarationSpan} = declaration;\n\n    const sf = host.getSourceFile(type.filePath);\n    if (!sf) {\n      host.error(`directive ${type.name} exists but has no source file`);\n      return [];\n    }\n    // TypeScript identifier of the directive declaration annotation (e.g. \"Component\" or\n    // \"Directive\") on a directive class.\n    const directiveIdentifier = findTightestNode(sf, declarationSpan.start);\n    if (!directiveIdentifier) {\n      host.error(`directive ${type.name} exists but has no identifier`);\n      return [];\n    }\n\n    for (const error of errors) {\n      results.push({\n        kind: ts.DiagnosticCategory.Error,\n        message: error.message,\n        span: error.span,\n      });\n    }\n    if (metadata.isComponent) {\n      if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {\n        results.push({\n          kind: ts.DiagnosticCategory.Suggestion,\n          message: missingDirective(type.name, metadata.isComponent),\n          span: declarationSpan,\n        });\n      }\n      const {template, templateUrl, styleUrls} = metadata.template !;\n      if (template === null && !templateUrl) {\n        results.push({\n          kind: ts.DiagnosticCategory.Error,\n          message: `Component '${type.name}' must have a template or templateUrl`,\n          span: declarationSpan,\n        });\n      } else if (templateUrl) {\n        if (template) {\n          results.push({\n            kind: ts.DiagnosticCategory.Error,\n            message: `Component '${type.name}' must not have both template and templateUrl`,\n            span: declarationSpan,\n          });\n        }\n\n        // Find templateUrl value from the directive call expression, which is the parent of the\n        // directive identifier.\n        //\n        // TODO: We should create an enum of the various properties a directive can have to use\n        // instead of string literals. We can then perform a mass migration of all literal usages.\n        const templateUrlNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);\n        if (!templateUrlNode) {\n          host.error(`templateUrl ${templateUrl} exists but its TypeScript node doesn't`);\n          return [];\n        }\n\n        results.push(...validateUrls([templateUrlNode], host.tsLsHost));\n      }\n\n      if (styleUrls.length > 0) {\n        // Find styleUrls value from the directive call expression, which is the parent of the\n        // directive identifier.\n        const styleUrlsNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);\n        if (!styleUrlsNode) {\n          host.error(`styleUrls property exists but its TypeScript node doesn't'`);\n          return [];\n        }\n\n        results.push(...validateUrls(styleUrlsNode.elements, host.tsLsHost));\n      }\n    } else if (!directives.has(declaration.type)) {\n      results.push({\n        kind: ts.DiagnosticCategory.Suggestion,\n        message: missingDirective(type.name, metadata.isComponent),\n        span: declarationSpan,\n      });\n    }\n  }\n\n  return results;\n}\n\n/**\n * Checks that URLs on a directive point to a valid file.\n * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other\n * checks.\n *\n * @param urls urls to check for validity\n * @param tsLsHost TS LS host used for querying filesystem information\n * @return diagnosed url errors, if any\n */\nfunction validateUrls(\n    urls: ArrayLike<ts.Expression>, tsLsHost: Readonly<ts.LanguageServiceHost>): ng.Diagnostic[] {\n  if (!tsLsHost.fileExists) {\n    return [];\n  }\n\n  const allErrors: ng.Diagnostic[] = [];\n  // TODO(ayazhafiz): most of this logic can be unified with the logic in\n  // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.\n  for (let i = 0; i < urls.length; ++i) {\n    const urlNode = urls[i];\n    if (!ts.isStringLiteralLike(urlNode)) {\n      // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be\n      // picked up by the TS Language Server.\n      continue;\n    }\n    const curPath = urlNode.getSourceFile().fileName;\n    const url = path.join(path.dirname(curPath), urlNode.text);\n    if (tsLsHost.fileExists(url)) continue;\n\n    allErrors.push({\n      kind: ts.DiagnosticCategory.Error,\n      message: `URL does not point to a valid file`,\n      // Exclude opening and closing quotes in the url span.\n      span: {start: urlNode.getStart() + 1, end: urlNode.end - 1},\n    });\n  }\n  return allErrors;\n}\n\n/**\n * Return a recursive data structure that chains diagnostic messages.\n * @param chain\n */\nfunction chainDiagnostics(chain: ng.DiagnosticMessageChain): ts.DiagnosticMessageChain {\n  return {\n    messageText: chain.message,\n    category: ts.DiagnosticCategory.Error,\n    code: 0,\n    next: chain.next ? chain.next.map(chainDiagnostics) : undefined\n  };\n}\n\n/**\n * Convert ng.Diagnostic to ts.Diagnostic.\n * @param d diagnostic\n * @param file\n */\nexport function ngDiagnosticToTsDiagnostic(\n    d: ng.Diagnostic, file: ts.SourceFile | undefined): ts.Diagnostic {\n  return {\n    file,\n    start: d.span.start,\n    length: d.span.end - d.span.start,\n    messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),\n    category: d.kind,\n    code: 0,\n    source: 'ng',\n  };\n}\n"]}
@@ -6,8 +6,7 @@
6
6
  * found in the LICENSE file at https://angular.io/license
7
7
  */
8
8
  /// <amd-module name="@angular/language-service/src/expression_diagnostics" />
9
- import { AST, Node, TemplateAst, TemplateAstPath } from '@angular/compiler';
10
- import { ExpressionDiagnosticsContext, TypeDiagnostic } from './expression_type';
9
+ import { Node, TemplateAst, TemplateAstPath } from '@angular/compiler';
11
10
  import { SymbolQuery, SymbolTable } from './symbols';
12
11
  import { Diagnostic } from './types';
13
12
  export interface DiagnosticTemplateInfo {
@@ -19,5 +18,10 @@ export interface DiagnosticTemplateInfo {
19
18
  templateAst: TemplateAst[];
20
19
  }
21
20
  export declare function getTemplateExpressionDiagnostics(info: DiagnosticTemplateInfo): Diagnostic[];
22
- export declare function getExpressionDiagnostics(scope: SymbolTable, ast: AST, query: SymbolQuery, context?: ExpressionDiagnosticsContext): TypeDiagnostic[];
21
+ /**
22
+ * Returns the symbols available in a particular scope of a template.
23
+ * @param info parsed template information
24
+ * @param path path of template nodes narrowing to the context the expression scope should be
25
+ * derived for.
26
+ */
23
27
  export declare function getExpressionScope(info: DiagnosticTemplateInfo, path: TemplateAstPath): SymbolTable;