/** * @license * Copyright Google Inc. 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 {NgForOfContext} from '@angular/common'; import {ElementRef, QueryList, TemplateRef, ViewContainerRef} from '@angular/core'; import {EventEmitter} from '../..'; import {AttributeMarker, detectChanges, ΔProvidersFeature, ΔdefineComponent, ΔdefineDirective} from '../../src/render3/index'; import {Δbind, Δcontainer, ΔcontainerRefreshEnd, ΔcontainerRefreshStart, ΔdirectiveInject, Δelement, ΔelementContainerEnd, ΔelementContainerStart, ΔelementEnd, ΔelementProperty, ΔelementStart, ΔembeddedViewEnd, ΔembeddedViewStart, Δload, Δreference, Δtemplate, Δtext} from '../../src/render3/instructions/all'; import {RenderFlags} from '../../src/render3/interfaces/definition'; import {query, ΔcontentQuery, ΔloadContentQuery, ΔloadViewQuery, ΔqueryRefresh, ΔviewQuery} from '../../src/render3/query'; import {getLView} from '../../src/render3/state'; import {getNativeByIndex} from '../../src/render3/util/view_utils'; import {ΔtemplateRefExtractor} from '../../src/render3/view_engine_compatibility_prebound'; import {NgForOf, NgIf, NgTemplateOutlet} from './common_with_def'; import {ComponentFixture, TemplateFixture, createComponent, createDirective, getDirectiveOnNode, renderComponent} from './render_util'; /** * Helper function to check if a given candidate object resembles ElementRef * @param candidate * @returns true if `ElementRef`. */ function isElementRef(candidate: any): boolean { return candidate.nativeElement != null; } /** * Helper function to check if a given candidate object resembles TemplateRef * @param candidate * @returns true if `TemplateRef`. */ function isTemplateRef(candidate: any): boolean { return candidate.createEmbeddedView != null && candidate.createComponent == null; } /** * Helper function to check if a given candidate object resembles ViewContainerRef * @param candidate * @returns true if `ViewContainerRef`. */ function isViewContainerRef(candidate: any): boolean { return candidate.createEmbeddedView != null && candidate.createComponent != null; } describe('query', () => { it('should match projected query children', () => { const Child = createComponent('child', function(rf: RenderFlags, ctx: any) {}); let child1 = null; let child2 = null; const Cmp = createComponent( 'cmp', function(rf: RenderFlags, ctx: any) { /** * * * * * class Cmp { * @ViewChildren(Child) query0; * @ViewChildren(Child, {descend: true}) query1; * } */ if (rf & RenderFlags.Create) { ΔelementStart(0, 'child'); { Δelement(1, 'child'); } ΔelementEnd(); } if (rf & RenderFlags.Update) { child1 = getDirectiveOnNode(0); child2 = getDirectiveOnNode(1); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(Child, false, null); ΔviewQuery(Child, true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query0 = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query1 = tmp as QueryList); } }); const parent = renderComponent(Cmp); expect((parent.query0 as QueryList).toArray()).toEqual([child1]); expect((parent.query1 as QueryList).toArray()).toEqual([child1, child2]); }); describe('predicate', () => { describe('types', () => { it('should query using type predicate and read a specified token', () => { const Child = createDirective('child'); let elToQuery; /** *
* class Cmpt { * @ViewChildren(Child, {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', '']); elToQuery = getNativeByIndex(0, getLView()); } }, 1, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(Child, false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isElementRef(qList.first)).toBeTruthy(); expect(qList.first.nativeElement).toBe(elToQuery); }); it('should query using type predicate and read another directive type', () => { const Child = createDirective('child'); const OtherChild = createDirective('otherChild'); let otherChildInstance; /** *
* class Cmpt { * @ViewChildren(Child, {read: OtherChild}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['child', '', 'otherChild', '']); { otherChildInstance = getDirectiveOnNode(0, 1); } ΔelementEnd(); } }, 1, 0, [Child, OtherChild], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(Child, false, OtherChild); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first).toBe(otherChildInstance); }); it('should not add results to query if a requested token cant be read', () => { const Child = createDirective('child'); const OtherChild = createDirective('otherChild'); /** *
* class Cmpt { * @ViewChildren(Child, {read: OtherChild}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', '']); } }, 1, 0, [Child, OtherChild], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(Child, false, OtherChild); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(0); }); }); describe('providers', () => { class Service {} class Alias {} let directive: MyDirective|null = null; class MyDirective { constructor(public service: Service) {} static ngDirectiveDef = ΔdefineDirective({ type: MyDirective, selectors: [['', 'myDir', '']], factory: function MyDirective_Factory() { return directive = new MyDirective(ΔdirectiveInject(Service)); }, features: [ΔProvidersFeature([Service, {provide: Alias, useExisting: Service}])], }); } beforeEach(() => directive = null); // https://stackblitz.com/edit/ng-viewengine-viewchild-providers?file=src%2Fapp%2Fapp.component.ts it('should query for providers that are present on a directive', () => { /** *
* class App { * @ViewChild(MyDirective) directive: MyDirective; * @ViewChild(Service) service: Service; * @ViewChild(Alias) alias: Alias; * } */ class App { directive?: MyDirective; service?: Service; alias?: Alias; static ngComponentDef = ΔdefineComponent({ type: App, selectors: [['app']], consts: 1, vars: 0, factory: function App_Factory() { return new App(); }, template: function App_Template(rf: RenderFlags, ctx: App) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['myDir']); } }, viewQuery: function(rf: RenderFlags, ctx: App) { if (rf & RenderFlags.Create) { ΔviewQuery(MyDirective, false, null); ΔviewQuery(Service, false, null); ΔviewQuery(Alias, false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.directive = tmp.first); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.service = tmp.first); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.alias = tmp.first); } }, directives: [MyDirective] }); } const componentFixture = new ComponentFixture(App); expect(componentFixture.component.directive).toBe(directive !); expect(componentFixture.component.service).toBe(directive !.service); expect(componentFixture.component.alias).toBe(directive !.service); }); it('should resolve a provider if given as read token', () => { /** *
* class App { * @ViewChild(MyDirective, {read: Alias}}) service: Service; * } */ class App { service?: Service; static ngComponentDef = ΔdefineComponent({ type: App, selectors: [['app']], consts: 1, vars: 0, factory: function App_Factory() { return new App(); }, template: function App_Template(rf: RenderFlags, ctx: App) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['myDir']); } }, viewQuery: function(rf: RenderFlags, ctx: App) { let tmp: any; if (rf & RenderFlags.Create) { ΔviewQuery(MyDirective, false, Alias); } if (rf & RenderFlags.Update) { ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.service = tmp.first); } }, directives: [MyDirective] }); } const componentFixture = new ComponentFixture(App); expect(componentFixture.component.service).toBe(directive !.service); }); }); describe('local names', () => { it('should query for a single element and read ElementRef by default', () => { let elToQuery; /** *
*
* class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); elToQuery = getNativeByIndex(0, getLView()); Δelement(2, 'div'); } }, 3, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first.nativeElement).toEqual(elToQuery); }); it('should query multiple locals on the same element', () => { let elToQuery; /** *
*
* class Cmpt { * @ViewChildren('foo') fooQuery; * @ViewChildren('bar') barQuery; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '', 'bar', '']); elToQuery = getNativeByIndex(0, getLView()); Δelement(3, 'div'); } }, 4, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, null); ΔviewQuery(['bar'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.fooQuery = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.barQuery = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const fooList = (cmptInstance.fooQuery as QueryList); expect(fooList.length).toBe(1); expect(fooList.first.nativeElement).toEqual(elToQuery); const barList = (cmptInstance.barQuery as QueryList); expect(barList.length).toBe(1); expect(barList.first.nativeElement).toEqual(elToQuery); }); it('should query for multiple elements and read ElementRef by default', () => { let el1ToQuery; let el2ToQuery; /** *
*
*
* class Cmpt { * @ViewChildren('foo,bar') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); el1ToQuery = getNativeByIndex(0, getLView()); Δelement(2, 'div'); Δelement(3, 'div', null, ['bar', '']); el2ToQuery = getNativeByIndex(3, getLView()); } }, 5, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo', 'bar'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toEqual(el1ToQuery); expect(qList.last.nativeElement).toEqual(el2ToQuery); }); it('should read ElementRef from an element when explicitly asked for', () => { let elToQuery; /** *
*
* class Cmpt { * @ViewChildren('foo', {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); elToQuery = getNativeByIndex(0, getLView()); Δelement(2, 'div'); } }, 3, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isElementRef(qList.first)).toBeTruthy(); expect(qList.first.nativeElement).toEqual(elToQuery); }); it('should query for and read ElementRef with a native element pointing to comment node', () => { let elToQuery; /** * * class Cmpt { * @ViewChildren('foo', {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementContainerStart(0, null, ['foo', '']); elToQuery = getNativeByIndex(0, getLView()); ΔelementContainerEnd(); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isElementRef(qList.first)).toBeTruthy(); expect(qList.first.nativeElement).toEqual(elToQuery); }); it('should query for and read ElementRef without explicit read option', () => { let elToQuery; /** * * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementContainerStart(0, null, ['foo', '']); elToQuery = getNativeByIndex(0, getLView()); ΔelementContainerEnd(); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isElementRef(qList.first)).toBeTruthy(); expect(qList.first.nativeElement).toEqual(elToQuery); }); /** * BREAKING CHANGE: this tests asserts different behavior as compared to Renderer2 when it * comes to descendants: false option and . * * Previous behavior: queries with descendants: false would descend into . * New behavior: queries with descendants: false would NOT descend into . * * Reasoning: the Renderer2 behavior is inconsistent and hard to explain to users when it * comes to descendants: false interpretation (see * https://github.com/angular/angular/issues/14769#issuecomment-356609267) so we are changing * it in ngIvy. * * In ngIvy implementation queries with the descendants: false option are interpreted as * "don't descend" into children of a given element when looking for matches. In other words * only direct children of a given component / directive are checked for matches. This applies * to both regular elements (ex.
) and grouping elements (, * )). * * Grouping elements (, ) are treated as regular elements since we * can query for and , so they behave like regular elements from * this point of view. */ it('should not descend into when descendants: false', () => { let elToQuery; /** * *
*
* class Cmpt { * @ViewChildren('foo') deep; * @ViewChildren('foo', {descendants: false}) shallow; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementContainerStart(0); { Δelement(1, 'div', null, ['foo', '']); elToQuery = getNativeByIndex(3, getLView()); } ΔelementContainerEnd(); } }, 3, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, ElementRef); ΔviewQuery(['foo'], false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.deep = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.shallow = tmp as QueryList); } }); const fixture = new ComponentFixture(Cmpt); const deepQList = fixture.component.deep; const shallowQList = fixture.component.shallow; expect(deepQList.length).toBe(1); expect(shallowQList.length).toBe(0); }); it('should read ViewContainerRef from element nodes when explicitly asked for', () => { /** *
* class Cmpt { * @ViewChildren('foo', {read: ViewContainerRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, ViewContainerRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isViewContainerRef(qList.first)).toBeTruthy(); }); it('should read ViewContainerRef from container nodes when explicitly asked for', () => { /** * * class Cmpt { * @ViewChildren('foo', {read: ViewContainerRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate(0, null, 0, 0, 'ng-template', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, ViewContainerRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isViewContainerRef(qList.first)).toBeTruthy(); }); it('should read ElementRef with a native element pointing to comment DOM node from containers', () => { /** * * class Cmpt { * @ViewChildren('foo', {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate(0, null, 0, 0, 'ng-template', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isElementRef(qList.first)).toBeTruthy(); expect(qList.first.nativeElement.nodeType).toBe(8); // Node.COMMENT_NODE = 8 }); it('should read TemplateRef from container nodes by default', () => { // http://plnkr.co/edit/BVpORly8wped9I3xUYsX?p=preview /** * * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate(0, null, 0, 0, 'ng-template', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isTemplateRef(qList.first)).toBeTruthy(); }); it('should read TemplateRef from container nodes when explicitly asked for', () => { /** * * class Cmpt { * @ViewChildren('foo', {read: TemplateRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate(0, null, 0, 0, 'ng-template', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, TemplateRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(isTemplateRef(qList.first)).toBeTruthy(); }); it('should read component instance if element queried for is a component host', () => { const Child = createComponent('child', function(rf: RenderFlags, ctx: any) {}); let childInstance; /** * * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'child', null, ['foo', '']); } if (rf & RenderFlags.Update) { childInstance = getDirectiveOnNode(0); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first).toBe(childInstance); }); it('should read component instance with explicit exportAs', () => { let childInstance: Child; class Child { static ngComponentDef = ΔdefineComponent({ type: Child, selectors: [['child']], factory: () => childInstance = new Child(), consts: 0, vars: 0, template: (rf: RenderFlags, ctx: Child) => {}, exportAs: ['child'] }); } /** * * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'child', null, ['foo', 'child']); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first).toBe(childInstance !); }); it('should read directive instance if element queried for has an exported directive with a matching name', () => { const Child = createDirective('child', {exportAs: ['child']}); let childInstance; /** *
* class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', ''], ['foo', 'child']); } if (rf & RenderFlags.Update) { childInstance = getDirectiveOnNode(0); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first).toBe(childInstance); }); it('should read all matching directive instances from a given element', () => { const Child1 = createDirective('child1', {exportAs: ['child1']}); const Child2 = createDirective('child2', {exportAs: ['child2']}); let child1Instance, child2Instance; /** *
* class Cmpt { * @ViewChildren('foo, bar') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement( 0, 'div', ['child1', '', 'child2', ''], ['foo', 'child1', 'bar', 'child2']); } if (rf & RenderFlags.Update) { child1Instance = getDirectiveOnNode(0, 0); child2Instance = getDirectiveOnNode(0, 1); } }, 3, 0, [Child1, Child2], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo', 'bar'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(2); expect(qList.first).toBe(child1Instance); expect(qList.last).toBe(child2Instance); }); it('should read multiple locals exporting the same directive from a given element', () => { const Child = createDirective('child', {exportAs: ['child']}); let childInstance; /** *
* class Cmpt { * @ViewChildren('foo') fooQuery; * @ViewChildren('bar') barQuery; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', ''], ['foo', 'child', 'bar', 'child']); } if (rf & RenderFlags.Update) { childInstance = getDirectiveOnNode(0); } }, 3, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); ΔviewQuery(['bar'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.fooQuery = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.barQuery = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const fooList = cmptInstance.fooQuery as QueryList; expect(fooList.length).toBe(1); expect(fooList.first).toBe(childInstance); const barList = cmptInstance.barQuery as QueryList; expect(barList.length).toBe(1); expect(barList.first).toBe(childInstance); }); it('should match on exported directive name and read a requested token', () => { const Child = createDirective('child', {exportAs: ['child']}); let div; /** *
* class Cmpt { * @ViewChildren('foo', {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', ''], ['foo', 'child']); div = getNativeByIndex(0, getLView()); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(1); expect(qList.first.nativeElement).toBe(div); }); it('should support reading a mix of ElementRef and directive instances', () => { const Child = createDirective('child', {exportAs: ['child']}); let childInstance, div; /** *
* class Cmpt { * @ViewChildren('foo, bar') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', ''], ['foo', '', 'bar', 'child']); div = getNativeByIndex(0, getLView()); } if (rf & RenderFlags.Update) { childInstance = getDirectiveOnNode(0); } }, 3, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo', 'bar'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as QueryList); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toBe(div); expect(qList.last).toBe(childInstance); }); it('should not add results to selector-based query if a requested token cant be read', () => { const Child = createDirective('child'); /** *
* class Cmpt { * @ViewChildren('foo', {read: Child}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['foo', '']); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, Child); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(0); }); it('should not add results to directive-based query if requested token cant be read', () => { const Child = createDirective('child'); const OtherChild = createDirective('otherchild'); /** *
* class Cmpt { * @ViewChildren(Child, {read: OtherChild}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', '']); } }, 1, 0, [Child, OtherChild], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(Child, false, OtherChild); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(0); }); it('should not add results to directive-based query if only read token matches', () => { const Child = createDirective('child'); const OtherChild = createDirective('otherchild'); /** *
* class Cmpt { * @ViewChildren(OtherChild, {read: Child}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', '']); } }, 1, 0, [Child, OtherChild], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(OtherChild, false, Child); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(0); }); it('should not add results to TemplateRef-based query if only read token matches', () => { /** *
* class Cmpt { * @ViewChildren(TemplateRef, {read: ElementRef}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div'); } }, 1, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(TemplateRef as any, false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(0); }); it('should match using string selector and directive as a read argument', () => { const Child = createDirective('child'); /** *
* class Cmpt { * @ViewChildren('foo', {read: Child}) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', ''], ['foo', '']); } }, 2, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, Child); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(1); expect(qList.first instanceof Child).toBeTruthy(); }); it('should not add results to the query in case no match found (via TemplateRef)', () => { const Child = createDirective('child'); /** *
* class Cmpt { * @ViewChildren(TemplateRef) query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['child', '']); } }, 1, 0, [Child], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(TemplateRef as any, false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); const qList = component.query; expect(qList.length).toBe(0); }); it('should query templates if the type is TemplateRef (and respect "read" option)', () => { function Cmpt_Template_1(rf: RenderFlags, ctx1: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div'); Δtext(1, 'Test'); ΔelementEnd(); } } /** *
Test
*
Test
*
Test
* class Cmpt { * @ViewChildren(TemplateRef) tmplQuery; * @ViewChildren(TemplateRef, {read: ElementRef}) elemQuery; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate( 0, Cmpt_Template_1, 2, 0, 'ng-template', null, ['foo', ''], ΔtemplateRefExtractor); Δtemplate( 2, Cmpt_Template_1, 2, 0, 'ng-template', null, ['bar', ''], ΔtemplateRefExtractor); Δtemplate( 4, Cmpt_Template_1, 2, 0, 'ng-template', null, ['baz', ''], ΔtemplateRefExtractor); } }, 6, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(TemplateRef as any, false, null); ΔviewQuery(TemplateRef as any, false, ElementRef); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.tmplQuery = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.elemQuery = tmp as QueryList); } }); const {component} = new ComponentFixture(Cmpt); // check template-based query set const tmplQList = component.tmplQuery; expect(tmplQList.length).toBe(3); expect(isTemplateRef(tmplQList.first)).toBeTruthy(); // check element-based query set const elemQList = component.elemQuery; expect(elemQList.length).toBe(3); expect(isElementRef(elemQList.first)).toBeTruthy(); }); }); }); describe('view boundaries', () => { describe('ViewContainerRef', () => { let directiveInstances: ViewContainerManipulatorDirective[] = []; class ViewContainerManipulatorDirective { static ngDirectiveDef = ΔdefineDirective({ type: ViewContainerManipulatorDirective, selectors: [['', 'vc', '']], factory: () => { const directiveInstance = new ViewContainerManipulatorDirective(ΔdirectiveInject(ViewContainerRef as any)); directiveInstances.push(directiveInstance); return directiveInstance; } }); constructor(private _vcRef: ViewContainerRef) {} insertTpl(tpl: TemplateRef<{}>, ctx: {}, idx?: number) { this._vcRef.createEmbeddedView(tpl, ctx, idx); } remove(index?: number) { this._vcRef.remove(index); } } beforeEach(() => { directiveInstances = []; }); it('should report results in views inserted / removed by ngIf', () => { function Cmpt_Template_1(rf: RenderFlags, ctx1: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } } /** * *
*
* class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate( 0, Cmpt_Template_1, 2, 0, 'ng-template', [AttributeMarker.Bindings, 'ngIf']); } if (rf & RenderFlags.Update) { ΔelementProperty(0, 'ngIf', Δbind(ctx.value)); } }, 2, 1, [NgIf], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const fixture = new ComponentFixture(Cmpt); const qList = fixture.component.query; expect(qList.length).toBe(0); fixture.component.value = true; fixture.update(); expect(qList.length).toBe(1); fixture.component.value = false; fixture.update(); expect(qList.length).toBe(0); }); it('should report results in views inserted / removed by ngFor', () => { function Cmpt_Template_1(rf1: RenderFlags, row: NgForOfContext) { if (rf1 & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } if (rf1 & RenderFlags.Update) { ΔelementProperty(0, 'id', Δbind(row.$implicit)); } } /** * *
*
* class Cmpt { * @ViewChildren('foo') query; * } */ class Cmpt { // TODO(issue/24571): remove '!'. value !: string[]; query: any; static ngComponentDef = ΔdefineComponent({ type: Cmpt, factory: () => new Cmpt(), selectors: [['my-app']], consts: 2, vars: 1, template: function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate(0, Cmpt_Template_1, 2, 1, 'ng-template', ['ngForOf', '']); } if (rf & RenderFlags.Update) { ΔelementProperty(0, 'ngForOf', Δbind(ctx.value)); } }, viewQuery: function(rf: RenderFlags, ctx: Cmpt) { let tmp: any; if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }, directives: () => [NgForOf] }); } const fixture = new ComponentFixture(Cmpt); const qList = fixture.component.query; expect(qList.length).toBe(0); fixture.component.value = ['a', 'b', 'c']; fixture.update(); expect(qList.length).toBe(3); fixture.component.value.splice(1, 1); // remove "b" fixture.update(); expect(qList.length).toBe(2); // make sure that a proper element was removed from query results expect(qList.first.nativeElement.id).toBe('a'); expect(qList.last.nativeElement.id).toBe('c'); }); // https://stackblitz.com/edit/angular-rrmmuf?file=src/app/app.component.ts it('should report results when different instances of TemplateRef are inserted into one ViewContainerRefs', () => { let tpl1: TemplateRef<{}>; let tpl2: TemplateRef<{}>; function Cmpt_Template_1(rf: RenderFlags, ctx: {idx: number}) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } if (rf & RenderFlags.Update) { ΔelementProperty(0, 'id', Δbind('foo1_' + ctx.idx)); } } function Cmpt_Template_5(rf: RenderFlags, ctx: {idx: number}) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } if (rf & RenderFlags.Update) { ΔelementProperty(0, 'id', Δbind('foo2_' + ctx.idx)); } } /** * *
*
* *
* * *
*
* * */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate( 0, Cmpt_Template_1, 2, 1, 'ng-template', null, ['tpl1', ''], ΔtemplateRefExtractor); Δelement(2, 'div', ['id', 'middle'], ['foo', '']); Δtemplate( 4, Cmpt_Template_5, 2, 1, 'ng-template', null, ['tpl2', ''], ΔtemplateRefExtractor); Δtemplate(6, null, 0, 0, 'ng-template', [AttributeMarker.Bindings, 'vc']); } if (rf & RenderFlags.Update) { tpl1 = Δreference(1); tpl2 = Δreference(5); } }, 8, 0, [ViewContainerManipulatorDirective], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const fixture = new ComponentFixture(Cmpt); const qList = fixture.component.query; expect(qList.length).toBe(1); expect(qList.first.nativeElement.getAttribute('id')).toBe('middle'); directiveInstances[0].insertTpl(tpl1 !, {idx: 0}, 0); directiveInstances[0].insertTpl(tpl2 !, {idx: 1}, 1); fixture.update(); expect(qList.length).toBe(3); let qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo1_0'); expect(qListArr[1].nativeElement.getAttribute('id')).toBe('middle'); expect(qListArr[2].nativeElement.getAttribute('id')).toBe('foo2_1'); directiveInstances[0].insertTpl(tpl1 !, {idx: 1}, 1); fixture.update(); expect(qList.length).toBe(4); qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo1_0'); expect(qListArr[1].nativeElement.getAttribute('id')).toBe('foo1_1'); expect(qListArr[2].nativeElement.getAttribute('id')).toBe('middle'); expect(qListArr[3].nativeElement.getAttribute('id')).toBe('foo2_1'); directiveInstances[0].remove(1); fixture.update(); expect(qList.length).toBe(3); qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo1_0'); expect(qListArr[1].nativeElement.getAttribute('id')).toBe('middle'); expect(qListArr[2].nativeElement.getAttribute('id')).toBe('foo2_1'); directiveInstances[0].remove(1); fixture.update(); expect(qList.length).toBe(2); qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo1_0'); expect(qListArr[1].nativeElement.getAttribute('id')).toBe('middle'); }); // https://stackblitz.com/edit/angular-7vvo9j?file=src%2Fapp%2Fapp.component.ts // https://stackblitz.com/edit/angular-xzwp6n it('should report results when the same TemplateRef is inserted into different ViewContainerRefs', () => { let tpl: TemplateRef<{}>; function Cmpt_Template_1(rf: RenderFlags, ctx: {idx: number, container_idx: number}) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } if (rf & RenderFlags.Update) { ΔelementProperty(0, 'id', Δbind('foo_' + ctx.container_idx + '_' + ctx.idx)); } } /** * *
*
* * * */ class Cmpt { query: any; static ngComponentDef = ΔdefineComponent({ type: Cmpt, factory: () => new Cmpt(), selectors: [['my-app']], consts: 4, vars: 0, template: function(rf: RenderFlags, ctx: any) { let tmp: any; if (rf & RenderFlags.Create) { Δtemplate( 0, Cmpt_Template_1, 2, 1, 'ng-template', [], ['tpl', ''], ΔtemplateRefExtractor); Δtemplate(2, null, 0, 0, 'ng-template', [AttributeMarker.Bindings, 'vc']); Δtemplate(3, null, 0, 0, 'ng-template', [AttributeMarker.Bindings, 'vc']); } if (rf & RenderFlags.Update) { tpl = Δreference(1); } }, viewQuery: (rf: RenderFlags, cmpt: Cmpt) => { let tmp: any; if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (cmpt.query = tmp as QueryList); } }, directives: () => [ViewContainerManipulatorDirective], }); } const fixture = new ComponentFixture(Cmpt); const qList = fixture.component.query; expect(qList.length).toBe(0); directiveInstances[0].insertTpl(tpl !, {idx: 0, container_idx: 0}, 0); directiveInstances[1].insertTpl(tpl !, {idx: 0, container_idx: 1}, 0); fixture.update(); expect(qList.length).toBe(2); let qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo_1_0'); expect(qListArr[1].nativeElement.getAttribute('id')).toBe('foo_0_0'); directiveInstances[0].remove(); fixture.update(); expect(qList.length).toBe(1); qListArr = qList.toArray(); expect(qListArr[0].nativeElement.getAttribute('id')).toBe('foo_1_0'); directiveInstances[1].remove(); fixture.update(); expect(qList.length).toBe(0); }); // https://stackblitz.com/edit/angular-wpd6gv?file=src%2Fapp%2Fapp.component.ts it('should report results from views inserted in a lifecycle hook', () => { function MyApp_Template_1(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'span', ['id', 'from_tpl'], ['foo', '']); } } class MyApp { show = false; query: any; static ngComponentDef = ΔdefineComponent({ type: MyApp, factory: () => new MyApp(), selectors: [['my-app']], consts: 4, vars: 1, /** * * */ template: (rf: RenderFlags, myApp: MyApp) => { if (rf & RenderFlags.Create) { Δtemplate( 0, MyApp_Template_1, 2, 0, 'ng-template', undefined, ['tpl', ''], ΔtemplateRefExtractor); Δtemplate( 2, null, 0, 0, 'ng-template', [AttributeMarker.Bindings, 'ngTemplateOutlet']); } if (rf & RenderFlags.Update) { const tplRef = Δreference(1); ΔelementProperty(2, 'ngTemplateOutlet', Δbind(myApp.show ? tplRef : null)); } }, directives: () => [NgTemplateOutlet], viewQuery: (rf: RenderFlags, myApp: MyApp) => { let tmp: any; if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (myApp.query = tmp as QueryList); } } }); } const fixture = new ComponentFixture(MyApp); const qList = fixture.component.query; expect(qList.length).toBe(0); fixture.component.show = true; fixture.update(); expect(qList.length).toBe(1); expect(qList.first.nativeElement.id).toBe('from_tpl'); fixture.component.show = false; fixture.update(); expect(qList.length).toBe(0); }); }); describe('JS blocks', () => { it('should report results in embedded views', () => { let firstEl; /** * % if (exp) { *
* % } * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δcontainer(0); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(0); { if (ctx.exp) { let rf1 = ΔembeddedViewStart(1, 2, 0); { if (rf1 & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); firstEl = getNativeByIndex(0, getLView()); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 1, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as any); expect(qList.length).toBe(0); cmptInstance.exp = true; detectChanges(cmptInstance); expect(qList.length).toBe(1); expect(qList.first.nativeElement).toBe(firstEl); cmptInstance.exp = false; detectChanges(cmptInstance); expect(qList.length).toBe(0); }); it('should add results from embedded views in the correct order - views and elements mix', () => { let firstEl, lastEl, viewEl; /** * * % if (exp) { *
* % } * * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'span', null, ['foo', '']); firstEl = getNativeByIndex(0, getLView()); Δcontainer(2); Δelement(3, 'span', null, ['foo', '']); lastEl = getNativeByIndex(3, getLView()); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(2); { if (ctx.exp) { let rf1 = ΔembeddedViewStart(1, 2, 0); { if (rf1 & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); viewEl = getNativeByIndex(0, getLView()); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 5, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as any); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toBe(firstEl); expect(qList.last.nativeElement).toBe(lastEl); cmptInstance.exp = true; detectChanges(cmptInstance); expect(qList.length).toBe(3); expect(qList.toArray()[0].nativeElement).toBe(firstEl); expect(qList.toArray()[1].nativeElement).toBe(viewEl); expect(qList.toArray()[2].nativeElement).toBe(lastEl); cmptInstance.exp = false; detectChanges(cmptInstance); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toBe(firstEl); expect(qList.last.nativeElement).toBe(lastEl); }); it('should add results from embedded views in the correct order - views side by side', () => { let firstEl, lastEl; /** * % if (exp1) { *
* % } if (exp2) { * * % } * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δcontainer(0); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(0); { if (ctx.exp1) { let rf0 = ΔembeddedViewStart(0, 2, 0); { if (rf0 & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); firstEl = getNativeByIndex(0, getLView()); } } ΔembeddedViewEnd(); } if (ctx.exp2) { let rf1 = ΔembeddedViewStart(1, 2, 0); { if (rf1 & RenderFlags.Create) { Δelement(0, 'span', null, ['foo', '']); lastEl = getNativeByIndex(0, getLView()); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 1, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as any); expect(qList.length).toBe(0); cmptInstance.exp2 = true; detectChanges(cmptInstance); expect(qList.length).toBe(1); expect(qList.last.nativeElement).toBe(lastEl); cmptInstance.exp1 = true; detectChanges(cmptInstance); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toBe(firstEl); expect(qList.last.nativeElement).toBe(lastEl); }); it('should add results from embedded views in the correct order - nested views', () => { let firstEl, lastEl; /** * % if (exp1) { *
* % if (exp2) { * * } * % } * class Cmpt { * @ViewChildren('foo') query; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δcontainer(0); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(0); { if (ctx.exp1) { let rf0 = ΔembeddedViewStart(0, 3, 0); { if (rf0 & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); firstEl = getNativeByIndex(0, getLView()); Δcontainer(2); } if (rf0 & RenderFlags.Update) { ΔcontainerRefreshStart(2); { if (ctx.exp2) { let rf2 = ΔembeddedViewStart(0, 2, 0); { if (rf2) { Δelement(0, 'span', null, ['foo', '']); lastEl = getNativeByIndex(0, getLView()); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 1, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const qList = (cmptInstance.query as any); expect(qList.length).toBe(0); cmptInstance.exp1 = true; detectChanges(cmptInstance); expect(qList.length).toBe(1); expect(qList.first.nativeElement).toBe(firstEl); cmptInstance.exp2 = true; detectChanges(cmptInstance); expect(qList.length).toBe(2); expect(qList.first.nativeElement).toBe(firstEl); expect(qList.last.nativeElement).toBe(lastEl); }); /** * What is tested here can't be achieved in the Renderer2 as all view queries are deep by * default and can't be marked as shallow by a user. */ it('should support combination of deep and shallow queries', () => { /** * % if (exp) { "> *
*
*
* % } * * class Cmpt { * @ViewChildren('foo') deep; * @ViewChildren('foo') shallow; * } */ const Cmpt = createComponent( 'cmpt', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δcontainer(0); Δelement(1, 'span', null, ['foo', '']); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(0); { if (ctx.exp) { let rf0 = ΔembeddedViewStart(0, 4, 0); { if (rf0 & RenderFlags.Create) { ΔelementStart(0, 'div', null, ['foo', '']); { Δelement(2, 'div', null, ['foo', '']); } ΔelementEnd(); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 3, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); ΔviewQuery(['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.deep = tmp as QueryList); ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.shallow = tmp as QueryList); } }); const cmptInstance = renderComponent(Cmpt); const deep = (cmptInstance.deep as any); const shallow = (cmptInstance.shallow as any); expect(deep.length).toBe(1); expect(shallow.length).toBe(1); cmptInstance.exp = true; detectChanges(cmptInstance); expect(deep.length).toBe(3); // embedded % if blocks should behave the same way as *ngIf, namely they // should match shallow queries on the first level of elements underneath // the embedded view boundary. expect(shallow.length).toBe(2); cmptInstance.exp = false; detectChanges(cmptInstance); expect(deep.length).toBe(1); expect(shallow.length).toBe(1); }); }); }); describe('queryList', () => { it('should be destroyed when the containing view is destroyed', () => { let queryInstance: QueryList; const SimpleComponentWithQuery = createComponent( 'some-component-with-query', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } }, 2, 0, [], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = queryInstance = tmp as QueryList); } }); function createTemplate() { Δcontainer(0); } function updateTemplate() { ΔcontainerRefreshStart(0); { if (condition) { let rf1 = ΔembeddedViewStart(1, 1, 0); { if (rf1 & RenderFlags.Create) { Δelement(0, 'some-component-with-query'); } } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } /** * % if (condition) { * * %} */ let condition = true; const t = new TemplateFixture(createTemplate, updateTemplate, 1, 0, [SimpleComponentWithQuery]); expect(t.html).toEqual('
'); expect((queryInstance !.changes as EventEmitter).closed).toBeFalsy(); condition = false; t.update(); expect(t.html).toEqual(''); expect((queryInstance !.changes as EventEmitter).closed).toBeTruthy(); }); }); it('should restore queries if view changes', () => { class SomeDir { constructor(public vcr: ViewContainerRef, public temp: TemplateRef) { this.vcr.createEmbeddedView(this.temp); } static ngDirectiveDef = ΔdefineDirective({ type: SomeDir, selectors: [['', 'someDir', '']], factory: () => new SomeDir( ΔdirectiveInject(ViewContainerRef as any), ΔdirectiveInject(TemplateRef as any)) }); } function AppComponent_Template_1(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div'); } } /** *
*
*/ const AppComponent = createComponent( 'app', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δtemplate( 0, AppComponent_Template_1, 1, 0, 'div', [AttributeMarker.Template, 'someDir']); Δelement(1, 'div', null, ['foo', '']); } }, 3, 0, [SomeDir], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.query = tmp as QueryList); } }); const fixture = new ComponentFixture(AppComponent); expect(fixture.component.query.length).toBe(1); }); describe('content', () => { let withContentInstance: WithContentDirective|null; let shallowCompInstance: ShallowComp|null; beforeEach(() => { withContentInstance = null; shallowCompInstance = null; }); class WithContentDirective { // @ContentChildren('foo') foos !: QueryList; contentInitQuerySnapshot = 0; contentCheckedQuerySnapshot = 0; ngAfterContentInit() { this.contentInitQuerySnapshot = this.foos ? this.foos.length : 0; } ngAfterContentChecked() { this.contentCheckedQuerySnapshot = this.foos ? this.foos.length : 0; } static ngDirectiveDef = ΔdefineDirective({ type: WithContentDirective, selectors: [['', 'with-content', '']], factory: () => withContentInstance = new WithContentDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.foos = tmp); } } }); } class ShallowComp { // @ContentChildren('foo', {descendants: false}) foos !: QueryList; static ngComponentDef = ΔdefineComponent({ type: ShallowComp, selectors: [['shallow-comp']], factory: () => shallowCompInstance = new ShallowComp(), template: function(rf: RenderFlags, ctx: any) {}, consts: 0, vars: 0, contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.foos = tmp); } } }); } it('should support content queries for directives', () => { /** *
* *
*/ const AppComponent = createComponent('app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', [AttributeMarker.Bindings, 'with-content']); { Δelement(1, 'span', null, ['foo', '']); } ΔelementEnd(); } }, 3, 0, [WithContentDirective]); const fixture = new ComponentFixture(AppComponent); expect(withContentInstance !.foos.length) .toBe(1, `Expected content query to match .`); expect(withContentInstance !.contentInitQuerySnapshot) .toBe( 1, `Expected content query results to be available when ngAfterContentInit was called.`); expect(withContentInstance !.contentCheckedQuerySnapshot) .toBe( 1, `Expected content query results to be available when ngAfterContentChecked was called.`); }); it('should support content queries for directives within repeated embedded views', () => { /** * % for (let i = 0; i < 3; i++) { *
* *
* % } */ const AppComponent = createComponent('app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δcontainer(0); } if (rf & RenderFlags.Update) { ΔcontainerRefreshStart(0); { for (let i = 0; i < 3; i++) { let rf = ΔembeddedViewStart(1, 3, 0); if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', [AttributeMarker.Bindings, 'with-content']); { Δelement(1, 'span', null, ['foo', '']); } ΔelementEnd(); } ΔembeddedViewEnd(); } } ΔcontainerRefreshEnd(); } }, 1, 0, [WithContentDirective]); const fixture = new ComponentFixture(AppComponent); expect(withContentInstance !.foos.length) .toBe(1, `Expected content query to match .`); expect(withContentInstance !.contentInitQuerySnapshot) .toBe( 1, `Expected content query results to be available when ngAfterContentInit was called.`); expect(withContentInstance !.contentCheckedQuerySnapshot) .toBe( 1, `Expected content query results to be available when ngAfterContentChecked was called.`); }); it('should not match directive host with content queries', () => { /** *
*
*/ const AppComponent = createComponent('app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', ['with-content', ''], ['foo', '']); } }, 2, 0, [WithContentDirective]); const fixture = new ComponentFixture(AppComponent); expect(withContentInstance !.foos.length) .toBe(0, `Expected content query not to match
.`); }); it('should match shallow content queries in views inserted / removed by ngIf', () => { function IfTemplate(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { Δelement(0, 'div', null, ['foo', '']); } } /** * *
*
*/ const AppComponent = createComponent('app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'shallow-comp'); { Δtemplate(1, IfTemplate, 2, 0, 'div', [AttributeMarker.Template, 'ngIf', '']); } ΔelementEnd(); } if (rf & RenderFlags.Update) { ΔelementProperty(1, 'ngIf', Δbind(ctx.showing)); } }, 2, 1, [ShallowComp, NgIf]); const fixture = new ComponentFixture(AppComponent); const qList = shallowCompInstance !.foos; expect(qList.length).toBe(0); fixture.component.showing = true; fixture.update(); expect(qList.length).toBe(1); fixture.component.showing = false; fixture.update(); expect(qList.length).toBe(0); }); // https://stackblitz.com/edit/angular-wlenwd?file=src%2Fapp%2Fapp.component.ts it('should support view and content queries matching the same element', () => { /** *
*
*
*
* class Cmpt { * @ViewChildren('foo, bar') foos; * } */ const AppComponent = createComponent( 'app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['with-content', '']); { Δelement(1, 'div', null, ['foo', '']); } ΔelementEnd(); Δelement(3, 'div', ['id', 'after'], ['bar', '']); } }, 5, 0, [WithContentDirective], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['foo', 'bar'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.foos = tmp as QueryList); } }); const fixture = new ComponentFixture(AppComponent); const viewQList = fixture.component.foos; expect(viewQList.length).toBe(2); expect(withContentInstance !.foos.length).toBe(1); expect(viewQList.first.nativeElement).toBe(withContentInstance !.foos.first.nativeElement); expect(viewQList.last.nativeElement.id).toBe('after'); }); it('should not report deep content query matches found above content children', () => { /** *
*
<-- should match content query *
*
<-- should not match content query * class AppComponent { * @ViewChildren('bar') bars: QueryList; * } */ const AppComponent = createComponent( 'app-component', function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['with-content', '']); { Δelement(1, 'div', ['id', 'yes'], ['foo', '']); } ΔelementEnd(); Δelement(3, 'div', null, ['foo', '']); } }, 5, 0, [WithContentDirective], [], function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔviewQuery(['bar'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.bars = tmp as QueryList); } }); const fixture = new ComponentFixture(AppComponent); expect(withContentInstance !.foos.length).toBe(1); expect(withContentInstance !.foos.first.nativeElement.id).toEqual('yes'); }); it('should report results to appropriate queries where deep content queries are nested', () => { class QueryDirective { fooBars: any; static ngDirectiveDef = ΔdefineDirective({ type: QueryDirective, selectors: [['', 'query', '']], exportAs: ['query'], factory: () => new QueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren('foo, bar, baz', {descendants: true}) // fooBars: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo', 'bar', 'baz'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.fooBars = tmp); } } }); } let outInstance: QueryDirective; let inInstance: QueryDirective; const AppComponent = createComponent( 'app-component', /** *
* *
* *
* *
*/ function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', [AttributeMarker.Bindings, 'query'], ['out', 'query']); { Δelement(2, 'span', ['id', 'foo'], ['foo', '']); ΔelementStart(4, 'div', [AttributeMarker.Bindings, 'query'], ['in', 'query']); { Δelement(6, 'span', ['id', 'bar'], ['bar', '']); } ΔelementEnd(); Δelement(8, 'span', ['id', 'baz'], ['baz', '']); } ΔelementEnd(); } if (rf & RenderFlags.Update) { outInstance = Δload(1); inInstance = Δload(5); } }, 10, 0, [QueryDirective]); const fixture = new ComponentFixture(AppComponent); expect(outInstance !.fooBars.length).toBe(3); expect(inInstance !.fooBars.length).toBe(1); }); it('should support nested shallow content queries ', () => { let outInstance: QueryDirective; let inInstance: QueryDirective; class QueryDirective { fooBars: any; static ngDirectiveDef = ΔdefineDirective({ type: QueryDirective, selectors: [['', 'query', '']], exportAs: ['query'], factory: () => new QueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren('foo', {descendants: true}) // fooBars: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.fooBars = tmp); } } }); } const AppComponent = createComponent( 'app-component', /** *
*
* *
*
*/ function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['query', ''], ['out', 'query']); { ΔelementStart(2, 'div', ['query', ''], ['in', 'query', 'foo', '']); { Δelement(5, 'span', ['id', 'bar'], ['foo', '']); } ΔelementEnd(); } ΔelementEnd(); } if (rf & RenderFlags.Update) { outInstance = Δload(1); inInstance = Δload(3); } }, 7, 0, [QueryDirective]); const fixture = new ComponentFixture(AppComponent); expect(outInstance !.fooBars.length).toBe(1); expect(inInstance !.fooBars.length).toBe(1); }); it('should support nested shallow content queries across multiple component instances', () => { let outInstance: QueryDirective; let inInstance: QueryDirective; class QueryDirective { fooBars: any; static ngDirectiveDef = ΔdefineDirective({ type: QueryDirective, selectors: [['', 'query', '']], exportAs: ['query'], factory: () => new QueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren('foo', {descendants: true}) // fooBars: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.fooBars = tmp); } } }); } const AppComponent = createComponent( 'app-component', /** *
*
* *
*
*/ function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['query', ''], ['out', 'query']); { ΔelementStart(2, 'div', ['query', ''], ['in', 'query', 'foo', '']); { Δelement(5, 'span', ['id', 'bar'], ['foo', '']); } ΔelementEnd(); } ΔelementEnd(); } if (rf & RenderFlags.Update) { outInstance = Δload(1); inInstance = Δload(3); } }, 7, 0, [QueryDirective]); const fixture1 = new ComponentFixture(AppComponent); expect(outInstance !.fooBars.length).toBe(1); expect(inInstance !.fooBars.length).toBe(1); outInstance = inInstance = null !; const fixture2 = new ComponentFixture(AppComponent); expect(outInstance !.fooBars.length).toBe(1); expect(inInstance !.fooBars.length).toBe(1); }); it('should respect shallow flag on content queries when mixing deep and shallow queries', () => { class ShallowQueryDirective { foos: any; static ngDirectiveDef = ΔdefineDirective({ type: ShallowQueryDirective, selectors: [['', 'shallow-query', '']], exportAs: ['shallow-query'], factory: () => new ShallowQueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren('foo', {descendants: false}) // foos: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], false, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.foos = tmp); } } }); } class DeepQueryDirective { foos: any; static ngDirectiveDef = ΔdefineDirective({ type: DeepQueryDirective, selectors: [['', 'deep-query', '']], exportAs: ['deep-query'], factory: () => new DeepQueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren('foo', {descendants: true}) // foos: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, ['foo'], true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.foos = tmp); } } }); } let shallowInstance: ShallowQueryDirective; let deepInstance: DeepQueryDirective; const AppComponent = createComponent( 'app-component', /** *
* *
* *
*
*/ function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart( 0, 'div', [AttributeMarker.Bindings, 'shallow-query', 'deep-query'], ['shallow', 'shallow-query', 'deep', 'deep-query']); { Δelement(3, 'span', null, ['foo', '']); ΔelementStart(5, 'div'); { Δelement(6, 'span', null, ['foo', '']); } ΔelementEnd(); } ΔelementEnd(); } if (rf & RenderFlags.Update) { shallowInstance = Δload(1); deepInstance = Δload(2); } }, 8, 0, [ShallowQueryDirective, DeepQueryDirective]); const fixture = new ComponentFixture(AppComponent); expect(shallowInstance !.foos.length).toBe(1); expect(deepInstance !.foos.length).toBe(2); }); }); describe('order', () => { class TextDirective { value !: string; static ngDirectiveDef = ΔdefineDirective({ type: TextDirective, selectors: [['', 'text', '']], factory: () => new TextDirective(), inputs: {value: 'text'} }); } it('should register content matches from top to bottom', () => { let contentQueryDirective: ContentQueryDirective; class ContentQueryDirective { // @ContentChildren(TextDirective) texts !: QueryList; static ngComponentDef = ΔdefineDirective({ type: ContentQueryDirective, selectors: [['', 'content-query', '']], factory: () => contentQueryDirective = new ContentQueryDirective(), contentQueries: (rf: RenderFlags, ctx: any, dirIndex: number) => { // @ContentChildren(TextDirective, {descendants: true}) // texts: QueryList; if (rf & RenderFlags.Create) { ΔcontentQuery(dirIndex, TextDirective, true, null); } if (rf & RenderFlags.Update) { let tmp: any; ΔqueryRefresh(tmp = ΔloadContentQuery()) && (ctx.texts = tmp); } } }); } const AppComponent = createComponent( 'app-component', /** *
* *
* * * *
* *
*/ function(rf: RenderFlags, ctx: any) { if (rf & RenderFlags.Create) { ΔelementStart(0, 'div', ['content-query']); { Δelement(1, 'span', ['text', 'A']); ΔelementStart(2, 'div', ['text', 'B']); ΔelementStart(3, 'span', ['text', 'C']); { Δelement(4, 'span', ['text', 'D']); } ΔelementEnd(); ΔelementEnd(); Δelement(5, 'span', ['text', 'E']); } ΔelementEnd(); } }, 6, 0, [TextDirective, ContentQueryDirective]); new ComponentFixture(AppComponent); expect(contentQueryDirective !.texts.map(item => item.value)).toEqual([ 'A', 'B', 'C', 'D', 'E' ]); }); it('should register view matches from top to bottom', () => { /** * *
* * * *
* */ class ViewQueryComponent { // @ViewChildren(TextDirective) texts !: QueryList; static ngComponentDef = ΔdefineComponent({ type: ViewQueryComponent, selectors: [['view-query']], factory: () => new ViewQueryComponent(), template: function(rf: RenderFlags, ctx: ViewQueryComponent) { if (rf & RenderFlags.Create) { Δelement(0, 'span', ['text', 'A']); ΔelementStart(1, 'div', ['text', 'B']); ΔelementStart(2, 'span', ['text', 'C']); { Δelement(3, 'span', ['text', 'D']); } ΔelementEnd(); ΔelementEnd(); Δelement(4, 'span', ['text', 'E']); } }, consts: 5, vars: 0, viewQuery: function(rf: RenderFlags, ctx: ViewQueryComponent) { let tmp: any; if (rf & RenderFlags.Create) { ΔviewQuery(TextDirective, true, null); } if (rf & RenderFlags.Update) { ΔqueryRefresh(tmp = ΔloadViewQuery>()) && (ctx.texts = tmp as QueryList); } }, directives: [TextDirective] }); } const fixture = new ComponentFixture(ViewQueryComponent); expect(fixture.component.texts.map(item => item.value)).toEqual(['A', 'B', 'C', 'D', 'E']); }); }); });