UNPKG

19.8 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_plugin", ["require", "exports", "tslib", "@angular/language-service/src/language_service", "@angular/language-service/src/typescript_host"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.create = exports.getExternalFiles = void 0;
20 var tslib_1 = require("tslib");
21 var language_service_1 = require("@angular/language-service/src/language_service");
22 var typescript_host_1 = require("@angular/language-service/src/typescript_host");
23 // Use a WeakMap to keep track of Project to Host mapping so that when Project
24 // is deleted Host could be garbage collected.
25 var PROJECT_MAP = new WeakMap();
26 /**
27 * This function is called by tsserver to retrieve the external (non-TS) files
28 * that should belong to the specified `project`. For Angular, these files are
29 * external templates. This is called once when the project is loaded, then
30 * every time when the program is updated.
31 * @param project Project for which external files should be retrieved.
32 */
33 function getExternalFiles(project) {
34 if (!project.hasRoots()) {
35 // During project initialization where there is no root files yet we should
36 // not do any work.
37 return [];
38 }
39 var ngLsHost = PROJECT_MAP.get(project);
40 if (ngLsHost === undefined) {
41 return [];
42 }
43 ngLsHost.getAnalyzedModules();
44 return ngLsHost.getExternalTemplates().filter(function (fileName) {
45 // TODO(kyliau): Remove this when the following PR lands on the version of
46 // TypeScript used in this repo.
47 // https://github.com/microsoft/TypeScript/pull/41737
48 return project.fileExists(fileName);
49 });
50 }
51 exports.getExternalFiles = getExternalFiles;
52 function create(info) {
53 var tsLS = info.languageService, tsLSHost = info.languageServiceHost, config = info.config, project = info.project;
54 // This plugin could operate under two different modes:
55 // 1. TS + Angular
56 // Plugin augments TS language service to provide additional Angular
57 // information. This only works with inline templates and is meant to be
58 // used as a local plugin (configured via tsconfig.json)
59 // 2. Angular only
60 // Plugin only provides information on Angular templates, no TS info at all.
61 // This effectively disables native TS features and is meant for internal
62 // use only.
63 var angularOnly = config ? config.angularOnly === true : false;
64 var ngLSHost = new typescript_host_1.TypeScriptServiceHost(tsLSHost, tsLS);
65 var ngLS = language_service_1.createLanguageService(ngLSHost);
66 PROJECT_MAP.set(project, ngLSHost);
67 function getCompletionsAtPosition(fileName, position, options) {
68 if (!angularOnly) {
69 var results = tsLS.getCompletionsAtPosition(fileName, position, options);
70 if (results && results.entries.length) {
71 // If TS could answer the query, then return results immediately.
72 return results;
73 }
74 }
75 return ngLS.getCompletionsAtPosition(fileName, position, options);
76 }
77 function getQuickInfoAtPosition(fileName, position) {
78 if (!angularOnly) {
79 var result = tsLS.getQuickInfoAtPosition(fileName, position);
80 if (result) {
81 // If TS could answer the query, then return results immediately.
82 return result;
83 }
84 }
85 return ngLS.getQuickInfoAtPosition(fileName, position);
86 }
87 function getSemanticDiagnostics(fileName) {
88 var results = [];
89 if (!angularOnly) {
90 results.push.apply(results, tslib_1.__spread(tsLS.getSemanticDiagnostics(fileName)));
91 }
92 // For semantic diagnostics we need to combine both TS + Angular results
93 results.push.apply(results, tslib_1.__spread(ngLS.getSemanticDiagnostics(fileName)));
94 return results;
95 }
96 function getDefinitionAtPosition(fileName, position) {
97 if (!angularOnly) {
98 var results = tsLS.getDefinitionAtPosition(fileName, position);
99 if (results) {
100 // If TS could answer the query, then return results immediately.
101 return results;
102 }
103 }
104 var result = ngLS.getDefinitionAndBoundSpan(fileName, position);
105 if (!result || !result.definitions || !result.definitions.length) {
106 return;
107 }
108 return result.definitions;
109 }
110 function getDefinitionAndBoundSpan(fileName, position) {
111 if (!angularOnly) {
112 var result = tsLS.getDefinitionAndBoundSpan(fileName, position);
113 if (result) {
114 // If TS could answer the query, then return results immediately.
115 return result;
116 }
117 }
118 return ngLS.getDefinitionAndBoundSpan(fileName, position);
119 }
120 function getTypeDefinitionAtPosition(fileName, position) {
121 // Not implemented in VE Language Service
122 return undefined;
123 }
124 function getReferencesAtPosition(fileName, position) {
125 // Not implemented in VE Language Service
126 return undefined;
127 }
128 function findRenameLocations(fileName, position, findInStrings, findInComments, providePrefixAndSuffixTextForRename) {
129 // not implemented in VE Language Service
130 return undefined;
131 }
132 function getTcb(fileName, position) {
133 // Not implemented in VE Language Service
134 return undefined;
135 }
136 return tslib_1.__assign(tslib_1.__assign({}, tsLS), {
137 // Then override the methods supported by Angular language service
138 getCompletionsAtPosition: getCompletionsAtPosition,
139 getQuickInfoAtPosition: getQuickInfoAtPosition,
140 getSemanticDiagnostics: getSemanticDiagnostics,
141 getDefinitionAtPosition: getDefinitionAtPosition,
142 getDefinitionAndBoundSpan: getDefinitionAndBoundSpan,
143 getTypeDefinitionAtPosition: getTypeDefinitionAtPosition,
144 getReferencesAtPosition: getReferencesAtPosition,
145 findRenameLocations: findRenameLocations,
146 getTcb: getTcb });
147 }
148 exports.create = create;
149});
150//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ts_plugin.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/ts_plugin.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 tss from 'typescript/lib/tsserverlibrary';\nimport {NgLanguageService} from '../api';\n\nimport {createLanguageService} from './language_service';\nimport {TypeScriptServiceHost} from './typescript_host';\n\n// Use a WeakMap to keep track of Project to Host mapping so that when Project\n// is deleted Host could be garbage collected.\nconst PROJECT_MAP = new WeakMap<tss.server.Project, TypeScriptServiceHost>();\n\n/**\n * This function is called by tsserver to retrieve the external (non-TS) files\n * that should belong to the specified `project`. For Angular, these files are\n * external templates. This is called once when the project is loaded, then\n * every time when the program is updated.\n * @param project Project for which external files should be retrieved.\n */\nexport function getExternalFiles(project: tss.server.Project): string[] {\n  if (!project.hasRoots()) {\n    // During project initialization where there is no root files yet we should\n    // not do any work.\n    return [];\n  }\n  const ngLsHost = PROJECT_MAP.get(project);\n  if (ngLsHost === undefined) {\n    return [];\n  }\n  ngLsHost.getAnalyzedModules();\n  return ngLsHost.getExternalTemplates().filter(fileName => {\n    // TODO(kyliau): Remove this when the following PR lands on the version of\n    // TypeScript used in this repo.\n    // https://github.com/microsoft/TypeScript/pull/41737\n    return project.fileExists(fileName);\n  });\n}\n\nexport function create(info: tss.server.PluginCreateInfo): NgLanguageService {\n  const {languageService: tsLS, languageServiceHost: tsLSHost, config, project} = info;\n  // This plugin could operate under two different modes:\n  // 1. TS + Angular\n  //    Plugin augments TS language service to provide additional Angular\n  //    information. This only works with inline templates and is meant to be\n  //    used as a local plugin (configured via tsconfig.json)\n  // 2. Angular only\n  //    Plugin only provides information on Angular templates, no TS info at all.\n  //    This effectively disables native TS features and is meant for internal\n  //    use only.\n  const angularOnly = config ? config.angularOnly === true : false;\n  const ngLSHost = new TypeScriptServiceHost(tsLSHost, tsLS);\n  const ngLS = createLanguageService(ngLSHost);\n  PROJECT_MAP.set(project, ngLSHost);\n\n  function getCompletionsAtPosition(\n      fileName: string, position: number, options: tss.GetCompletionsAtPositionOptions|undefined) {\n    if (!angularOnly) {\n      const results = tsLS.getCompletionsAtPosition(fileName, position, options);\n      if (results && results.entries.length) {\n        // If TS could answer the query, then return results immediately.\n        return results;\n      }\n    }\n    return ngLS.getCompletionsAtPosition(fileName, position, options);\n  }\n\n  function getQuickInfoAtPosition(fileName: string, position: number): tss.QuickInfo|undefined {\n    if (!angularOnly) {\n      const result = tsLS.getQuickInfoAtPosition(fileName, position);\n      if (result) {\n        // If TS could answer the query, then return results immediately.\n        return result;\n      }\n    }\n    return ngLS.getQuickInfoAtPosition(fileName, position);\n  }\n\n  function getSemanticDiagnostics(fileName: string): tss.Diagnostic[] {\n    const results: tss.Diagnostic[] = [];\n    if (!angularOnly) {\n      results.push(...tsLS.getSemanticDiagnostics(fileName));\n    }\n    // For semantic diagnostics we need to combine both TS + Angular results\n    results.push(...ngLS.getSemanticDiagnostics(fileName));\n    return results;\n  }\n\n  function getDefinitionAtPosition(\n      fileName: string, position: number): ReadonlyArray<tss.DefinitionInfo>|undefined {\n    if (!angularOnly) {\n      const results = tsLS.getDefinitionAtPosition(fileName, position);\n      if (results) {\n        // If TS could answer the query, then return results immediately.\n        return results;\n      }\n    }\n    const result = ngLS.getDefinitionAndBoundSpan(fileName, position);\n    if (!result || !result.definitions || !result.definitions.length) {\n      return;\n    }\n    return result.definitions;\n  }\n\n  function getDefinitionAndBoundSpan(\n      fileName: string, position: number): tss.DefinitionInfoAndBoundSpan|undefined {\n    if (!angularOnly) {\n      const result = tsLS.getDefinitionAndBoundSpan(fileName, position);\n      if (result) {\n        // If TS could answer the query, then return results immediately.\n        return result;\n      }\n    }\n    return ngLS.getDefinitionAndBoundSpan(fileName, position);\n  }\n\n  function getTypeDefinitionAtPosition(fileName: string, position: number) {\n    // Not implemented in VE Language Service\n    return undefined;\n  }\n\n  function getReferencesAtPosition(fileName: string, position: number) {\n    // Not implemented in VE Language Service\n    return undefined;\n  }\n\n  function findRenameLocations(\n      fileName: string, position: number, findInStrings: boolean, findInComments: boolean,\n      providePrefixAndSuffixTextForRename?: boolean): readonly ts.RenameLocation[]|undefined {\n    // not implemented in VE Language Service\n    return undefined;\n  }\n\n  function getTcb(fileName: string, position: number) {\n    // Not implemented in VE Language Service\n    return undefined;\n  }\n\n  return {\n    // First clone the original TS language service\n    ...tsLS,\n    // Then override the methods supported by Angular language service\n    getCompletionsAtPosition,\n    getQuickInfoAtPosition,\n    getSemanticDiagnostics,\n    getDefinitionAtPosition,\n    getDefinitionAndBoundSpan,\n    getTypeDefinitionAtPosition,\n    getReferencesAtPosition,\n    findRenameLocations,\n    getTcb,\n  };\n}\n"]}
\No newline at end of file