/** * @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 {CommonModule} from '@angular/common'; import {Attribute, ChangeDetectorRef, Component, ComponentFactoryResolver, ComponentRef, Directive, ElementRef, EventEmitter, forwardRef, Host, HostBinding, Inject, Injectable, InjectionToken, INJECTOR, Injector, Input, LOCALE_ID, NgModule, NgZone, Optional, Output, Pipe, PipeTransform, Self, SkipSelf, TemplateRef, ViewChild, ViewContainerRef, ViewRef, ɵDEFAULT_LOCALE_ID as DEFAULT_LOCALE_ID} from '@angular/core'; import {ɵINJECTOR_SCOPE} from '@angular/core/src/core'; import {ViewRef as ViewRefInternal} from '@angular/core/src/render3/view_ref'; import {TestBed} from '@angular/core/testing'; import {ivyEnabled, onlyInIvy} from '@angular/private/testing'; import {BehaviorSubject} from 'rxjs'; describe('di', () => { describe('no dependencies', () => { it('should create directive with no deps', () => { @Directive({selector: '[dir]', exportAs: 'dir'}) class MyDirective { value = 'Created'; } @Component({template: '
{{ dir.value }}
'}) class MyComp { } TestBed.configureTestingModule({declarations: [MyDirective, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toContain('Created'); }); }); describe('multi providers', () => { it('should process ModuleWithProvider providers after module imports', () => { const testToken = new InjectionToken('test-multi'); @NgModule({providers: [{provide: testToken, useValue: 'A', multi: true}]}) class TestModuleA { } @NgModule({providers: [{provide: testToken, useValue: 'B', multi: true}]}) class TestModuleB { } TestBed.configureTestingModule({ imports: [ { ngModule: TestModuleA, providers: [{provide: testToken, useValue: 'C', multi: true}], }, TestModuleB, ] }); expect(TestBed.inject(testToken)).toEqual(['A', 'B', 'C']); }); }); describe('directive injection', () => { let log: string[] = []; @Directive({selector: '[dirB]', exportAs: 'dirB'}) class DirectiveB { @Input() value = 'DirB'; constructor() { log.push(this.value); } } beforeEach(() => log = []); it('should create directive with intra view dependencies', () => { @Directive({selector: '[dirA]', exportAs: 'dirA'}) class DirectiveA { value = 'DirA'; } @Directive({selector: '[dirC]', exportAs: 'dirC'}) class DirectiveC { value: string; constructor(dirA: DirectiveA, dirB: DirectiveB) { this.value = dirA.value + dirB.value; } } @Component({ template: `
{{ dir.value }}
` }) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, DirectiveC, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('span'); expect(divElement.textContent).toContain('DirADirB'); }); it('should instantiate injected directives in dependency order', () => { @Directive({selector: '[dirA]'}) class DirectiveA { value = 'dirA'; constructor(dirB: DirectiveB) { log.push(`DirA (dep: ${dirB.value})`); } } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(log).toEqual(['DirB', 'DirA (dep: DirB)']); }); it('should fallback to the module injector', () => { @Directive({selector: '[dirA]'}) class DirectiveA { value = 'dirA'; constructor(dirB: DirectiveB) { log.push(`DirA (dep: ${dirB.value})`); } } // - dirB is know to the node injectors // - then when dirA tries to inject dirB, it will check the node injector first tree // - if not found, it will check the module injector tree @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({ declarations: [DirectiveA, DirectiveB, MyComp], providers: [{provide: DirectiveB, useValue: {value: 'module'}}] }); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(log).toEqual(['DirB', 'DirA (dep: module)']); }); it('should instantiate injected directives before components', () => { @Component({selector: 'my-comp', template: ''}) class MyComp { constructor(dirB: DirectiveB) { log.push(`Comp (dep: ${dirB.value})`); } } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect(log).toEqual(['DirB', 'Comp (dep: DirB)']); }); it('should inject directives in the correct order in a for loop', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(dir: DirectiveB) { log.push(`DirA (dep: ${dir.value})`); } } @Component({template: '
'}) class MyComp { array = [1, 2, 3]; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(log).toEqual( ['DirB', 'DirA (dep: DirB)', 'DirB', 'DirA (dep: DirB)', 'DirB', 'DirA (dep: DirB)']); }); it('should instantiate directives with multiple out-of-order dependencies', () => { @Directive({selector: '[dirA]'}) class DirectiveA { value = 'DirA'; constructor() { log.push(this.value); } } @Directive({selector: '[dirC]'}) class DirectiveC { value = 'DirC'; constructor() { log.push(this.value); } } @Directive({selector: '[dirB]'}) class DirectiveB { constructor(dirA: DirectiveA, dirC: DirectiveC) { log.push(`DirB (deps: ${dirA.value} and ${dirC.value})`); } } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, DirectiveC, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(log).toEqual(['DirA', 'DirC', 'DirB (deps: DirA and DirC)']); }); it('should instantiate in the correct order for complex case', () => { @Directive({selector: '[dirC]'}) class DirectiveC { value = 'DirC'; constructor(dirB: DirectiveB) { log.push(`DirC (dep: ${dirB.value})`); } } @Directive({selector: '[dirA]'}) class DirectiveA { value = 'DirA'; constructor(dirC: DirectiveC) { log.push(`DirA (dep: ${dirC.value})`); } } @Directive({selector: '[dirD]'}) class DirectiveD { value = 'DirD'; constructor(dirA: DirectiveA) { log.push(`DirD (dep: ${dirA.value})`); } } @Component({selector: 'my-comp', template: ''}) class MyComp { constructor(dirD: DirectiveD) { log.push(`Comp (dep: ${dirD.value})`); } } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule( {declarations: [DirectiveA, DirectiveB, DirectiveC, DirectiveD, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect(log).toEqual( ['DirB', 'DirC (dep: DirB)', 'DirA (dep: DirC)', 'DirD (dep: DirA)', 'Comp (dep: DirD)']); }); it('should instantiate in correct order with mixed parent and peer dependencies', () => { @Component({template: '
'}) class MyApp { value = 'App'; } @Directive({selector: '[dirA]'}) class DirectiveA { constructor(dirB: DirectiveB, app: MyApp) { log.push(`DirA (deps: ${dirB.value} and ${app.value})`); } } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect(log).toEqual(['DirB', 'DirA (deps: DirB and App)']); }); it('should not use a parent when peer dep is available', () => { let count = 1; @Directive({selector: '[dirB]'}) class DirectiveB { count: number; constructor() { log.push(`DirB`); this.count = count++; } } @Directive({selector: '[dirA]'}) class DirectiveA { constructor(dirB: DirectiveB) { log.push(`DirA (dep: DirB - ${dirB.count})`); } } @Component({selector: 'my-comp', template: '
'}) class MyComp { } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect(log).toEqual(['DirB', 'DirB', 'DirA (dep: DirB - 2)']); }); describe('dependencies in parent views', () => { @Directive({selector: '[dirA]', exportAs: 'dirA'}) class DirectiveA { injector: Injector; constructor(public dirB: DirectiveB, public vcr: ViewContainerRef) { this.injector = vcr.injector; } } @Component( {selector: 'my-comp', template: '
{{ dir.dirB.value }}
'}) class MyComp { } it('should find dependencies on component hosts', () => { @Component({template: ''}) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('DirB'); }); it('should find dependencies for directives in embedded views', () => { @Component({ template: `
{{ dir.dirB.value }}
` }) class MyApp { showing = false; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.componentInstance.showing = true; fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('DirB'); }); it('should find dependencies of directives nested deeply in inline views', () => { @Component({ template: `
{{ dir.dirB.value }}
` }) class MyApp { skipContent = false; skipContent2 = false; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('DirB'); }); it('should find dependencies in declaration tree of ng-template (not insertion tree)', () => { @Directive({selector: '[structuralDir]'}) class StructuralDirective { @Input() tmp!: TemplateRef; constructor(public vcr: ViewContainerRef) {} create() { this.vcr.createEmbeddedView(this.tmp); } } @Component({ template: `
{{ dir.dirB.value }}
` }) class MyComp { @ViewChild(StructuralDirective) structuralDir!: StructuralDirective; } TestBed.configureTestingModule( {declarations: [StructuralDirective, DirectiveA, DirectiveB, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); fixture.componentInstance.structuralDir.create(); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div[value=insertion]'); expect(divElement.textContent).toEqual('declaration'); }); it('should create injectors on second template pass', () => { @Component({ template: `
` }) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('DirBDirB'); }); it('should create injectors and host bindings in same view', () => { @Directive({selector: '[hostBindingDir]'}) class HostBindingDirective { @HostBinding('id') id = 'foo'; } @Component({ template: `

{{ dir.dirB.value }}

` }) class MyApp { @ViewChild(HostBindingDirective) hostBindingDir!: HostBindingDirective; @ViewChild(DirectiveA) dirA!: DirectiveA; } TestBed.configureTestingModule( {declarations: [DirectiveA, DirectiveB, HostBindingDirective, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('DirB'); expect(divElement.id).toEqual('foo'); const dirA = fixture.componentInstance.dirA; expect(dirA.vcr.injector).toEqual(dirA.injector); const hostBindingDir = fixture.componentInstance.hostBindingDir; hostBindingDir.id = 'bar'; fixture.detectChanges(); expect(divElement.id).toBe('bar'); }); it('dynamic components should find dependencies when parent is projected', () => { @Directive({selector: '[dirA]'}) class DirA { } @Directive({selector: '[dirB]'}) class DirB { } @Component({selector: 'child', template: ''}) class Child { constructor(@Optional() readonly dirA: DirA, @Optional() readonly dirB: DirB) {} } @Component({ selector: 'projector', template: '', }) class Projector { } @Component({ template: `
`, entryComponents: [Child] }) class MyApp { @ViewChild('childOrigin', {read: ViewContainerRef, static: true}) childOrigin!: ViewContainerRef; @ViewChild('childOriginWithDirB', {read: ViewContainerRef, static: true}) childOriginWithDirB!: ViewContainerRef; childFactory = this.resolver.resolveComponentFactory(Child); constructor(readonly resolver: ComponentFactoryResolver, readonly injector: Injector) {} addChild() { return this.childOrigin.createComponent(this.childFactory); } addChildWithDirB() { return this.childOriginWithDirB.createComponent(this.childFactory); } } const fixture = TestBed.configureTestingModule({declarations: [Child, DirA, DirB, Projector, MyApp]}) .createComponent(MyApp); const child = fixture.componentInstance.addChild(); expect(child).toBeDefined(); expect(child.instance.dirA) .not.toBeNull('dirA should be found. It is on the parent of the viewContainerRef.'); const child2 = fixture.componentInstance.addChildWithDirB(); expect(child2).toBeDefined(); expect(child2.instance.dirA) .not.toBeNull('dirA should be found. It is on the parent of the viewContainerRef.'); expect(child2.instance.dirB) .toBeNull( 'dirB appears on the ng-container and should not be found because the ' + 'viewContainerRef.createComponent node is inserted next to the container.'); }); }); it('should throw if directive is not found anywhere', () => { @Directive({selector: '[dirB]'}) class DirectiveB { constructor() {} } @Directive({selector: '[dirA]'}) class DirectiveA { constructor(siblingDir: DirectiveB) {} } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/No provider for DirectiveB/); }); it('should throw if directive is not found in ancestor tree', () => { @Directive({selector: '[dirB]'}) class DirectiveB { constructor() {} } @Directive({selector: '[dirA]'}) class DirectiveA { constructor(siblingDir: DirectiveB) {} } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/No provider for DirectiveB/); }); onlyInIvy('Ivy has different error message for circular dependency') .it('should throw if directives try to inject each other', () => { @Directive({selector: '[dirB]'}) class DirectiveB { constructor(@Inject(forwardRef(() => DirectiveA)) siblingDir: DirectiveA) {} } @Directive({selector: '[dirA]'}) class DirectiveA { constructor(siblingDir: DirectiveB) {} } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/Circular dep for/); }); onlyInIvy('Ivy has different error message for circular dependency') .it('should throw if directive tries to inject itself', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(siblingDir: DirectiveA) {} } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/Circular dep for/); }); describe('flags', () => { @Directive({selector: '[dirB]'}) class DirectiveB { @Input('dirB') value = ''; } describe('Optional', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(@Optional() public dirB: DirectiveB) {} } it('should not throw if dependency is @Optional (module injector)', () => { @Component({template: '
'}) class MyComp { @ViewChild(DirectiveA) dirA!: DirectiveA; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const dirA = fixture.componentInstance.dirA; expect(dirA.dirB).toBeNull(); }); it('should return null if @Optional dependency has @Self flag', () => { @Directive({selector: '[dirC]'}) class DirectiveC { constructor(@Optional() @Self() public dirB: DirectiveB) {} } @Component({template: '
'}) class MyComp { @ViewChild(DirectiveC) dirC!: DirectiveC; } TestBed.configureTestingModule({declarations: [DirectiveC, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const dirC = fixture.componentInstance.dirC; expect(dirC.dirB).toBeNull(); }); it('should not throw if dependency is @Optional but defined elsewhere', () => { @Directive({selector: '[dirC]'}) class DirectiveC { constructor(@Optional() public dirB: DirectiveB) {} } @Component({template: '
'}) class MyComp { @ViewChild(DirectiveC) dirC!: DirectiveC; } TestBed.configureTestingModule({declarations: [DirectiveB, DirectiveC, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const dirC = fixture.componentInstance.dirC; expect(dirC.dirB).toBeNull(); }); }); it('should skip the current node with @SkipSelf', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(@SkipSelf() public dirB: DirectiveB) {} } @Component({selector: 'my-comp', template: '
'}) class MyComp { @ViewChild(DirectiveA) dirA!: DirectiveA; } @Component({template: ''}) class MyApp { @ViewChild(MyComp) myComp!: MyComp; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const dirA = fixture.componentInstance.myComp.dirA; expect(dirA.dirB.value).toEqual('parent'); }); onlyInIvy('Ivy has different error message when dependency is not found') .it('should check only the current node with @Self', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(@Self() public dirB: DirectiveB) {} } @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyComp]}); expect(() => TestBed.createComponent(MyComp)) .toThrowError(/NodeInjector: NOT_FOUND \[DirectiveB]/); }); describe('@Host', () => { @Directive({selector: '[dirA]'}) class DirectiveA { constructor(@Host() public dirB: DirectiveB) {} } @Directive({selector: '[dirString]'}) class DirectiveString { constructor(@Host() public s: String) {} } it('should find viewProviders on the host itself', () => { @Component({ selector: 'my-comp', template: '
', viewProviders: [{provide: String, useValue: 'Foo'}] }) class MyComp { @ViewChild(DirectiveString) dirString!: DirectiveString; } @Component({template: ''}) class MyApp { @ViewChild(MyComp) myComp!: MyComp; } TestBed.configureTestingModule({declarations: [DirectiveString, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const dirString = fixture.componentInstance.myComp.dirString; expect(dirString.s).toBe('Foo'); }); it('should find host component on the host itself', () => { @Directive({selector: '[dirComp]'}) class DirectiveComp { constructor(@Inject(forwardRef(() => MyComp)) @Host() public comp: MyComp) {} } @Component({selector: 'my-comp', template: '
'}) class MyComp { @ViewChild(DirectiveComp) dirComp!: DirectiveComp; } @Component({template: ''}) class MyApp { @ViewChild(MyComp) myComp!: MyComp; } TestBed.configureTestingModule({declarations: [DirectiveComp, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const myComp = fixture.componentInstance.myComp; const dirComp = myComp.dirComp; expect(dirComp.comp).toBe(myComp); }); onlyInIvy('Ivy has different error message when dependency is not found') .it('should not find providers on the host itself', () => { @Component({ selector: 'my-comp', template: '
', providers: [{provide: String, useValue: 'Foo'}] }) class MyComp { } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveString, MyComp, MyApp]}); expect(() => TestBed.createComponent(MyApp)) .toThrowError(/NodeInjector: NOT_FOUND \[String]/); }); onlyInIvy('Ivy has different error message when dependency is not found') .it('should not find other directives on the host itself', () => { @Component({selector: 'my-comp', template: '
'}) class MyComp { } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule( {declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); expect(() => TestBed.createComponent(MyApp)) .toThrowError(/NodeInjector: NOT_FOUND \[DirectiveB]/); }); onlyInIvy('Ivy has different error message when dependency is not found') .it('should not find providers on the host itself if in inline view', () => { @Component({ selector: 'my-comp', template: '
' }) class MyComp { showing = false; } @Component({template: ''}) class MyApp { @ViewChild(MyComp) myComp!: MyComp; } TestBed.configureTestingModule( {declarations: [DirectiveA, DirectiveB, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect(() => { fixture.componentInstance.myComp.showing = true; fixture.detectChanges(); }).toThrowError(/NodeInjector: NOT_FOUND \[DirectiveB]/); }); it('should find providers across embedded views if not passing component boundary', () => { @Component({template: '
'}) class MyApp { showing = false; @ViewChild(DirectiveA) dirA!: DirectiveA; @ViewChild(DirectiveB) dirB!: DirectiveB; } TestBed.configureTestingModule({declarations: [DirectiveA, DirectiveB, MyApp]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); fixture.componentInstance.showing = true; fixture.detectChanges(); const dirA = fixture.componentInstance.dirA; const dirB = fixture.componentInstance.dirB; expect(dirA.dirB).toBe(dirB); }); onlyInIvy('Ivy has different error message when dependency is not found') .it('should not find component above the host', () => { @Directive({selector: '[dirComp]'}) class DirectiveComp { constructor(@Inject(forwardRef(() => MyApp)) @Host() public comp: MyApp) {} } @Component({selector: 'my-comp', template: '
'}) class MyComp { } @Component({template: ''}) class MyApp { } TestBed.configureTestingModule({declarations: [DirectiveComp, MyComp, MyApp]}); expect(() => TestBed.createComponent(MyApp)) .toThrowError(/NodeInjector: NOT_FOUND \[MyApp]/); }); describe('regression', () => { // based on https://stackblitz.com/edit/angular-riss8k?file=src/app/app.component.ts it('should allow directives with Host flag to inject view providers from containing component', () => { class ControlContainer {} let controlContainers: ControlContainer[] = []; let injectedControlContainer: ControlContainer|null = null; @Directive({ selector: '[group]', providers: [{provide: ControlContainer, useExisting: GroupDirective}] }) class GroupDirective { constructor() { controlContainers.push(this); } } @Directive({selector: '[control]'}) class ControlDirective { constructor(@Host() @SkipSelf() @Inject(ControlContainer) parent: ControlContainer) { injectedControlContainer = parent; } } @Component({ selector: 'my-comp', template: '', viewProviders: [{provide: ControlContainer, useExisting: GroupDirective}] }) class MyComp { } @Component({ template: `
` }) class MyApp { } TestBed.configureTestingModule( {declarations: [GroupDirective, ControlDirective, MyComp, MyApp]}); const fixture = TestBed.createComponent(MyApp); expect(fixture.nativeElement.innerHTML) .toBe('
'); expect(controlContainers).toEqual([injectedControlContainer!]); }); }); }); }); }); describe('Tree shakable injectors', () => { it('should support tree shakable injectors scopes', () => { @Injectable({providedIn: 'any'}) class AnyService { constructor(public injector: Injector) {} } @Injectable({providedIn: 'root'}) class RootService { constructor(public injector: Injector) {} } @Injectable({providedIn: 'platform'}) class PlatformService { constructor(public injector: Injector) {} } const testBedInjector: Injector = TestBed.get(Injector); const childInjector = Injector.create([], testBedInjector); const anyService = childInjector.get(AnyService); expect(anyService.injector).toBe(childInjector); const rootService = childInjector.get(RootService); expect(rootService.injector.get(ɵINJECTOR_SCOPE)).toBe('root'); const platformService = childInjector.get(PlatformService); expect(platformService.injector.get(ɵINJECTOR_SCOPE)).toBe('platform'); }); }); describe('service injection', () => { it('should create instance even when no injector present', () => { @Injectable({providedIn: 'root'}) class MyService { value = 'MyService'; } @Component({template: '
{{myService.value}}
'}) class MyComp { constructor(public myService: MyService) {} } TestBed.configureTestingModule({declarations: [MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toEqual('MyService'); }); it('should support sub-classes with no @Injectable decorator', () => { @Injectable() class Dependency { } @Injectable() class SuperClass { constructor(public dep: Dependency) {} } // Note, no @Injectable decorators for these two classes class SubClass extends SuperClass {} class SubSubClass extends SubClass {} @Component({template: ''}) class MyComp { constructor(public myService: SuperClass) {} } TestBed.configureTestingModule({ declarations: [MyComp], providers: [{provide: SuperClass, useClass: SubSubClass}, Dependency] }); const warnSpy = spyOn(console, 'warn'); const fixture = TestBed.createComponent(MyComp); expect(fixture.componentInstance.myService.dep instanceof Dependency).toBe(true); if (ivyEnabled) { expect(warnSpy).toHaveBeenCalledWith( `DEPRECATED: DI is instantiating a token "SubSubClass" that inherits its @Injectable decorator but does not provide one itself.\n` + `This will become an error in a future version of Angular. Please add @Injectable() to the "SubSubClass" class.`); } }); it('should instantiate correct class when undecorated class extends an injectable', () => { @Injectable() class MyService { id = 1; } class MyRootService extends MyService { id = 2; } @Component({template: ''}) class App { } TestBed.configureTestingModule({declarations: [App], providers: [MyRootService]}); const warnSpy = spyOn(console, 'warn'); const fixture = TestBed.createComponent(App); fixture.detectChanges(); const provider = TestBed.inject(MyRootService); expect(provider instanceof MyRootService).toBe(true); expect(provider.id).toBe(2); if (ivyEnabled) { expect(warnSpy).toHaveBeenCalledWith( `DEPRECATED: DI is instantiating a token "MyRootService" that inherits its @Injectable decorator but does not provide one itself.\n` + `This will become an error in a future version of Angular. Please add @Injectable() to the "MyRootService" class.`); } }); it('should inject services in constructor with overloads', () => { @Injectable({providedIn: 'root'}) class MyService { } @Injectable({providedIn: 'root'}) class MyOtherService { } @Component({template: ''}) class MyComp { constructor(myService: MyService); constructor( public myService: MyService, @Optional() public myOtherService?: MyOtherService) {} } TestBed.configureTestingModule({declarations: [MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.componentInstance.myService instanceof MyService).toBe(true); expect(fixture.componentInstance.myOtherService instanceof MyOtherService).toBe(true); }); }); describe('service injection with useClass', () => { @Injectable({providedIn: 'root'}) class BarServiceDep { name = 'BarServiceDep'; } @Injectable({providedIn: 'root'}) class BarService { constructor(public dep: BarServiceDep) {} getMessage() { return 'bar'; } } @Injectable({providedIn: 'root'}) class FooServiceDep { name = 'FooServiceDep'; } @Injectable({providedIn: 'root', useClass: BarService}) class FooService { constructor(public dep: FooServiceDep) {} getMessage() { return 'foo'; } } it('should use @Injectable useClass config when token is not provided', () => { let provider: FooService|BarService; @Component({template: ''}) class App { constructor(service: FooService) { provider = service; } } TestBed.configureTestingModule({declarations: [App]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(provider!.getMessage()).toBe('bar'); // ViewEngine incorrectly uses the original class DI config, instead of the one from useClass. if (ivyEnabled) { expect(provider!.dep.name).toBe('BarServiceDep'); } }); it('should use constructor config directly when token is explicitly provided via useClass', () => { let provider: FooService|BarService; @Component({template: ''}) class App { constructor(service: FooService) { provider = service; } } TestBed.configureTestingModule( {declarations: [App], providers: [{provide: FooService, useClass: FooService}]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(provider!.getMessage()).toBe('foo'); }); it('should inject correct provider when re-providing an injectable that has useClass', () => { let directProvider: FooService|BarService; let overriddenProvider: FooService|BarService; @Component({template: ''}) class App { constructor(@Inject('stringToken') overriddenService: FooService, service: FooService) { overriddenProvider = overriddenService; directProvider = service; } } TestBed.configureTestingModule( {declarations: [App], providers: [{provide: 'stringToken', useClass: FooService}]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(directProvider!.getMessage()).toBe('bar'); expect(overriddenProvider!.getMessage()).toBe('foo'); // ViewEngine incorrectly uses the original class DI config, instead of the one from useClass. if (ivyEnabled) { expect(directProvider!.dep.name).toBe('BarServiceDep'); expect(overriddenProvider!.dep.name).toBe('FooServiceDep'); } }); it('should use constructor config directly when token is explicitly provided as a type provider', () => { let provider: FooService|BarService; @Component({template: ''}) class App { constructor(service: FooService) { provider = service; } } TestBed.configureTestingModule({declarations: [App], providers: [FooService]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(provider!.getMessage()).toBe('foo'); expect(provider!.dep.name).toBe('FooServiceDep'); }); }); describe('inject', () => { it('should inject from parent view', () => { @Directive({selector: '[parentDir]'}) class ParentDirective { } @Directive({selector: '[childDir]', exportAs: 'childDir'}) class ChildDirective { value: string; constructor(public parent: ParentDirective) { this.value = parent.constructor.name; } } @Directive({selector: '[child2Dir]', exportAs: 'child2Dir'}) class Child2Directive { value: boolean; constructor(parent: ParentDirective, child: ChildDirective) { this.value = parent === child.parent; } } @Component({ template: `
{{ child1.value }}-{{ child2.value }}
` }) class MyComp { showing = true; } TestBed.configureTestingModule( {declarations: [ParentDirective, ChildDirective, Child2Directive, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); expect(divElement.textContent).toBe('ParentDirective-true'); }); }); describe('Special tokens', () => { describe('Injector', () => { it('should inject the injector', () => { @Directive({selector: '[injectorDir]'}) class InjectorDir { constructor(public injector: Injector) {} } @Directive({selector: '[otherInjectorDir]'}) class OtherInjectorDir { constructor(public otherDir: InjectorDir, public injector: Injector) {} } @Component({template: '
'}) class MyComp { @ViewChild(InjectorDir) injectorDir!: InjectorDir; @ViewChild(OtherInjectorDir) otherInjectorDir!: OtherInjectorDir; } TestBed.configureTestingModule({declarations: [InjectorDir, OtherInjectorDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); const injectorDir = fixture.componentInstance.injectorDir; const otherInjectorDir = fixture.componentInstance.otherInjectorDir; expect(injectorDir.injector.get(ElementRef).nativeElement).toBe(divElement); expect(otherInjectorDir.injector.get(ElementRef).nativeElement).toBe(divElement); expect(otherInjectorDir.injector.get(InjectorDir)).toBe(injectorDir); expect(injectorDir.injector).not.toBe(otherInjectorDir.injector); }); it('should inject INJECTOR', () => { @Directive({selector: '[injectorDir]'}) class InjectorDir { constructor(@Inject(INJECTOR) public injector: Injector) {} } @Component({template: '
'}) class MyComp { @ViewChild(InjectorDir) injectorDir!: InjectorDir; } TestBed.configureTestingModule({declarations: [InjectorDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); const injectorDir = fixture.componentInstance.injectorDir; expect(injectorDir.injector.get(ElementRef).nativeElement).toBe(divElement); expect(injectorDir.injector.get(Injector).get(ElementRef).nativeElement).toBe(divElement); expect(injectorDir.injector.get(INJECTOR).get(ElementRef).nativeElement).toBe(divElement); }); }); describe('ElementRef', () => { it('should create directive with ElementRef dependencies', () => { @Directive({selector: '[dir]'}) class MyDir { value: string; constructor(public elementRef: ElementRef) { this.value = (elementRef.constructor as any).name; } } @Directive({selector: '[otherDir]'}) class MyOtherDir { isSameInstance: boolean; constructor(public elementRef: ElementRef, public directive: MyDir) { this.isSameInstance = elementRef === directive.elementRef; } } @Component({template: '
'}) class MyComp { @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyDir, MyOtherDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const divElement = fixture.nativeElement.querySelector('div'); const directive = fixture.componentInstance.directive; const otherDirective = fixture.componentInstance.otherDirective; expect(directive.value).toContain('ElementRef'); expect(directive.elementRef.nativeElement).toEqual(divElement); expect(otherDirective.elementRef.nativeElement).toEqual(divElement); // Each ElementRef instance should be unique expect(otherDirective.isSameInstance).toBe(false); }); it('should create ElementRef with comment if requesting directive is on node', () => { @Directive({selector: '[dir]'}) class MyDir { value: string; constructor(public elementRef: ElementRef) { this.value = (elementRef.constructor as any).name; } } @Component({template: ''}) class MyComp { @ViewChild(MyDir) directive!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directive; expect(directive.value).toContain('ElementRef'); // the nativeElement should be a comment expect(directive.elementRef.nativeElement.nodeType).toEqual(Node.COMMENT_NODE); }); it('should be available if used in conjunction with other tokens', () => { @Injectable() class ServiceA { subject: any; constructor(protected zone: NgZone) { this.subject = new BehaviorSubject(1); // trigger change detection zone.run(() => { this.subject.next(2); }); } } @Directive({selector: '[dir]'}) class DirectiveA { constructor(public service: ServiceA, public elementRef: ElementRef) {} } @Component({ selector: 'child', template: `
`, }) class ChildComp { @ViewChild(DirectiveA) directive!: DirectiveA; } @Component({ selector: 'root', template: '...', }) class RootComp { public childCompRef!: ComponentRef; constructor( public factoryResolver: ComponentFactoryResolver, public vcr: ViewContainerRef) {} create() { const factory = this.factoryResolver.resolveComponentFactory(ChildComp); this.childCompRef = this.vcr.createComponent(factory); this.childCompRef.changeDetectorRef.detectChanges(); } } // this module is needed, so that View Engine can generate factory for ChildComp @NgModule({ declarations: [DirectiveA, RootComp, ChildComp], entryComponents: [RootComp, ChildComp], }) class ModuleA { } TestBed.configureTestingModule({ imports: [ModuleA], providers: [ServiceA], }); const fixture = TestBed.createComponent(RootComp); fixture.autoDetectChanges(); fixture.componentInstance.create(); const {elementRef} = fixture.componentInstance.childCompRef.instance.directive; expect(elementRef.nativeElement.id).toBe('test-id'); }); }); describe('TemplateRef', () => { @Directive({selector: '[dir]', exportAs: 'dir'}) class MyDir { value: string; constructor(public templateRef: TemplateRef) { this.value = (templateRef.constructor as any).name; } } onlyInIvy('Ivy creates a unique instance of TemplateRef for each directive') .it('should create directive with TemplateRef dependencies', () => { @Directive({selector: '[otherDir]', exportAs: 'otherDir'}) class MyOtherDir { isSameInstance: boolean; constructor(public templateRef: TemplateRef, public directive: MyDir) { this.isSameInstance = templateRef === directive.templateRef; } } @Component({ template: '' }) class MyComp { @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyDir, MyOtherDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directive; const otherDirective = fixture.componentInstance.otherDirective; expect(directive.value).toContain('TemplateRef'); expect(directive.templateRef).not.toBeNull(); expect(otherDirective.templateRef).not.toBeNull(); // Each TemplateRef instance should be unique expect(otherDirective.isSameInstance).toBe(false); }); it('should throw if injected on an element', () => { @Component({template: '
'}) class MyComp { } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/No provider for TemplateRef/); }); it('should throw if injected on an ng-container', () => { @Component({template: ''}) class MyComp { } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); expect(() => TestBed.createComponent(MyComp)).toThrowError(/No provider for TemplateRef/); }); it('should NOT throw if optional and injected on an element', () => { @Directive({selector: '[optionalDir]', exportAs: 'optionalDir'}) class OptionalDir { constructor(@Optional() public templateRef: TemplateRef) {} } @Component({template: '
'}) class MyComp { @ViewChild(OptionalDir) directive!: OptionalDir; } TestBed.configureTestingModule({declarations: [OptionalDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.componentInstance.directive.templateRef).toBeNull(); }); }); describe('ViewContainerRef', () => { onlyInIvy('Ivy creates a unique instance of ViewContainerRef for each directive') .it('should create directive with ViewContainerRef dependencies', () => { @Directive({selector: '[dir]', exportAs: 'dir'}) class MyDir { value: string; constructor(public viewContainerRef: ViewContainerRef) { this.value = (viewContainerRef.constructor as any).name; } } @Directive({selector: '[otherDir]', exportAs: 'otherDir'}) class MyOtherDir { isSameInstance: boolean; constructor(public viewContainerRef: ViewContainerRef, public directive: MyDir) { this.isSameInstance = viewContainerRef === directive.viewContainerRef; } } @Component({template: '
'}) class MyComp { @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyDir, MyOtherDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directive; const otherDirective = fixture.componentInstance.otherDirective; expect(directive.value).toContain('ViewContainerRef'); expect(directive.viewContainerRef).not.toBeNull(); expect(otherDirective.viewContainerRef).not.toBeNull(); // Each ViewContainerRef instance should be unique expect(otherDirective.isSameInstance).toBe(false); }); it('should sync ViewContainerRef state between all injected instances', () => { @Component({ selector: 'root', template: `Test`, }) class Root { @ViewChild(TemplateRef, {static: true}) tmpl!: TemplateRef; constructor(public vcr: ViewContainerRef, public vcr2: ViewContainerRef) {} ngOnInit(): void { this.vcr.createEmbeddedView(this.tmpl); } } TestBed.configureTestingModule({ declarations: [Root], }); const fixture = TestBed.createComponent(Root); fixture.detectChanges(); const cmp = fixture.componentInstance; expect(cmp.vcr.length).toBe(1); expect(cmp.vcr2.length).toBe(1); expect(cmp.vcr2.get(0)).toEqual(cmp.vcr.get(0)); cmp.vcr2.remove(0); expect(cmp.vcr.length).toBe(0); expect(cmp.vcr.get(0)).toBeNull(); }); }); describe('ChangeDetectorRef', () => { @Directive({selector: '[dir]', exportAs: 'dir'}) class MyDir { value: string; constructor(public cdr: ChangeDetectorRef) { this.value = (cdr.constructor as any).name; } } @Directive({selector: '[otherDir]', exportAs: 'otherDir'}) class MyOtherDir { constructor(public cdr: ChangeDetectorRef) {} } @Component({selector: 'my-comp', template: ''}) class MyComp { constructor(public cdr: ChangeDetectorRef) {} } it('should inject host component ChangeDetectorRef into directives on templates', () => { let pipeInstance: MyPipe; @Pipe({name: 'pipe'}) class MyPipe implements PipeTransform { constructor(public cdr: ChangeDetectorRef) { pipeInstance = this; } transform(value: any): any { return value; } } @Component({ selector: 'my-app', template: `
Visible
`, }) class MyApp { showing = true; constructor(public cdr: ChangeDetectorRef) {} } TestBed.configureTestingModule({declarations: [MyApp, MyPipe], imports: [CommonModule]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect((pipeInstance!.cdr as ViewRefInternal).context) .toBe(fixture.componentInstance); }); it('should inject current component ChangeDetectorRef into directives on the same node as components', () => { @Component({selector: 'my-app', template: ''}) class MyApp { @ViewChild(MyComp) component!: MyComp; @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyApp, MyComp, MyDir, MyOtherDir]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const app = fixture.componentInstance; const comp = fixture.componentInstance.component; expect((comp!.cdr as ViewRefInternal).context).toBe(comp); // ChangeDetectorRef is the token, ViewRef has historically been the constructor expect(app.directive.value).toContain('ViewRef'); // Each ChangeDetectorRef instance should be unique expect(app.directive!.cdr).not.toBe(comp!.cdr); expect(app.directive!.cdr).not.toBe(app.otherDirective!.cdr); }); it('should inject host component ChangeDetectorRef into directives on normal elements', () => { @Component({selector: 'my-comp', template: '
'}) class MyComp { constructor(public cdr: ChangeDetectorRef) {} @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyComp, MyDir, MyOtherDir]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const comp = fixture.componentInstance; expect((comp!.cdr as ViewRefInternal).context).toBe(comp); // ChangeDetectorRef is the token, ViewRef has historically been the constructor expect(comp.directive.value).toContain('ViewRef'); // Each ChangeDetectorRef instance should be unique expect(comp.directive!.cdr).not.toBe(comp.cdr); expect(comp.directive!.cdr).not.toBe(comp.otherDirective!.cdr); }); it('should inject host component ChangeDetectorRef into directives in a component\'s ContentChildren', () => { @Component({ selector: 'my-app', template: `
` }) class MyApp { constructor(public cdr: ChangeDetectorRef) {} @ViewChild(MyComp) component!: MyComp; @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyApp, MyComp, MyDir, MyOtherDir]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); const app = fixture.componentInstance; expect((app!.cdr as ViewRefInternal).context).toBe(app); const comp = fixture.componentInstance.component; // ChangeDetectorRef is the token, ViewRef has historically been the constructor expect(app.directive.value).toContain('ViewRef'); // Each ChangeDetectorRef instance should be unique expect(app.directive!.cdr).not.toBe(comp.cdr); expect(app.directive!.cdr).not.toBe(app.otherDirective!.cdr); }); it('should inject host component ChangeDetectorRef into directives in embedded views', () => { @Component({ selector: 'my-comp', template: `
` }) class MyComp { showing = true; constructor(public cdr: ChangeDetectorRef) {} @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyComp, MyDir, MyOtherDir]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const comp = fixture.componentInstance; expect((comp!.cdr as ViewRefInternal).context).toBe(comp); // ChangeDetectorRef is the token, ViewRef has historically been the constructor expect(comp.directive.value).toContain('ViewRef'); // Each ChangeDetectorRef instance should be unique expect(comp.directive!.cdr).not.toBe(comp.cdr); expect(comp.directive!.cdr).not.toBe(comp.otherDirective!.cdr); }); it('should inject host component ChangeDetectorRef into directives on containers', () => { @Component( {selector: 'my-comp', template: '
'}) class MyComp { showing = true; constructor(public cdr: ChangeDetectorRef) {} @ViewChild(MyDir) directive!: MyDir; @ViewChild(MyOtherDir) otherDirective!: MyOtherDir; } TestBed.configureTestingModule({declarations: [MyComp, MyDir, MyOtherDir]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const comp = fixture.componentInstance; expect((comp!.cdr as ViewRefInternal).context).toBe(comp); // ChangeDetectorRef is the token, ViewRef has historically been the constructor expect(comp.directive.value).toContain('ViewRef'); // Each ChangeDetectorRef instance should be unique expect(comp.directive!.cdr).not.toBe(comp.cdr); expect(comp.directive!.cdr).not.toBe(comp.otherDirective!.cdr); }); it('should inject host component ChangeDetectorRef into directives on ng-container', () => { let dirInstance: MyDirective; @Directive({selector: '[getCDR]'}) class MyDirective { constructor(public cdr: ChangeDetectorRef) { dirInstance = this; } } @Component({ selector: 'my-app', template: `Visible`, }) class MyApp { constructor(public cdr: ChangeDetectorRef) {} } TestBed.configureTestingModule({declarations: [MyApp, MyDirective]}); const fixture = TestBed.createComponent(MyApp); fixture.detectChanges(); expect((dirInstance!.cdr as ViewRefInternal).context) .toBe(fixture.componentInstance); }); }); }); describe('string tokens', () => { it('should be able to provide a string token', () => { @Directive({selector: '[injectorDir]', providers: [{provide: 'test', useValue: 'provided'}]}) class InjectorDir { constructor(@Inject('test') public value: string) {} } @Component({template: '
'}) class MyComp { @ViewChild(InjectorDir) injectorDirInstance!: InjectorDir; } TestBed.configureTestingModule({declarations: [InjectorDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const injectorDir = fixture.componentInstance.injectorDirInstance; expect(injectorDir.value).toBe('provided'); }); }); it('should not cause cyclic dependency if same token is requested in deps with @SkipSelf', () => { @Component({ selector: 'my-comp', template: '...', providers: [{ provide: LOCALE_ID, useFactory: () => 'ja-JP', // Note: `LOCALE_ID` is also provided within APPLICATION_MODULE_PROVIDERS, so we use it here // as a dep and making sure it doesn't cause cyclic dependency (since @SkipSelf is present) deps: [[new Inject(LOCALE_ID), new Optional(), new SkipSelf()]] }] }) class MyComp { constructor(@Inject(LOCALE_ID) public localeId: string) {} } TestBed.configureTestingModule({declarations: [MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.componentInstance.localeId).toBe('ja-JP'); }); it('module-level deps should not access Component/Directive providers', () => { @Component({ selector: 'my-comp', template: '...', providers: [{ provide: 'LOCALE_ID_DEP', // useValue: 'LOCALE_ID_DEP_VALUE' }] }) class MyComp { constructor(@Inject(LOCALE_ID) public localeId: string) {} } TestBed.configureTestingModule({ declarations: [MyComp], providers: [{ provide: LOCALE_ID, // we expect `localeDepValue` to be undefined, since it's not provided at a module level useFactory: (localeDepValue: any) => localeDepValue || 'en-GB', deps: [[new Inject('LOCALE_ID_DEP'), new Optional()]] }] }); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.componentInstance.localeId).toBe('en-GB'); }); it('should skip current level while retrieving tokens if @SkipSelf is defined', () => { @Component({ selector: 'my-comp', template: '...', providers: [{provide: LOCALE_ID, useFactory: () => 'en-GB'}] }) class MyComp { constructor(@SkipSelf() @Inject(LOCALE_ID) public localeId: string) {} } TestBed.configureTestingModule({declarations: [MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); // takes `LOCALE_ID` from module injector, since we skip Component level with @SkipSelf expect(fixture.componentInstance.localeId).toBe(DEFAULT_LOCALE_ID); }); it('should work when injecting dependency in Directives', () => { @Directive({ selector: '[dir]', // providers: [{provide: LOCALE_ID, useValue: 'ja-JP'}] }) class MyDir { constructor(@SkipSelf() @Inject(LOCALE_ID) public localeId: string) {} } @Component({ selector: 'my-comp', template: '
', providers: [{provide: LOCALE_ID, useValue: 'en-GB'}] }) class MyComp { @ViewChild(MyDir) myDir!: MyDir; constructor(@Inject(LOCALE_ID) public localeId: string) {} } TestBed.configureTestingModule({declarations: [MyDir, MyComp, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.componentInstance.myDir.localeId).toBe('en-GB'); }); describe('@Attribute', () => { it('should inject attributes', () => { @Directive({selector: '[dir]'}) class MyDir { constructor( @Attribute('exist') public exist: string, @Attribute('nonExist') public nonExist: string) {} } @Component({template: '
'}) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.exist).toBe('existValue'); expect(directive.nonExist).toBeNull(); }); it('should inject attributes on ', () => { @Directive({selector: '[dir]'}) class MyDir { constructor( @Attribute('exist') public exist: string, @Attribute('dir') public myDirectiveAttrValue: string) {} } @Component( {template: ''}) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.exist).toBe('existValue'); expect(directive.myDirectiveAttrValue).toBe('initial'); }); it('should inject attributes on ', () => { @Directive({selector: '[dir]'}) class MyDir { constructor( @Attribute('exist') public exist: string, @Attribute('dir') public myDirectiveAttrValue: string) {} } @Component({ template: '' }) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.exist).toBe('existValue'); expect(directive.myDirectiveAttrValue).toBe('initial'); }); it('should be able to inject different kinds of attributes', () => { @Directive({selector: '[dir]'}) class MyDir { constructor( @Attribute('class') public className: string, @Attribute('style') public inlineStyles: string, @Attribute('other-attr') public otherAttr: string) {} } @Component({ template: '
' }) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.otherAttr).toBe('value'); expect(directive.className).toBe('hello there'); expect(directive.inlineStyles).toMatch(/color:\s*red/); expect(directive.inlineStyles).toMatch(/margin:\s*1px/); }); it('should not inject attributes with namespace', () => { @Directive({selector: '[dir]'}) class MyDir { constructor( @Attribute('exist') public exist: string, @Attribute('svg:exist') public namespacedExist: string, @Attribute('other') public other: string) {} } @Component({ template: '
' }) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.exist).toBe('existValue'); expect(directive.namespacedExist).toBeNull(); expect(directive.other).toBe('otherValue'); }); it('should not inject attributes representing bindings and outputs', () => { @Directive({selector: '[dir]'}) class MyDir { @Input() binding!: string; @Output() output = new EventEmitter(); constructor( @Attribute('exist') public exist: string, @Attribute('binding') public bindingAttr: string, @Attribute('output') public outputAttr: string, @Attribute('other') public other: string) {} } @Component({ template: '
' }) class MyComp { @ViewChild(MyDir) directiveInstance!: MyDir; } TestBed.configureTestingModule({declarations: [MyDir, MyComp]}); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); const directive = fixture.componentInstance.directiveInstance; expect(directive.exist).toBe('existValue'); expect(directive.bindingAttr).toBeNull(); expect(directive.outputAttr).toBeNull(); expect(directive.other).toBe('otherValue'); }); }); it('should support dependencies in Pipes used inside ICUs', () => { @Injectable() class MyService { transform(value: string): string { return `${value} (transformed)`; } } @Pipe({name: 'somePipe'}) class MyPipe { constructor(private service: MyService) {} transform(value: any): any { return this.service.transform(value); } } @Component({ template: `
{ count, select, =1 {One} other {Other value is: {{count | somePipe}}} }
` }) class MyComp { count = '2'; } TestBed.configureTestingModule({ declarations: [MyPipe, MyComp], providers: [MyService], }); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); expect(fixture.nativeElement.innerHTML).toContain('Other value is: 2 (transformed)'); }); it('should support dependencies in Pipes used inside i18n blocks', () => { @Injectable() class MyService { transform(value: string): string { return `${value} (transformed)`; } } @Pipe({name: 'somePipe'}) class MyPipe { constructor(private service: MyService) {} transform(value: any): any { return this.service.transform(value); } } @Component({ template: ` {{count | somePipe}} items ` }) class MyComp { count = '2'; @ViewChild('target', {read: ViewContainerRef}) target!: ViewContainerRef; @ViewChild('source', {read: TemplateRef}) source!: TemplateRef; create() { this.target.createEmbeddedView(this.source); } } TestBed.configureTestingModule({ declarations: [MyPipe, MyComp], providers: [MyService], }); const fixture = TestBed.createComponent(MyComp); fixture.detectChanges(); fixture.componentInstance.create(); fixture.detectChanges(); expect(fixture.nativeElement.textContent.trim()).toBe('2 (transformed) items'); }); // TODO: https://angular-team.atlassian.net/browse/FW-1779 it('should prioritize useFactory over useExisting', () => { abstract class Base {} @Directive({selector: '[dirA]'}) class DirA implements Base { } @Directive({selector: '[dirB]'}) class DirB implements Base { } const PROVIDER = {provide: Base, useExisting: DirA, useFactory: () => new DirB()}; @Component({selector: 'child', template: '', providers: [PROVIDER]}) class Child { constructor(readonly base: Base) {} } @Component({template: `
`}) class App { @ViewChild(DirA) dirA!: DirA; @ViewChild(Child) child!: Child; } const fixture = TestBed.configureTestingModule({declarations: [DirA, DirB, App, Child]}) .createComponent(App); fixture.detectChanges(); expect(fixture.componentInstance.dirA) .not.toEqual( fixture.componentInstance.child.base, 'should not get dirA from parent, but create new dirB from the useFactory provider'); }); describe('provider access on the same node', () => { const token = new InjectionToken('token'); onlyInIvy('accessing providers on the same node through a pipe was not supported in ViewEngine') .it('pipes should access providers from the component they are on', () => { @Pipe({name: 'token'}) class TokenPipe { constructor(@Inject(token) private _token: string) {} transform(value: string): string { return value + this._token; } } @Component({ selector: 'child-comp', template: '{{value}}', providers: [{provide: token, useValue: 'child'}] }) class ChildComp { @Input() value: any; } @Component({ template: ``, providers: [{provide: token, useValue: 'parent'}] }) class App { } TestBed.configureTestingModule({declarations: [App, ChildComp, TokenPipe]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(fixture.nativeElement.textContent.trim()).toBe('child'); }); it('pipes should not access viewProviders from the component they are on', () => { @Pipe({name: 'token'}) class TokenPipe { constructor(@Inject(token) private _token: string) {} transform(value: string): string { return value + this._token; } } @Component({ selector: 'child-comp', template: '{{value}}', viewProviders: [{provide: token, useValue: 'child'}] }) class ChildComp { @Input() value: any; } @Component({ template: ``, viewProviders: [{provide: token, useValue: 'parent'}] }) class App { } TestBed.configureTestingModule({declarations: [App, ChildComp, TokenPipe]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(fixture.nativeElement.textContent.trim()).toBe('parent'); }); it('directives should access providers from the component they are on', () => { @Directive({selector: '[dir]'}) class Dir { constructor(@Inject(token) public token: string) {} } @Component({ selector: 'child-comp', template: '', providers: [{provide: token, useValue: 'child'}], }) class ChildComp { } @Component({ template: '', providers: [{provide: token, useValue: 'parent'}] }) class App { @ViewChild(Dir) dir!: Dir; } TestBed.configureTestingModule({declarations: [App, ChildComp, Dir]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(fixture.componentInstance.dir.token).toBe('child'); }); it('directives should not access viewProviders from the component they are on', () => { @Directive({selector: '[dir]'}) class Dir { constructor(@Inject(token) public token: string) {} } @Component({ selector: 'child-comp', template: '', viewProviders: [{provide: token, useValue: 'child'}] }) class ChildComp { } @Component({ template: '', viewProviders: [{provide: token, useValue: 'parent'}] }) class App { @ViewChild(Dir) dir!: Dir; } TestBed.configureTestingModule({declarations: [App, ChildComp, Dir]}); const fixture = TestBed.createComponent(App); fixture.detectChanges(); expect(fixture.componentInstance.dir.token).toBe('parent'); }); }); it('should not be able to inject ViewRef', () => { // If the current browser does not support `__proto__` natively, this test will never // result in an error as the `__NG_ELEMENT_ID__` from `ChangeDetectorRef` is directly // assigned to the `ViewRef` instance, due to TypeScript, and `setPrototypeOf` polyfills // not being able to simulate the prototype chain. This means that Angular's DI system // considers `ViewRef` as injectable due to it having a `__NG_ELEMENT_ID__` directly // assigned. We skip this test in such cases. This is currently the case in IE9 and // IE10 as those don't support `__proto__`. Related TypeScript issue: // https://github.com/microsoft/TypeScript/issues/1601#issuecomment-94892833. if (!('__proto__' in {})) { return; } @Component({template: ''}) class App { constructor(_viewRef: ViewRef) {} } TestBed.configureTestingModule({declarations: [App]}); expect(() => TestBed.createComponent(App)).toThrowError(/NullInjectorError/); }); });