Previously, injector definitions contained a `factory` property that
was used to create a new instance of the associated NgModule class.
Now this factory has been moved to its own `ɵfac` static property on the
NgModule class itself. This is inline with how directives, components and
pipes are created.
There is a small size increase to bundle sizes for each NgModule class,
because the `ɵfac` takes up a bit more space:
Before:
```js
let a = (() => {
  class n {}
  return n.\u0275mod = c.Cb({type: n}),
  n.\u0275inj = c.Bb({factory: function(t) { return new (t || n) }, imports: [[e.a.forChild(s)], e.a]}),
  n
})(),
```
After:
```js
let a = (() => {
  class n {}
  return n.\u0275fac = function(t) { return new (t || n) },
  n.\u0275mod = c.Cb({type: n}),
  n.\u0275inj = c.Bb({imports: [[r.a.forChild(s)], r.a]}),
  n
})(),
```
In other words `n.\u0275fac = ` is longer than `factory: ` (by 5 characters)
and only because the tooling insists on encoding `ɵ` as `\u0275`.
This can be mitigated in a future PR by only generating the `ɵfac` property
if it is actually needed.
PR Close #41022
		
	
			
		
			
				
	
	
		
			1553 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			1553 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
/**
 | 
						|
 * @license
 | 
						|
 * Copyright Google LLC All Rights Reserved.
 | 
						|
 *
 | 
						|
 * Use of this source code is governed by an MIT-style license that can be
 | 
						|
 * found in the LICENSE file at https://angular.io/license
 | 
						|
 */
 | 
						|
 | 
						|
import {Component as _Component, ComponentFactoryResolver, ElementRef, Injectable as _Injectable, InjectFlags, InjectionToken, InjectorType, Provider, RendererFactory2, Type, ViewContainerRef, ɵɵdefineInjectable, ɵɵdefineInjector, ɵɵdefineNgModule, ɵɵinject} from '../../src/core';
 | 
						|
import {forwardRef} from '../../src/di/forward_ref';
 | 
						|
import {createInjector} from '../../src/di/r3_injector';
 | 
						|
import {injectComponentFactoryResolver, ɵɵdefineComponent, ɵɵdefineDirective, ɵɵdirectiveInject, ɵɵelement, ɵɵelementEnd, ɵɵelementStart, ɵɵgetInheritedFactory, ɵɵProvidersFeature, ɵɵtext, ɵɵtextInterpolate1} from '../../src/render3/index';
 | 
						|
import {RenderFlags} from '../../src/render3/interfaces/definition';
 | 
						|
import {NgModuleFactory} from '../../src/render3/ng_module_ref';
 | 
						|
import {getInjector} from '../../src/render3/util/discovery_utils';
 | 
						|
 | 
						|
import {getRendererFactory2} from './imported_renderer2';
 | 
						|
import {ComponentFixture} from './render_util';
 | 
						|
 | 
						|
const Component: typeof _Component = function(...args: any[]): any {
 | 
						|
  // In test we use @Component for documentation only so it's safe to mock out the implementation.
 | 
						|
  return () => undefined;
 | 
						|
} as any;
 | 
						|
const Injectable: typeof _Injectable = function(...args: any[]): any {
 | 
						|
  // In test we use @Injectable for documentation only so it's safe to mock out the implementation.
 | 
						|
  return () => undefined;
 | 
						|
} as any;
 | 
						|
 | 
						|
 | 
						|
