refactor(compiler-cli): Store inline templates and styles in the resource registry (#39482)
The Language Service is not only interested in external resources, but also inline styles and templates. By storing the expression of the inline resources, we can more easily determine if a given position is part of the inline template/style expression. PR Close #39482
This commit is contained in:
parent
3241d922fc
commit
beb935613e
|
@ -262,7 +262,7 @@ export class ComponentDecoratorHandler implements
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const templateResource = template.isInline ?
|
const templateResource = template.isInline ?
|
||||||
null :
|
{path: null, expression: component.get('template')!} :
|
||||||
{path: absoluteFrom(template.templateUrl), expression: template.sourceMapping.node};
|
{path: absoluteFrom(template.templateUrl), expression: template.sourceMapping.node};
|
||||||
|
|
||||||
let diagnostics: ts.Diagnostic[]|undefined = undefined;
|
let diagnostics: ts.Diagnostic[]|undefined = undefined;
|
||||||
|
@ -666,21 +666,30 @@ export class ComponentDecoratorHandler implements
|
||||||
|
|
||||||
private _extractStyleResources(component: Map<string, ts.Expression>, containingFile: string):
|
private _extractStyleResources(component: Map<string, ts.Expression>, containingFile: string):
|
||||||
ReadonlySet<Resource> {
|
ReadonlySet<Resource> {
|
||||||
const styleUrlsExpr = component.get('styleUrls');
|
const styles = new Set<Resource>();
|
||||||
// If styleUrls is a literal array of strings, process each resource url individually and
|
function stringLiteralElements(array: ts.ArrayLiteralExpression): ts.StringLiteralLike[] {
|
||||||
// register them. Otherwise, give up and return an empty set.
|
return array.elements.filter(
|
||||||
if (styleUrlsExpr === undefined || !ts.isArrayLiteralExpression(styleUrlsExpr) ||
|
(e: ts.Expression): e is ts.StringLiteralLike => ts.isStringLiteralLike(e));
|
||||||
!styleUrlsExpr.elements.every(e => ts.isStringLiteralLike(e))) {
|
|
||||||
return new Set();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const externalStyles = new Set<Resource>();
|
// If styleUrls is a literal array, process each resource url individually and
|
||||||
for (const expression of styleUrlsExpr.elements) {
|
// register ones that are string literals.
|
||||||
const resourceUrl =
|
const styleUrlsExpr = component.get('styleUrls');
|
||||||
this.resourceLoader.resolve((expression as ts.StringLiteralLike).text, containingFile);
|
if (styleUrlsExpr !== undefined && ts.isArrayLiteralExpression(styleUrlsExpr)) {
|
||||||
externalStyles.add({path: absoluteFrom(resourceUrl), expression});
|
for (const expression of stringLiteralElements(styleUrlsExpr)) {
|
||||||
|
const resourceUrl = this.resourceLoader.resolve(expression.text, containingFile);
|
||||||
|
styles.add({path: absoluteFrom(resourceUrl), expression});
|
||||||
}
|
}
|
||||||
return externalStyles;
|
}
|
||||||
|
|
||||||
|
const stylesExpr = component.get('styles');
|
||||||
|
if (stylesExpr !== undefined && ts.isArrayLiteralExpression(stylesExpr)) {
|
||||||
|
for (const expression of stringLiteralElements(stylesExpr)) {
|
||||||
|
styles.add({path: null, expression});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return styles;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _preloadAndParseTemplate(
|
private _preloadAndParseTemplate(
|
||||||
|
|
|
@ -18,18 +18,49 @@ import {getDeclaration, makeProgram} from '../../testing';
|
||||||
import {ResourceLoader} from '../src/api';
|
import {ResourceLoader} from '../src/api';
|
||||||
import {ComponentDecoratorHandler} from '../src/component';
|
import {ComponentDecoratorHandler} from '../src/component';
|
||||||
|
|
||||||
export class NoopResourceLoader implements ResourceLoader {
|
export class StubResourceLoader implements ResourceLoader {
|
||||||
resolve(): string {
|
resolve(v: string): string {
|
||||||
throw new Error('Not implemented.');
|
return v;
|
||||||
}
|
}
|
||||||
canPreload = false;
|
canPreload = false;
|
||||||
load(): string {
|
load(v: string): string {
|
||||||
throw new Error('Not implemented');
|
return '';
|
||||||
}
|
}
|
||||||
preload(): Promise<void>|undefined {
|
preload(): Promise<void>|undefined {
|
||||||
throw new Error('Not implemented');
|
throw new Error('Not implemented');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function setup(program: ts.Program, options: ts.CompilerOptions, host: ts.CompilerHost) {
|
||||||
|
const checker = program.getTypeChecker();
|
||||||
|
const reflectionHost = new TypeScriptReflectionHost(checker);
|
||||||
|
const evaluator = new PartialEvaluator(reflectionHost, checker, /* dependencyTracker */ null);
|
||||||
|
const moduleResolver =
|
||||||
|
new ModuleResolver(program, options, host, /* moduleResolutionCache */ null);
|
||||||
|
const importGraph = new ImportGraph(moduleResolver);
|
||||||
|
const cycleAnalyzer = new CycleAnalyzer(importGraph);
|
||||||
|
const metaRegistry = new LocalMetadataRegistry();
|
||||||
|
const dtsReader = new DtsMetadataReader(checker, reflectionHost);
|
||||||
|
const scopeRegistry = new LocalModuleScopeRegistry(
|
||||||
|
metaRegistry, new MetadataDtsModuleScopeResolver(dtsReader, null), new ReferenceEmitter([]),
|
||||||
|
null);
|
||||||
|
const metaReader = new CompoundMetadataReader([metaRegistry, dtsReader]);
|
||||||
|
const refEmitter = new ReferenceEmitter([]);
|
||||||
|
const injectableRegistry = new InjectableClassRegistry(reflectionHost);
|
||||||
|
const resourceRegistry = new ResourceRegistry();
|
||||||
|
|
||||||
|
const handler = new ComponentDecoratorHandler(
|
||||||
|
reflectionHost, evaluator, metaRegistry, metaReader, scopeRegistry, scopeRegistry,
|
||||||
|
resourceRegistry,
|
||||||
|
/* isCore */ false, new StubResourceLoader(), /* rootDirs */['/'],
|
||||||
|
/* defaultPreserveWhitespaces */ false, /* i18nUseExternalIds */ true,
|
||||||
|
/* enableI18nLegacyMessageIdFormat */ false,
|
||||||
|
/* i18nNormalizeLineEndingsInICUs */ undefined, moduleResolver, cycleAnalyzer, refEmitter,
|
||||||
|
NOOP_DEFAULT_IMPORT_RECORDER, /* depTracker */ null, injectableRegistry,
|
||||||
|
/* annotateForClosureCompiler */ false);
|
||||||
|
return {reflectionHost, handler};
|
||||||
|
}
|
||||||
|
|
||||||
runInEachFileSystem(() => {
|
runInEachFileSystem(() => {
|
||||||
describe('ComponentDecoratorHandler', () => {
|
describe('ComponentDecoratorHandler', () => {
|
||||||
let _: typeof absoluteFrom;
|
let _: typeof absoluteFrom;
|
||||||
|
@ -51,31 +82,7 @@ runInEachFileSystem(() => {
|
||||||
`
|
`
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
const checker = program.getTypeChecker();
|
const {reflectionHost, handler} = setup(program, options, host);
|
||||||
const reflectionHost = new TypeScriptReflectionHost(checker);
|
|
||||||
const evaluator = new PartialEvaluator(reflectionHost, checker, /* dependencyTracker */ null);
|
|
||||||
const moduleResolver =
|
|
||||||
new ModuleResolver(program, options, host, /* moduleResolutionCache */ null);
|
|
||||||
const importGraph = new ImportGraph(moduleResolver);
|
|
||||||
const cycleAnalyzer = new CycleAnalyzer(importGraph);
|
|
||||||
const metaRegistry = new LocalMetadataRegistry();
|
|
||||||
const dtsReader = new DtsMetadataReader(checker, reflectionHost);
|
|
||||||
const scopeRegistry = new LocalModuleScopeRegistry(
|
|
||||||
metaRegistry, new MetadataDtsModuleScopeResolver(dtsReader, null),
|
|
||||||
new ReferenceEmitter([]), null);
|
|
||||||
const metaReader = new CompoundMetadataReader([metaRegistry, dtsReader]);
|
|
||||||
const refEmitter = new ReferenceEmitter([]);
|
|
||||||
const injectableRegistry = new InjectableClassRegistry(reflectionHost);
|
|
||||||
|
|
||||||
const handler = new ComponentDecoratorHandler(
|
|
||||||
reflectionHost, evaluator, metaRegistry, metaReader, scopeRegistry, scopeRegistry,
|
|
||||||
new ResourceRegistry(),
|
|
||||||
/* isCore */ false, new NoopResourceLoader(), /* rootDirs */[''],
|
|
||||||
/* defaultPreserveWhitespaces */ false, /* i18nUseExternalIds */ true,
|
|
||||||
/* enableI18nLegacyMessageIdFormat */ false,
|
|
||||||
/* i18nNormalizeLineEndingsInICUs */ undefined, moduleResolver, cycleAnalyzer, refEmitter,
|
|
||||||
NOOP_DEFAULT_IMPORT_RECORDER, /* depTracker */ null, injectableRegistry,
|
|
||||||
/* annotateForClosureCompiler */ false);
|
|
||||||
const TestCmp = getDeclaration(program, _('/entry.ts'), 'TestCmp', isNamedClassDeclaration);
|
const TestCmp = getDeclaration(program, _('/entry.ts'), 'TestCmp', isNamedClassDeclaration);
|
||||||
const detected = handler.detect(TestCmp, reflectionHost.getDecoratorsOfDeclaration(TestCmp));
|
const detected = handler.detect(TestCmp, reflectionHost.getDecoratorsOfDeclaration(TestCmp));
|
||||||
if (detected === undefined) {
|
if (detected === undefined) {
|
||||||
|
@ -94,6 +101,104 @@ runInEachFileSystem(() => {
|
||||||
expect(diag.start).toBe(detected.metadata.args![0].getStart());
|
expect(diag.start).toBe(detected.metadata.args![0].getStart());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should keep track of inline template', () => {
|
||||||
|
const template = '<span>inline</span>';
|
||||||
|
const {program, options, host} = makeProgram([
|
||||||
|
{
|
||||||
|
name: _('/node_modules/@angular/core/index.d.ts'),
|
||||||
|
contents: 'export const Component: any;',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/entry.ts'),
|
||||||
|
contents: `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: '${template}',
|
||||||
|
}) class TestCmp {}
|
||||||
|
`
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const {reflectionHost, handler} = setup(program, options, host);
|
||||||
|
const TestCmp = getDeclaration(program, _('/entry.ts'), 'TestCmp', isNamedClassDeclaration);
|
||||||
|
const detected = handler.detect(TestCmp, reflectionHost.getDecoratorsOfDeclaration(TestCmp));
|
||||||
|
if (detected === undefined) {
|
||||||
|
return fail('Failed to recognize @Component');
|
||||||
|
}
|
||||||
|
const {analysis} = handler.analyze(TestCmp, detected.metadata);
|
||||||
|
expect(analysis?.resources.template.path).toBeNull();
|
||||||
|
expect(analysis?.resources.template.expression.getText()).toEqual(`'${template}'`);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should keep track of external template', () => {
|
||||||
|
const templateUrl = '/myTemplate.ng.html';
|
||||||
|
const {program, options, host} = makeProgram([
|
||||||
|
{
|
||||||
|
name: _('/node_modules/@angular/core/index.d.ts'),
|
||||||
|
contents: 'export const Component: any;',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _(templateUrl),
|
||||||
|
contents: '<div>hello world</div>',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/entry.ts'),
|
||||||
|
contents: `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
templateUrl: '${templateUrl}',
|
||||||
|
}) class TestCmp {}
|
||||||
|
`
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const {reflectionHost, handler} = setup(program, options, host);
|
||||||
|
const TestCmp = getDeclaration(program, _('/entry.ts'), 'TestCmp', isNamedClassDeclaration);
|
||||||
|
const detected = handler.detect(TestCmp, reflectionHost.getDecoratorsOfDeclaration(TestCmp));
|
||||||
|
if (detected === undefined) {
|
||||||
|
return fail('Failed to recognize @Component');
|
||||||
|
}
|
||||||
|
const {analysis} = handler.analyze(TestCmp, detected.metadata);
|
||||||
|
expect(analysis?.resources.template.path).toContain(templateUrl);
|
||||||
|
expect(analysis?.resources.template.expression.getText()).toContain(`'${templateUrl}'`);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should keep track of internal and external styles', () => {
|
||||||
|
const {program, options, host} = makeProgram([
|
||||||
|
{
|
||||||
|
name: _('/node_modules/@angular/core/index.d.ts'),
|
||||||
|
contents: 'export const Component: any;',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/myStyle.css'),
|
||||||
|
contents: '<div>hello world</div>',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: _('/entry.ts'),
|
||||||
|
contents: `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
// These are ignored because we only attempt to store string literals
|
||||||
|
const ignoredStyleUrl = 'asdlfkj';
|
||||||
|
const ignoredStyle = '';
|
||||||
|
@Component({
|
||||||
|
template: '',
|
||||||
|
styleUrls: ['/myStyle.css', ignoredStyleUrl],
|
||||||
|
styles: ['a { color: red; }', 'b { color: blue; }', ignoredStyle, ...[ignoredStyle]],
|
||||||
|
}) class TestCmp {}
|
||||||
|
`
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const {reflectionHost, handler} = setup(program, options, host);
|
||||||
|
const TestCmp = getDeclaration(program, _('/entry.ts'), 'TestCmp', isNamedClassDeclaration);
|
||||||
|
const detected = handler.detect(TestCmp, reflectionHost.getDecoratorsOfDeclaration(TestCmp));
|
||||||
|
if (detected === undefined) {
|
||||||
|
return fail('Failed to recognize @Component');
|
||||||
|
}
|
||||||
|
const {analysis} = handler.analyze(TestCmp, detected.metadata);
|
||||||
|
expect(analysis?.resources.styles.size).toBe(3);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function ivyCode(code: ErrorCode): number {
|
function ivyCode(code: ErrorCode): number {
|
||||||
|
|
|
@ -261,6 +261,9 @@ export class NgCompiler {
|
||||||
const {resourceRegistry} = this.ensureAnalyzed();
|
const {resourceRegistry} = this.ensureAnalyzed();
|
||||||
const styles = resourceRegistry.getStyles(classDecl);
|
const styles = resourceRegistry.getStyles(classDecl);
|
||||||
const template = resourceRegistry.getTemplate(classDecl);
|
const template = resourceRegistry.getTemplate(classDecl);
|
||||||
|
if (template === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return {styles, template};
|
return {styles, template};
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,23 +12,24 @@ import {AbsoluteFsPath} from '../../file_system';
|
||||||
import {ClassDeclaration} from '../../reflection';
|
import {ClassDeclaration} from '../../reflection';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents an external resource for a component and contains the `AbsoluteFsPath`
|
* Represents an resource for a component and contains the `AbsoluteFsPath`
|
||||||
* to the file which was resolved by evaluating the `ts.Expression` (generally, a relative or
|
* to the file which was resolved by evaluating the `ts.Expression` (generally, a relative or
|
||||||
* absolute string path to the resource).
|
* absolute string path to the resource).
|
||||||
|
*
|
||||||
|
* If the resource is inline, the `path` will be `null`.
|
||||||
*/
|
*/
|
||||||
export interface Resource {
|
export interface Resource {
|
||||||
path: AbsoluteFsPath;
|
path: AbsoluteFsPath|null;
|
||||||
expression: ts.Expression;
|
expression: ts.Expression;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents the external resources of a component.
|
* Represents the either inline or external resources of a component.
|
||||||
*
|
*
|
||||||
* If the component uses an inline template, the template resource will be `null`.
|
* A resource with a `path` of `null` is considered inline.
|
||||||
* If the component does not have external styles, the `styles` `Set` will be empty.
|
|
||||||
*/
|
*/
|
||||||
export interface ComponentResources {
|
export interface ComponentResources {
|
||||||
template: Resource|null;
|
template: Resource;
|
||||||
styles: ReadonlySet<Resource>;
|
styles: ReadonlySet<Resource>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,17 +41,17 @@ export interface ComponentResources {
|
||||||
* assistance.
|
* assistance.
|
||||||
*/
|
*/
|
||||||
export class ResourceRegistry {
|
export class ResourceRegistry {
|
||||||
private templateToComponentsMap = new Map<AbsoluteFsPath, Set<ClassDeclaration>>();
|
private externalTemplateToComponentsMap = new Map<AbsoluteFsPath, Set<ClassDeclaration>>();
|
||||||
private componentToTemplateMap = new Map<ClassDeclaration, Resource>();
|
private componentToTemplateMap = new Map<ClassDeclaration, Resource>();
|
||||||
private componentToStylesMap = new Map<ClassDeclaration, Set<Resource>>();
|
private componentToStylesMap = new Map<ClassDeclaration, Set<Resource>>();
|
||||||
private styleToComponentsMap = new Map<AbsoluteFsPath, Set<ClassDeclaration>>();
|
private externalStyleToComponentsMap = new Map<AbsoluteFsPath, Set<ClassDeclaration>>();
|
||||||
|
|
||||||
getComponentsWithTemplate(template: AbsoluteFsPath): ReadonlySet<ClassDeclaration> {
|
getComponentsWithTemplate(template: AbsoluteFsPath): ReadonlySet<ClassDeclaration> {
|
||||||
if (!this.templateToComponentsMap.has(template)) {
|
if (!this.externalTemplateToComponentsMap.has(template)) {
|
||||||
return new Set();
|
return new Set();
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.templateToComponentsMap.get(template)!;
|
return this.externalTemplateToComponentsMap.get(template)!;
|
||||||
}
|
}
|
||||||
|
|
||||||
registerResources(resources: ComponentResources, component: ClassDeclaration) {
|
registerResources(resources: ComponentResources, component: ClassDeclaration) {
|
||||||
|
@ -64,10 +65,12 @@ export class ResourceRegistry {
|
||||||
|
|
||||||
registerTemplate(templateResource: Resource, component: ClassDeclaration): void {
|
registerTemplate(templateResource: Resource, component: ClassDeclaration): void {
|
||||||
const {path} = templateResource;
|
const {path} = templateResource;
|
||||||
if (!this.templateToComponentsMap.has(path)) {
|
if (path !== null) {
|
||||||
this.templateToComponentsMap.set(path, new Set());
|
if (!this.externalTemplateToComponentsMap.has(path)) {
|
||||||
|
this.externalTemplateToComponentsMap.set(path, new Set());
|
||||||
|
}
|
||||||
|
this.externalTemplateToComponentsMap.get(path)!.add(component);
|
||||||
}
|
}
|
||||||
this.templateToComponentsMap.get(path)!.add(component);
|
|
||||||
this.componentToTemplateMap.set(component, templateResource);
|
this.componentToTemplateMap.set(component, templateResource);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -83,10 +86,12 @@ export class ResourceRegistry {
|
||||||
if (!this.componentToStylesMap.has(component)) {
|
if (!this.componentToStylesMap.has(component)) {
|
||||||
this.componentToStylesMap.set(component, new Set());
|
this.componentToStylesMap.set(component, new Set());
|
||||||
}
|
}
|
||||||
if (!this.styleToComponentsMap.has(path)) {
|
if (path !== null) {
|
||||||
this.styleToComponentsMap.set(path, new Set());
|
if (!this.externalStyleToComponentsMap.has(path)) {
|
||||||
|
this.externalStyleToComponentsMap.set(path, new Set());
|
||||||
|
}
|
||||||
|
this.externalStyleToComponentsMap.get(path)!.add(component);
|
||||||
}
|
}
|
||||||
this.styleToComponentsMap.get(path)!.add(component);
|
|
||||||
this.componentToStylesMap.get(component)!.add(styleResource);
|
this.componentToStylesMap.get(component)!.add(styleResource);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,10 +103,10 @@ export class ResourceRegistry {
|
||||||
}
|
}
|
||||||
|
|
||||||
getComponentsWithStyle(styleUrl: AbsoluteFsPath): ReadonlySet<ClassDeclaration> {
|
getComponentsWithStyle(styleUrl: AbsoluteFsPath): ReadonlySet<ClassDeclaration> {
|
||||||
if (!this.styleToComponentsMap.has(styleUrl)) {
|
if (!this.externalStyleToComponentsMap.has(styleUrl)) {
|
||||||
return new Set();
|
return new Set();
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.styleToComponentsMap.get(styleUrl)!;
|
return this.externalStyleToComponentsMap.get(styleUrl)!;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue