fix(core): support read option when querying for types (#21187)

PR Close #21187
This commit is contained in:
Pawel Kozlowski 2017-12-20 12:19:59 +01:00 committed by Miško Hevery
parent a62371c0eb
commit afd89ed8d9
3 changed files with 40 additions and 4 deletions

View File

@ -154,6 +154,7 @@ function getIdxOfMatchingSelector(staticData: LNodeStatic, selector: string): nu
} }
function add(predicate: QueryPredicate<any>| null, node: LNode) { function add(predicate: QueryPredicate<any>| null, node: LNode) {
const nodeInjector = getOrCreateNodeInjectorForNode(node as LElement | LContainer);
while (predicate) { while (predicate) {
const type = predicate.type; const type = predicate.type;
if (type) { if (type) {
@ -164,18 +165,21 @@ function add(predicate: QueryPredicate<any>| null, node: LNode) {
i < ii; i++) { i < ii; i++) {
const def = ngStaticData[i] as DirectiveDef<any>; const def = ngStaticData[i] as DirectiveDef<any>;
if (def.diPublic && def.type === type) { if (def.diPublic && def.type === type) {
predicate.values.push(node.view.data[i]); if (predicate.read !== null) {
predicate.values.push(readFromNodeInjector(nodeInjector, node, predicate.read));
} else {
predicate.values.push(node.view.data[i]);
}
} }
} }
} else { } else {
const nodeInjector = getOrCreateNodeInjectorForNode(node as LElement | LContainer);
const selector = predicate.selector !; const selector = predicate.selector !;
for (let i = 0; i < selector.length; i++) { for (let i = 0; i < selector.length; i++) {
ngDevMode && assertNotNull(node.staticData, 'node.staticData'); ngDevMode && assertNotNull(node.staticData, 'node.staticData');
const directiveIdx = getIdxOfMatchingSelector(node.staticData !, selector[i]); const directiveIdx = getIdxOfMatchingSelector(node.staticData !, selector[i]);
// is anything on a node matching a selector? // is anything on a node matching a selector?
if (directiveIdx !== null) { if (directiveIdx !== null) {
if (predicate.read != null) { if (predicate.read !== null) {
predicate.values.push(readFromNodeInjector(nodeInjector, node, predicate.read)); predicate.values.push(readFromNodeInjector(nodeInjector, node, predicate.read));
} else { } else {
// is local name pointing to a directive? // is local name pointing to a directive?

View File

@ -77,7 +77,38 @@ describe('query', () => {
expect((parent.query1 as QueryList<any>).toArray()).toEqual([child1, child2]); expect((parent.query1 as QueryList<any>).toArray()).toEqual([child1, child2]);
}); });
describe('local names', () => { describe('types predicate', () => {
it('should query using type predicate and read specified token', () => {
const Child = createDirective();
let elToQuery;
/**
* <div child></div>
* class Cmpt {
* @ViewChildren(Child, {read: ElementRef}) query;
* }
*/
const Cmpt = createComponent('cmpt', function(ctx: any, cm: boolean) {
let tmp: any;
if (cm) {
m(0, Q(Child, false, QueryReadType.ElementRef));
elToQuery = E(1, 'div');
{ D(2, Child.ngDirectiveDef.n(), Child.ngDirectiveDef); }
e();
}
qR(tmp = m<QueryList<any>>(0)) && (ctx.query = tmp as QueryList<any>);
});
const cmptInstance = renderComponent(Cmpt);
const query = (cmptInstance.query as QueryList<any>);
expect(query.length).toBe(1);
expect(isElementRef(query.first)).toBeTruthy();
expect(query.first.nativeElement).toEqual(elToQuery);
});
});
describe('local names predicate', () => {
it('should query for a single element and read ElementRef', () => { it('should query for a single element and read ElementRef', () => {

View File

@ -85,6 +85,7 @@ export function createDirective(): DirectiveType<any> {
static ngDirectiveDef = defineDirective({ static ngDirectiveDef = defineDirective({
type: Directive, type: Directive,
factory: () => new Directive(), factory: () => new Directive(),
features: [PublicFeature],
}); });
}; };
} }