/** * @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 {CommonModule} from '@angular/common'; import {Component, Directive, ElementRef, EventEmitter, NgModule, Output, TemplateRef, ViewChild, ViewContainerRef} from '@angular/core'; import {Input} from '@angular/core/src/metadata'; import {TestBed} from '@angular/core/testing'; import {By} from '@angular/platform-browser'; describe('directives', () => { describe('matching', () => { @Directive({selector: 'ng-template[test]'}) class TestDirective { constructor(public templateRef: TemplateRef) {} } @Directive({selector: '[title]'}) class TitleDirective { } @Component({selector: 'test-cmpt', template: ''}) class TestComponent { } it('should match directives with attribute selectors on bindings', () => { @Directive({selector: '[test]'}) class TestDir { testValue: boolean|undefined; /** Setter to assert that a binding is not invoked with stringified attribute value */ @Input() set test(value: any) { // Assert that the binding is processed correctly. The property should be set // to a "false" boolean and never to the "false" string literal. this.testValue = value; if (value !== false) { fail('Should only be called with a false Boolean value, got a non-falsy value'); } } } TestBed.configureTestingModule({declarations: [TestComponent, TestDir]}); TestBed.overrideTemplate(TestComponent, ``); const fixture = TestBed.createComponent(TestComponent); const testDir = fixture.debugElement.query(By.directive(TestDir)).injector.get(TestDir); const spanEl = fixture.nativeElement.children[0]; fixture.detectChanges(); // the "test" attribute should not be reflected in the DOM as it is here only // for directive matching purposes expect(spanEl.hasAttribute('test')).toBe(false); expect(spanEl.getAttribute('class')).toBe('fade'); expect(testDir.testValue).toBe(false); }); it('should not accidentally set inputs from attributes extracted from bindings / outputs', () => { @Directive({selector: '[test]'}) class TestDir { @Input() prop1: boolean|undefined; @Input() prop2: boolean|undefined; testValue: boolean|undefined; /** Setter to assert that a binding is not invoked with stringified attribute value */ @Input() set test(value: any) { // Assert that the binding is processed correctly. The property should be set // to a "false" boolean and never to the "false" string literal. this.testValue = value; if (value !== false) { fail('Should only be called with a false Boolean value, got a non-falsy value'); } } } TestBed.configureTestingModule({declarations: [TestComponent, TestDir]}); TestBed.overrideTemplate( TestComponent, ``); const fixture = TestBed.createComponent(TestComponent); const testDir = fixture.debugElement.query(By.directive(TestDir)).injector.get(TestDir); const spanEl = fixture.nativeElement.children[0]; fixture.detectChanges(); // the "test" attribute should not be reflected in the DOM as it is here only // for directive matching purposes expect(spanEl.hasAttribute('test')).toBe(false); expect(spanEl.hasAttribute('prop1')).toBe(false); expect(spanEl.hasAttribute('prop2')).toBe(false); expect(spanEl.getAttribute('class')).toBe('fade'); expect(testDir.testValue).toBe(false); }); it('should match directives on ng-template', () => { TestBed.configureTestingModule({declarations: [TestComponent, TestDirective]}); TestBed.overrideTemplate(TestComponent, ``); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TestDirective)); expect(nodesWithDirective.length).toBe(1); expect(nodesWithDirective[0].injector.get(TestDirective).templateRef instanceof TemplateRef) .toBe(true); }); it('should match directives on ng-template created by * syntax', () => { TestBed.configureTestingModule({declarations: [TestComponent, TestDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TestDirective)); expect(nodesWithDirective.length).toBe(1); }); it('should match directives on ', () => { @Directive({selector: 'ng-container[directiveA]'}) class DirectiveA { constructor(public viewContainerRef: ViewContainerRef) {} } @Component({ selector: 'my-component', template: ` Some content ` }) class MyComponent { visible = true; } TestBed.configureTestingModule( {declarations: [MyComponent, DirectiveA], imports: [CommonModule]}); const fixture = TestBed.createComponent(MyComponent); fixture.detectChanges(); const directiveA = fixture.debugElement.query(By.css('span')).injector.get(DirectiveA); expect(directiveA.viewContainerRef).toBeTruthy(); }); it('should match directives on i18n-annotated attributes', () => { TestBed.configureTestingModule({declarations: [TestComponent, TitleDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TitleDirective)); expect(nodesWithDirective.length).toBe(1); }); it('should match a mix of bound directives and classes', () => { TestBed.configureTestingModule({declarations: [TestComponent, TitleDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TitleDirective)); expect(nodesWithDirective.length).toBe(1); }); it('should match classes to directive selectors without case sensitivity', () => { @Directive({selector: '.Titledir'}) class TitleClassDirective { } TestBed.configureTestingModule({declarations: [TestComponent, TitleClassDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TitleClassDirective)); expect(nodesWithDirective.length).toBe(1); }); it('should NOT match classes to directive selectors', () => { TestBed.configureTestingModule({declarations: [TestComponent, TitleDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TitleDirective)); expect(nodesWithDirective.length).toBe(0); }); it('should match attributes to directive selectors without case sensitivity', () => { @Directive({selector: '[title=Titledir]'}) class TitleAttributeDirective { } TestBed.configureTestingModule({declarations: [TestComponent, TitleAttributeDirective]}); TestBed.overrideTemplate(TestComponent, `
`); const fixture = TestBed.createComponent(TestComponent); const nodesWithDirective = fixture.debugElement.queryAllNodes(By.directive(TitleAttributeDirective)); expect(nodesWithDirective.length).toBe(1); }); it('should match directives with attribute selectors on outputs', () => { @Directive({selector: '[out]'}) class TestDir { @Output() out = new EventEmitter(); } TestBed.configureTestingModule({declarations: [TestComponent, TestDir]}); TestBed.overrideTemplate(TestComponent, ``); const fixture = TestBed.createComponent(TestComponent); const spanEl = fixture.nativeElement.children[0]; // "out" should not be part of reflected attributes expect(spanEl.hasAttribute('out')).toBe(false); expect(spanEl.getAttribute('class')).toBe('span'); expect(fixture.debugElement.query(By.directive(TestDir))).toBeTruthy(); }); it('should not match directives based on attribute bindings', () => { const calls: string[] = []; @Directive({selector: '[dir]'}) class MyDir { ngOnInit() { calls.push('MyDir.ngOnInit'); } } @Component({ selector: `my-comp`, template: `

`, }) class MyComp { direction = 'auto'; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); // Expect only one directive to be instantiated. expect(calls).toEqual(['MyDir.ngOnInit']); }); it('should match directives on elements with namespace', () => { const calls: string[] = []; @Directive({selector: 'svg[dir]'}) class MyDir { constructor(private el: ElementRef) {} ngOnInit() { calls.push(`MyDir.ngOnInit: ${this.el.nativeElement.tagName}`); } } @Component({ selector: `my-comp`, template: ``, }) class MyComp { } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(calls).toEqual(['MyDir.ngOnInit: svg']); }); it('should match directives on descendant elements with namespace', () => { const calls: string[] = []; @Directive({selector: 'text[dir]'}) class MyDir { constructor(private el: ElementRef) {} ngOnInit() { calls.push(`MyDir.ngOnInit: ${this.el.nativeElement.tagName}`); } } @Component({ selector: `my-comp`, template: ``, }) class MyComp { } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(calls).toEqual(['MyDir.ngOnInit: text']); }); it('should match directives when the node has "class", "style" and a binding', () => { const logs: string[] = []; @Directive({selector: '[test]'}) class MyDir { constructor() { logs.push('MyDir.contructor'); } @Input('test') myInput = ''; @Input('disabled') myInput2 = ''; } @Component({ // Note that below we're checking the case where the `test` attribute is after // one `class`, one `attribute` and one other binding. template: `
` }) class MyComp { test = ''; } TestBed.configureTestingModule({declarations: [MyComp, MyDir]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(logs).toEqual(['MyDir.contructor']); }); }); describe('inputs', () => { it('should allow directive inputs (as a prop binding) on ', () => { let dirInstance: WithInput; @Directive({selector: '[dir]'}) class WithInput { constructor() { dirInstance = this; } @Input() dir: string = ''; } @Component({ selector: 'my-app', template: '', }) class TestComp { message = 'Hello'; } TestBed.configureTestingModule({declarations: [TestComp, WithInput]}); const fixture = TestBed.createComponent(TestComp); fixture.detectChanges(); expect(dirInstance !.dir).toBe('Hello'); }); it('should allow directive inputs (as an interpolated prop) on ', () => { let dirInstance: WithInput; @Directive({selector: '[dir]'}) class WithInput { constructor() { dirInstance = this; } @Input() dir: string = ''; } @Component({ selector: 'my-app', template: '', }) class TestComp { message = 'Hello'; } TestBed.configureTestingModule({declarations: [TestComp, WithInput]}); const fixture = TestBed.createComponent(TestComp); fixture.detectChanges(); expect(dirInstance !.dir).toBe('Hello'); }); it('should allow directive inputs (as an interpolated prop) on with structural directives', () => { let dirInstance: WithInput; @Directive({selector: '[dir]'}) class WithInput { constructor() { dirInstance = this; } @Input() dir: string = ''; } @Component({ selector: 'my-app', template: '', }) class TestComp { message = 'Hello'; } TestBed.configureTestingModule({declarations: [TestComp, WithInput]}); const fixture = TestBed.createComponent(TestComp); fixture.detectChanges(); expect(dirInstance !.dir).toBe('Hello'); }); }); describe('outputs', () => { @Directive({selector: '[out]'}) class TestDir { @Output() out = new EventEmitter(); } it('should allow outputs of directive on ng-template', () => { @Component({template: ``}) class TestComp { @ViewChild(TestDir, {static: true}) testDir: TestDir|undefined; value = false; } TestBed.configureTestingModule({declarations: [TestComp, TestDir]}); const fixture = TestBed.createComponent(TestComp); fixture.detectChanges(); expect(fixture.componentInstance.testDir).toBeTruthy(); expect(fixture.componentInstance.value).toBe(false); fixture.componentInstance.testDir !.out.emit(); fixture.detectChanges(); expect(fixture.componentInstance.value).toBe(true); }); it('should allow outputs of directive on ng-container', () => { @Component({ template: ` Hello ` }) class TestComp { value = false; } TestBed.configureTestingModule({declarations: [TestComp, TestDir]}); const fixture = TestBed.createComponent(TestComp); const testDir = fixture.debugElement.query(By.css('span')).injector.get(TestDir); expect(fixture.componentInstance.value).toBe(false); testDir.out.emit(); fixture.detectChanges(); expect(fixture.componentInstance.value).toBeTruthy(); }); }); describe('attribute shadowing behaviors', () => { /** * To match ViewEngine, we need to ensure the following behaviors */ @Directive({ selector: '[dir-with-title]', }) class DirWithTitle { @Input() title = ''; } it('should set both the div attribute and the directive input for `title="value"`', () => { @Component({template: `
`}) class App { } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('title')).toBe('a'); }); it('should set the directive input only, shadowing the title property of the div, for `[title]="value"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); // We are checking the property here, not the attribute, because in the case of // [key]="value" we are always setting the property of the instance, and actually setting // the attribute is just a side-effect of the DOM implementation. expect(dirWithTitle.title).toBe('a'); expect(div.title).toBe(''); }); it('should allow setting directive `title` input with `[title]="value"` and a "attr.title" attribute with `attr.title="test"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('attr.title')).toBe('test'); expect(div.title).toBe(''); }); it('should allow setting directive `title` input with `[title]="value1"` and attribute with `[attr.title]="value2"`', () => { @Component({template: `
`}) class App { value1 = 'a'; value2 = 'b'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('title')).toBe('b'); }); it('should allow setting directive `title` input with `[title]="value1"` and attribute with `attr.title="{{value2}}"`', () => { @Component( {template: `
`}) class App { value1 = 'a'; value2 = 'b'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('title')).toBe('b'); }); it('should allow setting directive `title` input with `title="{{value}}"` and a "attr.title" attribute with `attr.title="test"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('attr.title')).toBe('test'); expect(div.title).toBe(''); }); it('should allow setting directive `title` input with `title="{{value1}}"` and attribute with `[attr.title]="value2"`', () => { @Component( {template: `
`}) class App { value1 = 'a'; value2 = 'b'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('title')).toBe('b'); }); it('should allow setting directive `title` input with `title="{{value1}}"` and attribute with `attr.title="{{value2}}"`', () => { @Component( {template: `
`}) class App { value1 = 'a'; value2 = 'b'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.getAttribute('title')).toBe('b'); }); it('should set the directive input only, shadowing the title property on the div, for `title="{{value}}"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe('a'); expect(div.title).toBe(''); }); it('should set the title attribute only, not directive input, for `attr.title="{{value}}"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe(''); expect(div.getAttribute('title')).toBe('a'); }); it('should set the title attribute only, not directive input, for `[attr.title]="value"`', () => { @Component({template: `
`}) class App { value = 'a'; } TestBed.configureTestingModule({ declarations: [App, DirWithTitle], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const dirWithTitle = fixture.debugElement.query(By.directive(DirWithTitle)).injector.get(DirWithTitle); const div = fixture.nativeElement.querySelector('div'); expect(dirWithTitle.title).toBe(''); expect(div.getAttribute('title')).toBe('a'); }); }); describe('directives with the same selector', () => { it('should process Directives from `declarations` list after imported ones', () => { const log: string[] = []; @Directive({selector: '[dir]'}) class DirectiveA { constructor() { log.push('DirectiveA.constructor'); } ngOnInit() { log.push('DirectiveA.ngOnInit'); } } @NgModule({ declarations: [DirectiveA], exports: [DirectiveA], }) class ModuleA { } @Directive({selector: '[dir]'}) class DirectiveB { constructor() { log.push('DirectiveB.constructor'); } ngOnInit() { log.push('DirectiveB.ngOnInit'); } } @Component({ selector: 'app', template: '
', }) class App { } TestBed.configureTestingModule({ imports: [ModuleA], declarations: [DirectiveB, App], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(log).toEqual([ 'DirectiveA.constructor', 'DirectiveB.constructor', 'DirectiveA.ngOnInit', 'DirectiveB.ngOnInit' ]); }); it('should respect imported module order', () => { const log: string[] = []; @Directive({selector: '[dir]'}) class DirectiveA { constructor() { log.push('DirectiveA.constructor'); } ngOnInit() { log.push('DirectiveA.ngOnInit'); } } @NgModule({ declarations: [DirectiveA], exports: [DirectiveA], }) class ModuleA { } @Directive({selector: '[dir]'}) class DirectiveB { constructor() { log.push('DirectiveB.constructor'); } ngOnInit() { log.push('DirectiveB.ngOnInit'); } } @NgModule({ declarations: [DirectiveB], exports: [DirectiveB], }) class ModuleB { } @Component({ selector: 'app', template: '
', }) class App { } TestBed.configureTestingModule({ imports: [ModuleA, ModuleB], declarations: [App], }); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(log).toEqual([ 'DirectiveA.constructor', 'DirectiveB.constructor', 'DirectiveA.ngOnInit', 'DirectiveB.ngOnInit' ]); }); }); });