import { MapWrapper, ListWrapper, StringMapWrapper, SetWrapper } from 'angular2/src/facade/collection'; import { RegExpWrapper, isPresent, StringWrapper, StringJoiner, stringify, assertionsEnabled, isBlank } from 'angular2/src/facade/lang'; import {Injectable} from 'angular2/src/core/di'; import {BaseException} from 'angular2/src/facade/exceptions'; import {Parser, AST, ASTWithSource} from 'angular2/src/core/change_detection/change_detection'; import {TemplateBinding} from 'angular2/src/core/change_detection/parser/ast'; import {CompileDirectiveMetadata} from './directive_metadata'; import {HtmlParser} from './html_parser'; import {ParseSourceSpan, ParseError, ParseLocation} from './parse_util'; import { ElementAst, BoundElementPropertyAst, BoundEventAst, VariableAst, TemplateAst, TextAst, BoundTextAst, EmbeddedTemplateAst, AttrAst, NgContentAst, PropertyBindingType, DirectiveAst, BoundDirectivePropertyAst } from './template_ast'; import {CssSelector, SelectorMatcher} from 'angular2/src/compiler/selector'; import {ElementSchemaRegistry} from 'angular2/src/compiler/schema/element_schema_registry'; import {preparseElement, PreparsedElement, PreparsedElementType} from './template_preparser'; import {isStyleUrlResolvable} from './style_url_resolver'; import { HtmlAstVisitor, HtmlAst, HtmlElementAst, HtmlAttrAst, HtmlTextAst, htmlVisitAll } from './html_ast'; import {dashCaseToCamelCase, camelCaseToDashCase, splitAtColon} from './util'; // Group 1 = "bind-" // Group 2 = "var-" or "#" // Group 3 = "on-" // Group 4 = "bindon-" // Group 5 = the identifier after "bind-", "var-/#", or "on-" // Group 6 = idenitifer inside [()] // Group 7 = idenitifer inside [] // Group 8 = identifier inside () var BIND_NAME_REGEXP = /^(?:(?:(?:(bind-)|(var-|#)|(on-)|(bindon-))(.+))|\[\(([^\)]+)\)\]|\[([^\]]+)\]|\(([^\)]+)\))$/ig; const TEMPLATE_ELEMENT = 'template'; const TEMPLATE_ATTR = 'template'; const TEMPLATE_ATTR_PREFIX = '*'; const CLASS_ATTR = 'class'; var PROPERTY_PARTS_SEPARATOR = '.'; const ATTRIBUTE_PREFIX = 'attr'; const CLASS_PREFIX = 'class'; const STYLE_PREFIX = 'style'; var TEXT_CSS_SELECTOR = CssSelector.parse('*')[0]; export class TemplateParseError extends ParseError { constructor(message: string, location: ParseLocation) { super(location, message); } } @Injectable() export class TemplateParser { constructor(private _exprParser: Parser, private _schemaRegistry: ElementSchemaRegistry, private _htmlParser: HtmlParser) {} parse(template: string, directives: CompileDirectiveMetadata[], templateUrl: string): TemplateAst[] { var parseVisitor = new TemplateParseVisitor(directives, this._exprParser, this._schemaRegistry); var htmlAstWithErrors = this._htmlParser.parse(template, templateUrl); var result = htmlVisitAll(parseVisitor, htmlAstWithErrors.rootNodes, EMPTY_COMPONENT); var errors: ParseError[] = htmlAstWithErrors.errors.concat(parseVisitor.errors); if (errors.length > 0) { var errorString = errors.join('\n'); throw new BaseException(`Template parse errors:\n${errorString}`); } return result; } } class TemplateParseVisitor implements HtmlAstVisitor { selectorMatcher: SelectorMatcher; errors: TemplateParseError[] = []; directivesIndex = new Map(); ngContentCount: number = 0; constructor(directives: CompileDirectiveMetadata[], private _exprParser: Parser, private _schemaRegistry: ElementSchemaRegistry) { this.selectorMatcher = new SelectorMatcher(); ListWrapper.forEachWithIndex(directives, (directive: CompileDirectiveMetadata, index: number) => { var selector = CssSelector.parse(directive.selector); this.selectorMatcher.addSelectables(selector, directive); this.directivesIndex.set(directive, index); }); } private _reportError(message: string, sourceSpan: ParseSourceSpan) { this.errors.push(new TemplateParseError(message, sourceSpan.start)); } private _parseInterpolation(value: string, sourceSpan: ParseSourceSpan): ASTWithSource { var sourceInfo = sourceSpan.start.toString(); try { return this._exprParser.parseInterpolation(value, sourceInfo); } catch (e) { this._reportError(`${e}`, sourceSpan); return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo); } } private _parseAction(value: string, sourceSpan: ParseSourceSpan): ASTWithSource { var sourceInfo = sourceSpan.start.toString(); try { return this._exprParser.parseAction(value, sourceInfo); } catch (e) { this._reportError(`${e}`, sourceSpan); return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo); } } private _parseBinding(value: string, sourceSpan: ParseSourceSpan): ASTWithSource { var sourceInfo = sourceSpan.start.toString(); try { return this._exprParser.parseBinding(value, sourceInfo); } catch (e) { this._reportError(`${e}`, sourceSpan); return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo); } } private _parseTemplateBindings(value: string, sourceSpan: ParseSourceSpan): TemplateBinding[] { var sourceInfo = sourceSpan.start.toString(); try { return this._exprParser.parseTemplateBindings(value, sourceInfo); } catch (e) { this._reportError(`${e}`, sourceSpan); return []; } } visitText(ast: HtmlTextAst, component: Component): any { var ngContentIndex = component.findNgContentIndex(TEXT_CSS_SELECTOR); var expr = this._parseInterpolation(ast.value, ast.sourceSpan); if (isPresent(expr)) { return new BoundTextAst(expr, ngContentIndex, ast.sourceSpan); } else { return new TextAst(ast.value, ngContentIndex, ast.sourceSpan); } } visitAttr(ast: HtmlAttrAst, contex: any): any { return new AttrAst(ast.name, ast.value, ast.sourceSpan); } visitElement(element: HtmlElementAst, component: Component): any { var nodeName = element.name; var preparsedElement = preparseElement(element); if (preparsedElement.type === PreparsedElementType.SCRIPT || preparsedElement.type === PreparsedElementType.STYLE) { // Skipping