This adds the feature for `@ViewChild`/`@ViewChildren`/`@ContentChild`/`@ContentChildren` to define what to read from the queried element. E.g. `@ViewChild(`someVar`, read: ViewContainerRef)` will locate the element with a variable `someVar` on it and return a `ViewContainerRef` for it. Background: With this change, Angular knows exactly at which elements there will be `ViewConainerRef`s as the user has to ask explicitly of them. This simplifies codegen and will make converting Angular templates into server side templates simpler as well. BREAKING CHANGE: - `DynamicComponentLoader.loadIntoLocation` has been removed. Use `@ViewChild(‘myVar’, read: ViewContainerRef)` to get hold of a `ViewContainerRef` at an element with variable `myVar`. - `DynamicComponentLoader.loadNextToLocation` now takes a `ViewContainerRef` instead of an `ElementRef`. - `AppViewManager` is renamed into `ViewUtils` and is a mere private utility service.
121 lines
4.4 KiB
TypeScript
121 lines
4.4 KiB
TypeScript
import {isPresent, isBlank} from 'angular2/src/facade/lang';
|
|
import {ListWrapper} from 'angular2/src/facade/collection';
|
|
|
|
import * as o from '../output/output_ast';
|
|
import {Identifiers} from '../identifiers';
|
|
|
|
import {
|
|
CompileQueryMetadata,
|
|
CompileIdentifierMetadata,
|
|
CompileTokenMap
|
|
} from '../compile_metadata';
|
|
|
|
import {CompileView} from './compile_view';
|
|
import {CompileElement} from './compile_element';
|
|
import {CompileMethod} from './compile_method';
|
|
import {getPropertyInView} from './util';
|
|
|
|
class ViewQueryValues {
|
|
constructor(public view: CompileView, public values: Array<o.Expression | ViewQueryValues>) {}
|
|
}
|
|
|
|
export class CompileQuery {
|
|
private _values: ViewQueryValues;
|
|
|
|
constructor(public meta: CompileQueryMetadata, public queryList: o.Expression,
|
|
public ownerDirectiveExpression: o.Expression, public view: CompileView) {
|
|
this._values = new ViewQueryValues(view, []);
|
|
}
|
|
|
|
addValue(value: o.Expression, view: CompileView) {
|
|
var currentView = view;
|
|
var elPath: CompileElement[] = [];
|
|
var viewPath: CompileView[] = [];
|
|
while (isPresent(currentView) && currentView !== this.view) {
|
|
var parentEl = currentView.declarationElement;
|
|
elPath.unshift(parentEl);
|
|
currentView = parentEl.view;
|
|
viewPath.push(currentView);
|
|
}
|
|
var queryListForDirtyExpr = getPropertyInView(this.queryList, viewPath);
|
|
|
|
var viewValues = this._values;
|
|
elPath.forEach((el) => {
|
|
var last =
|
|
viewValues.values.length > 0 ? viewValues.values[viewValues.values.length - 1] : null;
|
|
if (last instanceof ViewQueryValues && last.view === el.embeddedView) {
|
|
viewValues = last;
|
|
} else {
|
|
var newViewValues = new ViewQueryValues(el.embeddedView, []);
|
|
viewValues.values.push(newViewValues);
|
|
viewValues = newViewValues;
|
|
}
|
|
});
|
|
viewValues.values.push(value);
|
|
|
|
if (elPath.length > 0) {
|
|
view.dirtyParentQueriesMethod.addStmt(
|
|
queryListForDirtyExpr.callMethod('setDirty', []).toStmt());
|
|
}
|
|
}
|
|
|
|
afterChildren(targetMethod: CompileMethod) {
|
|
var values = createQueryValues(this._values);
|
|
var updateStmts = [this.queryList.callMethod('reset', [o.literalArr(values)]).toStmt()];
|
|
if (isPresent(this.ownerDirectiveExpression)) {
|
|
var valueExpr = this.meta.first ? this.queryList.prop('first') : this.queryList;
|
|
updateStmts.push(
|
|
this.ownerDirectiveExpression.prop(this.meta.propertyName).set(valueExpr).toStmt());
|
|
}
|
|
if (!this.meta.first) {
|
|
updateStmts.push(this.queryList.callMethod('notifyOnChanges', []).toStmt());
|
|
}
|
|
targetMethod.addStmt(new o.IfStmt(this.queryList.prop('dirty'), updateStmts));
|
|
}
|
|
}
|
|
|
|
function createQueryValues(viewValues: ViewQueryValues): o.Expression[] {
|
|
return ListWrapper.flatten(viewValues.values.map((entry) => {
|
|
if (entry instanceof ViewQueryValues) {
|
|
return mapNestedViews(entry.view.declarationElement.appElement, entry.view,
|
|
createQueryValues(entry));
|
|
} else {
|
|
return <o.Expression>entry;
|
|
}
|
|
}));
|
|
}
|
|
|
|
function mapNestedViews(declarationAppElement: o.Expression, view: CompileView,
|
|
expressions: o.Expression[]): o.Expression {
|
|
var adjustedExpressions: o.Expression[] = expressions.map((expr) => {
|
|
return o.replaceVarInExpression(o.THIS_EXPR.name, o.variable('nestedView'), expr);
|
|
});
|
|
return declarationAppElement.callMethod('mapNestedViews', [
|
|
o.variable(view.className),
|
|
o.fn([new o.FnParam('nestedView', view.classType)],
|
|
[new o.ReturnStatement(o.literalArr(adjustedExpressions))])
|
|
]);
|
|
}
|
|
|
|
export function createQueryList(query: CompileQueryMetadata, directiveInstance: o.Expression,
|
|
propertyName: string, compileView: CompileView): o.Expression {
|
|
compileView.fields.push(new o.ClassField(propertyName, o.importType(Identifiers.QueryList),
|
|
[o.StmtModifier.Private]));
|
|
var expr = o.THIS_EXPR.prop(propertyName);
|
|
compileView.createMethod.addStmt(o.THIS_EXPR.prop(propertyName)
|
|
.set(o.importExpr(Identifiers.QueryList).instantiate([]))
|
|
.toStmt());
|
|
return expr;
|
|
}
|
|
|
|
export function addQueryToTokenMap(map: CompileTokenMap<CompileQuery[]>, query: CompileQuery) {
|
|
query.meta.selectors.forEach((selector) => {
|
|
var entry = map.get(selector);
|
|
if (isBlank(entry)) {
|
|
entry = [];
|
|
map.add(selector, entry);
|
|
}
|
|
entry.push(query);
|
|
});
|
|
}
|