angular/packages/compiler-cli/test/ngtsc/authoring_queries_spec.ts
Joey Perrott 9dbe6fc18b refactor: update license text to point to angular.dev (#57901)
Update license text to point to angular.dev instead of angular.io

PR Close #57901
2024-09-24 15:33:00 +02:00

394 lines
11 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);`);
expect(js).toContain(`i0.ɵɵviewQuerySignal(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);
});
});
});
});