describe('providers', () => {
 | 
						|
  describe('should support all types of Provider:', () => {
 | 
						|
    abstract class Greeter {
 | 
						|
      abstract greet: string;
 | 
						|
    }
 | 
						|
 | 
						|
    const GREETER = new InjectionToken<Greeter>('greeter');
 | 
						|
 | 
						|
    class GreeterClass implements Greeter {
 | 
						|
      greet = 'Class';
 | 
						|
    }
 | 
						|
 | 
						|
    class GreeterDeps implements Greeter {
 | 
						|
      constructor(public greet: string) {}
 | 
						|
    }
 | 
						|
 | 
						|
    class GreeterBuiltInDeps implements Greeter {
 | 
						|
      public greet: string;
 | 
						|
      constructor(private message: string, private elementRef: ElementRef) {
 | 
						|
        this.greet = this.message + ' from ' + this.elementRef.nativeElement.tagName;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    class GreeterProvider {
 | 
						|
      provide() {
 | 
						|
        return 'Provided';
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    @Injectable()
 | 
						|
    class GreeterInj implements Greeter {
 | 
						|
      public greet: string;
 | 
						|
      constructor(private provider: GreeterProvider) {
 | 
						|
        this.greet = this.provider.provide();
 | 
						|
      }
 | 
						|
 | 
						|
      static ɵprov = ɵɵdefineInjectable({
 | 
						|
        token: GreeterInj,
 | 
						|
        factory: () => new GreeterInj(ɵɵinject(GreeterProvider as any)),
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    it('TypeProvider', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [GreeterClass],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GreeterClass).greet).toEqual('Class');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('ValueProvider', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [{provide: GREETER, useValue: {greet: 'Value'}}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Value');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('ClassProvider', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [{provide: GREETER, useClass: GreeterClass}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Class');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('ExistingProvider', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [GreeterClass, {provide: GREETER, useExisting: GreeterClass}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Class');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('FactoryProvider', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [GreeterClass, {provide: GREETER, useFactory: () => new GreeterClass()}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Class');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    const MESSAGE = new InjectionToken<string>('message');
 | 
						|
 | 
						|
    it('ClassProvider with deps', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [
 | 
						|
            {provide: MESSAGE, useValue: 'Message'},
 | 
						|
            {provide: GREETER, useClass: GreeterDeps, deps: [MESSAGE]}
 | 
						|
          ],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Message');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('ClassProvider with built-in deps', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [
 | 
						|
            {provide: MESSAGE, useValue: 'Message'},
 | 
						|
            {provide: GREETER, useClass: GreeterBuiltInDeps, deps: [MESSAGE, ElementRef]}
 | 
						|
          ],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Message from PARENT');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('FactoryProvider with deps', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [
 | 
						|
            {provide: MESSAGE, useValue: 'Message'},
 | 
						|
            {provide: GREETER, useFactory: (msg: string) => new GreeterDeps(msg), deps: [MESSAGE]}
 | 
						|
          ],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Message');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('FactoryProvider with built-in deps', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [
 | 
						|
            {provide: MESSAGE, useValue: 'Message'}, {
 | 
						|
              provide: GREETER,
 | 
						|
              useFactory: (msg: string, elementRef: ElementRef) =>
 | 
						|
                  new GreeterBuiltInDeps(msg, elementRef),
 | 
						|
              deps: [MESSAGE, ElementRef]
 | 
						|
            }
 | 
						|
          ],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Message from PARENT');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('ClassProvider with injectable', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [GreeterProvider, {provide: GREETER, useClass: GreeterInj}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Provided');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    describe('forwardRef', () => {
 | 
						|
      it('forwardRef resolves later', (done) => {
 | 
						|
        setTimeout(() => {
 | 
						|
          expectProvidersScenario({
 | 
						|
            parent: {
 | 
						|
              providers: [forwardRef(() => ForLater)],
 | 
						|
              componentAssertion: () => {
 | 
						|
                expect(ɵɵdirectiveInject(ForLater) instanceof ForLater).toBeTruthy();
 | 
						|
              }
 | 
						|
            }
 | 
						|
          });
 | 
						|
          done();
 | 
						|
        }, 0);
 | 
						|
      });
 | 
						|
 | 
						|
      class ForLater {}
 | 
						|
 | 
						|
      // The following test that forwardRefs are called, so we don't search for an anon fn
 | 
						|
      it('ValueProvider wrapped in forwardRef', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{
 | 
						|
              provide: GREETER,
 | 
						|
              useValue: forwardRef(() => {
 | 
						|
                return {greet: 'Value'};
 | 
						|
              })
 | 
						|
            }],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Value');
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('ClassProvider wrapped in forwardRef', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: GREETER, useClass: forwardRef(() => GreeterClass)}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Class');
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('ExistingProvider wrapped in forwardRef', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers:
 | 
						|
                [GreeterClass, {provide: GREETER, useExisting: forwardRef(() => GreeterClass)}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(GREETER).greet).toEqual('Class');
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('@Inject annotation wrapped in forwardRef', () => {
 | 
						|
        // @Inject(forwardRef(() => GREETER))
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: GREETER, useValue: {greet: 'Value'}}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(forwardRef(() => GREETER)).greet).toEqual('Value');
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  /*
 | 
						|
   * All tests below assume this structure:
 | 
						|
   * ```
 | 
						|
   * <parent>
 | 
						|
   *   <#VIEW#>
 | 
						|
   *     <view-child>
 | 
						|
   *     </view-child>
 | 
						|
   *   </#VIEW#>
 | 
						|
   *   <content-child>
 | 
						|
   *   </content-child>
 | 
						|
   * </parent>
 | 
						|
   * ```
 | 
						|
   */
 | 
						|
 | 
						|
  describe('override rules:', () => {
 | 
						|
    it('directiveProviders should override providers', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [{provide: String, useValue: 'Message 1'}],
 | 
						|
          directiveProviders: [{provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('viewProviders should override providers', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [{provide: String, useValue: 'Message 1'}],
 | 
						|
          viewProviders: [{provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('viewProviders should override directiveProviders', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          directiveProviders: [{provide: String, useValue: 'Message 1'}],
 | 
						|
          viewProviders: [{provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('last declared directive should override other directives', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          directive2Providers: [{provide: String, useValue: 'Message 1'}],
 | 
						|
          directiveProviders: [{provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('last provider should override previous one in component providers', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers:
 | 
						|
              [{provide: String, useValue: 'Message 1'}, {provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('last provider should override previous one in component view providers', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          viewProviders:
 | 
						|
              [{provide: String, useValue: 'Message 1'}, {provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('last provider should override previous one in directive providers', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          directiveProviders:
 | 
						|
              [{provide: String, useValue: 'Message 1'}, {provide: String, useValue: 'Message 2'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Message 2');
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('single', () => {
 | 
						|
    class MyModule {
 | 
						|
      static ɵinj = ɵɵdefineInjector({providers: [{provide: String, useValue: 'From module'}]});
 | 
						|
    }
 | 
						|
 | 
						|
    describe('without directives', () => {
 | 
						|
      it('should work without providers nor viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only providers in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From module');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with both providers and viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers'}],
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From providers');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    describe('with directives (order in ɵcmp.directives matters)', () => {
 | 
						|
      it('should work without providers nor viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive'}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'Never'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only providers in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers'}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive'}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'Never'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders'}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive'}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'Never'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with both providers and viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers'}],
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders'}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive'}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'Never'}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From viewProviders');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual('From directive');
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('multi', () => {
 | 
						|
    class MyModule {
 | 
						|
      static ɵinj =
 | 
						|
          ɵɵdefineInjector({providers: [{provide: String, useValue: 'From module', multi: true}]});
 | 
						|
    }
 | 
						|
 | 
						|
    describe('without directives', () => {
 | 
						|
      it('should work without providers nor viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only providers in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From viewProviders']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From viewProviders']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From viewProviders']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From module']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with both providers and viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers', multi: true}],
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers', 'From viewProviders']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers', 'From viewProviders']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers', 'From viewProviders']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From providers']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    describe('with directives (order in ɵcmp.directives matters)', () => {
 | 
						|
      it('should work without providers nor viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive 1', multi: true}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'From directive 2', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only providers in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers', multi: true}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive 1', multi: true}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'From directive 2', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with only viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders', multi: true}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive 1', multi: true}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'From directive 2', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual(['From directive 2', 'From directive 1']);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      it('should work with both providers and viewProviders in component', () => {
 | 
						|
        expectProvidersScenario({
 | 
						|
          parent: {
 | 
						|
            providers: [{provide: String, useValue: 'From providers', multi: true}],
 | 
						|
            viewProviders: [{provide: String, useValue: 'From viewProviders', multi: true}],
 | 
						|
            directiveProviders: [{provide: String, useValue: 'From directive 1', multi: true}],
 | 
						|
            directive2Providers: [{provide: String, useValue: 'From directive 2', multi: true}],
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          viewChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From viewProviders', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          contentChild: {
 | 
						|
            componentAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            },
 | 
						|
            directiveAssertion: () => {
 | 
						|
              expect(ɵɵdirectiveInject(String)).toEqual([
 | 
						|
                'From providers', 'From directive 2', 'From directive 1'
 | 
						|
              ]);
 | 
						|
            }
 | 
						|
          },
 | 
						|
          ngModule: MyModule
 | 
						|
        });
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('tree-shakable injectables', () => {
 | 
						|
    it('should work with root', () => {
 | 
						|
      @Injectable({providedIn: 'root'})
 | 
						|
      class FooForRoot {
 | 
						|
        static ɵprov = ɵɵdefineInjectable({
 | 
						|
          token: FooForRoot,
 | 
						|
          factory: () => new FooForRoot(),
 | 
						|
          providedIn: 'root',
 | 
						|
        });
 | 
						|
      }
 | 
						|
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(FooForRoot) instanceof FooForRoot).toBeTruthy();
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should work with a module', () => {
 | 
						|
      class MyModule {
 | 
						|
        static ɵinj = ɵɵdefineInjector({providers: [{provide: String, useValue: 'From module'}]});
 | 
						|
      }
 | 
						|
 | 
						|
      @Injectable({providedIn: MyModule})
 | 
						|
      class FooForModule {
 | 
						|
        static ɵprov = ɵɵdefineInjectable({
 | 
						|
          token: FooForModule,
 | 
						|
          factory: () => new FooForModule(),
 | 
						|
          providedIn: MyModule,
 | 
						|
        });
 | 
						|
      }
 | 
						|
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(FooForModule) instanceof FooForModule).toBeTruthy();
 | 
						|
          }
 | 
						|
        },
 | 
						|
        ngModule: MyModule
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('- dynamic components dependency resolution', () => {
 | 
						|
    let hostComponent: HostComponent|null = null;
 | 
						|
 | 
						|
    @Component({
 | 
						|
      template: `{{s}}`,
 | 
						|
    })
 | 
						|
    class EmbeddedComponent {
 | 
						|
      constructor(private s: String) {}
 | 
						|
 | 
						|
      static ɵfac = () => new EmbeddedComponent(ɵɵdirectiveInject(String));
 | 
						|
      static ɵcmp = ɵɵdefineComponent({
 | 
						|
        type: EmbeddedComponent,
 | 
						|
        selectors: [['embedded-cmp']],
 | 
						|
        decls: 1,
 | 
						|
        vars: 1,
 | 
						|
        template:
 | 
						|
            (rf: RenderFlags, cmp: EmbeddedComponent) => {
 | 
						|
              if (rf & RenderFlags.Create) {
 | 
						|
                ɵɵtext(0);
 | 
						|
              }
 | 
						|
              if (rf & RenderFlags.Update) {
 | 
						|
                ɵɵtextInterpolate1('', cmp.s, '');
 | 
						|
              }
 | 
						|
            }
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    @Component({template: `foo`, providers: [{provide: String, useValue: 'From host component'}]})
 | 
						|
    class HostComponent {
 | 
						|
      constructor(public vcref: ViewContainerRef, public cfr: ComponentFactoryResolver) {}
 | 
						|
 | 
						|
      static ɵfac = () => hostComponent = new HostComponent(
 | 
						|
          ɵɵdirectiveInject(ViewContainerRef as any), injectComponentFactoryResolver())
 | 
						|
 | 
						|
          static ɵcmp = ɵɵdefineComponent({
 | 
						|
            type: HostComponent,
 | 
						|
            selectors: [['host-cmp']],
 | 
						|
            decls: 1,
 | 
						|
            vars: 0,
 | 
						|
            template:
 | 
						|
                (rf: RenderFlags, cmp: HostComponent) => {
 | 
						|
                  if (rf & RenderFlags.Create) {
 | 
						|
                    ɵɵtext(0, 'foo');
 | 
						|
                  }
 | 
						|
                },
 | 
						|
            features:
 | 
						|
                [
 | 
						|
                  ɵɵProvidersFeature([{provide: String, useValue: 'From host component'}]),
 | 
						|
                ],
 | 
						|
          });
 | 
						|
    }
 | 
						|
 | 
						|
    @Component({
 | 
						|
      template: `<host-cmp></host-cmp>`,
 | 
						|
      providers: [{provide: String, useValue: 'From app component'}]
 | 
						|
    })
 | 
						|
    class AppComponent {
 | 
						|
      constructor() {}
 | 
						|
 | 
						|
      static ɵfac = () => new AppComponent();
 | 
						|
      static ɵcmp = ɵɵdefineComponent({
 | 
						|
        type: AppComponent,
 | 
						|
        selectors: [['app-cmp']],
 | 
						|
        decls: 1,
 | 
						|
        vars: 0,
 | 
						|
        template:
 | 
						|
            (rf: RenderFlags, cmp: AppComponent) => {
 | 
						|
              if (rf & RenderFlags.Create) {
 | 
						|
                ɵɵelement(0, 'host-cmp');
 | 
						|
              }
 | 
						|
            },
 | 
						|
        features:
 | 
						|
            [
 | 
						|
              ɵɵProvidersFeature([{provide: String, useValue: 'From app component'}]),
 | 
						|
            ],
 | 
						|
        directives: [HostComponent]
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    it('should not cross the root view boundary, and use the root view injector', () => {
 | 
						|
      const fixture = new ComponentFixture(AppComponent);
 | 
						|
      expect(fixture.html).toEqual('<host-cmp>foo</host-cmp>');
 | 
						|
 | 
						|
      hostComponent!.vcref.createComponent(
 | 
						|
          hostComponent!.cfr.resolveComponentFactory(EmbeddedComponent), undefined, {
 | 
						|
            get: (token: any, notFoundValue?: any) => {
 | 
						|
              return token === String ? 'From custom root view injector' : notFoundValue;
 | 
						|
            }
 | 
						|
          });
 | 
						|
      fixture.update();
 | 
						|
      expect(fixture.html)
 | 
						|
          .toEqual(
 | 
						|
              '<host-cmp>foo</host-cmp><embedded-cmp>From custom root view injector</embedded-cmp>');
 | 
						|
    });
 | 
						|
 | 
						|
    it('should not cross the root view boundary, and use the module injector if no root view injector',
 | 
						|
       () => {
 | 
						|
         const fixture = new ComponentFixture(AppComponent);
 | 
						|
         expect(fixture.html).toEqual('<host-cmp>foo</host-cmp>');
 | 
						|
 | 
						|
         class MyAppModule {
 | 
						|
           static ɵinj = ɵɵdefineInjector({
 | 
						|
             imports: [],
 | 
						|
             providers:
 | 
						|
                 [
 | 
						|
                   {provide: RendererFactory2, useValue: getRendererFactory2(document)},
 | 
						|
                   {provide: String, useValue: 'From module injector'}
 | 
						|
                 ]
 | 
						|
           });
 | 
						|
           static ɵmod = ɵɵdefineNgModule({type: MyAppModule});
 | 
						|
         }
 | 
						|
         const myAppModuleFactory = new NgModuleFactory(MyAppModule);
 | 
						|
         const ngModuleRef = myAppModuleFactory.create(null);
 | 
						|
 | 
						|
         hostComponent!.vcref.createComponent(
 | 
						|
             hostComponent!.cfr.resolveComponentFactory(EmbeddedComponent), undefined,
 | 
						|
             {get: (token: any, notFoundValue?: any) => notFoundValue}, undefined, ngModuleRef);
 | 
						|
         fixture.update();
 | 
						|
         expect(fixture.html)
 | 
						|
             .toMatch(
 | 
						|
                 /<host-cmp>foo<\/host-cmp><embedded-cmp _nghost-[a-z]+-c(\d+)="">From module injector<\/embedded-cmp>/);
 | 
						|
       });
 | 
						|
 | 
						|
    it('should cross the root view boundary to the parent of the host, thanks to the default root view injector',
 | 
						|
       () => {
 | 
						|
         const fixture = new ComponentFixture(AppComponent);
 | 
						|
         expect(fixture.html).toEqual('<host-cmp>foo</host-cmp>');
 | 
						|
 | 
						|
         hostComponent!.vcref.createComponent(
 | 
						|
             hostComponent!.cfr.resolveComponentFactory(EmbeddedComponent));
 | 
						|
         fixture.update();
 | 
						|
         expect(fixture.html)
 | 
						|
             .toEqual('<host-cmp>foo</host-cmp><embedded-cmp>From app component</embedded-cmp>');
 | 
						|
       });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('deps boundary:', () => {
 | 
						|
    it('the deps of a token declared in providers should not be resolved with tokens from viewProviders',
 | 
						|
       () => {
 | 
						|
         @Injectable()
 | 
						|
         class MyService {
 | 
						|
           constructor(public value: String) {}
 | 
						|
 | 
						|
           static ɵprov = ɵɵdefineInjectable({
 | 
						|
             token: MyService,
 | 
						|
             factory: () => new MyService(ɵɵinject(String)),
 | 
						|
           });
 | 
						|
         }
 | 
						|
 | 
						|
         expectProvidersScenario({
 | 
						|
           parent: {
 | 
						|
             providers: [MyService, {provide: String, useValue: 'providers'}],
 | 
						|
             viewProviders: [{provide: String, useValue: 'viewProviders'}],
 | 
						|
             componentAssertion: () => {
 | 
						|
               expect(ɵɵdirectiveInject(String)).toEqual('viewProviders');
 | 
						|
               expect(ɵɵdirectiveInject(MyService).value).toEqual('providers');
 | 
						|
             }
 | 
						|
           }
 | 
						|
         });
 | 
						|
       });
 | 
						|
 | 
						|
    it('should make sure that parent service does not see overrides in child directives', () => {
 | 
						|
      class Greeter {
 | 
						|
        static ɵprov = ɵɵdefineInjectable({
 | 
						|
          token: Greeter,
 | 
						|
          factory: () => new Greeter(ɵɵinject(String)),
 | 
						|
        });
 | 
						|
        constructor(public greeting: String) {}
 | 
						|
      }
 | 
						|
 | 
						|
      expectProvidersScenario({
 | 
						|
        parent: {
 | 
						|
          providers: [Greeter, {provide: String, useValue: 'parent'}],
 | 
						|
        },
 | 
						|
        viewChild: {
 | 
						|
          providers: [{provide: String, useValue: 'view'}],
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(Greeter).greeting).toEqual('parent');
 | 
						|
          },
 | 
						|
        },
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('injection flags', () => {
 | 
						|
    class MyModule {
 | 
						|
      static ɵinj = ɵɵdefineInjector({providers: [{provide: String, useValue: 'Module'}]});
 | 
						|
    }
 | 
						|
    it('should not fall through to ModuleInjector if flags limit the scope', () => {
 | 
						|
      expectProvidersScenario({
 | 
						|
        ngModule: MyModule,
 | 
						|
        parent: {
 | 
						|
          componentAssertion: () => {
 | 
						|
            expect(ɵɵdirectiveInject(String)).toEqual('Module');
 | 
						|
            expect(ɵɵdirectiveInject(String, InjectFlags.Optional | InjectFlags.Self)).toBeNull();
 | 
						|
            expect(ɵɵdirectiveInject(String, InjectFlags.Optional | InjectFlags.Host)).toBeNull();
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  describe('from a node without injector', () => {
 | 
						|
    abstract class Some {
 | 
						|
      abstract location: String;
 | 
						|
    }
 | 
						|
 | 
						|
    class SomeInj implements Some {
 | 
						|
      constructor(public location: String) {}
 | 
						|
 | 
						|
      static ɵprov = ɵɵdefineInjectable({
 | 
						|
        token: SomeInj,
 | 
						|
        factory: () => new SomeInj(ɵɵinject(String)),
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    @Component({
 | 
						|
      template: `<p></p>`,
 | 
						|
      providers: [{provide: String, useValue: 'From my component'}],
 | 
						|
      viewProviders: [{provide: Number, useValue: 123}]
 | 
						|
    })
 | 
						|
    class MyComponent {
 | 
						|
      constructor() {}
 | 
						|
 | 
						|
      static ɵfac = () => new MyComponent();
 | 
						|
      static ɵcmp = ɵɵdefineComponent({
 | 
						|
        type: MyComponent,
 | 
						|
        selectors: [['my-cmp']],
 | 
						|
        decls: 1,
 | 
						|
        vars: 0,
 | 
						|
        template:
 | 
						|
            (rf: RenderFlags, cmp: MyComponent) => {
 | 
						|
              if (rf & RenderFlags.Create) {
 | 
						|
                ɵɵelement(0, 'p');
 | 
						|
              }
 | 
						|
            },
 | 
						|
        features:
 | 
						|
            [
 | 
						|
              ɵɵProvidersFeature(
 | 
						|
                  [{provide: String, useValue: 'From my component'}],
 | 
						|
                  [{provide: Number, useValue: 123}]),
 | 
						|
            ],
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    @Component({
 | 
						|
      template: `<my-cmp></my-cmp>`,
 | 
						|
      providers:
 | 
						|
          [{provide: String, useValue: 'From app component'}, {provide: Some, useClass: SomeInj}]
 | 
						|
    })
 | 
						|
    class AppComponent {
 | 
						|
      constructor() {}
 | 
						|
 | 
						|
      static ɵfac = () => new AppComponent();
 | 
						|
      static ɵcmp = ɵɵdefineComponent({
 | 
						|
        type: AppComponent,
 | 
						|
        selectors: [['app-cmp']],
 | 
						|
        decls: 1,
 | 
						|
        vars: 0,
 | 
						|
        template:
 | 
						|
            (rf: RenderFlags, cmp: AppComponent) => {
 | 
						|
              if (rf & RenderFlags.Create) {
 | 
						|
                ɵɵelement(0, 'my-cmp');
 | 
						|
              }
 | 
						|
            },
 | 
						|
        features:
 | 
						|
            [
 | 
						|
              ɵɵProvidersFeature([
 | 
						|
                {provide: String, useValue: 'From app component'},
 | 
						|
                {provide: Some, useClass: SomeInj}
 | 
						|
              ]),
 | 
						|
            ],
 | 
						|
        directives: [MyComponent]
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    it('should work from within the template', () => {
 | 
						|
      const fixture = new ComponentFixture(AppComponent);
 | 
						|
      expect(fixture.html).toEqual('<my-cmp><p></p></my-cmp>');
 | 
						|
 | 
						|
      const p = fixture.hostElement.querySelector('p');
 | 
						|
      const injector = getInjector(p as any);
 | 
						|
      expect(injector.get(Number)).toEqual(123);
 | 
						|
      expect(injector.get(String)).toEqual('From my component');
 | 
						|
      expect(injector.get(Some).location).toEqual('From app component');
 | 
						|
    });
 | 
						|
 | 
						|
    it('should work from the host of the component', () => {
 | 
						|
      const fixture = new ComponentFixture(AppComponent);
 | 
						|
      expect(fixture.html).toEqual('<my-cmp><p></p></my-cmp>');
 | 
						|
 | 
						|
      const myCmp = fixture.hostElement.querySelector('my-cmp');
 | 
						|
      const injector = getInjector(myCmp as any);
 | 
						|
      expect(injector.get(Number)).toEqual(123);
 | 
						|
      expect(injector.get(String)).toEqual('From my component');
 | 
						|
      expect(injector.get(Some).location).toEqual('From app component');
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  // Note: these tests check the behavior of `getInheritedFactory` specifically.
 | 
						|
  // Since `getInheritedFactory` is only generated in AOT, the tests can't be
 | 
						|
  // ported directly to TestBed while running in JIT mode.
 | 
						|
  describe('getInheritedFactory on class with custom decorator', () => {
 | 
						|
    function addFoo() {
 | 
						|
      return (constructor: Type<any>): any => {
 | 
						|
        const decoratedClass = class Extender extends constructor { foo = 'bar'; };
 | 
						|
        return decoratedClass;
 | 
						|
      };
 | 
						|
    }
 | 
						|
 | 
						|
    it('should find the correct factories if a parent class has a custom decorator', () => {
 | 
						|
      class GrandParent {
 | 
						|
        static ɵfac = function GrandParent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      @addFoo()
 | 
						|
      class Parent extends GrandParent {
 | 
						|
        static ɵfac = function Parent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      class Child extends Parent {
 | 
						|
        static ɵfac = function Child_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      expect(ɵɵgetInheritedFactory(Child).name).toBe('Parent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(Parent).name).toBe('GrandParent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(GrandParent).name).toBeFalsy();
 | 
						|
    });
 | 
						|
 | 
						|
    it('should find the correct factories if a child class has a custom decorator', () => {
 | 
						|
      class GrandParent {
 | 
						|
        static ɵfac = function GrandParent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      class Parent extends GrandParent {
 | 
						|
        static ɵfac = function Parent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      @addFoo()
 | 
						|
      class Child extends Parent {
 | 
						|
        static ɵfac = function Child_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      expect(ɵɵgetInheritedFactory(Child).name).toBe('Parent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(Parent).name).toBe('GrandParent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(GrandParent).name).toBeFalsy();
 | 
						|
    });
 | 
						|
 | 
						|
    it('should find the correct factories if a grandparent class has a custom decorator', () => {
 | 
						|
      @addFoo()
 | 
						|
      class GrandParent {
 | 
						|
        static ɵfac = function GrandParent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      class Parent extends GrandParent {
 | 
						|
        static ɵfac = function Parent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      class Child extends Parent {
 | 
						|
        static ɵfac = function Child_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      expect(ɵɵgetInheritedFactory(Child).name).toBe('Parent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(Parent).name).toBe('GrandParent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(GrandParent).name).toBeFalsy();
 | 
						|
    });
 | 
						|
 | 
						|
    it('should find the correct factories if all classes have a custom decorator', () => {
 | 
						|
      @addFoo()
 | 
						|
      class GrandParent {
 | 
						|
        static ɵfac = function GrandParent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      @addFoo()
 | 
						|
      class Parent extends GrandParent {
 | 
						|
        static ɵfac = function Parent_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      @addFoo()
 | 
						|
      class Child extends Parent {
 | 
						|
        static ɵfac = function Child_Factory() {};
 | 
						|
      }
 | 
						|
 | 
						|
      expect(ɵɵgetInheritedFactory(Child).name).toBe('Parent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(Parent).name).toBe('GrandParent_Factory');
 | 
						|
      expect(ɵɵgetInheritedFactory(GrandParent).name).toBeFalsy();
 | 
						|
    });
 | 
						|
 | 
						|
    it('should find the correct factories if parent and grandparent classes have a custom decorator',
 | 
						|
       () => {
 | 
						|
         @addFoo()
 | 
						|
         class GrandParent {
 | 
						|
           static ɵfac = function GrandParent_Factory() {};
 | 
						|
         }
 | 
						|
 | 
						|
         @addFoo()
 | 
						|
         class Parent extends GrandParent {
 | 
						|
           static ɵfac = function Parent_Factory() {};
 | 
						|
         }
 | 
						|
 | 
						|
         class Child extends Parent {
 | 
						|
           static ɵfac = function Child_Factory() {};
 | 
						|
         }
 | 
						|
 | 
						|
         expect(ɵɵgetInheritedFactory(Child).name).toBe('Parent_Factory');
 | 
						|
         expect(ɵɵgetInheritedFactory(Parent).name).toBe('GrandParent_Factory');
 | 
						|
         expect(ɵɵgetInheritedFactory(GrandParent).name).toBeFalsy();
 | 
						|
       });
 | 
						|
  });
 | 
						|
});
 | 
						|
 | 
						|
interface ComponentTest {
 | 
						|
  providers?: Provider[];
 | 
						|
  viewProviders?: Provider[];
 | 
						|
  directiveProviders?: Provider[];
 | 
						|
  directive2Providers?: Provider[];
 | 
						|
  directiveAssertion?: () => void;
 | 
						|
  componentAssertion?: () => void;
 | 
						|
}
 | 
						|
 | 
						|
function expectProvidersScenario(defs: {
 | 
						|
  app?: ComponentTest,
 | 
						|
  parent?: ComponentTest,
 | 
						|
  viewChild?: ComponentTest,
 | 
						|
  contentChild?: ComponentTest,
 | 
						|
  ngModule?: InjectorType<any>,
 | 
						|
}): void {
 | 
						|
  function testComponentInjection<T>(def: ComponentTest|undefined, instance: T): T {
 | 
						|
    if (def) {
 | 
						|
      def.componentAssertion && def.componentAssertion();
 | 
						|
    }
 | 
						|
    return instance;
 | 
						|
  }
 | 
						|
 | 
						|
  function testDirectiveInjection<T>(def: ComponentTest|undefined, instance: T): T {
 | 
						|
    if (def) {
 | 
						|
      def.directiveAssertion && def.directiveAssertion();
 | 
						|
    }
 | 
						|
    return instance;
 | 
						|
  }
 | 
						|
 | 
						|
  class ViewChildComponent {
 | 
						|
    static ɵfac = () => testComponentInjection(defs.viewChild, new ViewChildComponent());
 | 
						|
    static ɵcmp = ɵɵdefineComponent({
 | 
						|
      type: ViewChildComponent,
 | 
						|
      selectors: [['view-child']],
 | 
						|
      decls: 1,
 | 
						|
      vars: 0,
 | 
						|
      template:
 | 
						|
          function(fs: RenderFlags, ctx: ViewChildComponent) {
 | 
						|
            if (fs & RenderFlags.Create) {
 | 
						|
              ɵɵtext(0, 'view-child');
 | 
						|
            }
 | 
						|
          },
 | 
						|
      features: defs.viewChild &&
 | 
						|
          [ɵɵProvidersFeature(defs.viewChild.providers || [], defs.viewChild.viewProviders || [])]
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  class ViewChildDirective {
 | 
						|
    static ɵfac = () => testDirectiveInjection(defs.viewChild, new ViewChildDirective());
 | 
						|
    static ɵdir = ɵɵdefineDirective({
 | 
						|
      type: ViewChildDirective,
 | 
						|
      selectors: [['view-child']],
 | 
						|
      features: defs.viewChild && [ɵɵProvidersFeature(defs.viewChild.directiveProviders || [])],
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  class ContentChildComponent {
 | 
						|
    static ɵfac =
 | 
						|
        () => {
 | 
						|
          return testComponentInjection(defs.contentChild, new ContentChildComponent());
 | 
						|
        }
 | 
						|
 | 
						|
    static ɵcmp = ɵɵdefineComponent({
 | 
						|
      type: ContentChildComponent,
 | 
						|
      selectors: [['content-child']],
 | 
						|
      decls: 1,
 | 
						|
      vars: 0,
 | 
						|
      template:
 | 
						|
          function(fs: RenderFlags, ctx: ParentComponent) {
 | 
						|
            if (fs & RenderFlags.Create) {
 | 
						|
              ɵɵtext(0, 'content-child');
 | 
						|
            }
 | 
						|
          },
 | 
						|
      features: defs.contentChild &&
 | 
						|
          [ɵɵProvidersFeature(
 | 
						|
              defs.contentChild.providers || [], defs.contentChild.viewProviders || [])],
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  class ContentChildDirective {
 | 
						|
    static ɵfac =
 | 
						|
        () => {
 | 
						|
          return testDirectiveInjection(defs.contentChild, new ContentChildDirective());
 | 
						|
        }
 | 
						|
 | 
						|
    static ɵdir = ɵɵdefineDirective({
 | 
						|
      type: ContentChildDirective,
 | 
						|
      selectors: [['content-child']],
 | 
						|
      features: defs.contentChild &&
 | 
						|
          [ɵɵProvidersFeature(defs.contentChild.directiveProviders || [])],
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  class ParentComponent {
 | 
						|
    static ɵfac = () => testComponentInjection(defs.parent, new ParentComponent());
 | 
						|
    static ɵcmp = ɵɵdefineComponent({
 | 
						|
      type: ParentComponent,
 | 
						|
      selectors: [['parent']],
 | 
						|
      decls: 1,
 | 
						|
      vars: 0,
 | 
						|
      template:
 | 
						|
          function(fs: RenderFlags, ctx: ParentComponent) {
 | 
						|
            if (fs & RenderFlags.Create) {
 | 
						|
              ɵɵelement(0, 'view-child');
 | 
						|
            }
 | 
						|
          },
 | 
						|
      features: defs.parent &&
 | 
						|
          [ɵɵProvidersFeature(defs.parent.providers || [], defs.parent.viewProviders || [])],
 | 
						|
      directives: [ViewChildComponent, ViewChildDirective]
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  class ParentDirective {
 | 
						|
    static ɵfac = () => testDirectiveInjection(defs.parent, new ParentDirective());
 | 
						|
    static ɵdir = ɵɵdefineDirective({
 | 
						|
      type: ParentDirective,
 | 
						|
      selectors: [['parent']],
 | 
						|
      features: defs.parent && [ɵɵProvidersFeature(defs.parent.directiveProviders || [])],
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  class ParentDirective2 {
 | 
						|
    static ɵfac = () => testDirectiveInjection(defs.parent, new ParentDirective2());
 | 
						|
    static ɵdir = ɵɵdefineDirective({
 | 
						|
      type: ParentDirective2,
 | 
						|
      selectors: [['parent']],
 | 
						|
      features: defs.parent && [ɵɵProvidersFeature(defs.parent.directive2Providers || [])],
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  class App {
 | 
						|
    static ɵfac = () => testComponentInjection(defs.app, new App());
 | 
						|
    static ɵcmp = ɵɵdefineComponent({
 | 
						|
      type: App,
 | 
						|
      selectors: [['app']],
 | 
						|
      decls: 2,
 | 
						|
      vars: 0,
 | 
						|
      template:
 | 
						|
          function(fs: RenderFlags, ctx: App) {
 | 
						|
            if (fs & RenderFlags.Create) {
 | 
						|
              ɵɵelementStart(0, 'parent');
 | 
						|
              ɵɵelement(1, 'content-child');
 | 
						|
              ɵɵelementEnd();
 | 
						|
            }
 | 
						|
          },
 | 
						|
      features: defs.app &&
 | 
						|
          [ɵɵProvidersFeature(defs.app.providers || [], defs.app.viewProviders || [])],
 | 
						|
      directives:
 | 
						|
          [
 | 
						|
            ParentComponent, ParentDirective2, ParentDirective, ContentChildComponent,
 | 
						|
            ContentChildDirective
 | 
						|
          ]
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  const fixture = new ComponentFixture(
 | 
						|
      App, {injector: defs.ngModule ? createInjector(defs.ngModule) : undefined});
 | 
						|
  expect(fixture.html).toEqual('<parent><view-child>view-child</view-child></parent>');
 | 
						|
}
 |