refactor(compiler-cli): remove the overrideComponentTemplate API (#40585)
The `TemplateTypeChecker.overrideComponentTemplate` operation was originally conceived as a "fast path" for the Language Service to react to a template change without needing to go through a full incremental compilation step. It served this purpose until the previous commit, which switches the LS to use the new resource-only incremental change operation provided by `NgCompiler`. `overrideComponentTemplate` is now no longer utilized, and is known to have several hard-to-overcome issues that prevent it from being useful in any other situations. As such, this commit removes it entirely. PR Close #40585
This commit is contained in:
parent
e3bd23c915
commit
a3b0864428
|
@ -881,9 +881,6 @@ export class ComponentDecoratorHandler implements
|
||||||
//
|
//
|
||||||
// In order to guarantee the correctness of diagnostics, templates are parsed a second time
|
// In order to guarantee the correctness of diagnostics, templates are parsed a second time
|
||||||
// with the above options set to preserve source mappings.
|
// with the above options set to preserve source mappings.
|
||||||
//
|
|
||||||
// Note: template parse options should be aligned with `template_target_spec.ts` and
|
|
||||||
// `TemplateTypeCheckerImpl.overrideComponentTemplate`.
|
|
||||||
|
|
||||||
const {nodes: diagNodes} = parseTemplate(templateStr, template.sourceMapUrl, {
|
const {nodes: diagNodes} = parseTemplate(templateStr, template.sourceMapUrl, {
|
||||||
preserveWhitespaces: true,
|
preserveWhitespaces: true,
|
||||||
|
|
|
@ -29,29 +29,11 @@ import {DirectiveSymbol, ElementSymbol, ShimLocation, Symbol, TemplateSymbol} fr
|
||||||
* query, depending on the method either `null` will be returned or an error will be thrown.
|
* query, depending on the method either `null` will be returned or an error will be thrown.
|
||||||
*/
|
*/
|
||||||
export interface TemplateTypeChecker {
|
export interface TemplateTypeChecker {
|
||||||
/**
|
|
||||||
* Clear all overrides and return the template type-checker to the original input program state.
|
|
||||||
*/
|
|
||||||
resetOverrides(): void;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve the template in use for the given component.
|
* Retrieve the template in use for the given component.
|
||||||
*
|
|
||||||
* If the template has been overridden via `overrideComponentTemplate`, this will retrieve the
|
|
||||||
* overridden template nodes.
|
|
||||||
*/
|
*/
|
||||||
getTemplate(component: ts.ClassDeclaration): TmplAstNode[]|null;
|
getTemplate(component: ts.ClassDeclaration): TmplAstNode[]|null;
|
||||||
|
|
||||||
/**
|
|
||||||
* Provide a new template string that will be used in place of the user-defined template when
|
|
||||||
* checking or operating on the given component.
|
|
||||||
*
|
|
||||||
* The compiler will parse this template for diagnostics, and will return any parsing errors if it
|
|
||||||
* is not valid. If the template cannot be parsed correctly, no override will occur.
|
|
||||||
*/
|
|
||||||
overrideComponentTemplate(component: ts.ClassDeclaration, template: string):
|
|
||||||
{nodes: TmplAstNode[], errors: ParseError[]|null};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get all `ts.Diagnostic`s currently available for the given `ts.SourceFile`.
|
* Get all `ts.Diagnostic`s currently available for the given `ts.SourceFile`.
|
||||||
*
|
*
|
||||||
|
|
|
@ -20,7 +20,7 @@ import {DirectiveInScope, ElementSymbol, FullTemplateMapping, GlobalCompletion,
|
||||||
import {TemplateDiagnostic} from '../diagnostics';
|
import {TemplateDiagnostic} from '../diagnostics';
|
||||||
|
|
||||||
import {CompletionEngine} from './completion';
|
import {CompletionEngine} from './completion';
|
||||||
import {InliningMode, ShimTypeCheckingData, TemplateData, TemplateOverride, TypeCheckContextImpl, TypeCheckingHost} from './context';
|
import {InliningMode, ShimTypeCheckingData, TemplateData, TypeCheckContextImpl, TypeCheckingHost} from './context';
|
||||||
import {shouldReportDiagnostic, translateDiagnostic} from './diagnostics';
|
import {shouldReportDiagnostic, translateDiagnostic} from './diagnostics';
|
||||||
import {TemplateSourceManager} from './source';
|
import {TemplateSourceManager} from './source';
|
||||||
import {findTypeCheckBlock, getTemplateMapping, TemplateSourceResolver} from './tcb_util';
|
import {findTypeCheckBlock, getTemplateMapping, TemplateSourceResolver} from './tcb_util';
|
||||||
|
@ -56,10 +56,9 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
/**
|
/**
|
||||||
* Stores directives and pipes that are in scope for each component.
|
* Stores directives and pipes that are in scope for each component.
|
||||||
*
|
*
|
||||||
* Unlike other caches, the scope of a component is not affected by its template, so this
|
* Unlike other caches, the scope of a component is not affected by its template. It will be
|
||||||
* cache does not need to be invalidate if the template is overridden. It will be destroyed when
|
* destroyed when the `ts.Program` changes and the `TemplateTypeCheckerImpl` as a whole is
|
||||||
* the `ts.Program` changes and the `TemplateTypeCheckerImpl` as a whole is destroyed and
|
* destroyed and replaced.
|
||||||
* replaced.
|
|
||||||
*/
|
*/
|
||||||
private scopeCache = new Map<ts.ClassDeclaration, ScopeData>();
|
private scopeCache = new Map<ts.ClassDeclaration, ScopeData>();
|
||||||
|
|
||||||
|
@ -67,10 +66,9 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
* Stores potential element tags for each component (a union of DOM tags as well as directive
|
* Stores potential element tags for each component (a union of DOM tags as well as directive
|
||||||
* tags).
|
* tags).
|
||||||
*
|
*
|
||||||
* Unlike other caches, the scope of a component is not affected by its template, so this
|
* Unlike other caches, the scope of a component is not affected by its template. It will be
|
||||||
* cache does not need to be invalidate if the template is overridden. It will be destroyed when
|
* destroyed when the `ts.Program` changes and the `TemplateTypeCheckerImpl` as a whole is
|
||||||
* the `ts.Program` changes and the `TemplateTypeCheckerImpl` as a whole is destroyed and
|
* destroyed and replaced.
|
||||||
* replaced.
|
|
||||||
*/
|
*/
|
||||||
private elementTagCache = new Map<ts.ClassDeclaration, Map<string, DirectiveInScope|null>>();
|
private elementTagCache = new Map<ts.ClassDeclaration, Map<string, DirectiveInScope|null>>();
|
||||||
|
|
||||||
|
@ -86,22 +84,6 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
private readonly componentScopeReader: ComponentScopeReader,
|
private readonly componentScopeReader: ComponentScopeReader,
|
||||||
private readonly typeCheckScopeRegistry: TypeCheckScopeRegistry) {}
|
private readonly typeCheckScopeRegistry: TypeCheckScopeRegistry) {}
|
||||||
|
|
||||||
resetOverrides(): void {
|
|
||||||
for (const fileRecord of this.state.values()) {
|
|
||||||
if (fileRecord.templateOverrides !== null) {
|
|
||||||
fileRecord.templateOverrides = null;
|
|
||||||
fileRecord.shimData.clear();
|
|
||||||
fileRecord.isComplete = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ideally only those components with overridden templates would have their caches invalidated,
|
|
||||||
// but the `TemplateTypeCheckerImpl` does not track the class for components with overrides. As
|
|
||||||
// a quick workaround, clear the entire cache instead.
|
|
||||||
this.completionCache.clear();
|
|
||||||
this.symbolBuilderCache.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
getTemplate(component: ts.ClassDeclaration): TmplAstNode[]|null {
|
getTemplate(component: ts.ClassDeclaration): TmplAstNode[]|null {
|
||||||
const {data} = this.getLatestComponentState(component);
|
const {data} = this.getLatestComponentState(component);
|
||||||
if (data === null) {
|
if (data === null) {
|
||||||
|
@ -151,42 +133,6 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
return {data, tcb, shimPath};
|
return {data, tcb, shimPath};
|
||||||
}
|
}
|
||||||
|
|
||||||
overrideComponentTemplate(component: ts.ClassDeclaration, template: string):
|
|
||||||
{nodes: TmplAstNode[], errors: ParseError[]|null} {
|
|
||||||
const {nodes, errors} = parseTemplate(template, 'override.html', {
|
|
||||||
// Set `leadingTriviaChars` and `preserveWhitespaces` such that whitespace is not stripped
|
|
||||||
// and fully accounted for in source spans. Without these flags the source spans can be
|
|
||||||
// inaccurate.
|
|
||||||
// Note: template parse options should be aligned with `template_target_spec.ts` and the
|
|
||||||
// `diagNodes` in `ComponentDecoratorHandler._parseTemplate`.
|
|
||||||
preserveWhitespaces: true,
|
|
||||||
leadingTriviaChars: [],
|
|
||||||
});
|
|
||||||
|
|
||||||
const filePath = absoluteFromSourceFile(component.getSourceFile());
|
|
||||||
|
|
||||||
const fileRecord = this.getFileData(filePath);
|
|
||||||
const id = fileRecord.sourceManager.getTemplateId(component);
|
|
||||||
|
|
||||||
if (fileRecord.templateOverrides === null) {
|
|
||||||
fileRecord.templateOverrides = new Map();
|
|
||||||
}
|
|
||||||
|
|
||||||
fileRecord.templateOverrides.set(id, {nodes, errors});
|
|
||||||
|
|
||||||
// Clear data for the shim in question, so it'll be regenerated on the next request.
|
|
||||||
const shimFile = this.typeCheckingStrategy.shimPathForComponent(component);
|
|
||||||
fileRecord.shimData.delete(shimFile);
|
|
||||||
fileRecord.isComplete = false;
|
|
||||||
this.isComplete = false;
|
|
||||||
|
|
||||||
// Overriding a component's template invalidates its cached results.
|
|
||||||
this.completionCache.delete(component);
|
|
||||||
this.symbolBuilderCache.delete(component);
|
|
||||||
|
|
||||||
return {nodes, errors};
|
|
||||||
}
|
|
||||||
|
|
||||||
isTrackedTypeCheckFile(filePath: AbsoluteFsPath): boolean {
|
isTrackedTypeCheckFile(filePath: AbsoluteFsPath): boolean {
|
||||||
return this.getFileAndShimRecordsForPath(filePath) !== null;
|
return this.getFileAndShimRecordsForPath(filePath) !== null;
|
||||||
}
|
}
|
||||||
|
@ -337,7 +283,6 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
|
|
||||||
fileData.shimData.delete(shimPath);
|
fileData.shimData.delete(shimPath);
|
||||||
fileData.isComplete = false;
|
fileData.isComplete = false;
|
||||||
fileData.templateOverrides?.delete(templateId);
|
|
||||||
|
|
||||||
this.isComplete = false;
|
this.isComplete = false;
|
||||||
}
|
}
|
||||||
|
@ -361,10 +306,6 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
const sfPath = absoluteFromSourceFile(sf);
|
const sfPath = absoluteFromSourceFile(sf);
|
||||||
if (this.state.has(sfPath)) {
|
if (this.state.has(sfPath)) {
|
||||||
const existingResults = this.state.get(sfPath)!;
|
const existingResults = this.state.get(sfPath)!;
|
||||||
if (existingResults.templateOverrides !== null) {
|
|
||||||
// Cannot adopt prior results if template overrides have been requested.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (existingResults.isComplete) {
|
if (existingResults.isComplete) {
|
||||||
// All data for this file has already been generated, so no need to adopt anything.
|
// All data for this file has already been generated, so no need to adopt anything.
|
||||||
|
@ -373,8 +314,7 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
}
|
}
|
||||||
|
|
||||||
const previousResults = this.priorBuild.priorTypeCheckingResultsFor(sf);
|
const previousResults = this.priorBuild.priorTypeCheckingResultsFor(sf);
|
||||||
if (previousResults === null || !previousResults.isComplete ||
|
if (previousResults === null || !previousResults.isComplete) {
|
||||||
previousResults.templateOverrides !== null) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -496,7 +436,6 @@ export class TemplateTypeCheckerImpl implements TemplateTypeChecker {
|
||||||
if (!this.state.has(path)) {
|
if (!this.state.has(path)) {
|
||||||
this.state.set(path, {
|
this.state.set(path, {
|
||||||
hasInlines: false,
|
hasInlines: false,
|
||||||
templateOverrides: null,
|
|
||||||
sourceManager: new TemplateSourceManager(),
|
sourceManager: new TemplateSourceManager(),
|
||||||
isComplete: false,
|
isComplete: false,
|
||||||
shimData: new Map(),
|
shimData: new Map(),
|
||||||
|
@ -679,11 +618,6 @@ export interface FileTypeCheckingData {
|
||||||
*/
|
*/
|
||||||
sourceManager: TemplateSourceManager;
|
sourceManager: TemplateSourceManager;
|
||||||
|
|
||||||
/**
|
|
||||||
* Map of template overrides applied to any components in this input file.
|
|
||||||
*/
|
|
||||||
templateOverrides: Map<TemplateId, TemplateOverride>|null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Data for each shim generated from this input file.
|
* Data for each shim generated from this input file.
|
||||||
*
|
*
|
||||||
|
@ -716,20 +650,6 @@ class WholeProgramTypeCheckingHost implements TypeCheckingHost {
|
||||||
return !fileData.shimData.has(shimPath);
|
return !fileData.shimData.has(shimPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
getTemplateOverride(sfPath: AbsoluteFsPath, node: ts.ClassDeclaration): TemplateOverride|null {
|
|
||||||
const fileData = this.impl.getFileData(sfPath);
|
|
||||||
if (fileData.templateOverrides === null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const templateId = fileData.sourceManager.getTemplateId(node);
|
|
||||||
if (fileData.templateOverrides.has(templateId)) {
|
|
||||||
return fileData.templateOverrides.get(templateId)!;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
recordShimData(sfPath: AbsoluteFsPath, data: ShimTypeCheckingData): void {
|
recordShimData(sfPath: AbsoluteFsPath, data: ShimTypeCheckingData): void {
|
||||||
const fileData = this.impl.getFileData(sfPath);
|
const fileData = this.impl.getFileData(sfPath);
|
||||||
fileData.shimData.set(data.path, data);
|
fileData.shimData.set(data.path, data);
|
||||||
|
@ -774,20 +694,6 @@ class SingleFileTypeCheckingHost implements TypeCheckingHost {
|
||||||
return !this.fileData.shimData.has(shimPath);
|
return !this.fileData.shimData.has(shimPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
getTemplateOverride(sfPath: AbsoluteFsPath, node: ts.ClassDeclaration): TemplateOverride|null {
|
|
||||||
this.assertPath(sfPath);
|
|
||||||
if (this.fileData.templateOverrides === null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const templateId = this.fileData.sourceManager.getTemplateId(node);
|
|
||||||
if (this.fileData.templateOverrides.has(templateId)) {
|
|
||||||
return this.fileData.templateOverrides.get(templateId)!;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
recordShimData(sfPath: AbsoluteFsPath, data: ShimTypeCheckingData): void {
|
recordShimData(sfPath: AbsoluteFsPath, data: ShimTypeCheckingData): void {
|
||||||
this.assertPath(sfPath);
|
this.assertPath(sfPath);
|
||||||
|
|
||||||
|
|
|
@ -51,11 +51,6 @@ export interface ShimTypeCheckingData {
|
||||||
templates: Map<TemplateId, TemplateData>;
|
templates: Map<TemplateId, TemplateData>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface TemplateOverride {
|
|
||||||
nodes: TmplAstNode[];
|
|
||||||
errors: ParseError[]|null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Data tracked for each template processed by the template type-checking system.
|
* Data tracked for each template processed by the template type-checking system.
|
||||||
*/
|
*/
|
||||||
|
@ -143,12 +138,6 @@ export interface TypeCheckingHost {
|
||||||
*/
|
*/
|
||||||
shouldCheckComponent(node: ts.ClassDeclaration): boolean;
|
shouldCheckComponent(node: ts.ClassDeclaration): boolean;
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the given component has had its template overridden, and retrieve the new template
|
|
||||||
* nodes if so.
|
|
||||||
*/
|
|
||||||
getTemplateOverride(sfPath: AbsoluteFsPath, node: ts.ClassDeclaration): TemplateOverride|null;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Report data from a shim generated from the given input file path.
|
* Report data from a shim generated from the given input file path.
|
||||||
*/
|
*/
|
||||||
|
@ -225,13 +214,6 @@ export class TypeCheckContextImpl implements TypeCheckContext {
|
||||||
|
|
||||||
const templateDiagnostics: TemplateDiagnostic[] = [];
|
const templateDiagnostics: TemplateDiagnostic[] = [];
|
||||||
|
|
||||||
const sfPath = absoluteFromSourceFile(ref.node.getSourceFile());
|
|
||||||
const overrideTemplate = this.host.getTemplateOverride(sfPath, ref.node);
|
|
||||||
if (overrideTemplate !== null) {
|
|
||||||
template = overrideTemplate.nodes;
|
|
||||||
parseErrors = overrideTemplate.errors;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parseErrors !== null) {
|
if (parseErrors !== null) {
|
||||||
templateDiagnostics.push(
|
templateDiagnostics.push(
|
||||||
...this.getTemplateDiagnostics(parseErrors, templateId, sourceMapping));
|
...this.getTemplateDiagnostics(parseErrors, templateId, sourceMapping));
|
||||||
|
|
|
@ -71,32 +71,6 @@ runInEachFileSystem(() => {
|
||||||
expect(userAtTopLevel.kind).toBe(CompletionKind.Reference);
|
expect(userAtTopLevel.kind).toBe(CompletionKind.Reference);
|
||||||
expect(userInNgFor.kind).toBe(CompletionKind.Variable);
|
expect(userInNgFor.kind).toBe(CompletionKind.Variable);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should invalidate cached completions when overrides change', () => {
|
|
||||||
// The template starts with a #foo local reference.
|
|
||||||
const {completions: before, templateTypeChecker, component} =
|
|
||||||
setupCompletions('<div #foo></div>');
|
|
||||||
expect(Array.from(before.templateContext.keys())).toEqual(['foo']);
|
|
||||||
|
|
||||||
// Override the template and change the name of the local reference to #bar. This should
|
|
||||||
// invalidate any cached completions.
|
|
||||||
templateTypeChecker.overrideComponentTemplate(component, '<div #bar></div>');
|
|
||||||
|
|
||||||
// Fresh completions should include the #bar reference instead.
|
|
||||||
const afterOverride =
|
|
||||||
templateTypeChecker.getGlobalCompletions(/* root template */ null, component)!;
|
|
||||||
expect(afterOverride).toBeDefined();
|
|
||||||
expect(Array.from(afterOverride.templateContext.keys())).toEqual(['bar']);
|
|
||||||
|
|
||||||
// Reset the template to its original. This should also invalidate any cached completions.
|
|
||||||
templateTypeChecker.resetOverrides();
|
|
||||||
|
|
||||||
// Fresh completions should include the original #foo now.
|
|
||||||
const afterReset =
|
|
||||||
templateTypeChecker.getGlobalCompletions(/* root template */ null, component)!;
|
|
||||||
expect(afterReset).toBeDefined();
|
|
||||||
expect(Array.from(afterReset.templateContext.keys())).toEqual(['foo']);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('TemplateTypeChecker scopes', () => {
|
describe('TemplateTypeChecker scopes', () => {
|
||||||
|
|
|
@ -40,33 +40,6 @@ runInEachFileSystem(() => {
|
||||||
assertElementSymbol(symbol.host);
|
assertElementSymbol(symbol.host);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should invalidate symbols when template overrides change', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
const templateString = `<div id="helloWorld"></div>`;
|
|
||||||
const {templateTypeChecker, program} = setup(
|
|
||||||
[
|
|
||||||
{
|
|
||||||
fileName,
|
|
||||||
templates: {'Cmp': templateString},
|
|
||||||
source: `export class Cmp {}`,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
);
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const cmp = getClass(sf, 'Cmp');
|
|
||||||
const {attributes: beforeAttributes} = getAstElements(templateTypeChecker, cmp)[0];
|
|
||||||
const beforeSymbol = templateTypeChecker.getSymbolOfNode(beforeAttributes[0], cmp)!;
|
|
||||||
|
|
||||||
// Replace the <div> with a <span>.
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp, '<span id="helloWorld"></span>');
|
|
||||||
|
|
||||||
const {attributes: afterAttributes} = getAstElements(templateTypeChecker, cmp)[0];
|
|
||||||
const afterSymbol = templateTypeChecker.getSymbolOfNode(afterAttributes[0], cmp)!;
|
|
||||||
|
|
||||||
// After the override, the symbol cache should have been invalidated.
|
|
||||||
expect(beforeSymbol).not.toBe(afterSymbol);
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('should get a symbol for text attributes corresponding with a directive input', () => {
|
describe('should get a symbol for text attributes corresponding with a directive input', () => {
|
||||||
let fileName: AbsoluteFsPath;
|
let fileName: AbsoluteFsPath;
|
||||||
let targets: TypeCheckingTarget[];
|
let targets: TypeCheckingTarget[];
|
||||||
|
|
|
@ -211,143 +211,6 @@ runInEachFileSystem(() => {
|
||||||
expect(absoluteFromSourceFile(diags[0].relatedInformation![0].file!)).toBe(dirSf.fileName);
|
expect(absoluteFromSourceFile(diags[0].relatedInformation![0].file!)).toBe(dirSf.fileName);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('template overrides', () => {
|
|
||||||
it('should override a simple template', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
const {program, templateTypeChecker} = setup([{
|
|
||||||
fileName,
|
|
||||||
templates: {'Cmp': '<div>{{original}}</div>'},
|
|
||||||
}]);
|
|
||||||
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const cmp = getClass(sf, 'Cmp');
|
|
||||||
|
|
||||||
const tcbReal = templateTypeChecker.getTypeCheckBlock(cmp)!;
|
|
||||||
expect(tcbReal.getText()).toContain('original');
|
|
||||||
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp, '<div>{{override}}</div>');
|
|
||||||
const tcbOverridden = templateTypeChecker.getTypeCheckBlock(cmp);
|
|
||||||
expect(tcbOverridden).not.toBeNull();
|
|
||||||
expect(tcbOverridden!.getText()).not.toContain('original');
|
|
||||||
expect(tcbOverridden!.getText()).toContain('override');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should clear overrides on request', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
const {program, templateTypeChecker} = setup([{
|
|
||||||
fileName,
|
|
||||||
templates: {'Cmp': '<div>{{original}}</div>'},
|
|
||||||
}]);
|
|
||||||
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const cmp = getClass(sf, 'Cmp');
|
|
||||||
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp, '<div>{{override}}</div>');
|
|
||||||
const tcbOverridden = templateTypeChecker.getTypeCheckBlock(cmp)!;
|
|
||||||
expect(tcbOverridden.getText()).not.toContain('original');
|
|
||||||
expect(tcbOverridden.getText()).toContain('override');
|
|
||||||
|
|
||||||
templateTypeChecker.resetOverrides();
|
|
||||||
|
|
||||||
// The template should be back to the original.
|
|
||||||
const tcbReal = templateTypeChecker.getTypeCheckBlock(cmp)!;
|
|
||||||
expect(tcbReal.getText()).toContain('original');
|
|
||||||
expect(tcbReal.getText()).not.toContain('override');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should override a template and make use of previously unused directives', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
const dirFile = absoluteFrom('/dir.ts');
|
|
||||||
const {program, templateTypeChecker} = setup(
|
|
||||||
[
|
|
||||||
{
|
|
||||||
fileName,
|
|
||||||
source: `export class Cmp {}`,
|
|
||||||
templates: {'Cmp': '<div></div>'},
|
|
||||||
declarations: [{
|
|
||||||
name: 'TestDir',
|
|
||||||
selector: '[dir]',
|
|
||||||
file: dirFile,
|
|
||||||
type: 'directive',
|
|
||||||
inputs: {'input': 'input'},
|
|
||||||
isGeneric: true,
|
|
||||||
}]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fileName: dirFile,
|
|
||||||
source: `export class TestDir<T> {}`,
|
|
||||||
templates: {},
|
|
||||||
}
|
|
||||||
],
|
|
||||||
{inlining: false});
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const cmp = getClass(sf, 'Cmp');
|
|
||||||
|
|
||||||
// TestDir is initially unused. Note that this checks the entire text of the ngtypecheck
|
|
||||||
// file, to ensure it captures not just the TCB function but also any inline type
|
|
||||||
// constructors.
|
|
||||||
const tcbReal = templateTypeChecker.getTypeCheckBlock(cmp)!;
|
|
||||||
expect(tcbReal.getSourceFile().text).not.toContain('TestDir');
|
|
||||||
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp, '<div dir [input]="value"></div>');
|
|
||||||
|
|
||||||
const tcbOverridden = templateTypeChecker.getTypeCheckBlock(cmp);
|
|
||||||
expect(tcbOverridden).not.toBeNull();
|
|
||||||
expect(tcbOverridden!.getSourceFile().text).toContain('TestDir');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should not invalidate other templates when an override is requested', () => {
|
|
||||||
const file1 = absoluteFrom('/file1.ts');
|
|
||||||
const file2 = absoluteFrom('/file2.ts');
|
|
||||||
const {program, templateTypeChecker, programStrategy} = setup([
|
|
||||||
{fileName: file1, templates: {'Cmp1': '<div></div>'}},
|
|
||||||
{fileName: file2, templates: {'Cmp2': '<span></span>'}}
|
|
||||||
]);
|
|
||||||
|
|
||||||
const cmp1 = getClass(getSourceFileOrError(program, file1), 'Cmp1');
|
|
||||||
const cmp2 = getClass(getSourceFileOrError(program, file2), 'Cmp2');
|
|
||||||
|
|
||||||
// To test this scenario, Cmp1's type check block will be captured, then Cmp2's template
|
|
||||||
// will be overridden. Cmp1's type check block should not change as a result.
|
|
||||||
const originalTcb = templateTypeChecker.getTypeCheckBlock(cmp1)!;
|
|
||||||
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp2, '<p></p>');
|
|
||||||
|
|
||||||
// Trigger generation of the TCB for Cmp2.
|
|
||||||
templateTypeChecker.getTypeCheckBlock(cmp2);
|
|
||||||
|
|
||||||
// Verify that Cmp1's TCB has not changed.
|
|
||||||
const currentTcb = templateTypeChecker.getTypeCheckBlock(cmp1)!;
|
|
||||||
expect(currentTcb).toBe(originalTcb);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should allow get diagnostics for a single component', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
|
|
||||||
const {program, templateTypeChecker} = setup([{
|
|
||||||
fileName,
|
|
||||||
templates: {
|
|
||||||
'Cmp1': '<invalid-element-a></invalid-element-a>',
|
|
||||||
'Cmp2': '<invalid-element-b></invalid-element-b>'
|
|
||||||
},
|
|
||||||
}]);
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const cmp1 = getClass(sf, 'Cmp1');
|
|
||||||
const cmp2 = getClass(sf, 'Cmp2');
|
|
||||||
|
|
||||||
const diags1 = templateTypeChecker.getDiagnosticsForComponent(cmp1);
|
|
||||||
expect(diags1.length).toBe(1);
|
|
||||||
expect(diags1[0].messageText).toContain('invalid-element-a');
|
|
||||||
expect(diags1[0].messageText).not.toContain('invalid-element-b');
|
|
||||||
|
|
||||||
const diags2 = templateTypeChecker.getDiagnosticsForComponent(cmp2);
|
|
||||||
expect(diags2.length).toBe(1);
|
|
||||||
expect(diags2[0].messageText).toContain('invalid-element-b');
|
|
||||||
expect(diags2[0].messageText).not.toContain('invalid-element-a');
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('getTemplateOfComponent()', () => {
|
describe('getTemplateOfComponent()', () => {
|
||||||
it('should provide access to a component\'s real template', () => {
|
it('should provide access to a component\'s real template', () => {
|
||||||
const fileName = absoluteFrom('/main.ts');
|
const fileName = absoluteFrom('/main.ts');
|
||||||
|
@ -363,24 +226,6 @@ runInEachFileSystem(() => {
|
||||||
expect(nodes).not.toBeNull();
|
expect(nodes).not.toBeNull();
|
||||||
expect(nodes[0].sourceSpan.start.file.content).toBe('<div>Template</div>');
|
expect(nodes[0].sourceSpan.start.file.content).toBe('<div>Template</div>');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should provide access to an overridden template', () => {
|
|
||||||
const fileName = absoluteFrom('/main.ts');
|
|
||||||
const {program, templateTypeChecker} = setup([{
|
|
||||||
fileName,
|
|
||||||
templates: {
|
|
||||||
'Cmp': '<div>Template</div>',
|
|
||||||
},
|
|
||||||
}]);
|
|
||||||
const cmp = getClass(getSourceFileOrError(program, fileName), 'Cmp');
|
|
||||||
|
|
||||||
templateTypeChecker.overrideComponentTemplate(cmp, '<div>Overridden</div>');
|
|
||||||
templateTypeChecker.getDiagnosticsForComponent(cmp);
|
|
||||||
|
|
||||||
const nodes = templateTypeChecker.getTemplate(cmp)!;
|
|
||||||
expect(nodes).not.toBeNull();
|
|
||||||
expect(nodes[0].sourceSpan.start.file.content).toBe('<div>Overridden</div>');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -32,8 +32,7 @@ function parse(template: string): ParseResult {
|
||||||
// and fully accounted for in source spans. Without these flags the source spans can be
|
// and fully accounted for in source spans. Without these flags the source spans can be
|
||||||
// inaccurate.
|
// inaccurate.
|
||||||
// Note: template parse options should be aligned with the `diagNodes` in
|
// Note: template parse options should be aligned with the `diagNodes` in
|
||||||
// `ComponentDecoratorHandler._parseTemplate`. and
|
// `ComponentDecoratorHandler._parseTemplate`.
|
||||||
// `TemplateTypeCheckerImpl.overrideComponentTemplate`.
|
|
||||||
leadingTriviaChars: [],
|
leadingTriviaChars: [],
|
||||||
preserveWhitespaces: true,
|
preserveWhitespaces: true,
|
||||||
}),
|
}),
|
||||||
|
|
Loading…
Reference in New Issue