2016-04-28 17:50:03 -07:00
|
|
|
import {ChangeDetectionStrategy, ViewEncapsulation} from '@angular/core';
|
|
|
|
import {isDefaultChangeDetectionStrategy, ViewType} from '../../core_private';
|
|
|
|
|
|
|
|
import {isPresent, StringWrapper} from '../../src/facade/lang';
|
|
|
|
import {ListWrapper, StringMapWrapper, SetWrapper} from '../../src/facade/collection';
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
import * as o from '../output/output_ast';
|
|
|
|
import {Identifiers, identifierToken} from '../identifiers';
|
|
|
|
import {
|
|
|
|
ViewConstructorVars,
|
|
|
|
InjectMethodVars,
|
|
|
|
DetectChangesVars,
|
|
|
|
ViewTypeEnum,
|
|
|
|
ViewEncapsulationEnum,
|
|
|
|
ChangeDetectionStrategyEnum,
|
|
|
|
ViewProperties
|
|
|
|
} from './constants';
|
|
|
|
|
|
|
|
import {CompileView} from './compile_view';
|
|
|
|
import {CompileElement, CompileNode} from './compile_element';
|
|
|
|
|
|
|
|
import {
|
|
|
|
TemplateAst,
|
|
|
|
TemplateAstVisitor,
|
|
|
|
NgContentAst,
|
|
|
|
EmbeddedTemplateAst,
|
|
|
|
ElementAst,
|
2016-04-25 19:52:24 -07:00
|
|
|
ReferenceAst,
|
2016-01-06 14:13:44 -08:00
|
|
|
VariableAst,
|
|
|
|
BoundEventAst,
|
|
|
|
BoundElementPropertyAst,
|
|
|
|
AttrAst,
|
|
|
|
BoundTextAst,
|
|
|
|
TextAst,
|
|
|
|
DirectiveAst,
|
|
|
|
BoundDirectivePropertyAst,
|
|
|
|
templateVisitAll,
|
|
|
|
} from '../template_ast';
|
|
|
|
|
|
|
|
import {getViewFactoryName, createFlatArray, createDiTokenExpression} from './util';
|
|
|
|
|
|
|
|
import {
|
|
|
|
CompileIdentifierMetadata,
|
|
|
|
CompileDirectiveMetadata,
|
|
|
|
CompileTokenMetadata
|
|
|
|
} from '../compile_metadata';
|
|
|
|
|
|
|
|
const IMPLICIT_TEMPLATE_VAR = '\$implicit';
|
|
|
|
const CLASS_ATTR = 'class';
|
|
|
|
const STYLE_ATTR = 'style';
|
|
|
|
|
|
|
|
var parentRenderNodeVar = o.variable('parentRenderNode');
|
|
|
|
var rootSelectorVar = o.variable('rootSelector');
|
|
|
|
|
|
|
|
export class ViewCompileDependency {
|
|
|
|
constructor(public comp: CompileDirectiveMetadata,
|
|
|
|
public factoryPlaceholder: CompileIdentifierMetadata) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function buildView(view: CompileView, template: TemplateAst[],
|
2016-04-27 11:22:44 -07:00
|
|
|
targetDependencies: ViewCompileDependency[]): number {
|
|
|
|
var builderVisitor = new ViewBuilderVisitor(view, targetDependencies);
|
2016-01-06 14:13:44 -08:00
|
|
|
templateVisitAll(builderVisitor, template, view.declarationElement.isNull() ?
|
|
|
|
view.declarationElement :
|
|
|
|
view.declarationElement.parent);
|
|
|
|
return builderVisitor.nestedViewCount;
|
|
|
|
}
|
|
|
|
|
2016-04-27 11:22:44 -07:00
|
|
|
export function finishView(view: CompileView, targetStatements: o.Statement[]) {
|
|
|
|
view.afterNodes();
|
|
|
|
createViewTopLevelStmts(view, targetStatements);
|
|
|
|
view.nodes.forEach((node) => {
|
|
|
|
if (node instanceof CompileElement && node.hasEmbeddedView) {
|
|
|
|
finishView(node.embeddedView, targetStatements);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
class ViewBuilderVisitor implements TemplateAstVisitor {
|
|
|
|
nestedViewCount: number = 0;
|
|
|
|
|
2016-04-27 11:22:44 -07:00
|
|
|
constructor(public view: CompileView, public targetDependencies: ViewCompileDependency[]) {}
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
private _isRootNode(parent: CompileElement): boolean { return parent.view !== this.view; }
|
|
|
|
|
|
|
|
private _addRootNodeAndProject(node: CompileNode, ngContentIndex: number,
|
|
|
|
parent: CompileElement) {
|
2016-04-18 13:24:42 -07:00
|
|
|
var vcAppEl =
|
|
|
|
(node instanceof CompileElement && node.hasViewContainer) ? node.appElement : null;
|
2016-01-06 14:13:44 -08:00
|
|
|
if (this._isRootNode(parent)) {
|
2016-04-18 13:24:42 -07:00
|
|
|
// store appElement as root node only for ViewContainers
|
2016-01-06 14:13:44 -08:00
|
|
|
if (this.view.viewType !== ViewType.COMPONENT) {
|
2016-04-18 13:24:42 -07:00
|
|
|
this.view.rootNodesOrAppElements.push(isPresent(vcAppEl) ? vcAppEl : node.renderNode);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
} else if (isPresent(parent.component) && isPresent(ngContentIndex)) {
|
2016-04-18 13:24:42 -07:00
|
|
|
parent.addContentNode(ngContentIndex, isPresent(vcAppEl) ? vcAppEl : node.renderNode);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _getParentRenderNode(parent: CompileElement): o.Expression {
|
|
|
|
if (this._isRootNode(parent)) {
|
|
|
|
if (this.view.viewType === ViewType.COMPONENT) {
|
|
|
|
return parentRenderNodeVar;
|
|
|
|
} else {
|
|
|
|
// root node of an embedded/host view
|
|
|
|
return o.NULL_EXPR;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return isPresent(parent.component) &&
|
|
|
|
parent.component.template.encapsulation !== ViewEncapsulation.Native ?
|
|
|
|
o.NULL_EXPR :
|
|
|
|
parent.renderNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
visitBoundText(ast: BoundTextAst, parent: CompileElement): any {
|
|
|
|
return this._visitText(ast, '', ast.ngContentIndex, parent);
|
|
|
|
}
|
|
|
|
visitText(ast: TextAst, parent: CompileElement): any {
|
|
|
|
return this._visitText(ast, ast.value, ast.ngContentIndex, parent);
|
|
|
|
}
|
|
|
|
private _visitText(ast: TemplateAst, value: string, ngContentIndex: number,
|
|
|
|
parent: CompileElement): o.Expression {
|
|
|
|
var fieldName = `_text_${this.view.nodes.length}`;
|
2016-04-30 16:13:03 -07:00
|
|
|
this.view.fields.push(
|
|
|
|
new o.ClassField(fieldName, o.importType(this.view.genConfig.renderTypes.renderText)));
|
2016-01-06 14:13:44 -08:00
|
|
|
var renderNode = o.THIS_EXPR.prop(fieldName);
|
|
|
|
var compileNode = new CompileNode(parent, this.view, this.view.nodes.length, renderNode, ast);
|
|
|
|
var createRenderNode =
|
|
|
|
o.THIS_EXPR.prop(fieldName)
|
|
|
|
.set(ViewProperties.renderer.callMethod(
|
|
|
|
'createText',
|
|
|
|
[
|
|
|
|
this._getParentRenderNode(parent),
|
|
|
|
o.literal(value),
|
|
|
|
this.view.createMethod.resetDebugInfoExpr(this.view.nodes.length, ast)
|
|
|
|
]))
|
|
|
|
.toStmt();
|
|
|
|
this.view.nodes.push(compileNode);
|
|
|
|
this.view.createMethod.addStmt(createRenderNode);
|
|
|
|
this._addRootNodeAndProject(compileNode, ngContentIndex, parent);
|
|
|
|
return renderNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitNgContent(ast: NgContentAst, parent: CompileElement): any {
|
|
|
|
// the projected nodes originate from a different view, so we don't
|
|
|
|
// have debug information for them...
|
|
|
|
this.view.createMethod.resetDebugInfo(null, ast);
|
|
|
|
var parentRenderNode = this._getParentRenderNode(parent);
|
|
|
|
var nodesExpression = ViewProperties.projectableNodes.key(
|
|
|
|
o.literal(ast.index),
|
|
|
|
new o.ArrayType(o.importType(this.view.genConfig.renderTypes.renderNode)));
|
|
|
|
if (parentRenderNode !== o.NULL_EXPR) {
|
|
|
|
this.view.createMethod.addStmt(
|
|
|
|
ViewProperties.renderer.callMethod(
|
|
|
|
'projectNodes',
|
|
|
|
[
|
|
|
|
parentRenderNode,
|
|
|
|
o.importExpr(Identifiers.flattenNestedViewRenderNodes)
|
|
|
|
.callFn([nodesExpression])
|
|
|
|
])
|
|
|
|
.toStmt());
|
|
|
|
} else if (this._isRootNode(parent)) {
|
|
|
|
if (this.view.viewType !== ViewType.COMPONENT) {
|
|
|
|
// store root nodes only for embedded/host views
|
|
|
|
this.view.rootNodesOrAppElements.push(nodesExpression);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (isPresent(parent.component) && isPresent(ast.ngContentIndex)) {
|
|
|
|
parent.addContentNode(ast.ngContentIndex, nodesExpression);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitElement(ast: ElementAst, parent: CompileElement): any {
|
|
|
|
var nodeIndex = this.view.nodes.length;
|
|
|
|
var createRenderNodeExpr;
|
|
|
|
var debugContextExpr = this.view.createMethod.resetDebugInfoExpr(nodeIndex, ast);
|
|
|
|
if (nodeIndex === 0 && this.view.viewType === ViewType.HOST) {
|
2016-04-13 17:05:17 -07:00
|
|
|
createRenderNodeExpr = o.THIS_EXPR.callMethod(
|
|
|
|
'selectOrCreateHostElement', [o.literal(ast.name), rootSelectorVar, debugContextExpr]);
|
2016-01-06 14:13:44 -08:00
|
|
|
} else {
|
2016-04-13 17:05:17 -07:00
|
|
|
createRenderNodeExpr = ViewProperties.renderer.callMethod(
|
|
|
|
'createElement',
|
|
|
|
[this._getParentRenderNode(parent), o.literal(ast.name), debugContextExpr]);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
var fieldName = `_el_${nodeIndex}`;
|
|
|
|
this.view.fields.push(
|
2016-04-30 16:13:03 -07:00
|
|
|
new o.ClassField(fieldName, o.importType(this.view.genConfig.renderTypes.renderElement)));
|
2016-04-18 13:24:42 -07:00
|
|
|
this.view.createMethod.addStmt(o.THIS_EXPR.prop(fieldName).set(createRenderNodeExpr).toStmt());
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
var renderNode = o.THIS_EXPR.prop(fieldName);
|
|
|
|
|
|
|
|
var directives = ast.directives.map(directiveAst => directiveAst.directive);
|
2016-04-29 17:07:28 -07:00
|
|
|
var component = directives.find(directive => directive.isComponent);
|
2016-01-06 14:13:44 -08:00
|
|
|
var htmlAttrs = _readHtmlAttrs(ast.attrs);
|
|
|
|
var attrNameAndValues = _mergeHtmlAndDirectiveAttrs(htmlAttrs, directives);
|
|
|
|
for (var i = 0; i < attrNameAndValues.length; i++) {
|
|
|
|
var attrName = attrNameAndValues[i][0];
|
|
|
|
var attrValue = attrNameAndValues[i][1];
|
|
|
|
this.view.createMethod.addStmt(
|
|
|
|
ViewProperties.renderer.callMethod(
|
|
|
|
'setElementAttribute',
|
|
|
|
[renderNode, o.literal(attrName), o.literal(attrValue)])
|
|
|
|
.toStmt());
|
|
|
|
}
|
2016-04-18 13:24:42 -07:00
|
|
|
var compileElement =
|
|
|
|
new CompileElement(parent, this.view, nodeIndex, renderNode, ast, component, directives,
|
2016-04-25 19:52:24 -07:00
|
|
|
ast.providers, ast.hasViewContainer, false, ast.references);
|
2016-01-06 14:13:44 -08:00
|
|
|
this.view.nodes.push(compileElement);
|
|
|
|
var compViewExpr: o.ReadVarExpr = null;
|
|
|
|
if (isPresent(component)) {
|
|
|
|
var nestedComponentIdentifier =
|
|
|
|
new CompileIdentifierMetadata({name: getViewFactoryName(component, 0)});
|
|
|
|
this.targetDependencies.push(new ViewCompileDependency(component, nestedComponentIdentifier));
|
2016-05-03 18:49:59 -07:00
|
|
|
compViewExpr = o.variable(`compView_${nodeIndex}`); // fix highlighting: `
|
2016-04-18 13:24:42 -07:00
|
|
|
compileElement.setComponentView(compViewExpr);
|
2016-01-06 14:13:44 -08:00
|
|
|
this.view.createMethod.addStmt(compViewExpr.set(o.importExpr(nestedComponentIdentifier)
|
|
|
|
.callFn([
|
2016-04-18 13:24:42 -07:00
|
|
|
ViewProperties.viewUtils,
|
|
|
|
compileElement.injector,
|
|
|
|
compileElement.appElement
|
2016-01-06 14:13:44 -08:00
|
|
|
]))
|
|
|
|
.toDeclStmt());
|
|
|
|
}
|
|
|
|
compileElement.beforeChildren();
|
|
|
|
this._addRootNodeAndProject(compileElement, ast.ngContentIndex, parent);
|
|
|
|
templateVisitAll(this, ast.children, compileElement);
|
|
|
|
compileElement.afterChildren(this.view.nodes.length - nodeIndex - 1);
|
|
|
|
|
|
|
|
if (isPresent(compViewExpr)) {
|
|
|
|
var codeGenContentNodes;
|
|
|
|
if (this.view.component.type.isHost) {
|
|
|
|
codeGenContentNodes = ViewProperties.projectableNodes;
|
|
|
|
} else {
|
|
|
|
codeGenContentNodes = o.literalArr(
|
|
|
|
compileElement.contentNodesByNgContentIndex.map(nodes => createFlatArray(nodes)));
|
|
|
|
}
|
|
|
|
this.view.createMethod.addStmt(
|
2016-04-28 14:00:31 -07:00
|
|
|
compViewExpr.callMethod('create',
|
|
|
|
[compileElement.getComponent(), codeGenContentNodes, o.NULL_EXPR])
|
|
|
|
.toStmt());
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitEmbeddedTemplate(ast: EmbeddedTemplateAst, parent: CompileElement): any {
|
|
|
|
var nodeIndex = this.view.nodes.length;
|
|
|
|
var fieldName = `_anchor_${nodeIndex}`;
|
|
|
|
this.view.fields.push(
|
2016-04-30 16:13:03 -07:00
|
|
|
new o.ClassField(fieldName, o.importType(this.view.genConfig.renderTypes.renderComment)));
|
2016-04-18 13:24:42 -07:00
|
|
|
this.view.createMethod.addStmt(
|
|
|
|
o.THIS_EXPR.prop(fieldName)
|
|
|
|
.set(ViewProperties.renderer.callMethod(
|
|
|
|
'createTemplateAnchor',
|
|
|
|
[
|
|
|
|
this._getParentRenderNode(parent),
|
|
|
|
this.view.createMethod.resetDebugInfoExpr(nodeIndex, ast)
|
|
|
|
]))
|
|
|
|
.toStmt());
|
2016-01-06 14:13:44 -08:00
|
|
|
var renderNode = o.THIS_EXPR.prop(fieldName);
|
|
|
|
|
2016-04-25 19:52:24 -07:00
|
|
|
var templateVariableBindings = ast.variables.map(
|
2016-01-06 14:13:44 -08:00
|
|
|
varAst => [varAst.value.length > 0 ? varAst.value : IMPLICIT_TEMPLATE_VAR, varAst.name]);
|
|
|
|
|
|
|
|
var directives = ast.directives.map(directiveAst => directiveAst.directive);
|
2016-04-18 13:24:42 -07:00
|
|
|
var compileElement =
|
|
|
|
new CompileElement(parent, this.view, nodeIndex, renderNode, ast, null, directives,
|
2016-04-25 19:52:24 -07:00
|
|
|
ast.providers, ast.hasViewContainer, true, ast.references);
|
2016-01-06 14:13:44 -08:00
|
|
|
this.view.nodes.push(compileElement);
|
|
|
|
|
|
|
|
this.nestedViewCount++;
|
|
|
|
var embeddedView = new CompileView(
|
|
|
|
this.view.component, this.view.genConfig, this.view.pipeMetas, o.NULL_EXPR,
|
|
|
|
this.view.viewIndex + this.nestedViewCount, compileElement, templateVariableBindings);
|
2016-04-27 11:22:44 -07:00
|
|
|
this.nestedViewCount += buildView(embeddedView, ast.children, this.targetDependencies);
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
compileElement.beforeChildren();
|
|
|
|
this._addRootNodeAndProject(compileElement, ast.ngContentIndex, parent);
|
|
|
|
compileElement.afterChildren(0);
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitAttr(ast: AttrAst, ctx: any): any { return null; }
|
|
|
|
visitDirective(ast: DirectiveAst, ctx: any): any { return null; }
|
|
|
|
visitEvent(ast: BoundEventAst, eventTargetAndNames: Map<string, BoundEventAst>): any {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-04-25 19:52:24 -07:00
|
|
|
visitReference(ast: ReferenceAst, ctx: any): any { return null; }
|
2016-01-06 14:13:44 -08:00
|
|
|
visitVariable(ast: VariableAst, ctx: any): any { return null; }
|
|
|
|
visitDirectiveProperty(ast: BoundDirectivePropertyAst, context: any): any { return null; }
|
|
|
|
visitElementProperty(ast: BoundElementPropertyAst, context: any): any { return null; }
|
|
|
|
}
|
|
|
|
|
|
|
|
function _mergeHtmlAndDirectiveAttrs(declaredHtmlAttrs: {[key: string]: string},
|
|
|
|
directives: CompileDirectiveMetadata[]): string[][] {
|
|
|
|
var result: {[key: string]: string} = {};
|
|
|
|
StringMapWrapper.forEach(declaredHtmlAttrs, (value, key) => { result[key] = value; });
|
|
|
|
directives.forEach(directiveMeta => {
|
|
|
|
StringMapWrapper.forEach(directiveMeta.hostAttributes, (value, name) => {
|
|
|
|
var prevValue = result[name];
|
|
|
|
result[name] = isPresent(prevValue) ? mergeAttributeValue(name, prevValue, value) : value;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return mapToKeyValueArray(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _readHtmlAttrs(attrs: AttrAst[]): {[key: string]: string} {
|
|
|
|
var htmlAttrs: {[key: string]: string} = {};
|
|
|
|
attrs.forEach((ast) => { htmlAttrs[ast.name] = ast.value; });
|
|
|
|
return htmlAttrs;
|
|
|
|
}
|
|
|
|
|
|
|
|
function mergeAttributeValue(attrName: string, attrValue1: string, attrValue2: string): string {
|
|
|
|
if (attrName == CLASS_ATTR || attrName == STYLE_ATTR) {
|
|
|
|
return `${attrValue1} ${attrValue2}`;
|
|
|
|
} else {
|
|
|
|
return attrValue2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function mapToKeyValueArray(data: {[key: string]: string}): string[][] {
|
|
|
|
var entryArray = [];
|
|
|
|
StringMapWrapper.forEach(data, (value, name) => { entryArray.push([name, value]); });
|
|
|
|
// We need to sort to get a defined output order
|
|
|
|
// for tests and for caching generated artifacts...
|
|
|
|
ListWrapper.sort(entryArray, (entry1, entry2) => StringWrapper.compare(entry1[0], entry2[0]));
|
|
|
|
var keyValueArray = [];
|
|
|
|
entryArray.forEach((entry) => { keyValueArray.push([entry[0], entry[1]]); });
|
|
|
|
return keyValueArray;
|
|
|
|
}
|
|
|
|
|
|
|
|
function createViewTopLevelStmts(view: CompileView, targetStatements: o.Statement[]) {
|
|
|
|
var nodeDebugInfosVar: o.Expression = o.NULL_EXPR;
|
|
|
|
if (view.genConfig.genDebugInfo) {
|
feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
platform-browser. `SecurityContext` communicates what context a value is used
in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
determines the security context for an attribute or property (it turns out
attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
Value*s, i.e. the ability to mark a value as safe and not requiring further
sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
(surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
2016-04-29 16:04:08 -07:00
|
|
|
nodeDebugInfosVar = o.variable(
|
2016-05-03 18:49:59 -07:00
|
|
|
`nodeDebugInfos_${view.component.type.name}${view.viewIndex}`); // fix highlighting: `
|
2016-01-06 14:13:44 -08:00
|
|
|
targetStatements.push(
|
|
|
|
(<o.ReadVarExpr>nodeDebugInfosVar)
|
|
|
|
.set(o.literalArr(view.nodes.map(createStaticNodeDebugInfo),
|
|
|
|
new o.ArrayType(new o.ExternalType(Identifiers.StaticNodeDebugInfo),
|
|
|
|
[o.TypeModifier.Const])))
|
|
|
|
.toDeclStmt(null, [o.StmtModifier.Final]));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
feat: security implementation in Angular 2.
Summary:
This adds basic security hooks to Angular 2.
* `SecurityContext` is a private API between core, compiler, and
platform-browser. `SecurityContext` communicates what context a value is used
in across template parser, compiler, and sanitization at runtime.
* `SanitizationService` is the bare bones interface to sanitize values for a
particular context.
* `SchemaElementRegistry.securityContext(tagName, attributeOrPropertyName)`
determines the security context for an attribute or property (it turns out
attributes and properties match for the purposes of sanitization).
Based on these hooks:
* `DomSchemaElementRegistry` decides what sanitization applies in a particular
context.
* `DomSanitizationService` implements `SanitizationService` and adds *Safe
Value*s, i.e. the ability to mark a value as safe and not requiring further
sanitization.
* `url_sanitizer` and `style_sanitizer` sanitize URLs and Styles, respectively
(surprise!).
`DomSanitizationService` is the default implementation bound for browser
applications, in the three contexts (browser rendering, web worker rendering,
server side rendering).
BREAKING CHANGES:
*** SECURITY WARNING ***
Angular 2 Release Candidates do not implement proper contextual escaping yet.
Make sure to correctly escape all values that go into the DOM.
*** SECURITY WARNING ***
Reviewers: IgorMinar
Differential Revision: https://reviews.angular.io/D103
2016-04-29 16:04:08 -07:00
|
|
|
var renderCompTypeVar: o.ReadVarExpr =
|
|
|
|
o.variable(`renderType_${view.component.type.name}`); // fix highlighting: `
|
2016-01-06 14:13:44 -08:00
|
|
|
if (view.viewIndex === 0) {
|
|
|
|
targetStatements.push(renderCompTypeVar.set(o.NULL_EXPR)
|
|
|
|
.toDeclStmt(o.importType(Identifiers.RenderComponentType)));
|
|
|
|
}
|
|
|
|
|
|
|
|
var viewClass = createViewClass(view, renderCompTypeVar, nodeDebugInfosVar);
|
|
|
|
targetStatements.push(viewClass);
|
|
|
|
targetStatements.push(createViewFactory(view, viewClass, renderCompTypeVar));
|
|
|
|
}
|
|
|
|
|
|
|
|
function createStaticNodeDebugInfo(node: CompileNode): o.Expression {
|
|
|
|
var compileElement = node instanceof CompileElement ? node : null;
|
|
|
|
var providerTokens: o.Expression[] = [];
|
|
|
|
var componentToken: o.Expression = o.NULL_EXPR;
|
|
|
|
var varTokenEntries = [];
|
|
|
|
if (isPresent(compileElement)) {
|
|
|
|
providerTokens = compileElement.getProviderTokens();
|
|
|
|
if (isPresent(compileElement.component)) {
|
|
|
|
componentToken = createDiTokenExpression(identifierToken(compileElement.component.type));
|
|
|
|
}
|
2016-04-25 19:52:24 -07:00
|
|
|
StringMapWrapper.forEach(compileElement.referenceTokens, (token, varName) => {
|
2016-01-06 14:13:44 -08:00
|
|
|
varTokenEntries.push(
|
|
|
|
[varName, isPresent(token) ? createDiTokenExpression(token) : o.NULL_EXPR]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return o.importExpr(Identifiers.StaticNodeDebugInfo)
|
|
|
|
.instantiate(
|
|
|
|
[
|
|
|
|
o.literalArr(providerTokens, new o.ArrayType(o.DYNAMIC_TYPE, [o.TypeModifier.Const])),
|
|
|
|
componentToken,
|
|
|
|
o.literalMap(varTokenEntries, new o.MapType(o.DYNAMIC_TYPE, [o.TypeModifier.Const]))
|
|
|
|
],
|
|
|
|
o.importType(Identifiers.StaticNodeDebugInfo, null, [o.TypeModifier.Const]));
|
|
|
|
}
|
|
|
|
|
|
|
|
function createViewClass(view: CompileView, renderCompTypeVar: o.ReadVarExpr,
|
|
|
|
nodeDebugInfosVar: o.Expression): o.ClassStmt {
|
|
|
|
var viewConstructorArgs = [
|
2016-04-18 13:24:42 -07:00
|
|
|
new o.FnParam(ViewConstructorVars.viewUtils.name, o.importType(Identifiers.ViewUtils)),
|
2016-01-06 14:13:44 -08:00
|
|
|
new o.FnParam(ViewConstructorVars.parentInjector.name, o.importType(Identifiers.Injector)),
|
|
|
|
new o.FnParam(ViewConstructorVars.declarationEl.name, o.importType(Identifiers.AppElement))
|
|
|
|
];
|
2016-04-29 09:11:57 -07:00
|
|
|
var superConstructorArgs = [
|
|
|
|
o.variable(view.className),
|
|
|
|
renderCompTypeVar,
|
|
|
|
ViewTypeEnum.fromValue(view.viewType),
|
|
|
|
ViewConstructorVars.viewUtils,
|
|
|
|
ViewConstructorVars.parentInjector,
|
|
|
|
ViewConstructorVars.declarationEl,
|
|
|
|
ChangeDetectionStrategyEnum.fromValue(getChangeDetectionMode(view))
|
|
|
|
];
|
|
|
|
if (view.genConfig.genDebugInfo) {
|
|
|
|
superConstructorArgs.push(nodeDebugInfosVar);
|
|
|
|
}
|
|
|
|
var viewConstructor = new o.ClassMethod(null, viewConstructorArgs,
|
|
|
|
[o.SUPER_EXPR.callFn(superConstructorArgs).toStmt()]);
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
var viewMethods = [
|
|
|
|
new o.ClassMethod('createInternal', [new o.FnParam(rootSelectorVar.name, o.STRING_TYPE)],
|
2016-04-13 17:05:17 -07:00
|
|
|
generateCreateMethod(view), o.importType(Identifiers.AppElement)),
|
2016-01-06 14:13:44 -08:00
|
|
|
new o.ClassMethod(
|
|
|
|
'injectorGetInternal',
|
|
|
|
[
|
|
|
|
new o.FnParam(InjectMethodVars.token.name, o.DYNAMIC_TYPE),
|
|
|
|
// Note: Can't use o.INT_TYPE here as the method in AppView uses number
|
|
|
|
new o.FnParam(InjectMethodVars.requestNodeIndex.name, o.NUMBER_TYPE),
|
|
|
|
new o.FnParam(InjectMethodVars.notFoundResult.name, o.DYNAMIC_TYPE)
|
|
|
|
],
|
|
|
|
addReturnValuefNotEmpty(view.injectorGetMethod.finish(), InjectMethodVars.notFoundResult),
|
|
|
|
o.DYNAMIC_TYPE),
|
|
|
|
new o.ClassMethod('detectChangesInternal',
|
|
|
|
[new o.FnParam(DetectChangesVars.throwOnChange.name, o.BOOL_TYPE)],
|
|
|
|
generateDetectChangesMethod(view)),
|
|
|
|
new o.ClassMethod('dirtyParentQueriesInternal', [], view.dirtyParentQueriesMethod.finish()),
|
|
|
|
new o.ClassMethod('destroyInternal', [], view.destroyMethod.finish())
|
|
|
|
].concat(view.eventHandlerMethods);
|
2016-04-29 09:11:57 -07:00
|
|
|
var superClass = view.genConfig.genDebugInfo ? Identifiers.DebugAppView : Identifiers.AppView;
|
|
|
|
var viewClass = new o.ClassStmt(view.className, o.importExpr(superClass, [getContextType(view)]),
|
|
|
|
view.fields, view.getters, viewConstructor,
|
|
|
|
viewMethods.filter((method) => method.body.length > 0));
|
2016-01-06 14:13:44 -08:00
|
|
|
return viewClass;
|
|
|
|
}
|
|
|
|
|
|
|
|
function createViewFactory(view: CompileView, viewClass: o.ClassStmt,
|
|
|
|
renderCompTypeVar: o.ReadVarExpr): o.Statement {
|
|
|
|
var viewFactoryArgs = [
|
2016-04-18 13:24:42 -07:00
|
|
|
new o.FnParam(ViewConstructorVars.viewUtils.name, o.importType(Identifiers.ViewUtils)),
|
2016-01-06 14:13:44 -08:00
|
|
|
new o.FnParam(ViewConstructorVars.parentInjector.name, o.importType(Identifiers.Injector)),
|
|
|
|
new o.FnParam(ViewConstructorVars.declarationEl.name, o.importType(Identifiers.AppElement))
|
|
|
|
];
|
|
|
|
var initRenderCompTypeStmts = [];
|
|
|
|
var templateUrlInfo;
|
|
|
|
if (view.component.template.templateUrl == view.component.type.moduleUrl) {
|
|
|
|
templateUrlInfo =
|
|
|
|
`${view.component.type.moduleUrl} class ${view.component.type.name} - inline template`;
|
|
|
|
} else {
|
|
|
|
templateUrlInfo = view.component.template.templateUrl;
|
|
|
|
}
|
|
|
|
if (view.viewIndex === 0) {
|
|
|
|
initRenderCompTypeStmts = [
|
|
|
|
new o.IfStmt(renderCompTypeVar.identical(o.NULL_EXPR),
|
|
|
|
[
|
2016-04-18 13:24:42 -07:00
|
|
|
renderCompTypeVar.set(ViewConstructorVars
|
|
|
|
.viewUtils.callMethod('createRenderComponentType',
|
|
|
|
[
|
|
|
|
o.literal(templateUrlInfo),
|
|
|
|
o.literal(view.component
|
|
|
|
.template.ngContentSelectors.length),
|
|
|
|
ViewEncapsulationEnum
|
|
|
|
.fromValue(view.component.template.encapsulation),
|
|
|
|
view.styles
|
|
|
|
]))
|
2016-01-06 14:13:44 -08:00
|
|
|
.toStmt()
|
|
|
|
])
|
|
|
|
];
|
|
|
|
}
|
|
|
|
return o.fn(viewFactoryArgs, initRenderCompTypeStmts.concat([
|
|
|
|
new o.ReturnStatement(o.variable(viewClass.name)
|
|
|
|
.instantiate(viewClass.constructorMethod.params.map(
|
|
|
|
(param) => o.variable(param.name))))
|
|
|
|
]),
|
|
|
|
o.importType(Identifiers.AppView, [getContextType(view)]))
|
|
|
|
.toDeclStmt(view.viewFactory.name, [o.StmtModifier.Final]);
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateCreateMethod(view: CompileView): o.Statement[] {
|
|
|
|
var parentRenderNodeExpr: o.Expression = o.NULL_EXPR;
|
|
|
|
var parentRenderNodeStmts = [];
|
|
|
|
if (view.viewType === ViewType.COMPONENT) {
|
|
|
|
parentRenderNodeExpr = ViewProperties.renderer.callMethod(
|
|
|
|
'createViewRoot', [o.THIS_EXPR.prop('declarationAppElement').prop('nativeElement')]);
|
|
|
|
parentRenderNodeStmts = [
|
|
|
|
parentRenderNodeVar.set(parentRenderNodeExpr)
|
|
|
|
.toDeclStmt(o.importType(view.genConfig.renderTypes.renderNode), [o.StmtModifier.Final])
|
|
|
|
];
|
|
|
|
}
|
2016-04-13 17:05:17 -07:00
|
|
|
var resultExpr: o.Expression;
|
|
|
|
if (view.viewType === ViewType.HOST) {
|
2016-04-18 13:24:42 -07:00
|
|
|
resultExpr = (<CompileElement>view.nodes[0]).appElement;
|
2016-04-13 17:05:17 -07:00
|
|
|
} else {
|
|
|
|
resultExpr = o.NULL_EXPR;
|
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
return parentRenderNodeStmts.concat(view.createMethod.finish())
|
|
|
|
.concat([
|
|
|
|
o.THIS_EXPR.callMethod('init',
|
|
|
|
[
|
|
|
|
createFlatArray(view.rootNodesOrAppElements),
|
|
|
|
o.literalArr(view.nodes.map(node => node.renderNode)),
|
|
|
|
o.literalArr(view.disposables),
|
|
|
|
o.literalArr(view.subscriptions)
|
|
|
|
])
|
2016-04-13 17:05:17 -07:00
|
|
|
.toStmt(),
|
|
|
|
new o.ReturnStatement(resultExpr)
|
2016-01-06 14:13:44 -08:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateDetectChangesMethod(view: CompileView): o.Statement[] {
|
|
|
|
var stmts = [];
|
|
|
|
if (view.detectChangesInInputsMethod.isEmpty() && view.updateContentQueriesMethod.isEmpty() &&
|
|
|
|
view.afterContentLifecycleCallbacksMethod.isEmpty() &&
|
2016-04-20 18:10:19 -07:00
|
|
|
view.detectChangesRenderPropertiesMethod.isEmpty() &&
|
|
|
|
view.updateViewQueriesMethod.isEmpty() && view.afterViewLifecycleCallbacksMethod.isEmpty()) {
|
2016-01-06 14:13:44 -08:00
|
|
|
return stmts;
|
|
|
|
}
|
|
|
|
ListWrapper.addAll(stmts, view.detectChangesInInputsMethod.finish());
|
|
|
|
stmts.push(
|
|
|
|
o.THIS_EXPR.callMethod('detectContentChildrenChanges', [DetectChangesVars.throwOnChange])
|
|
|
|
.toStmt());
|
|
|
|
var afterContentStmts = view.updateContentQueriesMethod.finish().concat(
|
|
|
|
view.afterContentLifecycleCallbacksMethod.finish());
|
|
|
|
if (afterContentStmts.length > 0) {
|
|
|
|
stmts.push(new o.IfStmt(o.not(DetectChangesVars.throwOnChange), afterContentStmts));
|
|
|
|
}
|
2016-04-20 18:10:19 -07:00
|
|
|
ListWrapper.addAll(stmts, view.detectChangesRenderPropertiesMethod.finish());
|
2016-01-06 14:13:44 -08:00
|
|
|
stmts.push(o.THIS_EXPR.callMethod('detectViewChildrenChanges', [DetectChangesVars.throwOnChange])
|
|
|
|
.toStmt());
|
|
|
|
var afterViewStmts =
|
|
|
|
view.updateViewQueriesMethod.finish().concat(view.afterViewLifecycleCallbacksMethod.finish());
|
|
|
|
if (afterViewStmts.length > 0) {
|
|
|
|
stmts.push(new o.IfStmt(o.not(DetectChangesVars.throwOnChange), afterViewStmts));
|
|
|
|
}
|
|
|
|
|
|
|
|
var varStmts = [];
|
|
|
|
var readVars = o.findReadVarNames(stmts);
|
|
|
|
if (SetWrapper.has(readVars, DetectChangesVars.changed.name)) {
|
|
|
|
varStmts.push(DetectChangesVars.changed.set(o.literal(true)).toDeclStmt(o.BOOL_TYPE));
|
|
|
|
}
|
|
|
|
if (SetWrapper.has(readVars, DetectChangesVars.changes.name)) {
|
|
|
|
varStmts.push(DetectChangesVars.changes.set(o.NULL_EXPR)
|
|
|
|
.toDeclStmt(new o.MapType(o.importType(Identifiers.SimpleChange))));
|
|
|
|
}
|
|
|
|
if (SetWrapper.has(readVars, DetectChangesVars.valUnwrapper.name)) {
|
|
|
|
varStmts.push(
|
|
|
|
DetectChangesVars.valUnwrapper.set(o.importExpr(Identifiers.ValueUnwrapper).instantiate([]))
|
|
|
|
.toDeclStmt(null, [o.StmtModifier.Final]));
|
|
|
|
}
|
|
|
|
return varStmts.concat(stmts);
|
|
|
|
}
|
|
|
|
|
|
|
|
function addReturnValuefNotEmpty(statements: o.Statement[], value: o.Expression): o.Statement[] {
|
|
|
|
if (statements.length > 0) {
|
|
|
|
return statements.concat([new o.ReturnStatement(value)]);
|
|
|
|
} else {
|
|
|
|
return statements;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getContextType(view: CompileView): o.Type {
|
2016-04-28 14:00:31 -07:00
|
|
|
if (view.viewType === ViewType.COMPONENT) {
|
|
|
|
return o.importType(view.component.type);
|
|
|
|
}
|
|
|
|
return o.DYNAMIC_TYPE;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
function getChangeDetectionMode(view: CompileView): ChangeDetectionStrategy {
|
|
|
|
var mode: ChangeDetectionStrategy;
|
|
|
|
if (view.viewType === ViewType.COMPONENT) {
|
|
|
|
mode = isDefaultChangeDetectionStrategy(view.component.changeDetection) ?
|
|
|
|
ChangeDetectionStrategy.CheckAlways :
|
|
|
|
ChangeDetectionStrategy.CheckOnce;
|
|
|
|
} else {
|
|
|
|
mode = ChangeDetectionStrategy.CheckAlways;
|
|
|
|
}
|
|
|
|
return mode;
|
2016-04-22 15:33:32 -07:00
|
|
|
}
|