angular/packages/compiler-cli/test/ngtsc/authoring_queries_spec.ts
Paul Gschwendtner 2564b45b47 test: replace fake_core with real @angular/core output (#54650)
This commit replaces `fake_core` with the real `@angular/core`
output. See previous commit for reasons.

Overall, this commit:

* Replaces references of `fake_core`
* Fixes tests that were testing Angular compiler detection that _would_
  already be flagged by type-checking of TS directly. We keep these
  tests for now, and add `@ts-ignore` to verify the Angular checks, in
  case type checking is disabled in user applications- but it's worth
  considering to remove these tests. Follow-up question/non-priority.
* Adds `@ts-ignore` to the tests for `defer` 1P because the property is
  marked as `@internal` and now is (correctly) causing failures in the
  compiler test environment.
* Fixes a couple of tests with typos, wrong properties etc that
  previously weren't detected! A good sign.

PR Close #54650
2024-03-06 12:34:38 +01:00

279 lines
9.1 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.io/license
*/
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, 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.`,
})]);
});
});
});
});