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/ivy/language_service", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler-cli", "@angular/compiler-cli/src/ngtsc/diagnostics", "@angular/compiler-cli/src/ngtsc/file_system", "@angular/compiler-cli/src/ngtsc/typecheck", "@angular/compiler-cli/src/ngtsc/typecheck/api", "@angular/compiler-cli/src/ngtsc/typecheck/src/comments", "typescript/lib/tsserverlibrary", "@angular/language-service/ivy/adapters", "@angular/language-service/ivy/compiler_factory", "@angular/language-service/ivy/completions", "@angular/language-service/ivy/definitions", "@angular/language-service/ivy/quick_info", "@angular/language-service/ivy/references", "@angular/language-service/ivy/template_target", "@angular/language-service/ivy/utils"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.LanguageService = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var compiler_cli_1 = require("@angular/compiler-cli");
|
23 | var diagnostics_1 = require("@angular/compiler-cli/src/ngtsc/diagnostics");
|
24 | var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
|
25 | var typecheck_1 = require("@angular/compiler-cli/src/ngtsc/typecheck");
|
26 | var api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api");
|
27 | var comments_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/src/comments");
|
28 | var ts = require("typescript/lib/tsserverlibrary");
|
29 | var adapters_1 = require("@angular/language-service/ivy/adapters");
|
30 | var compiler_factory_1 = require("@angular/language-service/ivy/compiler_factory");
|
31 | var completions_1 = require("@angular/language-service/ivy/completions");
|
32 | var definitions_1 = require("@angular/language-service/ivy/definitions");
|
33 | var quick_info_1 = require("@angular/language-service/ivy/quick_info");
|
34 | var references_1 = require("@angular/language-service/ivy/references");
|
35 | var template_target_1 = require("@angular/language-service/ivy/template_target");
|
36 | var utils_1 = require("@angular/language-service/ivy/utils");
|
37 | var LanguageService = /** @class */ (function () {
|
38 | function LanguageService(project, tsLS) {
|
39 | this.project = project;
|
40 | this.tsLS = tsLS;
|
41 | this.parseConfigHost = new adapters_1.LSParseConfigHost(project.projectService.host);
|
42 | this.options = parseNgCompilerOptions(project, this.parseConfigHost);
|
43 | logCompilerOptions(project, this.options);
|
44 | this.strategy = createTypeCheckingProgramStrategy(project);
|
45 | this.adapter = new adapters_1.LanguageServiceAdapter(project);
|
46 | this.compilerFactory = new compiler_factory_1.CompilerFactory(this.adapter, this.strategy, this.options);
|
47 | this.watchConfigFile(project);
|
48 | }
|
49 | LanguageService.prototype.getCompilerOptions = function () {
|
50 | return this.options;
|
51 | };
|
52 | LanguageService.prototype.getSemanticDiagnostics = function (fileName) {
|
53 | var e_1, _a;
|
54 | var compiler = this.compilerFactory.getOrCreate();
|
55 | var ttc = compiler.getTemplateTypeChecker();
|
56 | var diagnostics = [];
|
57 | if (utils_1.isTypeScriptFile(fileName)) {
|
58 | var program = compiler.getNextProgram();
|
59 | var sourceFile = program.getSourceFile(fileName);
|
60 | if (sourceFile) {
|
61 | diagnostics.push.apply(diagnostics, tslib_1.__spread(compiler.getDiagnosticsForFile(sourceFile, api_1.OptimizeFor.SingleFile)));
|
62 | }
|
63 | }
|
64 | else {
|
65 | var components = compiler.getComponentsWithTemplateFile(fileName);
|
66 | try {
|
67 | for (var components_1 = tslib_1.__values(components), components_1_1 = components_1.next(); !components_1_1.done; components_1_1 = components_1.next()) {
|
68 | var component = components_1_1.value;
|
69 | if (ts.isClassDeclaration(component)) {
|
70 | diagnostics.push.apply(diagnostics, tslib_1.__spread(ttc.getDiagnosticsForComponent(component)));
|
71 | }
|
72 | }
|
73 | }
|
74 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
75 | finally {
|
76 | try {
|
77 | if (components_1_1 && !components_1_1.done && (_a = components_1.return)) _a.call(components_1);
|
78 | }
|
79 | finally { if (e_1) throw e_1.error; }
|
80 | }
|
81 | }
|
82 | this.compilerFactory.registerLastKnownProgram();
|
83 | return diagnostics;
|
84 | };
|
85 | LanguageService.prototype.getDefinitionAndBoundSpan = function (fileName, position) {
|
86 | var compiler = this.compilerFactory.getOrCreate();
|
87 | var results = new definitions_1.DefinitionBuilder(this.tsLS, compiler).getDefinitionAndBoundSpan(fileName, position);
|
88 | this.compilerFactory.registerLastKnownProgram();
|
89 | return results;
|
90 | };
|
91 | LanguageService.prototype.getTypeDefinitionAtPosition = function (fileName, position) {
|
92 | var compiler = this.compilerFactory.getOrCreate();
|
93 | var results = new definitions_1.DefinitionBuilder(this.tsLS, compiler).getTypeDefinitionsAtPosition(fileName, position);
|
94 | this.compilerFactory.registerLastKnownProgram();
|
95 | return results;
|
96 | };
|
97 | LanguageService.prototype.getQuickInfoAtPosition = function (fileName, position) {
|
98 | var compiler = this.compilerFactory.getOrCreate();
|
99 | var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, compiler);
|
100 | if (templateInfo === undefined) {
|
101 | return undefined;
|
102 | }
|
103 | var positionDetails = template_target_1.getTargetAtPosition(templateInfo.template, position);
|
104 | if (positionDetails === null) {
|
105 | return undefined;
|
106 | }
|
107 | // Because we can only show 1 quick info, just use the bound attribute if the target is a two
|
108 | // way binding. We may consider concatenating additional display parts from the other target
|
109 | // nodes or representing the two way binding in some other manner in the future.
|
110 | var node = positionDetails.context.kind === template_target_1.TargetNodeKind.TwoWayBindingContext ?
|
111 | positionDetails.context.nodes[0] :
|
112 | positionDetails.context.node;
|
113 | var results = new quick_info_1.QuickInfoBuilder(this.tsLS, compiler, templateInfo.component, node).get();
|
114 | this.compilerFactory.registerLastKnownProgram();
|
115 | return results;
|
116 | };
|
117 | LanguageService.prototype.getReferencesAtPosition = function (fileName, position) {
|
118 | var compiler = this.compilerFactory.getOrCreate();
|
119 | var results = new references_1.ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
120 | .getReferencesAtPosition(fileName, position);
|
121 | this.compilerFactory.registerLastKnownProgram();
|
122 | return results;
|
123 | };
|
124 | LanguageService.prototype.getRenameInfo = function (fileName, position) {
|
125 | var _a, _b, _c;
|
126 | var compiler = this.compilerFactory.getOrCreate();
|
127 | var renameInfo = new references_1.ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
128 | .getRenameInfo(file_system_1.absoluteFrom(fileName), position);
|
129 | if (!renameInfo.canRename) {
|
130 | return renameInfo;
|
131 | }
|
132 | var quickInfo = (_a = this.getQuickInfoAtPosition(fileName, position)) !== null && _a !== void 0 ? _a : this.tsLS.getQuickInfoAtPosition(fileName, position);
|
133 | var kind = (_b = quickInfo === null || quickInfo === void 0 ? void 0 : quickInfo.kind) !== null && _b !== void 0 ? _b : ts.ScriptElementKind.unknown;
|
134 | var kindModifiers = (_c = quickInfo === null || quickInfo === void 0 ? void 0 : quickInfo.kindModifiers) !== null && _c !== void 0 ? _c : ts.ScriptElementKind.unknown;
|
135 | return tslib_1.__assign(tslib_1.__assign({}, renameInfo), { kind: kind, kindModifiers: kindModifiers });
|
136 | };
|
137 | LanguageService.prototype.findRenameLocations = function (fileName, position) {
|
138 | var compiler = this.compilerFactory.getOrCreate();
|
139 | var results = new references_1.ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
140 | .findRenameLocations(fileName, position);
|
141 | this.compilerFactory.registerLastKnownProgram();
|
142 | return results;
|
143 | };
|
144 | LanguageService.prototype.getCompletionBuilder = function (fileName, position) {
|
145 | var compiler = this.compilerFactory.getOrCreate();
|
146 | var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, compiler);
|
147 | if (templateInfo === undefined) {
|
148 | return null;
|
149 | }
|
150 | var positionDetails = template_target_1.getTargetAtPosition(templateInfo.template, position);
|
151 | if (positionDetails === null) {
|
152 | return null;
|
153 | }
|
154 | // For two-way bindings, we actually only need to be concerned with the bound attribute because
|
155 | // the bindings in the template are written with the attribute name, not the event name.
|
156 | var node = positionDetails.context.kind === template_target_1.TargetNodeKind.TwoWayBindingContext ?
|
157 | positionDetails.context.nodes[0] :
|
158 | positionDetails.context.node;
|
159 | return new completions_1.CompletionBuilder(this.tsLS, compiler, templateInfo.component, node, nodeContextFromTarget(positionDetails.context), positionDetails.parent, positionDetails.template);
|
160 | };
|
161 | LanguageService.prototype.getCompletionsAtPosition = function (fileName, position, options) {
|
162 | var builder = this.getCompletionBuilder(fileName, position);
|
163 | if (builder === null) {
|
164 | return undefined;
|
165 | }
|
166 | var result = builder.getCompletionsAtPosition(options);
|
167 | this.compilerFactory.registerLastKnownProgram();
|
168 | return result;
|
169 | };
|
170 | LanguageService.prototype.getCompletionEntryDetails = function (fileName, position, entryName, formatOptions, preferences) {
|
171 | var builder = this.getCompletionBuilder(fileName, position);
|
172 | if (builder === null) {
|
173 | return undefined;
|
174 | }
|
175 | var result = builder.getCompletionEntryDetails(entryName, formatOptions, preferences);
|
176 | this.compilerFactory.registerLastKnownProgram();
|
177 | return result;
|
178 | };
|
179 | LanguageService.prototype.getCompletionEntrySymbol = function (fileName, position, entryName) {
|
180 | var builder = this.getCompletionBuilder(fileName, position);
|
181 | if (builder === null) {
|
182 | return undefined;
|
183 | }
|
184 | var result = builder.getCompletionEntrySymbol(entryName);
|
185 | this.compilerFactory.registerLastKnownProgram();
|
186 | return result;
|
187 | };
|
188 | LanguageService.prototype.getTcb = function (fileName, position) {
|
189 | return this.withCompiler(function (compiler) {
|
190 | var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, compiler);
|
191 | if (templateInfo === undefined) {
|
192 | return undefined;
|
193 | }
|
194 | var tcb = compiler.getTemplateTypeChecker().getTypeCheckBlock(templateInfo.component);
|
195 | if (tcb === null) {
|
196 | return undefined;
|
197 | }
|
198 | var sf = tcb.getSourceFile();
|
199 | var selections = [];
|
200 | var target = template_target_1.getTargetAtPosition(templateInfo.template, position);
|
201 | if (target !== null) {
|
202 | var selectionSpans = void 0;
|
203 | if ('nodes' in target.context) {
|
204 | selectionSpans = target.context.nodes.map(function (n) { return n.sourceSpan; });
|
205 | }
|
206 | else {
|
207 | selectionSpans = [target.context.node.sourceSpan];
|
208 | }
|
209 | var selectionNodes = selectionSpans
|
210 | .map(function (s) { return comments_1.findFirstMatchingNode(tcb, {
|
211 | withSpan: s,
|
212 | filter: function (node) { return true; },
|
213 | }); })
|
214 | .filter(function (n) { return n !== null; });
|
215 | selections = selectionNodes.map(function (n) {
|
216 | return {
|
217 | start: n.getStart(sf),
|
218 | length: n.getEnd() - n.getStart(sf),
|
219 | };
|
220 | });
|
221 | }
|
222 | return {
|
223 | fileName: sf.fileName,
|
224 | content: sf.getFullText(),
|
225 | selections: selections,
|
226 | };
|
227 | });
|
228 | };
|
229 | LanguageService.prototype.withCompiler = function (p) {
|
230 | var compiler = this.compilerFactory.getOrCreate();
|
231 | var result = p(compiler);
|
232 | this.compilerFactory.registerLastKnownProgram();
|
233 | return result;
|
234 | };
|
235 | LanguageService.prototype.getCompilerOptionsDiagnostics = function () {
|
236 | var project = this.project;
|
237 | if (!(project instanceof ts.server.ConfiguredProject)) {
|
238 | return [];
|
239 | }
|
240 | var diagnostics = [];
|
241 | var configSourceFile = ts.readJsonConfigFile(project.getConfigFilePath(), function (path) { return project.readFile(path); });
|
242 | if (!this.options.strictTemplates && !this.options.fullTemplateTypeCheck) {
|
243 | diagnostics.push({
|
244 | messageText: 'Some language features are not available. ' +
|
245 | 'To access all features, enable `strictTemplates` in `angularCompilerOptions`.',
|
246 | category: ts.DiagnosticCategory.Suggestion,
|
247 | code: diagnostics_1.ngErrorCode(diagnostics_1.ErrorCode.SUGGEST_STRICT_TEMPLATES),
|
248 | file: configSourceFile,
|
249 | start: undefined,
|
250 | length: undefined,
|
251 | });
|
252 | }
|
253 | var compiler = this.compilerFactory.getOrCreate();
|
254 | diagnostics.push.apply(diagnostics, tslib_1.__spread(compiler.getOptionDiagnostics()));
|
255 | return diagnostics;
|
256 | };
|
257 | LanguageService.prototype.watchConfigFile = function (project) {
|
258 | var _this = this;
|
259 | // TODO: Check the case when the project is disposed. An InferredProject
|
260 | // could be disposed when a tsconfig.json is added to the workspace,
|
261 | // in which case it becomes a ConfiguredProject (or vice-versa).
|
262 | // We need to make sure that the FileWatcher is closed.
|
263 | if (!(project instanceof ts.server.ConfiguredProject)) {
|
264 | return;
|
265 | }
|
266 | var host = project.projectService.host;
|
267 | host.watchFile(project.getConfigFilePath(), function (fileName, eventKind) {
|
268 | project.log("Config file changed: " + fileName);
|
269 | if (eventKind === ts.FileWatcherEventKind.Changed) {
|
270 | _this.options = parseNgCompilerOptions(project, _this.parseConfigHost);
|
271 | logCompilerOptions(project, _this.options);
|
272 | }
|
273 | });
|
274 | };
|
275 | return LanguageService;
|
276 | }());
|
277 | exports.LanguageService = LanguageService;
|
278 | function logCompilerOptions(project, options) {
|
279 | var logger = project.projectService.logger;
|
280 | var projectName = project.getProjectName();
|
281 | logger.info("Angular compiler options for " + projectName + ": " + JSON.stringify(options, null, 2));
|
282 | }
|
283 | function parseNgCompilerOptions(project, host) {
|
284 | if (!(project instanceof ts.server.ConfiguredProject)) {
|
285 | return {};
|
286 | }
|
287 | var _a = compiler_cli_1.readConfiguration(project.getConfigFilePath(), /* existingOptions */ undefined, host), options = _a.options, errors = _a.errors;
|
288 | if (errors.length > 0) {
|
289 | project.setProjectErrors(errors);
|
290 | }
|
291 | // Projects loaded into the Language Service often include test files which are not part of the
|
292 | // app's main compilation unit, and these test files often include inline NgModules that declare
|
293 | // components from the app. These declarations conflict with the main declarations of such
|
294 | // components in the app's NgModules. This conflict is not normally present during regular
|
295 | // compilation because the app and the tests are part of separate compilation units.
|
296 | //
|
297 | // As a temporary mitigation of this problem, we instruct the compiler to ignore classes which
|
298 | // are not exported. In many cases, this ensures the test NgModules are ignored by the compiler
|
299 | // and only the real component declaration is used.
|
300 | options.compileNonExportedClasses = false;
|
301 | return options;
|
302 | }
|
303 | function createTypeCheckingProgramStrategy(project) {
|
304 | return {
|
305 | supportsInlineOperations: false,
|
306 | shimPathForComponent: function (component) {
|
307 | return typecheck_1.TypeCheckShimGenerator.shimFor(file_system_1.absoluteFromSourceFile(component.getSourceFile()));
|
308 | },
|
309 | getProgram: function () {
|
310 | var program = project.getLanguageService().getProgram();
|
311 | if (!program) {
|
312 | throw new Error('Language service does not have a program!');
|
313 | }
|
314 | return program;
|
315 | },
|
316 | updateFiles: function (contents) {
|
317 | var e_2, _a;
|
318 | try {
|
319 | for (var contents_1 = tslib_1.__values(contents), contents_1_1 = contents_1.next(); !contents_1_1.done; contents_1_1 = contents_1.next()) {
|
320 | var _b = tslib_1.__read(contents_1_1.value, 2), fileName = _b[0], newText = _b[1];
|
321 | var scriptInfo = getOrCreateTypeCheckScriptInfo(project, fileName);
|
322 | var snapshot = scriptInfo.getSnapshot();
|
323 | var length_1 = snapshot.getLength();
|
324 | scriptInfo.editContent(0, length_1, newText);
|
325 | }
|
326 | }
|
327 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
328 | finally {
|
329 | try {
|
330 | if (contents_1_1 && !contents_1_1.done && (_a = contents_1.return)) _a.call(contents_1);
|
331 | }
|
332 | finally { if (e_2) throw e_2.error; }
|
333 | }
|
334 | },
|
335 | };
|
336 | }
|
337 | function getOrCreateTypeCheckScriptInfo(project, tcf) {
|
338 | // First check if there is already a ScriptInfo for the tcf
|
339 | var projectService = project.projectService;
|
340 | var scriptInfo = projectService.getScriptInfo(tcf);
|
341 | if (!scriptInfo) {
|
342 | // ScriptInfo needs to be opened by client to be able to set its user-defined
|
343 | // content. We must also provide file content, otherwise the service will
|
344 | // attempt to fetch the content from disk and fail.
|
345 | scriptInfo = projectService.getOrCreateScriptInfoForNormalizedPath(ts.server.toNormalizedPath(tcf), true, // openedByClient
|
346 | '', // fileContent
|
347 | // script info added by plugins should be marked as external, see
|
348 | // https://github.com/microsoft/TypeScript/blob/b217f22e798c781f55d17da72ed099a9dee5c650/src/compiler/program.ts#L1897-L1899
|
349 | ts.ScriptKind.External);
|
350 | if (!scriptInfo) {
|
351 | throw new Error("Failed to create script info for " + tcf);
|
352 | }
|
353 | }
|
354 | // Add ScriptInfo to project if it's missing. A ScriptInfo needs to be part of
|
355 | // the project so that it becomes part of the program.
|
356 | if (!project.containsScriptInfo(scriptInfo)) {
|
357 | project.addRoot(scriptInfo);
|
358 | }
|
359 | return scriptInfo;
|
360 | }
|
361 | function nodeContextFromTarget(target) {
|
362 | switch (target.kind) {
|
363 | case template_target_1.TargetNodeKind.ElementInTagContext:
|
364 | return completions_1.CompletionNodeContext.ElementTag;
|
365 | case template_target_1.TargetNodeKind.ElementInBodyContext:
|
366 | // Completions in element bodies are for new attributes.
|
367 | return completions_1.CompletionNodeContext.ElementAttributeKey;
|
368 | case template_target_1.TargetNodeKind.TwoWayBindingContext:
|
369 | return completions_1.CompletionNodeContext.TwoWayBinding;
|
370 | case template_target_1.TargetNodeKind.AttributeInKeyContext:
|
371 | return completions_1.CompletionNodeContext.ElementAttributeKey;
|
372 | case template_target_1.TargetNodeKind.AttributeInValueContext:
|
373 | if (target.node instanceof compiler_1.TmplAstBoundEvent) {
|
374 | return completions_1.CompletionNodeContext.EventValue;
|
375 | }
|
376 | else {
|
377 | return completions_1.CompletionNodeContext.None;
|
378 | }
|
379 | default:
|
380 | // No special context is available.
|
381 | return completions_1.CompletionNodeContext.None;
|
382 | }
|
383 | }
|
384 | });
|
385 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"language_service.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/language_service.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 {AbsoluteSourceSpan, AST, ParseSourceSpan, TmplAstBoundEvent, TmplAstNode} from '@angular/compiler';\nimport {CompilerOptions, ConfigurationHost, readConfiguration} from '@angular/compiler-cli';\nimport {NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';\nimport {ErrorCode, ngErrorCode} from '@angular/compiler-cli/src/ngtsc/diagnostics';\nimport {absoluteFrom, absoluteFromSourceFile, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';\nimport {TypeCheckShimGenerator} from '@angular/compiler-cli/src/ngtsc/typecheck';\nimport {OptimizeFor, TypeCheckingProgramStrategy} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport {findFirstMatchingNode} from '@angular/compiler-cli/src/ngtsc/typecheck/src/comments';\nimport * as ts from 'typescript/lib/tsserverlibrary';\nimport {GetTcbResponse} from '../api';\n\nimport {LanguageServiceAdapter, LSParseConfigHost} from './adapters';\nimport {CompilerFactory} from './compiler_factory';\nimport {CompletionBuilder, CompletionNodeContext} from './completions';\nimport {DefinitionBuilder} from './definitions';\nimport {QuickInfoBuilder} from './quick_info';\nimport {ReferencesAndRenameBuilder} from './references';\nimport {getTargetAtPosition, TargetContext, TargetNodeKind} from './template_target';\nimport {getTemplateInfoAtPosition, isTypeScriptFile} from './utils';\n\nexport class LanguageService {\n  private options: CompilerOptions;\n  readonly compilerFactory: CompilerFactory;\n  private readonly strategy: TypeCheckingProgramStrategy;\n  private readonly adapter: LanguageServiceAdapter;\n  private readonly parseConfigHost: LSParseConfigHost;\n\n  constructor(\n      private readonly project: ts.server.Project, private readonly tsLS: ts.LanguageService) {\n    this.parseConfigHost = new LSParseConfigHost(project.projectService.host);\n    this.options = parseNgCompilerOptions(project, this.parseConfigHost);\n    logCompilerOptions(project, this.options);\n    this.strategy = createTypeCheckingProgramStrategy(project);\n    this.adapter = new LanguageServiceAdapter(project);\n    this.compilerFactory = new CompilerFactory(this.adapter, this.strategy, this.options);\n    this.watchConfigFile(project);\n  }\n\n  getCompilerOptions(): CompilerOptions {\n    return this.options;\n  }\n\n  getSemanticDiagnostics(fileName: string): ts.Diagnostic[] {\n    const compiler = this.compilerFactory.getOrCreate();\n    const ttc = compiler.getTemplateTypeChecker();\n    const diagnostics: ts.Diagnostic[] = [];\n    if (isTypeScriptFile(fileName)) {\n      const program = compiler.getNextProgram();\n      const sourceFile = program.getSourceFile(fileName);\n      if (sourceFile) {\n        diagnostics.push(...compiler.getDiagnosticsForFile(sourceFile, OptimizeFor.SingleFile));\n      }\n    } else {\n      const components = compiler.getComponentsWithTemplateFile(fileName);\n      for (const component of components) {\n        if (ts.isClassDeclaration(component)) {\n          diagnostics.push(...ttc.getDiagnosticsForComponent(component));\n        }\n      }\n    }\n    this.compilerFactory.registerLastKnownProgram();\n    return diagnostics;\n  }\n\n  getDefinitionAndBoundSpan(fileName: string, position: number): ts.DefinitionInfoAndBoundSpan\n      |undefined {\n    const compiler = this.compilerFactory.getOrCreate();\n    const results =\n        new DefinitionBuilder(this.tsLS, compiler).getDefinitionAndBoundSpan(fileName, position);\n    this.compilerFactory.registerLastKnownProgram();\n    return results;\n  }\n\n  getTypeDefinitionAtPosition(fileName: string, position: number):\n      readonly ts.DefinitionInfo[]|undefined {\n    const compiler = this.compilerFactory.getOrCreate();\n    const results =\n        new DefinitionBuilder(this.tsLS, compiler).getTypeDefinitionsAtPosition(fileName, position);\n    this.compilerFactory.registerLastKnownProgram();\n    return results;\n  }\n\n  getQuickInfoAtPosition(fileName: string, position: number): ts.QuickInfo|undefined {\n    const compiler = this.compilerFactory.getOrCreate();\n    const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);\n    if (templateInfo === undefined) {\n      return undefined;\n    }\n    const positionDetails = getTargetAtPosition(templateInfo.template, position);\n    if (positionDetails === null) {\n      return undefined;\n    }\n\n    // Because we can only show 1 quick info, just use the bound attribute if the target is a two\n    // way binding. We may consider concatenating additional display parts from the other target\n    // nodes or representing the two way binding in some other manner in the future.\n    const node = positionDetails.context.kind === TargetNodeKind.TwoWayBindingContext ?\n        positionDetails.context.nodes[0] :\n        positionDetails.context.node;\n    const results = new QuickInfoBuilder(this.tsLS, compiler, templateInfo.component, node).get();\n    this.compilerFactory.registerLastKnownProgram();\n    return results;\n  }\n\n  getReferencesAtPosition(fileName: string, position: number): ts.ReferenceEntry[]|undefined {\n    const compiler = this.compilerFactory.getOrCreate();\n    const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)\n                        .getReferencesAtPosition(fileName, position);\n    this.compilerFactory.registerLastKnownProgram();\n    return results;\n  }\n\n  getRenameInfo(fileName: string, position: number): ts.RenameInfo {\n    const compiler = this.compilerFactory.getOrCreate();\n    const renameInfo = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)\n                           .getRenameInfo(absoluteFrom(fileName), position);\n    if (!renameInfo.canRename) {\n      return renameInfo;\n    }\n\n    const quickInfo = this.getQuickInfoAtPosition(fileName, position) ??\n        this.tsLS.getQuickInfoAtPosition(fileName, position);\n    const kind = quickInfo?.kind ?? ts.ScriptElementKind.unknown;\n    const kindModifiers = quickInfo?.kindModifiers ?? ts.ScriptElementKind.unknown;\n    return {...renameInfo, kind, kindModifiers};\n  }\n\n  findRenameLocations(fileName: string, position: number): readonly ts.RenameLocation[]|undefined {\n    const compiler = this.compilerFactory.getOrCreate();\n    const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)\n                        .findRenameLocations(fileName, position);\n    this.compilerFactory.registerLastKnownProgram();\n    return results;\n  }\n\n  private getCompletionBuilder(fileName: string, position: number):\n      CompletionBuilder<TmplAstNode|AST>|null {\n    const compiler = this.compilerFactory.getOrCreate();\n    const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);\n    if (templateInfo === undefined) {\n      return null;\n    }\n    const positionDetails = getTargetAtPosition(templateInfo.template, position);\n    if (positionDetails === null) {\n      return null;\n    }\n\n    // For two-way bindings, we actually only need to be concerned with the bound attribute because\n    // the bindings in the template are written with the attribute name, not the event name.\n    const node = positionDetails.context.kind === TargetNodeKind.TwoWayBindingContext ?\n        positionDetails.context.nodes[0] :\n        positionDetails.context.node;\n    return new CompletionBuilder(\n        this.tsLS, compiler, templateInfo.component, node,\n        nodeContextFromTarget(positionDetails.context), positionDetails.parent,\n        positionDetails.template);\n  }\n\n  getCompletionsAtPosition(\n      fileName: string, position: number, options: ts.GetCompletionsAtPositionOptions|undefined):\n      ts.WithMetadata<ts.CompletionInfo>|undefined {\n    const builder = this.getCompletionBuilder(fileName, position);\n    if (builder === null) {\n      return undefined;\n    }\n    const result = builder.getCompletionsAtPosition(options);\n    this.compilerFactory.registerLastKnownProgram();\n    return result;\n  }\n\n  getCompletionEntryDetails(\n      fileName: string, position: number, entryName: string,\n      formatOptions: ts.FormatCodeOptions|ts.FormatCodeSettings|undefined,\n      preferences: ts.UserPreferences|undefined): ts.CompletionEntryDetails|undefined {\n    const builder = this.getCompletionBuilder(fileName, position);\n    if (builder === null) {\n      return undefined;\n    }\n    const result = builder.getCompletionEntryDetails(entryName, formatOptions, preferences);\n    this.compilerFactory.registerLastKnownProgram();\n    return result;\n  }\n\n  getCompletionEntrySymbol(fileName: string, position: number, entryName: string): ts.Symbol\n      |undefined {\n    const builder = this.getCompletionBuilder(fileName, position);\n    if (builder === null) {\n      return undefined;\n    }\n    const result = builder.getCompletionEntrySymbol(entryName);\n    this.compilerFactory.registerLastKnownProgram();\n    return result;\n  }\n\n  getTcb(fileName: string, position: number): GetTcbResponse {\n    return this.withCompiler<GetTcbResponse>(compiler => {\n      const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);\n      if (templateInfo === undefined) {\n        return undefined;\n      }\n      const tcb = compiler.getTemplateTypeChecker().getTypeCheckBlock(templateInfo.component);\n      if (tcb === null) {\n        return undefined;\n      }\n      const sf = tcb.getSourceFile();\n\n      let selections: ts.TextSpan[] = [];\n      const target = getTargetAtPosition(templateInfo.template, position);\n      if (target !== null) {\n        let selectionSpans: Array<ParseSourceSpan|AbsoluteSourceSpan>;\n        if ('nodes' in target.context) {\n          selectionSpans = target.context.nodes.map(n => n.sourceSpan);\n        } else {\n          selectionSpans = [target.context.node.sourceSpan];\n        }\n        const selectionNodes: ts.Node[] =\n            selectionSpans\n                .map(s => findFirstMatchingNode(tcb, {\n                       withSpan: s,\n                       filter: (node: ts.Node): node is ts.Node => true,\n                     }))\n                .filter((n): n is ts.Node => n !== null);\n\n        selections = selectionNodes.map(n => {\n          return {\n            start: n.getStart(sf),\n            length: n.getEnd() - n.getStart(sf),\n          };\n        });\n      }\n\n      return {\n        fileName: sf.fileName,\n        content: sf.getFullText(),\n        selections,\n      };\n    });\n  }\n\n  private withCompiler<T>(p: (compiler: NgCompiler) => T): T {\n    const compiler = this.compilerFactory.getOrCreate();\n    const result = p(compiler);\n    this.compilerFactory.registerLastKnownProgram();\n    return result;\n  }\n\n  getCompilerOptionsDiagnostics(): ts.Diagnostic[] {\n    const project = this.project;\n    if (!(project instanceof ts.server.ConfiguredProject)) {\n      return [];\n    }\n\n    const diagnostics: ts.Diagnostic[] = [];\n    const configSourceFile = ts.readJsonConfigFile(\n        project.getConfigFilePath(), (path: string) => project.readFile(path));\n\n    if (!this.options.strictTemplates && !this.options.fullTemplateTypeCheck) {\n      diagnostics.push({\n        messageText: 'Some language features are not available. ' +\n            'To access all features, enable `strictTemplates` in `angularCompilerOptions`.',\n        category: ts.DiagnosticCategory.Suggestion,\n        code: ngErrorCode(ErrorCode.SUGGEST_STRICT_TEMPLATES),\n        file: configSourceFile,\n        start: undefined,\n        length: undefined,\n      });\n    }\n\n    const compiler = this.compilerFactory.getOrCreate();\n    diagnostics.push(...compiler.getOptionDiagnostics());\n\n    return diagnostics;\n  }\n\n  private watchConfigFile(project: ts.server.Project) {\n    // TODO: Check the case when the project is disposed. An InferredProject\n    // could be disposed when a tsconfig.json is added to the workspace,\n    // in which case it becomes a ConfiguredProject (or vice-versa).\n    // We need to make sure that the FileWatcher is closed.\n    if (!(project instanceof ts.server.ConfiguredProject)) {\n      return;\n    }\n    const {host} = project.projectService;\n    host.watchFile(\n        project.getConfigFilePath(), (fileName: string, eventKind: ts.FileWatcherEventKind) => {\n          project.log(`Config file changed: ${fileName}`);\n          if (eventKind === ts.FileWatcherEventKind.Changed) {\n            this.options = parseNgCompilerOptions(project, this.parseConfigHost);\n            logCompilerOptions(project, this.options);\n          }\n        });\n  }\n}\n\nfunction logCompilerOptions(project: ts.server.Project, options: CompilerOptions) {\n  const {logger} = project.projectService;\n  const projectName = project.getProjectName();\n  logger.info(`Angular compiler options for ${projectName}: ` + JSON.stringify(options, null, 2));\n}\n\nfunction parseNgCompilerOptions(\n    project: ts.server.Project, host: ConfigurationHost): CompilerOptions {\n  if (!(project instanceof ts.server.ConfiguredProject)) {\n    return {};\n  }\n  const {options, errors} =\n      readConfiguration(project.getConfigFilePath(), /* existingOptions */ undefined, host);\n  if (errors.length > 0) {\n    project.setProjectErrors(errors);\n  }\n\n  // Projects loaded into the Language Service often include test files which are not part of the\n  // app's main compilation unit, and these test files often include inline NgModules that declare\n  // components from the app. These declarations conflict with the main declarations of such\n  // components in the app's NgModules. This conflict is not normally present during regular\n  // compilation because the app and the tests are part of separate compilation units.\n  //\n  // As a temporary mitigation of this problem, we instruct the compiler to ignore classes which\n  // are not exported. In many cases, this ensures the test NgModules are ignored by the compiler\n  // and only the real component declaration is used.\n  options.compileNonExportedClasses = false;\n\n  return options;\n}\n\nfunction createTypeCheckingProgramStrategy(project: ts.server.Project):\n    TypeCheckingProgramStrategy {\n  return {\n    supportsInlineOperations: false,\n    shimPathForComponent(component: ts.ClassDeclaration): AbsoluteFsPath {\n      return TypeCheckShimGenerator.shimFor(absoluteFromSourceFile(component.getSourceFile()));\n    },\n    getProgram(): ts.Program {\n      const program = project.getLanguageService().getProgram();\n      if (!program) {\n        throw new Error('Language service does not have a program!');\n      }\n      return program;\n    },\n    updateFiles(contents: Map<AbsoluteFsPath, string>) {\n      for (const [fileName, newText] of contents) {\n        const scriptInfo = getOrCreateTypeCheckScriptInfo(project, fileName);\n        const snapshot = scriptInfo.getSnapshot();\n        const length = snapshot.getLength();\n        scriptInfo.editContent(0, length, newText);\n      }\n    },\n  };\n}\n\nfunction getOrCreateTypeCheckScriptInfo(\n    project: ts.server.Project, tcf: string): ts.server.ScriptInfo {\n  // First check if there is already a ScriptInfo for the tcf\n  const {projectService} = project;\n  let scriptInfo = projectService.getScriptInfo(tcf);\n  if (!scriptInfo) {\n    // ScriptInfo needs to be opened by client to be able to set its user-defined\n    // content. We must also provide file content, otherwise the service will\n    // attempt to fetch the content from disk and fail.\n    scriptInfo = projectService.getOrCreateScriptInfoForNormalizedPath(\n        ts.server.toNormalizedPath(tcf),\n        true,  // openedByClient\n        '',    // fileContent\n        // script info added by plugins should be marked as external, see\n        // https://github.com/microsoft/TypeScript/blob/b217f22e798c781f55d17da72ed099a9dee5c650/src/compiler/program.ts#L1897-L1899\n        ts.ScriptKind.External,  // scriptKind\n    );\n    if (!scriptInfo) {\n      throw new Error(`Failed to create script info for ${tcf}`);\n    }\n  }\n  // Add ScriptInfo to project if it's missing. A ScriptInfo needs to be part of\n  // the project so that it becomes part of the program.\n  if (!project.containsScriptInfo(scriptInfo)) {\n    project.addRoot(scriptInfo);\n  }\n  return scriptInfo;\n}\n\nfunction nodeContextFromTarget(target: TargetContext): CompletionNodeContext {\n  switch (target.kind) {\n    case TargetNodeKind.ElementInTagContext:\n      return CompletionNodeContext.ElementTag;\n    case TargetNodeKind.ElementInBodyContext:\n      // Completions in element bodies are for new attributes.\n      return CompletionNodeContext.ElementAttributeKey;\n    case TargetNodeKind.TwoWayBindingContext:\n      return CompletionNodeContext.TwoWayBinding;\n    case TargetNodeKind.AttributeInKeyContext:\n      return CompletionNodeContext.ElementAttributeKey;\n    case TargetNodeKind.AttributeInValueContext:\n      if (target.node instanceof TmplAstBoundEvent) {\n        return CompletionNodeContext.EventValue;\n      } else {\n        return CompletionNodeContext.None;\n      }\n    default:\n      // No special context is available.\n      return CompletionNodeContext.None;\n  }\n}\n"]} |
\ | No newline at end of file |