2017-12-01 14:23:03 -08: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
|
|
|
|
*/
|
|
|
|
|
2017-12-20 10:47:22 -08:00
|
|
|
// We are temporarily importing the existing viewEngine_from core so we can be sure we are
|
2017-12-14 18:05:41 -08:00
|
|
|
// correctly implementing its interfaces for backwards compatibility.
|
2017-12-01 14:23:03 -08:00
|
|
|
import {Observable} from 'rxjs/Observable';
|
2017-12-14 15:03:46 -08:00
|
|
|
|
2017-12-20 10:47:22 -08:00
|
|
|
import {ElementRef as viewEngine_ElementRef} from '../linker/element_ref';
|
|
|
|
import {QueryList as viewEngine_QueryList} from '../linker/query_list';
|
|
|
|
import {TemplateRef as viewEngine_TemplateRef} from '../linker/template_ref';
|
|
|
|
import {Type} from '../type';
|
2017-12-14 15:03:46 -08:00
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
import {assertNotNull} from './assert';
|
2017-12-18 15:14:09 +01:00
|
|
|
import {getOrCreateContainerRef, getOrCreateElementRef, getOrCreateNodeInjectorForNode, getOrCreateTemplateRef} from './di';
|
2018-01-09 16:43:12 -08:00
|
|
|
import {DirectiveDef, TypedDirectiveDef} from './interfaces/definition';
|
2018-01-09 18:38:17 -08:00
|
|
|
import {LInjector} from './interfaces/injector';
|
|
|
|
import {LContainerNode, LElementNode, LNode, LNodeFlags, LViewNode, TNode} from './interfaces/node';
|
|
|
|
import {LQuery, QueryReadType} from './interfaces/query';
|
2017-12-18 15:14:09 +01:00
|
|
|
import {assertNodeOfPossibleTypes} from './node_assert';
|
2018-01-08 20:17:13 -08:00
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-09 16:43:12 -08:00
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
/**
|
|
|
|
* A predicate which determines if a given element/directive should be included in the query
|
|
|
|
*/
|
|
|
|
export interface QueryPredicate<T> {
|
|
|
|
/**
|
|
|
|
* Next predicate
|
|
|
|
*/
|
|
|
|
next: QueryPredicate<any>|null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Destination to which the value should be added.
|
|
|
|
*/
|
|
|
|
list: QueryList<T>;
|
|
|
|
|
|
|
|
/**
|
2018-01-08 20:17:13 -08:00
|
|
|
* If looking for directives then it contains the directive type.
|
2017-12-01 14:23:03 -08:00
|
|
|
*/
|
2017-12-20 10:47:22 -08:00
|
|
|
type: Type<T>|null;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
|
|
|
/**
|
2017-12-12 14:42:28 +01:00
|
|
|
* If selector then contains local names to query for.
|
2017-12-01 14:23:03 -08:00
|
|
|
*/
|
2017-12-12 14:42:28 +01:00
|
|
|
selector: string[]|null;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2017-12-18 15:14:09 +01:00
|
|
|
/**
|
|
|
|
* Indicates which token should be read from DI for this query.
|
|
|
|
*/
|
|
|
|
read: QueryReadType|null;
|
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
/**
|
|
|
|
* Values which have been located.
|
|
|
|
*
|
|
|
|
* this is what builds up the `QueryList._valuesTree`.
|
|
|
|
*/
|
|
|
|
values: any[];
|
|
|
|
}
|
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
export class LQuery_ implements LQuery {
|
2017-12-01 14:23:03 -08:00
|
|
|
shallow: QueryPredicate<any>|null = null;
|
|
|
|
deep: QueryPredicate<any>|null = null;
|
|
|
|
|
|
|
|
constructor(deep?: QueryPredicate<any>) { this.deep = deep == null ? null : deep; }
|
|
|
|
|
2017-12-14 15:03:46 -08:00
|
|
|
track<T>(
|
2017-12-20 10:47:22 -08:00
|
|
|
queryList: viewEngine_QueryList<T>, predicate: Type<T>|string[], descend?: boolean,
|
2017-12-18 15:14:09 +01:00
|
|
|
read?: QueryReadType): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
// TODO(misko): This is not right. In case of inherited state, a calling track will incorrectly
|
|
|
|
// mutate parent.
|
|
|
|
if (descend) {
|
2017-12-18 15:14:09 +01:00
|
|
|
this.deep = createPredicate(this.deep, queryList, predicate, read != null ? read : null);
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
2017-12-18 15:14:09 +01:00
|
|
|
this.shallow =
|
|
|
|
createPredicate(this.shallow, queryList, predicate, read != null ? read : null);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
child(): LQuery|null {
|
2017-12-01 14:23:03 -08:00
|
|
|
if (this.deep === null) {
|
2018-01-08 20:17:13 -08:00
|
|
|
// if we don't have any deep queries then no need to track anything more.
|
2017-12-01 14:23:03 -08:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (this.shallow === null) {
|
|
|
|
// DeepQuery: We can reuse the current state if the child state would be same as current
|
|
|
|
// state.
|
|
|
|
return this;
|
|
|
|
} else {
|
|
|
|
// We need to create new state
|
2018-01-08 20:17:13 -08:00
|
|
|
return new LQuery_(this.deep);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 19:34:46 -08:00
|
|
|
addNode(node: LNode): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
add(this.shallow, node);
|
|
|
|
add(this.deep, node);
|
|
|
|
}
|
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
insertView(container: LContainerNode, view: LViewNode, index: number): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
removeView(container: LContainerNode, view: LViewNode, index: number): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-20 17:35:03 +01:00
|
|
|
/**
|
|
|
|
* Iterates over local names for a given node and returns directive index
|
|
|
|
* (or -1 if a local name points to an element).
|
|
|
|
*
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param tNode static data of a node to check
|
2017-12-20 17:35:03 +01:00
|
|
|
* @param selector selector to match
|
|
|
|
* @returns directive index, -1 or null if a selector didn't match any of the local names
|
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
function getIdxOfMatchingSelector(tNode: TNode, selector: string): number|null {
|
|
|
|
const localNames = tNode.localNames;
|
2017-12-20 17:35:03 +01:00
|
|
|
if (localNames) {
|
|
|
|
for (let i = 0; i < localNames.length; i += 2) {
|
|
|
|
if (localNames[i] === selector) {
|
|
|
|
return localNames[i + 1] as number;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterates over all the directives for a node and returns index of a directive for a given type.
|
|
|
|
*
|
|
|
|
* @param node Node on which directives are present.
|
|
|
|
* @param type Type of a directive to look for.
|
|
|
|
* @returns Index of a found directive or null when none found.
|
|
|
|
*/
|
|
|
|
function geIdxOfMatchingDirective(node: LNode, type: Type<any>): number|null {
|
|
|
|
const ngStaticData = node.view.ngStaticData;
|
|
|
|
const flags = node.flags;
|
|
|
|
for (let i = flags >> LNodeFlags.INDX_SHIFT,
|
|
|
|
ii = i + ((flags & LNodeFlags.SIZE_MASK) >> LNodeFlags.SIZE_SHIFT);
|
|
|
|
i < ii; i++) {
|
2018-01-09 16:43:12 -08:00
|
|
|
const def = ngStaticData[i] as TypedDirectiveDef<any>;
|
2017-12-20 17:35:03 +01:00
|
|
|
if (def.diPublic && def.type === type) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
function readDefaultInjectable(nodeInjector: LInjector, node: LNode): viewEngine_ElementRef|
|
2017-12-20 10:47:22 -08:00
|
|
|
viewEngine_TemplateRef<any>|undefined {
|
2017-12-18 15:14:09 +01:00
|
|
|
ngDevMode && assertNodeOfPossibleTypes(node, LNodeFlags.Container, LNodeFlags.Element);
|
|
|
|
if ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element) {
|
|
|
|
return getOrCreateElementRef(nodeInjector);
|
|
|
|
} else if ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Container) {
|
|
|
|
return getOrCreateTemplateRef(nodeInjector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-20 17:35:03 +01:00
|
|
|
function readFromNodeInjector(
|
2018-01-08 20:17:13 -08:00
|
|
|
nodeInjector: LInjector, node: LNode, read: QueryReadType | Type<any>): any {
|
2017-12-19 16:51:42 +01:00
|
|
|
if (read === QueryReadType.ElementRef) {
|
2017-12-18 15:14:09 +01:00
|
|
|
return getOrCreateElementRef(nodeInjector);
|
|
|
|
} else if (read === QueryReadType.ViewContainerRef) {
|
|
|
|
return getOrCreateContainerRef(nodeInjector);
|
|
|
|
} else if (read === QueryReadType.TemplateRef) {
|
|
|
|
return getOrCreateTemplateRef(nodeInjector);
|
2017-12-20 17:35:03 +01:00
|
|
|
} else {
|
|
|
|
const matchingIdx = geIdxOfMatchingDirective(node, read);
|
|
|
|
if (matchingIdx !== null) {
|
|
|
|
return node.view.data[matchingIdx];
|
2017-12-19 16:51:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
function add(predicate: QueryPredicate<any>| null, node: LNode) {
|
2018-01-08 20:17:13 -08:00
|
|
|
const nodeInjector = getOrCreateNodeInjectorForNode(node as LElementNode | LContainerNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
while (predicate) {
|
|
|
|
const type = predicate.type;
|
|
|
|
if (type) {
|
2017-12-20 17:35:03 +01:00
|
|
|
const directiveIdx = geIdxOfMatchingDirective(node, type);
|
|
|
|
if (directiveIdx !== null) {
|
|
|
|
if (predicate.read !== null) {
|
|
|
|
const requestedRead = readFromNodeInjector(nodeInjector, node, predicate.read);
|
|
|
|
if (requestedRead !== null) {
|
|
|
|
predicate.values.push(requestedRead);
|
2017-12-20 12:19:59 +01:00
|
|
|
}
|
2017-12-20 17:35:03 +01:00
|
|
|
} else {
|
|
|
|
predicate.values.push(node.view.data[directiveIdx]);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
2017-12-12 14:42:28 +01:00
|
|
|
} else {
|
2017-12-19 16:51:42 +01:00
|
|
|
const selector = predicate.selector !;
|
|
|
|
for (let i = 0; i < selector.length; i++) {
|
2018-01-08 20:17:13 -08:00
|
|
|
ngDevMode && assertNotNull(node.tNode, 'node.tNode');
|
|
|
|
const directiveIdx = getIdxOfMatchingSelector(node.tNode !, selector[i]);
|
2017-12-19 16:51:42 +01:00
|
|
|
// is anything on a node matching a selector?
|
|
|
|
if (directiveIdx !== null) {
|
2017-12-20 12:19:59 +01:00
|
|
|
if (predicate.read !== null) {
|
2017-12-20 17:35:03 +01:00
|
|
|
const requestedRead = readFromNodeInjector(nodeInjector, node, predicate.read);
|
|
|
|
if (requestedRead !== null) {
|
|
|
|
predicate.values.push(requestedRead);
|
|
|
|
}
|
2017-12-19 16:51:42 +01:00
|
|
|
} else {
|
|
|
|
// is local name pointing to a directive?
|
|
|
|
if (directiveIdx > -1) {
|
|
|
|
predicate.values.push(node.view.data[directiveIdx]);
|
|
|
|
} else {
|
|
|
|
predicate.values.push(readDefaultInjectable(nodeInjector, node));
|
|
|
|
}
|
2017-12-12 14:42:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
predicate = predicate.next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createPredicate<T>(
|
2017-12-20 10:47:22 -08:00
|
|
|
previous: QueryPredicate<any>| null, queryList: QueryList<T>, predicate: Type<T>| string[],
|
|
|
|
read: QueryReadType | null): QueryPredicate<T> {
|
2017-12-01 14:23:03 -08:00
|
|
|
const isArray = Array.isArray(predicate);
|
|
|
|
const values = <any>[];
|
|
|
|
if ((queryList as any as QueryList_<T>)._valuesTree === null) {
|
|
|
|
(queryList as any as QueryList_<T>)._valuesTree = values;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
next: previous,
|
|
|
|
list: queryList,
|
2017-12-20 10:47:22 -08:00
|
|
|
type: isArray ? null : predicate as Type<T>,
|
2017-12-12 14:42:28 +01:00
|
|
|
selector: isArray ? predicate as string[] : null,
|
2017-12-18 15:14:09 +01:00
|
|
|
read: read,
|
2017-12-01 14:23:03 -08:00
|
|
|
values: values
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-12-20 10:47:22 -08:00
|
|
|
class QueryList_<T>/* implements viewEngine_QueryList<T> */ {
|
2017-12-01 14:23:03 -08:00
|
|
|
dirty: boolean = false;
|
|
|
|
changes: Observable<T>;
|
|
|
|
|
|
|
|
get length(): number {
|
|
|
|
ngDevMode && assertNotNull(this._values, 'refreshed');
|
|
|
|
return this._values !.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
get first(): T|null {
|
|
|
|
ngDevMode && assertNotNull(this._values, 'refreshed');
|
|
|
|
let values = this._values !;
|
|
|
|
return values.length ? values[0] : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
get last(): T|null {
|
|
|
|
ngDevMode && assertNotNull(this._values, 'refreshed');
|
|
|
|
let values = this._values !;
|
|
|
|
return values.length ? values[values.length - 1] : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal */
|
|
|
|
_valuesTree: any[]|null = null;
|
|
|
|
/** @internal */
|
|
|
|
_values: T[]|null = null;
|
|
|
|
|
|
|
|
/** @internal */
|
|
|
|
_refresh(): boolean {
|
|
|
|
// TODO(misko): needs more logic to flatten tree.
|
|
|
|
if (this._values === null) {
|
|
|
|
this._values = this._valuesTree;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
map<U>(fn: (item: T, index: number, array: T[]) => U): U[] {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
filter(fn: (item: T, index: number, array: T[]) => boolean): T[] {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
find(fn: (item: T, index: number, array: T[]) => boolean): T|undefined {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
reduce<U>(fn: (prevValue: U, curValue: T, curIndex: number, array: T[]) => U, init: U): U {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
forEach(fn: (item: T, index: number, array: T[]) => void): void {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
some(fn: (value: T, index: number, array: T[]) => boolean): boolean {
|
|
|
|
throw new Error('Method not implemented.');
|
|
|
|
}
|
|
|
|
toArray(): T[] {
|
|
|
|
ngDevMode && assertNotNull(this._values, 'refreshed');
|
|
|
|
return this._values !;
|
|
|
|
}
|
|
|
|
toString(): string { throw new Error('Method not implemented.'); }
|
|
|
|
reset(res: (any[]|T)[]): void { throw new Error('Method not implemented.'); }
|
|
|
|
notifyOnChanges(): void { throw new Error('Method not implemented.'); }
|
|
|
|
setDirty(): void { throw new Error('Method not implemented.'); }
|
|
|
|
destroy(): void { throw new Error('Method not implemented.'); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: this hack is here because IQueryList has private members and therefore
|
|
|
|
// it can't be implemented only extended.
|
2017-12-20 10:47:22 -08:00
|
|
|
export type QueryList<T> = viewEngine_QueryList<T>;
|
|
|
|
export const QueryList: typeof viewEngine_QueryList = QueryList_ as any;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2017-12-14 16:26:28 -08:00
|
|
|
export function queryRefresh(query: QueryList<any>): boolean {
|
2017-12-01 14:23:03 -08:00
|
|
|
return (query as any as QueryList_<any>)._refresh();
|
|
|
|
}
|