refactor(ElementInjector): support components
- Allow to access containing component directive instance from the shadow DOM. - Allow to access app services of the app level injector of the component when the component is instantiated.
This commit is contained in:
parent
c68e78075a
commit
7308a3acc7
|
@ -10,8 +10,8 @@ export function run () {
|
||||||
var bindings = [A, B, C];
|
var bindings = [A, B, C];
|
||||||
var proto = new ProtoElementInjector(null, 0, bindings);
|
var proto = new ProtoElementInjector(null, 0, bindings);
|
||||||
for (var i = 0; i < ITERATIONS; ++i) {
|
for (var i = 0; i < ITERATIONS; ++i) {
|
||||||
var ei = proto.instantiate(null,null);
|
var ei = proto.instantiate(null,null,null);
|
||||||
ei.instantiateDirectives(appInjector);
|
ei.instantiateDirectives(appInjector, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -18,8 +18,8 @@ export function run () {
|
||||||
|
|
||||||
var proto = new ProtoElementInjector(null, 0, bindings);
|
var proto = new ProtoElementInjector(null, 0, bindings);
|
||||||
for (var i = 0; i < ITERATIONS; ++i) {
|
for (var i = 0; i < ITERATIONS; ++i) {
|
||||||
var ei = proto.instantiate(null,null);
|
var ei = proto.instantiate(null,null,null);
|
||||||
ei.instantiateDirectives(appInjector);
|
ei.instantiateDirectives(appInjector, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,11 +9,11 @@ export function run () {
|
||||||
|
|
||||||
var bindings = [A, B, C];
|
var bindings = [A, B, C];
|
||||||
var proto = new ProtoElementInjector(null, 0, bindings);
|
var proto = new ProtoElementInjector(null, 0, bindings);
|
||||||
var ei = proto.instantiate(null,null);
|
var ei = proto.instantiate(null,null,null);
|
||||||
|
|
||||||
for (var i = 0; i < ITERATIONS; ++i) {
|
for (var i = 0; i < ITERATIONS; ++i) {
|
||||||
ei.clearDirectives();
|
ei.clearDirectives();
|
||||||
ei.instantiateDirectives(appInjector);
|
ei.instantiateDirectives(appInjector, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import {FIELD, isPresent, isBlank, Type, int} from 'facade/lang';
|
import {FIELD, isPresent, isBlank, Type, int, BaseException} from 'facade/lang';
|
||||||
import {Math} from 'facade/math';
|
import {Math} from 'facade/math';
|
||||||
import {List, ListWrapper} from 'facade/collection';
|
import {List, ListWrapper} from 'facade/collection';
|
||||||
import {Injector, Key, Dependency, bind, Binding, NoProviderError, ProviderError, CyclicDependencyError} from 'di/di';
|
import {Injector, Key, Dependency, bind, Binding, NoProviderError, ProviderError, CyclicDependencyError} from 'di/di';
|
||||||
|
@ -106,6 +106,7 @@ export class ProtoElementInjector {
|
||||||
@FIELD('_binding7:Binding')
|
@FIELD('_binding7:Binding')
|
||||||
@FIELD('_binding8:Binding')
|
@FIELD('_binding8:Binding')
|
||||||
@FIELD('_binding9:Binding')
|
@FIELD('_binding9:Binding')
|
||||||
|
@FIELD('_binding0IsComponent:int')
|
||||||
@FIELD('_key0:int')
|
@FIELD('_key0:int')
|
||||||
@FIELD('_key1:int')
|
@FIELD('_key1:int')
|
||||||
@FIELD('_key2:int')
|
@FIELD('_key2:int')
|
||||||
|
@ -118,10 +119,11 @@ export class ProtoElementInjector {
|
||||||
@FIELD('_key9:int')
|
@FIELD('_key9:int')
|
||||||
@FIELD('final parent:ProtoElementInjector')
|
@FIELD('final parent:ProtoElementInjector')
|
||||||
@FIELD('final index:int')
|
@FIELD('final index:int')
|
||||||
constructor(parent:ProtoElementInjector, index:int, bindings:List) {
|
constructor(parent:ProtoElementInjector, index:int, bindings:List, firstBindingIsComponent:boolean = false) {
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
this.index = index;
|
this.index = index;
|
||||||
|
|
||||||
|
this._binding0IsComponent = firstBindingIsComponent;
|
||||||
this._binding0 = null; this._keyId0 = null;
|
this._binding0 = null; this._keyId0 = null;
|
||||||
this._binding1 = null; this._keyId1 = null;
|
this._binding1 = null; this._keyId1 = null;
|
||||||
this._binding2 = null; this._keyId2 = null;
|
this._binding2 = null; this._keyId2 = null;
|
||||||
|
@ -150,8 +152,8 @@ export class ProtoElementInjector {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
instantiate(parent:ElementInjector, view):ElementInjector {
|
instantiate(parent:ElementInjector, host:ElementInjector, view):ElementInjector {
|
||||||
return new ElementInjector(this, parent, view);
|
return new ElementInjector(this, parent, host, view);
|
||||||
}
|
}
|
||||||
|
|
||||||
_createBinding(bindingOrType) {
|
_createBinding(bindingOrType) {
|
||||||
|
@ -212,7 +214,9 @@ export class ElementInjector extends TreeNode {
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@FIELD('_proto:ProtoElementInjector')
|
@FIELD('_proto:ProtoElementInjector')
|
||||||
@FIELD('_appInjector:Injector')
|
@FIELD('_lightDomAppInjector:Injector')
|
||||||
|
@FIELD('_shadowDomAppInjector:Injector')
|
||||||
|
@FIELD('_host:ElementInjector')
|
||||||
@FIELD('_obj0:Object')
|
@FIELD('_obj0:Object')
|
||||||
@FIELD('_obj1:Object')
|
@FIELD('_obj1:Object')
|
||||||
@FIELD('_obj2:Object')
|
@FIELD('_obj2:Object')
|
||||||
|
@ -224,13 +228,24 @@ export class ElementInjector extends TreeNode {
|
||||||
@FIELD('_obj8:Object')
|
@FIELD('_obj8:Object')
|
||||||
@FIELD('_obj9:Object')
|
@FIELD('_obj9:Object')
|
||||||
@FIELD('_view:View')
|
@FIELD('_view:View')
|
||||||
constructor(proto:ProtoElementInjector, parent:ElementInjector, view) {
|
constructor(proto:ProtoElementInjector, parent:ElementInjector, host:ElementInjector, view) {
|
||||||
super(parent);
|
super(parent);
|
||||||
|
if (isPresent(parent) && isPresent(host)) {
|
||||||
|
throw new BaseException('Only either parent or host is allowed');
|
||||||
|
}
|
||||||
|
this._host = null; // needed to satisfy Dart
|
||||||
|
if (isPresent(parent)) {
|
||||||
|
this._host = parent._host;
|
||||||
|
} else {
|
||||||
|
this._host = host;
|
||||||
|
}
|
||||||
|
|
||||||
this._proto = proto;
|
this._proto = proto;
|
||||||
this._view = view;
|
this._view = view;
|
||||||
|
|
||||||
//we cannot call clearDirectives because fields won't be detected
|
//we cannot call clearDirectives because fields won't be detected
|
||||||
this._appInjector = null;
|
this._lightDomAppInjector = null;
|
||||||
|
this._shadowDomAppInjector = null;
|
||||||
this._obj0 = null;
|
this._obj0 = null;
|
||||||
this._obj1 = null;
|
this._obj1 = null;
|
||||||
this._obj2 = null;
|
this._obj2 = null;
|
||||||
|
@ -245,7 +260,9 @@ export class ElementInjector extends TreeNode {
|
||||||
}
|
}
|
||||||
|
|
||||||
clearDirectives() {
|
clearDirectives() {
|
||||||
this._appInjector = null;
|
this._lightDomAppInjector = null;
|
||||||
|
this._shadowDomAppInjector = null;
|
||||||
|
|
||||||
this._obj0 = null;
|
this._obj0 = null;
|
||||||
this._obj1 = null;
|
this._obj1 = null;
|
||||||
this._obj2 = null;
|
this._obj2 = null;
|
||||||
|
@ -259,24 +276,42 @@ export class ElementInjector extends TreeNode {
|
||||||
this._constructionCounter = 0;
|
this._constructionCounter = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
instantiateDirectives(appInjector:Injector) {
|
instantiateDirectives(lightDomAppInjector:Injector, shadowDomAppInjector:Injector) {
|
||||||
this._appInjector = appInjector;
|
|
||||||
|
|
||||||
var p = this._proto;
|
var p = this._proto;
|
||||||
|
if (this._proto._binding0IsComponent && isBlank(shadowDomAppInjector)) {
|
||||||
|
throw new BaseException('A shadowDomAppInjector is required as this ElementInjector contains a component');
|
||||||
|
} else if (!this._proto._binding0IsComponent && isPresent(shadowDomAppInjector)) {
|
||||||
|
throw new BaseException('No shadowDomAppInjector allowed as there is not component stored in this ElementInjector');
|
||||||
|
}
|
||||||
|
this._lightDomAppInjector = lightDomAppInjector;
|
||||||
|
this._shadowDomAppInjector = shadowDomAppInjector;
|
||||||
|
|
||||||
if (isPresent(p._keyId0)) this._getDirectiveByKeyId(p._keyId0);
|
if (isPresent(p._keyId0)) this._getDirectiveByKeyId(p._keyId0);
|
||||||
if (isPresent(p._keyId1)) this._getDirectiveByKeyId(p._keyId1);
|
if (isPresent(p._keyId1)) this._getDirectiveByKeyId(p._keyId1);
|
||||||
if (isPresent(p._keyId2)) this._getDirectiveByKeyId(p._keyId2);;
|
if (isPresent(p._keyId2)) this._getDirectiveByKeyId(p._keyId2);
|
||||||
if (isPresent(p._keyId3)) this._getDirectiveByKeyId(p._keyId3);;
|
if (isPresent(p._keyId3)) this._getDirectiveByKeyId(p._keyId3);
|
||||||
if (isPresent(p._keyId4)) this._getDirectiveByKeyId(p._keyId4);;
|
if (isPresent(p._keyId4)) this._getDirectiveByKeyId(p._keyId4);
|
||||||
if (isPresent(p._keyId5)) this._getDirectiveByKeyId(p._keyId5);;
|
if (isPresent(p._keyId5)) this._getDirectiveByKeyId(p._keyId5);
|
||||||
if (isPresent(p._keyId6)) this._getDirectiveByKeyId(p._keyId6);;
|
if (isPresent(p._keyId6)) this._getDirectiveByKeyId(p._keyId6);
|
||||||
if (isPresent(p._keyId7)) this._getDirectiveByKeyId(p._keyId7);;
|
if (isPresent(p._keyId7)) this._getDirectiveByKeyId(p._keyId7);
|
||||||
if (isPresent(p._keyId8)) this._getDirectiveByKeyId(p._keyId8);;
|
if (isPresent(p._keyId8)) this._getDirectiveByKeyId(p._keyId8);
|
||||||
if (isPresent(p._keyId9)) this._getDirectiveByKeyId(p._keyId9);;
|
if (isPresent(p._keyId9)) this._getDirectiveByKeyId(p._keyId9);
|
||||||
}
|
}
|
||||||
|
|
||||||
get(token) {
|
get(token) {
|
||||||
return this._getByKey(Key.get(token), 0);
|
return this._getByKey(Key.get(token), 0, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
getComponent() {
|
||||||
|
if (this._proto._binding0IsComponent) {
|
||||||
|
return this._obj0;
|
||||||
|
} else {
|
||||||
|
throw new BaseException('There is not component stored in this ElementInjector');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_isComponentKey(key:Key) {
|
||||||
|
return this._proto._binding0IsComponent && key.id === this._proto._keyId0;
|
||||||
}
|
}
|
||||||
|
|
||||||
_new(binding:Binding) {
|
_new(binding:Binding) {
|
||||||
|
@ -290,16 +325,16 @@ export class ElementInjector extends TreeNode {
|
||||||
|
|
||||||
var d0,d1,d2,d3,d4,d5,d6,d7,d8,d9;
|
var d0,d1,d2,d3,d4,d5,d6,d7,d8,d9;
|
||||||
try {
|
try {
|
||||||
d0 = length > 0 ? this._getByDependency(deps[0]) : null;
|
d0 = length > 0 ? this._getByDependency(deps[0], binding.key) : null;
|
||||||
d1 = length > 1 ? this._getByDependency(deps[1]) : null;
|
d1 = length > 1 ? this._getByDependency(deps[1], binding.key) : null;
|
||||||
d2 = length > 2 ? this._getByDependency(deps[2]) : null;
|
d2 = length > 2 ? this._getByDependency(deps[2], binding.key) : null;
|
||||||
d3 = length > 3 ? this._getByDependency(deps[3]) : null;
|
d3 = length > 3 ? this._getByDependency(deps[3], binding.key) : null;
|
||||||
d4 = length > 4 ? this._getByDependency(deps[4]) : null;
|
d4 = length > 4 ? this._getByDependency(deps[4], binding.key) : null;
|
||||||
d5 = length > 5 ? this._getByDependency(deps[5]) : null;
|
d5 = length > 5 ? this._getByDependency(deps[5], binding.key) : null;
|
||||||
d6 = length > 6 ? this._getByDependency(deps[6]) : null;
|
d6 = length > 6 ? this._getByDependency(deps[6], binding.key) : null;
|
||||||
d7 = length > 7 ? this._getByDependency(deps[7]) : null;
|
d7 = length > 7 ? this._getByDependency(deps[7], binding.key) : null;
|
||||||
d8 = length > 8 ? this._getByDependency(deps[8]) : null;
|
d8 = length > 8 ? this._getByDependency(deps[8], binding.key) : null;
|
||||||
d9 = length > 9 ? this._getByDependency(deps[9]) : null;
|
d9 = length > 9 ? this._getByDependency(deps[9], binding.key) : null;
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
if (e instanceof ProviderError) e.addKey(binding.key);
|
if (e instanceof ProviderError) e.addKey(binding.key);
|
||||||
throw e;
|
throw e;
|
||||||
|
@ -324,8 +359,8 @@ export class ElementInjector extends TreeNode {
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
_getByDependency(dep:DirectiveDependency) {
|
_getByDependency(dep:DirectiveDependency, requestor:Key) {
|
||||||
return this._getByKey(dep.key, dep.depth);
|
return this._getByKey(dep.key, dep.depth, requestor);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -340,7 +375,7 @@ export class ElementInjector extends TreeNode {
|
||||||
*
|
*
|
||||||
* Write benchmarks before doing this optimization.
|
* Write benchmarks before doing this optimization.
|
||||||
*/
|
*/
|
||||||
_getByKey(key:Key, depth:int) {
|
_getByKey(key:Key, depth:int, requestor:Key) {
|
||||||
var ei = this;
|
var ei = this;
|
||||||
while (ei != null && depth >= 0) {
|
while (ei != null && depth >= 0) {
|
||||||
var specObj = ei._getSpecialObjectByKeyId(key.id);
|
var specObj = ei._getSpecialObjectByKeyId(key.id);
|
||||||
|
@ -352,7 +387,17 @@ export class ElementInjector extends TreeNode {
|
||||||
ei = ei._parent;
|
ei = ei._parent;
|
||||||
depth -= 1;
|
depth -= 1;
|
||||||
}
|
}
|
||||||
return this._appInjector.get(key);
|
if (isPresent(this._host) && this._host._isComponentKey(key)) {
|
||||||
|
return this._host.getComponent();
|
||||||
|
} else {
|
||||||
|
var appInjector;
|
||||||
|
if (isPresent(requestor) && this._isComponentKey(requestor)) {
|
||||||
|
appInjector = this._shadowDomAppInjector;
|
||||||
|
} else {
|
||||||
|
appInjector = this._lightDomAppInjector;
|
||||||
|
}
|
||||||
|
return appInjector.get(key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_getSpecialObjectByKeyId(keyId:int) {
|
_getSpecialObjectByKeyId(keyId:int) {
|
||||||
|
@ -364,6 +409,7 @@ export class ElementInjector extends TreeNode {
|
||||||
|
|
||||||
_getDirectiveByKeyId(keyId:int) {
|
_getDirectiveByKeyId(keyId:int) {
|
||||||
var p = this._proto;
|
var p = this._proto;
|
||||||
|
|
||||||
if (p._keyId0 === keyId) {if (isBlank(this._obj0)){this._obj0 = this._new(p._binding0);} return this._obj0;}
|
if (p._keyId0 === keyId) {if (isBlank(this._obj0)){this._obj0 = this._new(p._binding0);} return this._obj0;}
|
||||||
if (p._keyId1 === keyId) {if (isBlank(this._obj1)){this._obj1 = this._new(p._binding1);} return this._obj1;}
|
if (p._keyId1 === keyId) {if (isBlank(this._obj1)){this._obj1 = this._new(p._binding1);} return this._obj1;}
|
||||||
if (p._keyId2 === keyId) {if (isBlank(this._obj2)){this._obj2 = this._new(p._binding2);} return this._obj2;}
|
if (p._keyId2 === keyId) {if (isBlank(this._obj2)){this._obj2 = this._new(p._binding2);} return this._obj2;}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
import {isPresent,} from 'facade/lang';
|
import {isPresent, isBlank} from 'facade/lang';
|
||||||
import {ListWrapper} from 'facade/collection';
|
import {ListWrapper} from 'facade/collection';
|
||||||
|
|
||||||
import {ProtoElementInjector} from '../element_injector';
|
import {Key} from 'di/di';
|
||||||
|
import {ProtoElementInjector, ComponentKeyMetaData} from '../element_injector';
|
||||||
|
|
||||||
import {CompileStep} from './compile_step';
|
import {CompileStep} from './compile_step';
|
||||||
import {CompileElement} from './compile_element';
|
import {CompileElement} from './compile_element';
|
||||||
|
@ -23,22 +24,23 @@ import {CompileControl} from './compile_control';
|
||||||
*/
|
*/
|
||||||
export class ProtoElementInjectorBuilder extends CompileStep {
|
export class ProtoElementInjectorBuilder extends CompileStep {
|
||||||
// public so that we can overwrite it in tests
|
// public so that we can overwrite it in tests
|
||||||
internalCreateProtoElementInjector(parent, index, directives) {
|
internalCreateProtoElementInjector(parent, index, directives, firstBindingIsComponent) {
|
||||||
return new ProtoElementInjector(parent, index, directives);
|
return new ProtoElementInjector(parent, index, directives, firstBindingIsComponent);
|
||||||
}
|
}
|
||||||
|
|
||||||
process(parent:CompileElement, current:CompileElement, control:CompileControl) {
|
process(parent:CompileElement, current:CompileElement, control:CompileControl) {
|
||||||
var inheritedProtoElementInjector = null;
|
var inheritedProtoElementInjector = null;
|
||||||
var parentProtoElementInjector = this._getParentProtoElementInjector(parent, current);
|
var parentProtoElementInjector = this._getParentProtoElementInjector(parent, current);
|
||||||
var injectorBindings = this._collectDirectiveTypes(current);
|
var injectorBindings = this._collectDirectiveBindings(current);
|
||||||
// TODO: add lightDomServices as well,
|
// TODO: add lightDomServices as well,
|
||||||
// but after the directives as we rely on that order
|
// but after the directives as we rely on that order
|
||||||
// in the element_binder_builder.
|
// in the element_binder_builder.
|
||||||
|
|
||||||
if (injectorBindings.length > 0) {
|
if (injectorBindings.length > 0) {
|
||||||
var protoView = current.inheritedProtoView;
|
var protoView = current.inheritedProtoView;
|
||||||
|
var hasComponent = isPresent(current.componentDirective);
|
||||||
inheritedProtoElementInjector = this.internalCreateProtoElementInjector(
|
inheritedProtoElementInjector = this.internalCreateProtoElementInjector(
|
||||||
parentProtoElementInjector, protoView.elementBinders.length, injectorBindings
|
parentProtoElementInjector, protoView.elementBinders.length, injectorBindings, hasComponent
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
inheritedProtoElementInjector = parentProtoElementInjector;
|
inheritedProtoElementInjector = parentProtoElementInjector;
|
||||||
|
@ -56,19 +58,19 @@ export class ProtoElementInjectorBuilder extends CompileStep {
|
||||||
return parentProtoElementInjector;
|
return parentProtoElementInjector;
|
||||||
}
|
}
|
||||||
|
|
||||||
_collectDirectiveTypes(pipelineElement) {
|
_collectDirectiveBindings(pipelineElement) {
|
||||||
var directiveTypes = [];
|
var directiveTypes = [];
|
||||||
|
if (isPresent(pipelineElement.componentDirective)) {
|
||||||
|
ListWrapper.push(directiveTypes, pipelineElement.componentDirective.type);
|
||||||
|
}
|
||||||
|
if (isPresent(pipelineElement.templateDirective)) {
|
||||||
|
ListWrapper.push(directiveTypes, pipelineElement.templateDirective.type);
|
||||||
|
}
|
||||||
if (isPresent(pipelineElement.decoratorDirectives)) {
|
if (isPresent(pipelineElement.decoratorDirectives)) {
|
||||||
for (var i=0; i<pipelineElement.decoratorDirectives.length; i++) {
|
for (var i=0; i<pipelineElement.decoratorDirectives.length; i++) {
|
||||||
ListWrapper.push(directiveTypes, pipelineElement.decoratorDirectives[i].type);
|
ListWrapper.push(directiveTypes, pipelineElement.decoratorDirectives[i].type);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (isPresent(pipelineElement.templateDirective)) {
|
|
||||||
ListWrapper.push(directiveTypes, pipelineElement.templateDirective.type);
|
|
||||||
}
|
|
||||||
if (isPresent(pipelineElement.componentDirective)) {
|
|
||||||
ListWrapper.push(directiveTypes, pipelineElement.componentDirective.type);
|
|
||||||
}
|
|
||||||
return directiveTypes;
|
return directiveTypes;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -167,13 +167,13 @@ export class ProtoView {
|
||||||
static _instantiateDirectives(
|
static _instantiateDirectives(
|
||||||
injectors:List<ElementInjectors>, appInjector:Injector) {
|
injectors:List<ElementInjectors>, appInjector:Injector) {
|
||||||
for (var i = 0; i < injectors.length; ++i) {
|
for (var i = 0; i < injectors.length; ++i) {
|
||||||
if (injectors[i] != null) injectors[i].instantiateDirectives(appInjector);
|
if (injectors[i] != null) injectors[i].instantiateDirectives(appInjector, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static _createElementInjector(element, parent:ElementInjector, proto:ProtoElementInjector) {
|
static _createElementInjector(element, parent:ElementInjector, proto:ProtoElementInjector) {
|
||||||
//TODO: vsavkin: pass element to `proto.instantiate()` once https://github.com/angular/angular/pull/98 is merged
|
//TODO: vsavkin: pass element to `proto.instantiate()` once https://github.com/angular/angular/pull/98 is merged
|
||||||
return proto.instantiate(parent, null);
|
return proto.instantiate(parent, null, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
static _rootElementInjectors(injectors) {
|
static _rootElementInjectors(injectors) {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import {describe, ddescribe, it, iit, xit, xdescribe, expect, beforeEach} from 'test_lib/test_lib';
|
import {describe, ddescribe, it, iit, xit, xdescribe, expect, beforeEach} from 'test_lib/test_lib';
|
||||||
import {isBlank, FIELD, IMPLEMENTS} from 'facade/lang';
|
import {isBlank, isPresent, FIELD, IMPLEMENTS} from 'facade/lang';
|
||||||
import {ListWrapper, MapWrapper, List} from 'facade/collection';
|
import {ListWrapper, MapWrapper, List} from 'facade/collection';
|
||||||
import {ProtoElementInjector, VIEW_KEY} from 'core/compiler/element_injector';
|
import {ProtoElementInjector, VIEW_KEY} from 'core/compiler/element_injector';
|
||||||
import {Parent, Ancestor} from 'core/annotations/visibility';
|
import {Parent, Ancestor} from 'core/annotations/visibility';
|
||||||
|
@ -12,6 +12,10 @@ class DummyView {}
|
||||||
class Directive {
|
class Directive {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class SomeOtherDirective {
|
||||||
|
}
|
||||||
|
|
||||||
class NeedsDirective {
|
class NeedsDirective {
|
||||||
@FIELD("dependency:Directive")
|
@FIELD("dependency:Directive")
|
||||||
constructor(dependency:Directive){
|
constructor(dependency:Directive){
|
||||||
|
@ -66,13 +70,13 @@ export function main() {
|
||||||
return [lookupName(tree), children];
|
return [lookupName(tree), children];
|
||||||
}
|
}
|
||||||
|
|
||||||
function injector(bindings, appInjector = null, props = null) {
|
function injector(bindings, lightDomAppInjector = null, shadowDomAppInjector = null, props = null) {
|
||||||
if (isBlank(appInjector)) appInjector = new Injector([]);
|
if (isBlank(lightDomAppInjector)) lightDomAppInjector = new Injector([]);
|
||||||
if (isBlank(props)) props = {"view" : null};
|
if (isBlank(props)) props = {"view": null};
|
||||||
|
|
||||||
var proto = new ProtoElementInjector(null, 0, bindings);
|
var proto = new ProtoElementInjector(null, 0, bindings, isPresent(shadowDomAppInjector));
|
||||||
var inj = proto.instantiate(null, props["view"]);
|
var inj = proto.instantiate(null, null, props["view"]);
|
||||||
inj.instantiateDirectives(appInjector);
|
inj.instantiateDirectives(lightDomAppInjector, shadowDomAppInjector);
|
||||||
return inj;
|
return inj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -80,16 +84,31 @@ export function main() {
|
||||||
var inj = new Injector([]);
|
var inj = new Injector([]);
|
||||||
|
|
||||||
var protoParent = new ProtoElementInjector(null, 0, parentBindings);
|
var protoParent = new ProtoElementInjector(null, 0, parentBindings);
|
||||||
var parent = protoParent.instantiate(null, null);
|
var parent = protoParent.instantiate(null, null, null);
|
||||||
parent.instantiateDirectives(inj);
|
parent.instantiateDirectives(inj, null);
|
||||||
|
|
||||||
var protoChild = new ProtoElementInjector(protoParent, 1, childBindings);
|
var protoChild = new ProtoElementInjector(protoParent, 1, childBindings);
|
||||||
var child = protoChild.instantiate(parent, null);
|
var child = protoChild.instantiate(parent, null, null);
|
||||||
child.instantiateDirectives(inj);
|
child.instantiateDirectives(inj, null);
|
||||||
|
|
||||||
return child;
|
return child;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function hostShadowInjectors(hostBindings, shadowBindings) {
|
||||||
|
var inj = new Injector([]);
|
||||||
|
var shadowInj = inj.createChild([]);
|
||||||
|
|
||||||
|
var protoParent = new ProtoElementInjector(null, 0, hostBindings, true);
|
||||||
|
var host = protoParent.instantiate(null, null, null);
|
||||||
|
host.instantiateDirectives(inj, shadowInj);
|
||||||
|
|
||||||
|
var protoChild = new ProtoElementInjector(protoParent, 0, shadowBindings, false);
|
||||||
|
var shadow = protoChild.instantiate(null, host, null);
|
||||||
|
shadow.instantiateDirectives(shadowInj, null);
|
||||||
|
|
||||||
|
return shadow;
|
||||||
|
}
|
||||||
|
|
||||||
describe("ElementInjector", function () {
|
describe("ElementInjector", function () {
|
||||||
describe("instantiate", function () {
|
describe("instantiate", function () {
|
||||||
it("should create an element injector", function () {
|
it("should create an element injector", function () {
|
||||||
|
@ -97,9 +116,9 @@ export function main() {
|
||||||
var protoChild1 = new ProtoElementInjector(protoParent, 1, []);
|
var protoChild1 = new ProtoElementInjector(protoParent, 1, []);
|
||||||
var protoChild2 = new ProtoElementInjector(protoParent, 2, []);
|
var protoChild2 = new ProtoElementInjector(protoParent, 2, []);
|
||||||
|
|
||||||
var p = protoParent.instantiate(null, null);
|
var p = protoParent.instantiate(null, null, null);
|
||||||
var c1 = protoChild1.instantiate(p, null);
|
var c1 = protoChild1.instantiate(p, null, null);
|
||||||
var c2 = protoChild2.instantiate(p, null);
|
var c2 = protoChild2.instantiate(p, null, null);
|
||||||
|
|
||||||
expect(humanize(p, [
|
expect(humanize(p, [
|
||||||
[p, 'parent'],
|
[p, 'parent'],
|
||||||
|
@ -147,13 +166,47 @@ export function main() {
|
||||||
expect(d.service).toEqual("service");
|
expect(d.service).toEqual("service");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should instantiate directives that depend on speical objects", function () {
|
it("should instantiate directives that depend on special objects", function () {
|
||||||
var view = new DummyView();
|
var view = new DummyView();
|
||||||
var inj = injector([NeedsView], null, {"view" : view});
|
var inj = injector([NeedsView], null, null, {'view':view});
|
||||||
|
|
||||||
expect(inj.get(NeedsView).view).toBe(view);
|
expect(inj.get(NeedsView).view).toBe(view);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should instantiate directives that depend on the containing component", function () {
|
||||||
|
var shadow = hostShadowInjectors([Directive], [NeedsDirective]);
|
||||||
|
|
||||||
|
var d = shadow.get(NeedsDirective);
|
||||||
|
expect(d).toBeAnInstanceOf(NeedsDirective);
|
||||||
|
expect(d.dependency).toBeAnInstanceOf(Directive);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not instantiate directives that depend on other directives in the containing component's ElementInjector", () => {
|
||||||
|
expect( () => {
|
||||||
|
hostShadowInjectors([SomeOtherDirective, Directive], [NeedsDirective]);
|
||||||
|
}).toThrowError('No provider for Directive! (NeedsDirective -> Directive)')
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should instantiate component directives that depend on app services in the shadow app injector", () => {
|
||||||
|
var shadowAppInjector = new Injector([
|
||||||
|
bind("service").toValue("service")
|
||||||
|
]);
|
||||||
|
var inj = injector([NeedsService], null, shadowAppInjector);
|
||||||
|
|
||||||
|
var d = inj.get(NeedsService);
|
||||||
|
expect(d).toBeAnInstanceOf(NeedsService);
|
||||||
|
expect(d.service).toEqual("service");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not instantiate other directives that depend on app services in the shadow app injector", () => {
|
||||||
|
var shadowAppInjector = new Injector([
|
||||||
|
bind("service").toValue("service")
|
||||||
|
]);
|
||||||
|
expect( () => {
|
||||||
|
injector([SomeOtherDirective, NeedsService], null, shadowAppInjector);
|
||||||
|
}).toThrowError('No provider for service! (NeedsService -> service)');
|
||||||
|
});
|
||||||
|
|
||||||
it("should return app services", function () {
|
it("should return app services", function () {
|
||||||
var appInjector = new Injector([
|
var appInjector = new Injector([
|
||||||
bind("service").toValue("service")
|
bind("service").toValue("service")
|
||||||
|
@ -204,7 +257,7 @@ export function main() {
|
||||||
describe("special objects", function () {
|
describe("special objects", function () {
|
||||||
it("should return view", function () {
|
it("should return view", function () {
|
||||||
var view = new DummyView();
|
var view = new DummyView();
|
||||||
var inj = injector([], null, {"view" : view});
|
var inj = injector([], null, null, {"view" : view});
|
||||||
|
|
||||||
expect(inj.get(View)).toEqual(view);
|
expect(inj.get(View)).toEqual(view);
|
||||||
});
|
});
|
||||||
|
|
|
@ -41,9 +41,8 @@ export function main() {
|
||||||
}), protoElementInjectorBuilder]);
|
}), protoElementInjectorBuilder]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function assertProtoElementInjector(protoElementInjector, parent, index, bindings) {
|
function getCreationArgs(protoElementInjector) {
|
||||||
var args = protoElementInjectorBuilder.findArgsFor(protoElementInjector);
|
return protoElementInjectorBuilder.findArgsFor(protoElementInjector);
|
||||||
expect(args).toEqual([parent, index, bindings]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
it('should not create a ProtoElementInjector for elements without directives', () => {
|
it('should not create a ProtoElementInjector for elements without directives', () => {
|
||||||
|
@ -51,10 +50,19 @@ export function main() {
|
||||||
expect(results[0].inheritedProtoElementInjector).toBe(null);
|
expect(results[0].inheritedProtoElementInjector).toBe(null);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create a ProtoElementInjector for elements with directives', () => {
|
it('should create a ProtoElementInjector for elements directives', () => {
|
||||||
var directives = [SomeDecoratorDirective, SomeTemplateDirective, SomeComponentDirective];
|
var directives = [SomeComponentDirective, SomeTemplateDirective, SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(createElement('<div directives></div>'));
|
var results = createPipeline(directives).process(createElement('<div directives></div>'));
|
||||||
assertProtoElementInjector(results[0].inheritedProtoElementInjector, null, 0, directives);
|
var creationArgs = getCreationArgs(results[0].inheritedProtoElementInjector);
|
||||||
|
expect(creationArgs['bindings']).toEqual(directives);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should mark ProtoElementInjector for elements with component directives and use the ComponentDirective as first binding', () => {
|
||||||
|
var directives = [SomeDecoratorDirective, SomeComponentDirective];
|
||||||
|
var results = createPipeline(directives).process(createElement('<div directives></div>'));
|
||||||
|
var creationArgs = getCreationArgs(results[0].inheritedProtoElementInjector);
|
||||||
|
expect(creationArgs['firstBindingIsComponent']).toBe(true);
|
||||||
|
expect(creationArgs['bindings']).toEqual([SomeComponentDirective, SomeDecoratorDirective]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use the next ElementBinder index as index of the ProtoElementInjector', () => {
|
it('should use the next ElementBinder index as index of the ProtoElementInjector', () => {
|
||||||
|
@ -63,37 +71,36 @@ export function main() {
|
||||||
ListWrapper.push(protoView.elementBinders, null);
|
ListWrapper.push(protoView.elementBinders, null);
|
||||||
var directives = [SomeDecoratorDirective];
|
var directives = [SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(createElement('<div directives></div>'));
|
var results = createPipeline(directives).process(createElement('<div directives></div>'));
|
||||||
assertProtoElementInjector(
|
var creationArgs = getCreationArgs(results[0].inheritedProtoElementInjector);
|
||||||
results[0].inheritedProtoElementInjector, null, protoView.elementBinders.length, directives);
|
expect(creationArgs['index']).toBe(protoView.elementBinders.length);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should inherit the ProtoElementInjector down to children without directives', () => {
|
it('should inherit the ProtoElementInjector down to children without directives', () => {
|
||||||
var directives = [SomeDecoratorDirective, SomeTemplateDirective, SomeComponentDirective];
|
var directives = [SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(createElement('<div directives><span></span></div>'));
|
var results = createPipeline(directives).process(createElement('<div directives><span></span></div>'));
|
||||||
assertProtoElementInjector(results[0].inheritedProtoElementInjector, null, 0, directives);
|
expect(results[1].inheritedProtoElementInjector).toBe(results[0].inheritedProtoElementInjector);
|
||||||
assertProtoElementInjector(results[1].inheritedProtoElementInjector, null, 0, directives);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use the ProtoElementInjector of the parent element as parent', () => {
|
it('should use the ProtoElementInjector of the parent element as parent', () => {
|
||||||
var el = createElement('<div directives><span><a directives></a></span></div>');
|
var el = createElement('<div directives><span><a directives></a></span></div>');
|
||||||
var directives = [SomeDecoratorDirective, SomeTemplateDirective, SomeComponentDirective];
|
var directives = [SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(el);
|
var results = createPipeline(directives).process(el);
|
||||||
assertProtoElementInjector(results[2].inheritedProtoElementInjector,
|
expect(results[2].inheritedProtoElementInjector.parent).toBe(
|
||||||
results[0].inheritedProtoElementInjector, 0, directives);
|
results[0].inheritedProtoElementInjector);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use a null parent for viewRoots', () => {
|
it('should use a null parent for viewRoots', () => {
|
||||||
var el = createElement('<div directives><span viewroot directives></span></div>');
|
var el = createElement('<div directives><span viewroot directives></span></div>');
|
||||||
var directives = [SomeDecoratorDirective, SomeTemplateDirective, SomeComponentDirective];
|
var directives = [SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(el);
|
var results = createPipeline(directives).process(el);
|
||||||
assertProtoElementInjector(results[1].inheritedProtoElementInjector, null, 0, directives);
|
expect(results[1].inheritedProtoElementInjector.parent).toBe(null);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use a null parent if there is an intermediate viewRoot', () => {
|
it('should use a null parent if there is an intermediate viewRoot', () => {
|
||||||
var el = createElement('<div directives><span viewroot><a directives></a></span></div>');
|
var el = createElement('<div directives><span viewroot><a directives></a></span></div>');
|
||||||
var directives = [SomeDecoratorDirective, SomeTemplateDirective, SomeComponentDirective];
|
var directives = [SomeDecoratorDirective];
|
||||||
var results = createPipeline(directives).process(el);
|
var results = createPipeline(directives).process(el);
|
||||||
assertProtoElementInjector(results[2].inheritedProtoElementInjector, null, 0, directives);
|
expect(results[2].inheritedProtoElementInjector.parent).toBe(null);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -111,10 +118,10 @@ class TestableProtoElementInjectorBuilder extends ProtoElementInjectorBuilder {
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
internalCreateProtoElementInjector(parent, index, directives) {
|
internalCreateProtoElementInjector(parent, index, bindings, firstBindingIsComponent) {
|
||||||
var result = new ProtoElementInjector(parent, index, directives);
|
var result = new ProtoElementInjector(parent, index, bindings, firstBindingIsComponent);
|
||||||
ListWrapper.push(this.debugObjects, result);
|
ListWrapper.push(this.debugObjects, result);
|
||||||
ListWrapper.push(this.debugObjects, [parent, index, directives]);
|
ListWrapper.push(this.debugObjects, {'parent': parent, 'index': index, 'bindings': bindings, 'firstBindingIsComponent': firstBindingIsComponent});
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -128,10 +135,14 @@ class MockStep extends CompileStep {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class SomeComponentService {}
|
||||||
|
|
||||||
@Template()
|
@Template()
|
||||||
class SomeTemplateDirective {}
|
class SomeTemplateDirective {}
|
||||||
|
|
||||||
@Component()
|
@Component({
|
||||||
|
componentServices: [SomeComponentService]
|
||||||
|
})
|
||||||
class SomeComponentDirective {}
|
class SomeComponentDirective {}
|
||||||
|
|
||||||
@Decorator()
|
@Decorator()
|
||||||
|
|
Loading…
Reference in New Issue