angular/packages/compiler-cli/test/ngtsc/authoring_queries_spec.ts
Kristiyan Kostadinov ae1c0dc490 perf(compiler): chain query creation instructions
We always emit the query creation instructions in a group which makes them good candidates for chaining.
2025-12-09 09:24:36 -08:00

535 lines
16 KiB
TypeScript

/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.dev/license
*/
import ts from 'typescript';
import {runInEachFileSystem} from '../../src/ngtsc/file_system/testing';
import {loadStandardTestFiles} from '../../src/ngtsc/testing';
import {NgtscTestEnvironment} from './env';
const testFiles = loadStandardTestFiles();
runInEachFileSystem(() => {
describe('signal-based queries', () => {
let env!: NgtscTestEnvironment;
beforeEach(() => {
env = NgtscTestEnvironment.setup(testFiles);
env.tsconfig({strictTemplates: true});
});
it('should handle a basic viewChild', () => {
env.write(
'test.ts',
`
import {Component, viewChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should support viewChild with `read` options', () => {
env.write('other-file.ts', `export class X {}`);
env.write(
'test.ts',
`
import {Component, viewChild} from '@angular/core';
import * as fromOtherFile from './other-file';
class X {}
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator', {read: X});
el2 = viewChild('myLocator', {read: fromOtherFile.X});
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(
`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5, X)(ctx.el2, _c0, 5, fromOtherFile.X);`,
);
expect(js).toContain(`i0.ɵɵqueryAdvance(2);`);
});
it('should support viewChild with `read` pointing to an expression with a generic', () => {
env.write(
'test.ts',
`
import {Component, viewChild, ElementRef} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator', {read: ElementRef<HTMLElement>});
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5, ElementRef);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should support viewChild with `read` pointing to a parenthesized expression', () => {
env.write(
'test.ts',
`
import {Component, viewChild, ElementRef} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator', {read: ((((ElementRef))))});
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5, ElementRef);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should support viewChild with `read` pointing to an `as` expression', () => {
env.write(
'test.ts',
`
import {Component, viewChild, ElementRef} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator', {read: ElementRef as any});
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5, ElementRef);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should handle a basic viewChildren', () => {
env.write(
'test.ts',
`
import {Component, viewChildren} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChildren('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵviewQuerySignal(ctx.el, _c0, 5);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should handle a basic contentChild', () => {
env.write(
'test.ts',
`
import {Component, contentChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = contentChild('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵcontentQuerySignal(dirIndex, ctx.el, _c0, 5);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
it('should handle a basic contentChildren', () => {
env.write(
'test.ts',
`
import {Component, contentChildren} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = contentChildren('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain(`i0.ɵɵcontentQuerySignal(dirIndex, ctx.el, _c0, 4);`);
expect(js).toContain(`i0.ɵɵqueryAdvance();`);
});
describe('diagnostics', () => {
it('should report an error when used with query decorator', () => {
env.write(
'test.ts',
`
import {Component, viewChild, ViewChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
@ViewChild('myLocator') el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Using @ViewChild with a signal-based query is not allowed.`,
}),
]);
});
it('should report an error when used on a static field', () => {
env.write(
'test.ts',
`
import {Component, viewChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
static el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Query is incorrectly declared on a static class member.`,
}),
]);
});
it('should report an error when declared in @Directive metadata', () => {
env.write(
'test.ts',
`
import {Directive, ViewChild, viewChild} from '@angular/core';
@Directive({
selector: 'test',
queries: {
el: new ViewChild('myLocator'),
},
})
export class TestDir {
el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Query is declared multiple times. "@Directive" declares a query for the same property.`,
}),
]);
});
it('should report an error when declared in @Component metadata', () => {
env.write(
'test.ts',
`
import {Component, ViewChild, viewChild} from '@angular/core';
@Component({
selector: 'test',
template: '',
queries: {
el: new ViewChild('myLocator'),
},
})
export class TestComp {
el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Query is declared multiple times. "@Component" declares a query for the same property.`,
}),
]);
});
it('should report an error when a signal-based query function is used in metadata', () => {
env.write(
'test.ts',
`
import {Component, viewChild} from '@angular/core';
@Component({
selector: 'test',
template: '',
queries: {
// @ts-ignore
el: new viewChild('myLocator'),
},
})
export class TestComp {}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Decorator query metadata must be an instance of a query type`,
}),
]);
});
it('should report an error when `read` option is complex', () => {
env.write(
'test.ts',
`
import {Directive, viewChild} from '@angular/core';
@Directive({
selector: 'test',
})
export class TestDir {
something = null!;
el = viewChild('myLocator', {read: this.something});
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining({
messageText: `Query "read" option expected a literal class reference.`,
}),
]);
});
it('should error when a query is declared using an ES private field', () => {
env.write(
'test.ts',
`
import {Directive, viewChild} from '@angular/core';
@Directive({
selector: 'test',
})
export class TestDir {
#el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(1);
expect(diagnostics).toEqual([
jasmine.objectContaining<ts.Diagnostic>({
messageText: jasmine.objectContaining<ts.DiagnosticMessageChain>({
messageText: `Cannot use "viewChild" on a class member that is declared as ES private.`,
}),
}),
]);
});
it('should allow query is declared on a `private` field', () => {
env.write(
'test.ts',
`
import {Directive, viewChild} from '@angular/core';
@Directive({
selector: 'test',
})
export class TestDir {
private el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(0);
});
it('should allow query is declared on a `protected` field', () => {
env.write(
'test.ts',
`
import {Directive, viewChild} from '@angular/core';
@Directive({
selector: 'test',
})
export class TestDir {
protected el = viewChild('myLocator');
}
`,
);
const diagnostics = env.driveDiagnostics();
expect(diagnostics.length).toBe(0);
});
it('should capture a viewChild query in the setClasMetadata call', () => {
env.write(
'test.ts',
`
import {Component, viewChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ViewChild, args: ['myLocator', { isSignal: true }] }] });`);
});
it('should capture a viewChildren query in the setClasMetadata call', () => {
env.write(
'test.ts',
`
import {Component, viewChildren} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChildren('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ViewChildren, args: ['myLocator', { isSignal: true }] }] });`);
});
it('should capture a contentChild query in the setClasMetadata call', () => {
env.write(
'test.ts',
`
import {Component, contentChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = contentChild('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ContentChild, args: ['myLocator', { isSignal: true }] }] });`);
});
it('should capture a contentChildren query in the setClasMetadata call', () => {
env.write(
'test.ts',
`
import {Component, contentChildren} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = contentChildren('myLocator');
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ContentChildren, args: ['myLocator', { isSignal: true }] }] });`);
});
it('should capture a query with options in a setClassMetadata call', () => {
env.write(
'test.ts',
`
import {Component, viewChild, ElementRef} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild('myLocator', {read: ElementRef});
}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(
`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ViewChild, ` +
`args: ['myLocator', Object.assign({ read: ElementRef }, { isSignal: true })] }] });`,
);
});
it('should wrap reference in query as a forwardRef in the setClassMetadata call', () => {
env.write(
'test.ts',
`
import {Component, Directive, viewChild} from '@angular/core';
@Component({selector: 'test', template: ''})
export class TestDir {
el = viewChild(Dep);
}
@Directive({selector: '[dep]'})
export class Dep {}
`,
);
env.driveMain();
const js = env.getContents('test.js');
expect(js).toContain('import * as i0 from "@angular/core";');
expect(js).toContain(
`i0.ɵsetClassMetadata(TestDir, [{
type: Component,
args: [{ selector: 'test', template: '' }]
}], null, { el: [{ type: i0.ViewChild, args: [i0.forwardRef(() => Dep), { isSignal: true }] }] });`,
);
});
});
});
});