2016-06-23 12:47:54 -04:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
import {AnimationEntryMetadata, ChangeDetectorRef, ComponentFactory, ComponentRef, ComponentResolver, DebugElement, ElementRef, Injectable, Injector, NgZone, NgZoneError, OpaqueToken, ViewMetadata, getDebugNode} from '@angular/core';
|
2016-06-24 15:41:49 -04:00
|
|
|
import {ComponentFixture, tick} from '@angular/core/testing';
|
2016-04-28 20:50:03 -04:00
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
import {DirectiveResolver, ViewResolver} from '../index';
|
|
|
|
import {ObservableWrapper, PromiseCompleter, PromiseWrapper} from '../src/facade/async';
|
2016-04-28 20:50:03 -04:00
|
|
|
import {ListWrapper, MapWrapper} from '../src/facade/collection';
|
2016-06-08 19:38:52 -04:00
|
|
|
import {BaseException} from '../src/facade/exceptions';
|
|
|
|
import {IS_DART, Type, isBlank, isPresent, scheduleMicroTask} from '../src/facade/lang';
|
2016-04-28 20:50:03 -04:00
|
|
|
|
2016-06-24 15:41:49 -04:00
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
* Import ComponentFixture from @angular/core/testing instead.
|
|
|
|
*/
|
|
|
|
export {ComponentFixture} from '@angular/core/testing';
|
2016-04-28 20:50:03 -04:00
|
|
|
/**
|
|
|
|
* An abstract class for inserting the root test component element in a platform independent way.
|
|
|
|
*/
|
|
|
|
export class TestComponentRenderer {
|
|
|
|
insertRootElement(rootElementId: string) {}
|
|
|
|
}
|
2015-11-03 16:51:33 -05:00
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
export var ComponentFixtureAutoDetect = new OpaqueToken('ComponentFixtureAutoDetect');
|
|
|
|
export var ComponentFixtureNoNgZone = new OpaqueToken('ComponentFixtureNoNgZone');
|
2016-04-26 19:38:54 -04:00
|
|
|
|
2015-05-28 18:02:20 -04:00
|
|
|
var _nextRootElementId = 0;
|
2015-05-15 19:42:52 -04:00
|
|
|
|
|
|
|
/**
|
2015-10-31 12:50:19 -04:00
|
|
|
* Builds a ComponentFixture for use in component level tests.
|
2015-05-15 19:42:52 -04:00
|
|
|
*/
|
|
|
|
@Injectable()
|
|
|
|
export class TestComponentBuilder {
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2015-09-29 14:11:06 -04:00
|
|
|
_bindingsOverrides = new Map<Type, any[]>();
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2015-09-29 14:11:06 -04:00
|
|
|
_directiveOverrides = new Map<Type, Map<Type, Type>>();
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2015-09-29 14:11:06 -04:00
|
|
|
_templateOverrides = new Map<Type, string>();
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2016-05-25 15:46:22 -04:00
|
|
|
_animationOverrides = new Map<Type, AnimationEntryMetadata[]>();
|
|
|
|
/** @internal */
|
2015-09-29 14:11:06 -04:00
|
|
|
_viewBindingsOverrides = new Map<Type, any[]>();
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2015-09-29 14:11:06 -04:00
|
|
|
_viewOverrides = new Map<Type, ViewMetadata>();
|
|
|
|
|
|
|
|
|
|
|
|
constructor(private _injector: Injector) {}
|
2015-05-15 19:42:52 -04:00
|
|
|
|
2015-10-09 20:21:25 -04:00
|
|
|
/** @internal */
|
2015-05-15 19:42:52 -04:00
|
|
|
_clone(): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = new TestComponentBuilder(this._injector);
|
2015-05-15 19:42:52 -04:00
|
|
|
clone._viewOverrides = MapWrapper.clone(this._viewOverrides);
|
|
|
|
clone._directiveOverrides = MapWrapper.clone(this._directiveOverrides);
|
|
|
|
clone._templateOverrides = MapWrapper.clone(this._templateOverrides);
|
2016-01-06 17:13:44 -05:00
|
|
|
clone._bindingsOverrides = MapWrapper.clone(this._bindingsOverrides);
|
|
|
|
clone._viewBindingsOverrides = MapWrapper.clone(this._viewBindingsOverrides);
|
2015-05-15 19:42:52 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-08-14 13:03:45 -04:00
|
|
|
* Overrides only the html of a {@link ComponentMetadata}.
|
|
|
|
* All the other properties of the component's {@link ViewMetadata} are preserved.
|
2015-05-15 19:42:52 -04:00
|
|
|
*/
|
|
|
|
overrideTemplate(componentType: Type, template: string): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = this._clone();
|
2015-06-17 19:21:40 -04:00
|
|
|
clone._templateOverrides.set(componentType, template);
|
2015-05-15 19:42:52 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
2016-06-08 19:38:52 -04:00
|
|
|
overrideAnimations(componentType: Type, animations: AnimationEntryMetadata[]):
|
|
|
|
TestComponentBuilder {
|
2016-05-25 15:46:22 -04:00
|
|
|
var clone = this._clone();
|
|
|
|
clone._animationOverrides.set(componentType, animations);
|
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
2015-05-15 19:42:52 -04:00
|
|
|
/**
|
2015-08-14 13:03:45 -04:00
|
|
|
* Overrides a component's {@link ViewMetadata}.
|
2015-05-15 19:42:52 -04:00
|
|
|
*/
|
2015-08-14 13:03:45 -04:00
|
|
|
overrideView(componentType: Type, view: ViewMetadata): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = this._clone();
|
2015-06-17 19:21:40 -04:00
|
|
|
clone._viewOverrides.set(componentType, view);
|
2015-05-15 19:42:52 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-08-14 13:03:45 -04:00
|
|
|
* Overrides the directives from the component {@link ViewMetadata}.
|
2015-05-15 19:42:52 -04:00
|
|
|
*/
|
|
|
|
overrideDirective(componentType: Type, from: Type, to: Type): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = this._clone();
|
|
|
|
let overridesForComponent = clone._directiveOverrides.get(componentType);
|
2015-05-15 19:42:52 -04:00
|
|
|
if (!isPresent(overridesForComponent)) {
|
2015-09-29 14:11:06 -04:00
|
|
|
clone._directiveOverrides.set(componentType, new Map<Type, Type>());
|
2015-06-17 19:21:40 -04:00
|
|
|
overridesForComponent = clone._directiveOverrides.get(componentType);
|
2015-05-15 19:42:52 -04:00
|
|
|
}
|
2015-06-17 19:21:40 -04:00
|
|
|
overridesForComponent.set(from, to);
|
2015-05-15 19:42:52 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
2015-09-08 13:14:57 -04:00
|
|
|
/**
|
2015-10-11 01:11:13 -04:00
|
|
|
* Overrides one or more injectables configured via `providers` metadata property of a directive
|
|
|
|
* or
|
2015-09-08 13:14:57 -04:00
|
|
|
* component.
|
2015-10-11 01:11:13 -04:00
|
|
|
* Very useful when certain providers need to be mocked out.
|
2015-09-08 13:14:57 -04:00
|
|
|
*
|
2015-10-11 01:11:13 -04:00
|
|
|
* The providers specified via this method are appended to the existing `providers` causing the
|
|
|
|
* duplicated providers to
|
2015-09-08 13:14:57 -04:00
|
|
|
* be overridden.
|
|
|
|
*/
|
2015-10-11 01:11:13 -04:00
|
|
|
overrideProviders(type: Type, providers: any[]): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = this._clone();
|
2015-10-11 01:11:13 -04:00
|
|
|
clone._bindingsOverrides.set(type, providers);
|
2015-09-08 13:14:57 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-10-11 01:11:13 -04:00
|
|
|
* @deprecated
|
|
|
|
*/
|
|
|
|
overrideBindings(type: Type, providers: any[]): TestComponentBuilder {
|
|
|
|
return this.overrideProviders(type, providers);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Overrides one or more injectables configured via `providers` metadata property of a directive
|
|
|
|
* or
|
2015-09-08 13:14:57 -04:00
|
|
|
* component.
|
2015-10-11 01:11:13 -04:00
|
|
|
* Very useful when certain providers need to be mocked out.
|
2015-09-08 13:14:57 -04:00
|
|
|
*
|
2015-10-11 01:11:13 -04:00
|
|
|
* The providers specified via this method are appended to the existing `providers` causing the
|
|
|
|
* duplicated providers to
|
2015-09-08 13:14:57 -04:00
|
|
|
* be overridden.
|
|
|
|
*/
|
2015-10-11 01:11:13 -04:00
|
|
|
overrideViewProviders(type: Type, providers: any[]): TestComponentBuilder {
|
2016-04-26 19:38:54 -04:00
|
|
|
let clone = this._clone();
|
2015-10-11 01:11:13 -04:00
|
|
|
clone._viewBindingsOverrides.set(type, providers);
|
2015-09-08 13:14:57 -04:00
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
2015-10-11 01:11:13 -04:00
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
|
|
|
overrideViewBindings(type: Type, providers: any[]): TestComponentBuilder {
|
|
|
|
return this.overrideViewProviders(type, providers);
|
|
|
|
}
|
|
|
|
|
2016-04-30 13:52:04 -04:00
|
|
|
private _create<C>(ngZone: NgZone, componentFactory: ComponentFactory<C>): ComponentFixture<C> {
|
|
|
|
let rootElId = `root${_nextRootElementId++}`;
|
2016-04-28 20:50:03 -04:00
|
|
|
var testComponentRenderer: TestComponentRenderer = this._injector.get(TestComponentRenderer);
|
|
|
|
testComponentRenderer.insertRootElement(rootElId);
|
2016-04-30 13:52:04 -04:00
|
|
|
|
|
|
|
var componentRef = componentFactory.create(this._injector, [], `#${rootElId}`);
|
|
|
|
let autoDetect: boolean = this._injector.get(ComponentFixtureAutoDetect, false);
|
|
|
|
return new ComponentFixture<any /*C*/>(componentRef, ngZone, autoDetect);
|
|
|
|
}
|
|
|
|
|
2015-05-15 19:42:52 -04:00
|
|
|
/**
|
2015-10-31 12:50:19 -04:00
|
|
|
* Builds and returns a ComponentFixture.
|
2015-05-15 19:42:52 -04:00
|
|
|
*/
|
2016-04-30 13:52:04 -04:00
|
|
|
createAsync(rootComponentType: Type): Promise<ComponentFixture<any>> {
|
2016-04-26 19:38:54 -04:00
|
|
|
let noNgZone = IS_DART || this._injector.get(ComponentFixtureNoNgZone, false);
|
|
|
|
let ngZone: NgZone = noNgZone ? null : this._injector.get(NgZone, null);
|
|
|
|
|
|
|
|
let initComponent = () => {
|
|
|
|
let mockDirectiveResolver = this._injector.get(DirectiveResolver);
|
|
|
|
let mockViewResolver = this._injector.get(ViewResolver);
|
|
|
|
this._viewOverrides.forEach((view, type) => mockViewResolver.setView(type, view));
|
2016-06-08 19:38:52 -04:00
|
|
|
this._templateOverrides.forEach(
|
|
|
|
(template, type) => mockViewResolver.setInlineTemplate(type, template));
|
|
|
|
this._animationOverrides.forEach(
|
|
|
|
(animationsEntry, type) => mockViewResolver.setAnimations(type, animationsEntry));
|
2016-04-26 19:38:54 -04:00
|
|
|
this._directiveOverrides.forEach((overrides, component) => {
|
|
|
|
overrides.forEach(
|
|
|
|
(to, from) => { mockViewResolver.overrideViewDirective(component, from, to); });
|
|
|
|
});
|
|
|
|
this._bindingsOverrides.forEach(
|
2016-03-21 06:57:17 -04:00
|
|
|
(bindings, type) => mockDirectiveResolver.setProvidersOverride(type, bindings));
|
2016-04-26 19:38:54 -04:00
|
|
|
this._viewBindingsOverrides.forEach(
|
2016-03-21 06:57:17 -04:00
|
|
|
(bindings, type) => mockDirectiveResolver.setViewProvidersOverride(type, bindings));
|
2016-04-26 19:38:54 -04:00
|
|
|
|
2016-04-30 13:52:04 -04:00
|
|
|
let promise: Promise<ComponentFactory<any>> =
|
|
|
|
this._injector.get(ComponentResolver).resolveComponent(rootComponentType);
|
|
|
|
return promise.then(componentFactory => this._create(ngZone, componentFactory));
|
2016-04-26 19:38:54 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
return ngZone == null ? initComponent() : ngZone.run(initComponent);
|
2015-05-15 19:42:52 -04:00
|
|
|
}
|
2016-01-06 17:13:44 -05:00
|
|
|
|
2016-04-30 13:52:04 -04:00
|
|
|
createFakeAsync(rootComponentType: Type): ComponentFixture<any> {
|
2016-06-08 18:45:15 -04:00
|
|
|
let result: any /** TODO #9100 */;
|
|
|
|
let error: any /** TODO #9100 */;
|
2016-06-08 19:38:52 -04:00
|
|
|
PromiseWrapper.then(
|
|
|
|
this.createAsync(rootComponentType), (_result) => { result = _result; },
|
|
|
|
(_error) => { error = _error; });
|
2016-01-06 17:13:44 -05:00
|
|
|
tick();
|
|
|
|
if (isPresent(error)) {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2016-04-30 13:52:04 -04:00
|
|
|
|
|
|
|
createSync<C>(componentFactory: ComponentFactory<C>): ComponentFixture<C> {
|
|
|
|
let noNgZone = IS_DART || this._injector.get(ComponentFixtureNoNgZone, false);
|
|
|
|
let ngZone: NgZone = noNgZone ? null : this._injector.get(NgZone, null);
|
|
|
|
|
|
|
|
let initComponent = () => this._create(ngZone, componentFactory);
|
|
|
|
return ngZone == null ? initComponent() : ngZone.run(initComponent);
|
|
|
|
}
|
2015-05-15 19:42:52 -04:00
|
|
|
}
|