2016-06-23 09:47:54 -07: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-02-17 12:55:55 -08:00
import { Inject , InjectionToken , Optional , SchemaMetadata , ɵ Console as Console } from '@angular/core' ;
2017-04-13 10:37:45 -07:00
import { CompileDirectiveMetadata , CompileDirectiveSummary , CompilePipeSummary , CompileTokenMetadata , CompileTypeMetadata , identifierName } from '../compile_metadata' ;
2017-05-18 13:46:51 -07:00
import { CompileReflector } from '../compile_reflector' ;
2017-02-17 08:56:36 -08:00
import { CompilerConfig } from '../config' ;
2017-02-09 14:59:57 -08:00
import { AST , ASTWithSource , EmptyExpr } from '../expression_parser/ast' ;
2016-07-21 11:41:25 -07:00
import { Parser } from '../expression_parser/parser' ;
2016-08-30 18:07:40 -07:00
import { I18NHtmlParser } from '../i18n/i18n_html_parser' ;
2017-05-18 13:46:51 -07:00
import { Identifiers , createTokenForExternalReference , createTokenForReference } from '../identifiers' ;
2016-12-15 09:12:40 -08:00
import { CompilerInjectable } from '../injectable' ;
2016-08-01 12:19:09 -07:00
import * as html from '../ml_parser/ast' ;
2016-08-11 21:00:35 -07:00
import { ParseTreeResult } from '../ml_parser/html_parser' ;
2016-08-01 12:19:09 -07:00
import { expandNodes } from '../ml_parser/icu_ast_expander' ;
import { InterpolationConfig } from '../ml_parser/interpolation_config' ;
2017-04-13 10:37:45 -07:00
import { isNgTemplate , splitNsName } from '../ml_parser/tags' ;
2016-08-02 15:53:34 -07:00
import { ParseError , ParseErrorLevel , ParseSourceSpan } from '../parse_util' ;
import { ProviderElementContext , ProviderViewContext } from '../provider_analyzer' ;
2016-07-21 11:41:25 -07:00
import { ElementSchemaRegistry } from '../schema/element_schema_registry' ;
2016-08-02 15:53:34 -07:00
import { CssSelector , SelectorMatcher } from '../selector' ;
2016-07-21 11:41:25 -07:00
import { isStyleUrlResolvable } from '../style_url_resolver' ;
2017-01-27 13:19:00 -08:00
import { syntaxError } from '../util' ;
2017-04-13 10:37:45 -07:00
2016-10-21 11:41:14 -07:00
import { BindingParser , BoundProperty } from './binding_parser' ;
2016-08-02 15:53:34 -07:00
import { AttrAst , BoundDirectivePropertyAst , BoundElementPropertyAst , BoundEventAst , BoundTextAst , DirectiveAst , ElementAst , EmbeddedTemplateAst , NgContentAst , PropertyBindingType , ReferenceAst , TemplateAst , TemplateAstVisitor , TextAst , VariableAst , templateVisitAll } from './template_ast' ;
import { PreparsedElementType , preparseElement } from './template_preparser' ;
2016-07-08 10:05:27 -07:00
const BIND_NAME_REGEXP =
2016-08-25 15:21:33 -07:00
/^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.+))|\[\(([^\)]+)\)\]|\[([^\]]+)\]|\(([^\)]+)\))$/ ;
2017-01-09 13:16:37 -08:00
// Group 1 = "bind-"
2016-08-25 15:21:33 -07:00
const KW_BIND_IDX = 1 ;
2017-01-09 13:16:37 -08:00
// Group 2 = "let-"
2016-08-25 15:21:33 -07:00
const KW_LET_IDX = 2 ;
2017-01-09 13:16:37 -08:00
// Group 3 = "ref-/#"
2016-08-25 15:21:33 -07:00
const KW_REF_IDX = 3 ;
2017-01-09 13:16:37 -08:00
// Group 4 = "on-"
2016-08-25 15:21:33 -07:00
const KW_ON_IDX = 4 ;
2017-01-09 13:16:37 -08:00
// Group 5 = "bindon-"
2016-08-25 15:21:33 -07:00
const KW_BINDON_IDX = 5 ;
2017-01-09 13:16:37 -08:00
// Group 6 = "@"
2016-08-25 15:21:33 -07:00
const KW_AT_IDX = 6 ;
2017-01-09 13:16:37 -08:00
// Group 7 = the identifier after "bind-", "let-", "ref-/#", "on-", "bindon-" or "@"
2016-08-25 15:21:33 -07:00
const IDENT_KW_IDX = 7 ;
2017-01-09 13:16:37 -08:00
// Group 8 = identifier inside [()]
2016-08-25 15:21:33 -07:00
const IDENT_BANANA_BOX_IDX = 8 ;
2017-01-09 13:16:37 -08:00
// Group 9 = identifier inside []
2016-08-25 15:21:33 -07:00
const IDENT_PROPERTY_IDX = 9 ;
2017-01-09 13:16:37 -08:00
// Group 10 = identifier inside ()
2016-08-25 15:21:33 -07:00
const IDENT_EVENT_IDX = 10 ;
2015-08-25 15:36:02 -07:00
2017-01-09 13:16:46 -08:00
// deprecated in 4.x
2015-08-25 15:36:02 -07:00
const TEMPLATE_ELEMENT = 'template' ;
2017-01-09 13:16:46 -08:00
// deprecated in 4.x
2015-08-25 15:36:02 -07:00
const TEMPLATE_ATTR = 'template' ;
const TEMPLATE_ATTR_PREFIX = '*' ;
const CLASS_ATTR = 'class' ;
2016-07-08 10:05:27 -07:00
const TEXT_CSS_SELECTOR = CssSelector . parse ( '*' ) [ 0 ] ;
2015-09-11 13:37:05 -07:00
2017-03-22 21:26:53 -07:00
const TEMPLATE_ELEMENT_DEPRECATION_WARNING =
'The <template> element is deprecated. Use <ng-template> instead' ;
const TEMPLATE_ATTR_DEPRECATION_WARNING =
'The template attribute is deprecated. Use an ng-template element instead.' ;
let warningCounts : { [ warning : string ] : number } = { } ;
function warnOnlyOnce ( warnings : string [ ] ) : ( warning : ParseError ) = > boolean {
return ( error : ParseError ) = > {
if ( warnings . indexOf ( error . msg ) !== - 1 ) {
warningCounts [ error . msg ] = ( warningCounts [ error . msg ] || 0 ) + 1 ;
return warningCounts [ error . msg ] <= 1 ;
}
return true ;
} ;
}
2015-12-03 15:49:09 -08:00
/ * *
* Provides an array of { @link TemplateAstVisitor } s which will be used to transform
* parsed templates before compilation is invoked , allowing custom expression syntax
* and other advanced transformations .
*
* This is currently an internal - only feature and not meant for general use .
* /
2017-01-03 16:54:46 -08:00
export const TEMPLATE_TRANSFORMS = new InjectionToken ( 'TemplateTransforms' ) ;
2015-11-19 10:51:16 -08:00
2015-10-07 09:34:21 -07:00
export class TemplateParseError extends ParseError {
2016-04-25 19:52:24 -07:00
constructor ( message : string , span : ParseSourceSpan , level : ParseErrorLevel ) {
super ( span , message , level ) ;
}
2015-10-07 09:34:21 -07:00
}
2016-03-31 12:14:08 -07:00
export class TemplateParseResult {
2017-02-09 14:59:57 -08:00
constructor (
public templateAst? : TemplateAst [ ] , public usedPipes? : CompilePipeSummary [ ] ,
public errors? : ParseError [ ] ) { }
2016-03-31 12:14:08 -07:00
}
2016-12-15 09:12:40 -08:00
@CompilerInjectable ( )
2015-08-25 15:36:02 -07:00
export class TemplateParser {
2016-06-08 16:38:52 -07:00
constructor (
2017-05-18 13:46:51 -07:00
private _config : CompilerConfig , private _reflector : CompileReflector ,
private _exprParser : Parser , private _schemaRegistry : ElementSchemaRegistry ,
private _htmlParser : I18NHtmlParser , private _console : Console ,
2016-06-08 16:38:52 -07:00
@Optional ( ) @Inject ( TEMPLATE_TRANSFORMS ) public transforms : TemplateAstVisitor [ ] ) { }
parse (
2016-11-10 16:27:53 -08:00
component : CompileDirectiveMetadata , template : string , directives : CompileDirectiveSummary [ ] ,
2017-02-09 14:59:57 -08:00
pipes : CompilePipeSummary [ ] , schemas : SchemaMetadata [ ] ,
templateUrl : string ) : { template : TemplateAst [ ] , pipes : CompilePipeSummary [ ] } {
2016-07-25 03:02:57 -07:00
const result = this . tryParse ( component , template , directives , pipes , schemas , templateUrl ) ;
2017-03-22 21:26:53 -07:00
const warnings =
2017-03-24 09:59:58 -07:00
result . errors ! . filter ( error = > error . level === ParseErrorLevel . WARNING )
. filter ( warnOnlyOnce (
[ TEMPLATE_ATTR_DEPRECATION_WARNING , TEMPLATE_ELEMENT_DEPRECATION_WARNING ] ) ) ;
2017-03-22 21:26:53 -07:00
2017-03-24 09:59:58 -07:00
const errors = result . errors ! . filter ( error = > error . level === ParseErrorLevel . ERROR ) ;
2016-08-23 10:52:40 -07:00
2016-04-25 19:52:24 -07:00
if ( warnings . length > 0 ) {
this . _console . warn ( ` Template parse warnings: \ n ${ warnings . join ( '\n' ) } ` ) ;
}
2016-08-23 10:52:40 -07:00
2016-04-25 19:52:24 -07:00
if ( errors . length > 0 ) {
2016-07-08 10:05:27 -07:00
const errorString = errors . join ( '\n' ) ;
2017-01-27 13:19:00 -08:00
throw syntaxError ( ` Template parse errors: \ n ${ errorString } ` ) ;
2016-03-31 12:14:08 -07:00
}
2016-06-22 17:25:42 -07:00
2017-03-24 09:59:58 -07:00
return { template : result.templateAst ! , pipes : result.usedPipes ! } ;
2016-03-31 12:14:08 -07:00
}
2016-06-08 16:38:52 -07:00
tryParse (
2016-11-10 16:27:53 -08:00
component : CompileDirectiveMetadata , template : string , directives : CompileDirectiveSummary [ ] ,
pipes : CompilePipeSummary [ ] , schemas : SchemaMetadata [ ] ,
2016-07-25 03:02:57 -07:00
templateUrl : string ) : TemplateParseResult {
2016-10-11 15:45:27 -07:00
return this . tryParseHtml (
2017-03-24 09:59:58 -07:00
this . expandHtml ( this . _htmlParser ! . parse (
2016-10-11 15:45:27 -07:00
template , templateUrl , true , this . getInterpolationConfig ( component ) ) ) ,
2017-04-13 11:56:00 -07:00
component , directives , pipes , schemas ) ;
2016-10-11 15:45:27 -07:00
}
2016-06-30 14:59:23 -07:00
2016-10-11 15:45:27 -07:00
tryParseHtml (
2017-04-13 11:56:00 -07:00
htmlAstWithErrors : ParseTreeResult , component : CompileDirectiveMetadata ,
directives : CompileDirectiveSummary [ ] , pipes : CompilePipeSummary [ ] ,
schemas : SchemaMetadata [ ] ) : TemplateParseResult {
2016-11-12 14:08:58 +01:00
let result : TemplateAst [ ] ;
const errors = htmlAstWithErrors . errors ;
2017-02-09 14:59:57 -08:00
const usedPipes : CompilePipeSummary [ ] = [ ] ;
2016-01-06 14:13:44 -08:00
if ( htmlAstWithErrors . rootNodes . length > 0 ) {
2016-11-10 16:27:53 -08:00
const uniqDirectives = removeSummaryDuplicates ( directives ) ;
const uniqPipes = removeSummaryDuplicates ( pipes ) ;
2017-05-18 13:46:51 -07:00
const providerViewContext = new ProviderViewContext ( this . _reflector , component ) ;
2017-03-24 09:59:58 -07:00
let interpolationConfig : InterpolationConfig = undefined ! ;
2016-10-21 11:41:14 -07:00
if ( component . template && component . template . interpolation ) {
interpolationConfig = {
start : component.template.interpolation [ 0 ] ,
end : component.template.interpolation [ 1 ]
} ;
}
2016-10-24 11:11:31 -07:00
const bindingParser = new BindingParser (
2017-03-24 09:59:58 -07:00
this . _exprParser , interpolationConfig ! , this . _schemaRegistry , uniqPipes , errors ) ;
2016-07-08 10:05:27 -07:00
const parseVisitor = new TemplateParseVisitor (
2017-05-18 13:46:51 -07:00
this . _reflector , this . _config , providerViewContext , uniqDirectives , bindingParser ,
this . _schemaRegistry , schemas , errors ) ;
2016-07-21 13:56:58 -07:00
result = html . visitAll ( parseVisitor , htmlAstWithErrors . rootNodes , EMPTY_ELEMENT_CONTEXT ) ;
2016-10-24 11:11:31 -07:00
errors . push ( . . . providerViewContext . errors ) ;
2017-02-09 14:59:57 -08:00
usedPipes . push ( . . . bindingParser . getUsedPipes ( ) ) ;
2016-01-06 14:13:44 -08:00
} else {
result = [ ] ;
}
2016-05-26 23:04:17 +03:00
this . _assertNoReferenceDuplicationOnTemplate ( result , errors ) ;
2015-10-07 09:34:21 -07:00
if ( errors . length > 0 ) {
2017-02-09 14:59:57 -08:00
return new TemplateParseResult ( result , usedPipes , errors ) ;
2015-08-27 16:29:02 -07:00
}
2016-06-30 14:59:23 -07:00
2016-12-27 15:23:49 -08:00
if ( this . transforms ) {
2015-11-19 10:51:16 -08:00
this . transforms . forEach (
( transform : TemplateAstVisitor ) = > { result = templateVisitAll ( transform , result ) ; } ) ;
}
2016-06-30 14:59:23 -07:00
2017-02-09 14:59:57 -08:00
return new TemplateParseResult ( result , usedPipes , errors ) ;
2015-08-25 15:36:02 -07:00
}
2016-05-26 23:04:17 +03:00
2016-10-11 15:45:27 -07:00
expandHtml ( htmlAstWithErrors : ParseTreeResult , forced : boolean = false ) : ParseTreeResult {
const errors : ParseError [ ] = htmlAstWithErrors . errors ;
if ( errors . length == 0 || forced ) {
// Transform ICU messages to angular directives
const expandedHtmlAst = expandNodes ( htmlAstWithErrors . rootNodes ) ;
errors . push ( . . . expandedHtmlAst . errors ) ;
htmlAstWithErrors = new ParseTreeResult ( expandedHtmlAst . nodes , errors ) ;
}
return htmlAstWithErrors ;
}
2017-03-24 09:59:58 -07:00
getInterpolationConfig ( component : CompileDirectiveMetadata ) : InterpolationConfig | undefined {
2016-10-11 15:45:27 -07:00
if ( component . template ) {
return InterpolationConfig . fromArray ( component . template . interpolation ) ;
}
2017-03-24 09:59:58 -07:00
return undefined ;
2016-10-11 15:45:27 -07:00
}
2016-05-27 09:16:46 -07:00
/** @internal */
2016-06-22 17:25:42 -07:00
_assertNoReferenceDuplicationOnTemplate ( result : TemplateAst [ ] , errors : TemplateParseError [ ] ) :
void {
const existingReferences : string [ ] = [ ] ;
result . filter ( element = > ! ! ( < any > element ) . references )
. forEach ( element = > ( < any > element ) . references . forEach ( ( reference : ReferenceAst ) = > {
2016-06-08 16:38:52 -07:00
const name = reference . name ;
if ( existingReferences . indexOf ( name ) < 0 ) {
existingReferences . push ( name ) ;
} else {
const error = new TemplateParseError (
` Reference "# ${ name } " is defined several times ` , reference . sourceSpan ,
2017-03-14 17:12:44 -07:00
ParseErrorLevel . ERROR ) ;
2016-06-08 16:38:52 -07:00
errors . push ( error ) ;
}
} ) ) ;
2016-05-26 23:04:17 +03:00
}
2015-08-25 15:36:02 -07:00
}
2016-10-24 11:11:31 -07:00
class TemplateParseVisitor implements html . Visitor {
2016-08-23 10:52:40 -07:00
selectorMatcher = new SelectorMatcher ( ) ;
2016-11-10 16:27:53 -08:00
directivesIndex = new Map < CompileDirectiveSummary , number > ( ) ;
2017-02-15 08:36:49 -08:00
ngContentCount = 0 ;
contentQueryStartId : number ;
2015-10-07 17:15:12 -07:00
2016-06-08 16:38:52 -07:00
constructor (
2017-05-18 13:46:51 -07:00
private reflector : CompileReflector , private config : CompilerConfig ,
public providerViewContext : ProviderViewContext , directives : CompileDirectiveSummary [ ] ,
private _bindingParser : BindingParser , private _schemaRegistry : ElementSchemaRegistry ,
private _schemas : SchemaMetadata [ ] , private _targetErrors : TemplateParseError [ ] ) {
2017-02-15 08:36:49 -08:00
// Note: queries start with id 1 so we can use the number in a Bloom filter!
this . contentQueryStartId = providerViewContext . component . viewQueries . length + 1 ;
2016-11-10 16:27:53 -08:00
directives . forEach ( ( directive , index ) = > {
2017-03-24 09:59:58 -07:00
const selector = CssSelector . parse ( directive . selector ! ) ;
2016-08-23 10:52:40 -07:00
this . selectorMatcher . addSelectables ( selector , directive ) ;
this . directivesIndex . set ( directive , index ) ;
} ) ;
2015-12-02 10:35:51 -08:00
}
2016-07-21 13:56:58 -07:00
visitExpansion ( expansion : html.Expansion , context : any ) : any { return null ; }
2016-04-13 16:01:25 -07:00
2016-07-21 13:56:58 -07:00
visitExpansionCase ( expansionCase : html.ExpansionCase , context : any ) : any { return null ; }
2016-04-13 16:01:25 -07:00
2016-07-21 13:56:58 -07:00
visitText ( text : html.Text , parent : ElementContext ) : any {
2017-03-24 09:59:58 -07:00
const ngContentIndex = parent . findNgContentIndex ( TEXT_CSS_SELECTOR ) ! ;
const expr = this . _bindingParser . parseInterpolation ( text . value , text . sourceSpan ! ) ;
return expr ? new BoundTextAst ( expr , ngContentIndex , text . sourceSpan ! ) :
new TextAst ( text . value , ngContentIndex , text . sourceSpan ! ) ;
2015-08-25 15:36:02 -07:00
}
2016-08-23 10:52:40 -07:00
visitAttribute ( attribute : html.Attribute , context : any ) : any {
2016-07-21 13:56:58 -07:00
return new AttrAst ( attribute . name , attribute . value , attribute . sourceSpan ) ;
2015-09-11 13:37:05 -07:00
}
2015-08-25 15:36:02 -07:00
2016-07-21 13:56:58 -07:00
visitComment ( comment : html.Comment , context : any ) : any { return null ; }
2016-03-06 20:21:20 -08:00
2016-07-21 13:56:58 -07:00
visitElement ( element : html.Element , parent : ElementContext ) : any {
2017-02-15 08:36:49 -08:00
const queryStartIndex = this . contentQueryStartId ;
2016-07-08 10:05:27 -07:00
const nodeName = element . name ;
const preparsedElement = preparseElement ( element ) ;
2015-09-18 10:33:23 -07:00
if ( preparsedElement . type === PreparsedElementType . SCRIPT ||
2015-10-14 09:39:40 -07:00
preparsedElement . type === PreparsedElementType . STYLE ) {
2015-09-18 10:33:23 -07:00
// Skipping <script> for security reasons
2015-10-14 09:39:40 -07:00
// Skipping <style> as we already processed them
// in the StyleCompiler
return null ;
}
if ( preparsedElement . type === PreparsedElementType . STYLESHEET &&
isStyleUrlResolvable ( preparsedElement . hrefAttr ) ) {
// Skipping stylesheets with either relative urls or package scheme as we already processed
2015-10-07 09:34:21 -07:00
// them in the StyleCompiler
2015-09-18 10:33:23 -07:00
return null ;
}
2016-12-27 15:23:49 -08:00
const matchableAttrs : [ string , string ] [ ] = [ ] ;
2016-10-21 11:41:14 -07:00
const elementOrDirectiveProps : BoundProperty [ ] = [ ] ;
2016-07-08 10:05:27 -07:00
const elementOrDirectiveRefs : ElementOrDirectiveRef [ ] = [ ] ;
const elementVars : VariableAst [ ] = [ ] ;
const events : BoundEventAst [ ] = [ ] ;
2015-08-25 15:36:02 -07:00
2016-10-21 11:41:14 -07:00
const templateElementOrDirectiveProps : BoundProperty [ ] = [ ] ;
2016-12-27 15:23:49 -08:00
const templateMatchableAttrs : [ string , string ] [ ] = [ ] ;
2016-07-08 10:05:27 -07:00
const templateElementVars : VariableAst [ ] = [ ] ;
2016-04-25 19:52:24 -07:00
2016-07-08 10:05:27 -07:00
let hasInlineTemplates = false ;
const attrs : AttrAst [ ] = [ ] ;
2017-01-09 13:16:46 -08:00
const isTemplateElement = isTemplate (
2017-01-10 19:07:03 -08:00
element , this . config . enableLegacyTemplate ,
2017-01-09 13:16:46 -08:00
( m : string , span : ParseSourceSpan ) = > this . _reportError ( m , span , ParseErrorLevel . WARNING ) ) ;
2015-10-07 09:34:21 -07:00
2015-08-25 15:36:02 -07:00
element . attrs . forEach ( attr = > {
2016-07-08 10:05:27 -07:00
const hasBinding = this . _parseAttr (
2016-10-21 11:41:14 -07:00
isTemplateElement , attr , matchableAttrs , elementOrDirectiveProps , events ,
2016-06-08 16:38:52 -07:00
elementOrDirectiveRefs , elementVars ) ;
2016-08-23 10:52:40 -07:00
2016-12-27 15:23:49 -08:00
let templateBindingsSource : string | undefined ;
let prefixToken : string | undefined ;
let normalizedName = this . _normalizeAttributeName ( attr . name ) ;
2017-01-10 19:07:03 -08:00
if ( this . config . enableLegacyTemplate && normalizedName == TEMPLATE_ATTR ) {
2017-01-09 13:16:46 -08:00
this . _reportError (
2017-03-22 21:26:53 -07:00
TEMPLATE_ATTR_DEPRECATION_WARNING , attr . sourceSpan , ParseErrorLevel . WARNING ) ;
2016-10-21 11:41:14 -07:00
templateBindingsSource = attr . value ;
2016-12-27 15:23:49 -08:00
} else if ( normalizedName . startsWith ( TEMPLATE_ATTR_PREFIX ) ) {
2016-11-10 13:15:09 -08:00
templateBindingsSource = attr . value ;
2017-01-05 17:25:41 -08:00
prefixToken = normalizedName . substring ( TEMPLATE_ATTR_PREFIX . length ) + ':' ;
2016-10-21 11:41:14 -07:00
}
2016-12-27 15:23:49 -08:00
2017-03-02 09:37:01 -08:00
const hasTemplateBinding = templateBindingsSource != null ;
2016-10-21 11:41:14 -07:00
if ( hasTemplateBinding ) {
if ( hasInlineTemplates ) {
2016-10-24 11:11:31 -07:00
this . _reportError (
2016-10-21 11:41:14 -07:00
` Can't have multiple template bindings on one element. Use only one attribute named 'template' or prefixed with * ` ,
attr . sourceSpan ) ;
}
hasInlineTemplates = true ;
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseInlineTemplateBinding (
2017-03-24 09:59:58 -07:00
prefixToken ! , templateBindingsSource ! , attr . sourceSpan , templateMatchableAttrs ,
2016-10-21 11:41:14 -07:00
templateElementOrDirectiveProps , templateElementVars ) ;
2016-06-22 17:43:27 +02:00
}
2015-08-25 15:36:02 -07:00
if ( ! hasBinding && ! hasTemplateBinding ) {
// don't include the bindings as attributes as well in the AST
2016-07-21 13:56:58 -07:00
attrs . push ( this . visitAttribute ( attr , null ) ) ;
2015-12-15 09:39:07 -08:00
matchableAttrs . push ( [ attr . name , attr . value ] ) ;
2015-08-25 15:36:02 -07:00
}
} ) ;
2015-10-07 09:34:21 -07:00
2016-07-08 10:05:27 -07:00
const elementCssSelector = createElementCssSelector ( nodeName , matchableAttrs ) ;
2016-08-23 10:52:40 -07:00
const { directives : directiveMetas , matchElement } =
this . _parseDirectives ( this . selectorMatcher , elementCssSelector ) ;
2016-07-08 10:05:27 -07:00
const references : ReferenceAst [ ] = [ ] ;
2017-02-09 14:59:57 -08:00
const boundDirectivePropNames = new Set < string > ( ) ;
2016-07-08 10:05:27 -07:00
const directiveAsts = this . _createDirectiveAsts (
2016-06-08 16:38:52 -07:00
isTemplateElement , element . name , directiveMetas , elementOrDirectiveProps ,
2017-03-24 09:59:58 -07:00
elementOrDirectiveRefs , element . sourceSpan ! , references , boundDirectivePropNames ) ;
2017-02-09 14:59:57 -08:00
const elementProps : BoundElementPropertyAst [ ] = this . _createElementPropertyAsts (
element . name , elementOrDirectiveProps , boundDirectivePropNames ) ;
2016-07-08 10:05:27 -07:00
const isViewRoot = parent . isTemplateElement || hasInlineTemplates ;
2017-01-04 13:59:43 -08:00
2016-07-08 10:05:27 -07:00
const providerContext = new ProviderElementContext (
2017-03-24 09:59:58 -07:00
this . providerViewContext , parent . providerContext ! , isViewRoot , directiveAsts , attrs ,
references , isTemplateElement , queryStartIndex , element . sourceSpan ! ) ;
2017-01-04 13:59:43 -08:00
2017-02-15 08:36:49 -08:00
const children : TemplateAst [ ] = html . visitAll (
2016-01-06 14:13:44 -08:00
preparsedElement . nonBindable ? NON_BINDABLE_VISITOR : this , element . children ,
2016-06-08 16:38:52 -07:00
ElementContext . create (
isTemplateElement , directiveAsts ,
2017-03-24 09:59:58 -07:00
isTemplateElement ? parent . providerContext ! : providerContext ) ) ;
2016-01-06 14:13:44 -08:00
providerContext . afterElement ( ) ;
2016-03-23 14:15:05 -07:00
// Override the actual selector when the `ngProjectAs` attribute is provided
2017-03-02 09:37:01 -08:00
const projectionSelector = preparsedElement . projectAs != null ?
2016-06-08 16:38:52 -07:00
CssSelector . parse ( preparsedElement . projectAs ) [ 0 ] :
elementCssSelector ;
2017-03-24 09:59:58 -07:00
const ngContentIndex = parent . findNgContentIndex ( projectionSelector ) ! ;
2016-07-08 10:05:27 -07:00
let parsedElement : TemplateAst ;
2016-03-23 14:15:05 -07:00
2015-09-18 10:33:23 -07:00
if ( preparsedElement . type === PreparsedElementType . NG_CONTENT ) {
2016-10-12 07:58:06 -07:00
if ( element . children && ! element . children . every ( _isEmptyTextNode ) ) {
2017-03-24 09:59:58 -07:00
this . _reportError ( ` <ng-content> element cannot have content. ` , element . sourceSpan ! ) ;
2015-12-03 14:20:00 -08:00
}
2016-03-23 14:15:05 -07:00
parsedElement = new NgContentAst (
2017-03-24 09:59:58 -07:00
this . ngContentCount ++ , hasInlineTemplates ? null ! : ngContentIndex ,
element . sourceSpan ! ) ;
2015-09-18 10:33:23 -07:00
} else if ( isTemplateElement ) {
2016-01-06 14:13:44 -08:00
this . _assertAllEventsPublishedByDirectives ( directiveAsts , events ) ;
2016-06-08 16:38:52 -07:00
this . _assertNoComponentsNorElementBindingsOnTemplate (
2017-03-24 09:59:58 -07:00
directiveAsts , elementProps , element . sourceSpan ! ) ;
2016-03-23 14:15:05 -07:00
2016-04-18 13:24:42 -07:00
parsedElement = new EmbeddedTemplateAst (
2016-04-25 19:52:24 -07:00
attrs , events , references , elementVars , providerContext . transformedDirectiveAsts ,
2017-02-02 15:01:35 -08:00
providerContext . transformProviders , providerContext . transformedHasViewContainer ,
2017-03-24 09:59:58 -07:00
providerContext . queryMatches , children , hasInlineTemplates ? null ! : ngContentIndex ,
element . sourceSpan ! ) ;
2015-08-25 15:36:02 -07:00
} else {
2016-08-23 10:52:40 -07:00
this . _assertElementExists ( matchElement , element ) ;
2017-03-24 09:59:58 -07:00
this . _assertOnlyOneComponent ( directiveAsts , element . sourceSpan ! ) ;
2016-08-23 10:52:40 -07:00
2016-07-08 10:05:27 -07:00
const ngContentIndex =
2016-01-06 14:13:44 -08:00
hasInlineTemplates ? null : parent . findNgContentIndex ( projectionSelector ) ;
parsedElement = new ElementAst (
2016-04-25 19:52:24 -07:00
nodeName , attrs , elementProps , events , references ,
2016-04-18 13:24:42 -07:00
providerContext . transformedDirectiveAsts , providerContext . transformProviders ,
2017-02-02 15:01:35 -08:00
providerContext . transformedHasViewContainer , providerContext . queryMatches , children ,
2017-03-24 09:57:32 -07:00
hasInlineTemplates ? null : ngContentIndex , element . sourceSpan ,
2017-03-24 09:59:58 -07:00
element . endSourceSpan || null ) ;
2015-08-25 15:36:02 -07:00
}
2016-08-23 10:52:40 -07:00
2015-08-25 15:36:02 -07:00
if ( hasInlineTemplates ) {
2017-02-15 08:36:49 -08:00
const templateQueryStartIndex = this . contentQueryStartId ;
2017-01-09 13:16:46 -08:00
const templateSelector = createElementCssSelector ( TEMPLATE_ELEMENT , templateMatchableAttrs ) ;
2016-08-23 10:52:40 -07:00
const { directives : templateDirectiveMetas } =
2017-01-09 13:16:46 -08:00
this . _parseDirectives ( this . selectorMatcher , templateSelector ) ;
2017-02-09 14:59:57 -08:00
const templateBoundDirectivePropNames = new Set < string > ( ) ;
2016-07-08 10:05:27 -07:00
const templateDirectiveAsts = this . _createDirectiveAsts (
2016-06-08 16:38:52 -07:00
true , element . name , templateDirectiveMetas , templateElementOrDirectiveProps , [ ] ,
2017-03-24 09:59:58 -07:00
element . sourceSpan ! , [ ] , templateBoundDirectivePropNames ) ;
2016-07-08 10:05:27 -07:00
const templateElementProps : BoundElementPropertyAst [ ] = this . _createElementPropertyAsts (
2017-02-09 14:59:57 -08:00
element . name , templateElementOrDirectiveProps , templateBoundDirectivePropNames ) ;
2016-01-06 14:13:44 -08:00
this . _assertNoComponentsNorElementBindingsOnTemplate (
2017-03-24 09:59:58 -07:00
templateDirectiveAsts , templateElementProps , element . sourceSpan ! ) ;
2016-07-08 10:05:27 -07:00
const templateProviderContext = new ProviderElementContext (
2017-03-24 09:59:58 -07:00
this . providerViewContext , parent . providerContext ! , parent . isTemplateElement ,
templateDirectiveAsts , [ ] , [ ] , true , templateQueryStartIndex , element . sourceSpan ! ) ;
2016-01-06 14:13:44 -08:00
templateProviderContext . afterElement ( ) ;
2016-06-08 16:38:52 -07:00
parsedElement = new EmbeddedTemplateAst (
[ ] , [ ] , [ ] , templateElementVars , templateProviderContext . transformedDirectiveAsts ,
templateProviderContext . transformProviders ,
2017-02-02 15:01:35 -08:00
templateProviderContext . transformedHasViewContainer , templateProviderContext . queryMatches ,
2017-03-24 09:59:58 -07:00
[ parsedElement ] , ngContentIndex , element . sourceSpan ! ) ;
2015-08-25 15:36:02 -07:00
}
2016-09-23 16:37:04 -04:00
2015-08-25 15:36:02 -07:00
return parsedElement ;
}
2016-06-08 16:38:52 -07:00
private _parseAttr (
2016-07-21 13:56:58 -07:00
isTemplateElement : boolean , attr : html.Attribute , targetMatchableAttrs : string [ ] [ ] ,
2016-10-21 11:41:14 -07:00
targetProps : BoundProperty [ ] , targetEvents : BoundEventAst [ ] ,
2016-06-08 16:38:52 -07:00
targetRefs : ElementOrDirectiveRef [ ] , targetVars : VariableAst [ ] ) : boolean {
2016-08-25 15:21:33 -07:00
const name = this . _normalizeAttributeName ( attr . name ) ;
const value = attr . value ;
const srcSpan = attr . sourceSpan ;
const bindParts = name . match ( BIND_NAME_REGEXP ) ;
2016-07-08 10:05:27 -07:00
let hasBinding = false ;
2016-08-25 15:21:33 -07:00
2016-08-05 09:50:49 -07:00
if ( bindParts !== null ) {
2015-08-25 15:36:02 -07:00
hasBinding = true ;
2017-03-02 09:37:01 -08:00
if ( bindParts [ KW_BIND_IDX ] != null ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parsePropertyBinding (
2016-10-21 11:41:14 -07:00
bindParts [ IDENT_KW_IDX ] , value , false , srcSpan , targetMatchableAttrs , targetProps ) ;
2015-08-25 15:36:02 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ KW_LET_IDX ] ) {
2016-04-25 19:52:24 -07:00
if ( isTemplateElement ) {
2016-08-25 15:21:33 -07:00
const identifier = bindParts [ IDENT_KW_IDX ] ;
this . _parseVariable ( identifier , value , srcSpan , targetVars ) ;
2016-04-25 19:52:24 -07:00
} else {
2016-10-24 11:11:31 -07:00
this . _reportError ( ` "let-" is only supported on template elements. ` , srcSpan ) ;
2016-04-25 19:52:24 -07:00
}
2015-08-25 15:36:02 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ KW_REF_IDX ] ) {
const identifier = bindParts [ IDENT_KW_IDX ] ;
this . _parseReference ( identifier , value , srcSpan , targetRefs ) ;
2016-04-25 19:52:24 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ KW_ON_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseEvent (
2016-08-25 15:21:33 -07:00
bindParts [ IDENT_KW_IDX ] , value , srcSpan , targetMatchableAttrs , targetEvents ) ;
2015-08-25 15:36:02 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ KW_BINDON_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parsePropertyBinding (
2016-10-21 11:41:14 -07:00
bindParts [ IDENT_KW_IDX ] , value , false , srcSpan , targetMatchableAttrs , targetProps ) ;
2016-06-08 16:38:52 -07:00
this . _parseAssignmentEvent (
2016-08-25 15:21:33 -07:00
bindParts [ IDENT_KW_IDX ] , value , srcSpan , targetMatchableAttrs , targetEvents ) ;
2015-08-25 15:36:02 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ KW_AT_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseLiteralAttr (
name , value , srcSpan , targetMatchableAttrs , targetProps ) ;
2016-10-21 11:41:14 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ IDENT_BANANA_BOX_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parsePropertyBinding (
2016-10-21 11:41:14 -07:00
bindParts [ IDENT_BANANA_BOX_IDX ] , value , false , srcSpan , targetMatchableAttrs ,
targetProps ) ;
2016-06-08 16:38:52 -07:00
this . _parseAssignmentEvent (
2016-08-25 15:21:33 -07:00
bindParts [ IDENT_BANANA_BOX_IDX ] , value , srcSpan , targetMatchableAttrs , targetEvents ) ;
2016-05-25 12:46:22 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ IDENT_PROPERTY_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parsePropertyBinding (
2016-10-21 11:41:14 -07:00
bindParts [ IDENT_PROPERTY_IDX ] , value , false , srcSpan , targetMatchableAttrs ,
targetProps ) ;
2015-08-25 15:36:02 -07:00
2016-08-25 15:21:33 -07:00
} else if ( bindParts [ IDENT_EVENT_IDX ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseEvent (
2016-08-25 15:21:33 -07:00
bindParts [ IDENT_EVENT_IDX ] , value , srcSpan , targetMatchableAttrs , targetEvents ) ;
2015-08-25 15:36:02 -07:00
}
} else {
2016-10-24 11:11:31 -07:00
hasBinding = this . _bindingParser . parsePropertyInterpolation (
name , value , srcSpan , targetMatchableAttrs , targetProps ) ;
2015-08-27 16:29:02 -07:00
}
2016-08-25 15:21:33 -07:00
2015-08-27 16:29:02 -07:00
if ( ! hasBinding ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseLiteralAttr ( name , value , srcSpan , targetMatchableAttrs , targetProps ) ;
2015-08-25 15:36:02 -07:00
}
2016-08-25 15:21:33 -07:00
2015-08-25 15:36:02 -07:00
return hasBinding ;
}
private _normalizeAttributeName ( attrName : string ) : string {
2016-08-25 15:21:33 -07:00
return /^data-/i . test ( attrName ) ? attrName . substring ( 5 ) : attrName ;
2015-08-25 15:36:02 -07:00
}
2016-06-08 16:38:52 -07:00
private _parseVariable (
identifier : string , value : string , sourceSpan : ParseSourceSpan , targetVars : VariableAst [ ] ) {
2015-11-23 16:02:19 -08:00
if ( identifier . indexOf ( '-' ) > - 1 ) {
2016-10-24 11:11:31 -07:00
this . _reportError ( ` "-" is not allowed in variable names ` , sourceSpan ) ;
2015-11-23 16:02:19 -08:00
}
2016-05-26 23:04:17 +03:00
2015-11-23 16:02:19 -08:00
targetVars . push ( new VariableAst ( identifier , value , sourceSpan ) ) ;
2015-08-25 15:36:02 -07:00
}
2016-06-08 16:38:52 -07:00
private _parseReference (
identifier : string , value : string , sourceSpan : ParseSourceSpan ,
targetRefs : ElementOrDirectiveRef [ ] ) {
2016-04-25 19:52:24 -07:00
if ( identifier . indexOf ( '-' ) > - 1 ) {
2016-10-24 11:11:31 -07:00
this . _reportError ( ` "-" is not allowed in reference names ` , sourceSpan ) ;
2016-04-25 19:52:24 -07:00
}
2016-05-26 23:04:17 +03:00
2016-04-25 19:52:24 -07:00
targetRefs . push ( new ElementOrDirectiveRef ( identifier , value , sourceSpan ) ) ;
}
2016-06-08 16:38:52 -07:00
private _parseAssignmentEvent (
name : string , expression : string , sourceSpan : ParseSourceSpan ,
targetMatchableAttrs : string [ ] [ ] , targetEvents : BoundEventAst [ ] ) {
2016-10-24 11:11:31 -07:00
this . _bindingParser . parseEvent (
2016-06-08 16:38:52 -07:00
` ${ name } Change ` , ` ${ expression } = $ event ` , sourceSpan , targetMatchableAttrs , targetEvents ) ;
2015-08-25 15:36:02 -07:00
}
2016-06-08 16:38:52 -07:00
private _parseDirectives ( selectorMatcher : SelectorMatcher , elementCssSelector : CssSelector ) :
2016-11-10 16:27:53 -08:00
{ directives : CompileDirectiveSummary [ ] , matchElement : boolean } {
2015-08-25 15:36:02 -07:00
// Need to sort the directives so that we get consistent results throughout,
// as selectorMatcher uses Maps inside.
2016-08-23 10:52:40 -07:00
// Also deduplicate directives as they might match more than one time!
const directives = new Array ( this . directivesIndex . size ) ;
// Whether any directive selector matches on the element name
let matchElement = false ;
2016-04-25 19:52:24 -07:00
selectorMatcher . match ( elementCssSelector , ( selector , directive ) = > {
2017-03-24 09:59:58 -07:00
directives [ this . directivesIndex . get ( directive ) ! ] = directive ;
2016-08-23 10:52:40 -07:00
matchElement = matchElement || selector . hasElementSelector ( ) ;
2016-04-25 19:52:24 -07:00
} ) ;
2016-08-23 10:52:40 -07:00
return {
directives : directives.filter ( dir = > ! ! dir ) ,
matchElement ,
} ;
2016-04-25 19:52:24 -07:00
}
2016-06-08 16:38:52 -07:00
private _createDirectiveAsts (
2016-11-10 16:27:53 -08:00
isTemplateElement : boolean , elementName : string , directives : CompileDirectiveSummary [ ] ,
2016-10-21 11:41:14 -07:00
props : BoundProperty [ ] , elementOrDirectiveRefs : ElementOrDirectiveRef [ ] ,
2017-02-09 14:59:57 -08:00
elementSourceSpan : ParseSourceSpan , targetReferences : ReferenceAst [ ] ,
targetBoundDirectivePropNames : Set < string > ) : DirectiveAst [ ] {
2016-07-08 10:05:27 -07:00
const matchedReferences = new Set < string > ( ) ;
2017-03-24 09:59:58 -07:00
let component : CompileDirectiveSummary = null ! ;
2016-12-27 15:23:49 -08:00
2016-11-10 16:27:53 -08:00
const directiveAsts = directives . map ( ( directive ) = > {
2016-07-28 02:27:07 -07:00
const sourceSpan = new ParseSourceSpan (
2016-11-23 09:42:19 -08:00
elementSourceSpan . start , elementSourceSpan . end ,
` Directive ${ identifierName ( directive . type ) } ` ) ;
2016-12-27 15:23:49 -08:00
2016-04-25 19:52:24 -07:00
if ( directive . isComponent ) {
component = directive ;
}
2016-07-08 10:05:27 -07:00
const directiveProperties : BoundDirectivePropertyAst [ ] = [ ] ;
2017-02-27 23:08:19 -08:00
let hostProperties =
2017-03-24 09:59:58 -07:00
this . _bindingParser . createDirectiveHostPropertyAsts ( directive , elementName , sourceSpan ) ! ;
2016-10-24 09:58:52 -07:00
// Note: We need to check the host properties here as well,
// as we don't know the element name in the DirectiveWrapperCompiler yet.
2017-02-09 14:59:57 -08:00
hostProperties = this . _checkPropertiesInSchema ( elementName , hostProperties ) ;
2017-03-24 09:59:58 -07:00
const hostEvents = this . _bindingParser . createDirectiveHostEventAsts ( directive , sourceSpan ) ! ;
2017-02-09 14:59:57 -08:00
this . _createDirectivePropertyAsts (
directive . inputs , props , directiveProperties , targetBoundDirectivePropNames ) ;
2016-04-25 19:52:24 -07:00
elementOrDirectiveRefs . forEach ( ( elOrDirRef ) = > {
if ( ( elOrDirRef . value . length === 0 && directive . isComponent ) ||
( directive . exportAs == elOrDirRef . value ) ) {
2016-06-08 16:38:52 -07:00
targetReferences . push ( new ReferenceAst (
2017-05-18 13:46:51 -07:00
elOrDirRef . name , createTokenForReference ( directive . type . reference ) ,
elOrDirRef . sourceSpan ) ) ;
2016-04-25 19:52:24 -07:00
matchedReferences . add ( elOrDirRef . name ) ;
2015-09-18 10:33:23 -07:00
}
} ) ;
2017-02-15 08:36:49 -08:00
const contentQueryStartId = this . contentQueryStartId ;
this . contentQueryStartId += directive . queries . length ;
2016-06-08 16:38:52 -07:00
return new DirectiveAst (
2017-02-15 08:36:49 -08:00
directive , directiveProperties , hostProperties , hostEvents , contentQueryStartId ,
sourceSpan ) ;
2015-09-18 10:33:23 -07:00
} ) ;
2016-12-27 15:23:49 -08:00
2016-04-25 19:52:24 -07:00
elementOrDirectiveRefs . forEach ( ( elOrDirRef ) = > {
if ( elOrDirRef . value . length > 0 ) {
2016-08-23 10:52:40 -07:00
if ( ! matchedReferences . has ( elOrDirRef . name ) ) {
2016-10-24 11:11:31 -07:00
this . _reportError (
2016-06-08 16:38:52 -07:00
` There is no directive with "exportAs" set to " ${ elOrDirRef . value } " ` ,
elOrDirRef . sourceSpan ) ;
2016-07-08 10:05:27 -07:00
}
2016-08-23 10:52:40 -07:00
} else if ( ! component ) {
2017-03-24 09:59:58 -07:00
let refToken : CompileTokenMetadata = null ! ;
2016-04-25 19:52:24 -07:00
if ( isTemplateElement ) {
2017-05-18 13:46:51 -07:00
refToken = createTokenForExternalReference ( this . reflector , Identifiers . TemplateRef ) ;
2016-04-25 19:52:24 -07:00
}
targetReferences . push ( new ReferenceAst ( elOrDirRef . name , refToken , elOrDirRef . sourceSpan ) ) ;
2015-09-18 10:33:23 -07:00
}
2016-12-27 15:23:49 -08:00
} ) ;
2015-09-18 10:33:23 -07:00
return directiveAsts ;
2015-08-27 16:29:02 -07:00
}
2016-06-08 16:38:52 -07:00
private _createDirectivePropertyAsts (
2016-10-21 11:41:14 -07:00
directiveProperties : { [ key : string ] : string } , boundProps : BoundProperty [ ] ,
2017-02-09 14:59:57 -08:00
targetBoundDirectiveProps : BoundDirectivePropertyAst [ ] ,
targetBoundDirectivePropNames : Set < string > ) {
2016-08-23 10:52:40 -07:00
if ( directiveProperties ) {
2016-10-21 11:41:14 -07:00
const boundPropsByName = new Map < string , BoundProperty > ( ) ;
2015-09-18 10:33:23 -07:00
boundProps . forEach ( boundProp = > {
2016-07-08 10:05:27 -07:00
const prevValue = boundPropsByName . get ( boundProp . name ) ;
2016-09-30 09:26:53 -07:00
if ( ! prevValue || prevValue . isLiteral ) {
2015-11-23 16:02:19 -08:00
// give [a]="b" a higher precedence than a="b" on the same element
boundPropsByName . set ( boundProp . name , boundProp ) ;
2015-09-18 10:33:23 -07:00
}
} ) ;
2015-08-27 16:29:02 -07:00
2016-10-03 16:46:05 -07:00
Object . keys ( directiveProperties ) . forEach ( dirProp = > {
const elProp = directiveProperties [ dirProp ] ;
2016-07-08 10:05:27 -07:00
const boundProp = boundPropsByName . get ( elProp ) ;
2015-08-27 16:29:02 -07:00
// Bindings are optional, so this binding only needs to be set up if an expression is given.
2016-08-23 10:52:40 -07:00
if ( boundProp ) {
2017-02-09 14:59:57 -08:00
targetBoundDirectivePropNames . add ( boundProp . name ) ;
if ( ! isEmptyExpression ( boundProp . expression ) ) {
targetBoundDirectiveProps . push ( new BoundDirectivePropertyAst (
dirProp , boundProp . name , boundProp . expression , boundProp . sourceSpan ) ) ;
}
2015-08-27 16:29:02 -07:00
}
} ) ;
}
}
2016-06-08 16:38:52 -07:00
private _createElementPropertyAsts (
2016-10-21 11:41:14 -07:00
elementName : string , props : BoundProperty [ ] ,
2017-02-09 14:59:57 -08:00
boundDirectivePropNames : Set < string > ) : BoundElementPropertyAst [ ] {
2016-07-08 10:05:27 -07:00
const boundElementProps : BoundElementPropertyAst [ ] = [ ] ;
2016-08-23 10:52:40 -07:00
2016-10-21 11:41:14 -07:00
props . forEach ( ( prop : BoundProperty ) = > {
2017-02-09 14:59:57 -08:00
if ( ! prop . isLiteral && ! boundDirectivePropNames . has ( prop . name ) ) {
2016-10-24 11:11:31 -07:00
boundElementProps . push ( this . _bindingParser . createElementPropertyAst ( elementName , prop ) ) ;
2015-08-27 16:29:02 -07:00
}
} ) ;
2017-02-09 14:59:57 -08:00
return this . _checkPropertiesInSchema ( elementName , boundElementProps ) ;
2015-08-27 16:29:02 -07:00
}
2016-09-23 16:37:04 -04:00
private _findComponentDirectives ( directives : DirectiveAst [ ] ) : DirectiveAst [ ] {
return directives . filter ( directive = > directive . directive . isComponent ) ;
}
2015-08-27 16:29:02 -07:00
private _findComponentDirectiveNames ( directives : DirectiveAst [ ] ) : string [ ] {
2016-09-23 16:37:04 -04:00
return this . _findComponentDirectives ( directives )
2017-03-24 09:59:58 -07:00
. map ( directive = > identifierName ( directive . directive . type ) ! ) ;
2015-08-27 16:29:02 -07:00
}
2015-10-07 09:34:21 -07:00
private _assertOnlyOneComponent ( directives : DirectiveAst [ ] , sourceSpan : ParseSourceSpan ) {
2016-07-08 10:05:27 -07:00
const componentTypeNames = this . _findComponentDirectiveNames ( directives ) ;
2015-08-27 16:29:02 -07:00
if ( componentTypeNames . length > 1 ) {
2016-10-28 20:59:00 -07:00
this . _reportError (
` More than one component matched on this element. \ n ` +
` Make sure that only one component's selector can match a given element. \ n ` +
` Conflicting components: ${ componentTypeNames . join ( ',' ) } ` ,
sourceSpan ) ;
2015-08-27 16:29:02 -07:00
}
}
2016-08-23 10:52:40 -07:00
/ * *
* Make sure that non - angular tags conform to the schemas .
*
* Note : An element is considered an angular tag when at least one directive selector matches the
* tag name .
*
* @param matchElement Whether any directive has matched on the tag name
* @param element the html element
* /
private _assertElementExists ( matchElement : boolean , element : html.Element ) {
const elName = element . name . replace ( /^:xhtml:/ , '' ) ;
if ( ! matchElement && ! this . _schemaRegistry . hasElement ( elName , this . _schemas ) ) {
2017-02-09 12:29:14 +03:00
let errorMsg = ` ' ${ elName } ' is not a known element: \ n ` ;
errorMsg +=
` 1. If ' ${ elName } ' is an Angular component, then verify that it is part of this module. \ n ` ;
if ( elName . indexOf ( '-' ) > - 1 ) {
errorMsg +=
` 2. If ' ${ elName } ' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message. ` ;
} else {
errorMsg +=
` 2. To allow any element add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component. ` ;
}
2017-03-24 09:59:58 -07:00
this . _reportError ( errorMsg , element . sourceSpan ! ) ;
2016-08-23 10:52:40 -07:00
}
}
2016-06-08 16:38:52 -07:00
private _assertNoComponentsNorElementBindingsOnTemplate (
directives : DirectiveAst [ ] , elementProps : BoundElementPropertyAst [ ] ,
sourceSpan : ParseSourceSpan ) {
2016-07-08 10:05:27 -07:00
const componentTypeNames : string [ ] = this . _findComponentDirectiveNames ( directives ) ;
2015-08-27 16:29:02 -07:00
if ( componentTypeNames . length > 0 ) {
2016-10-24 11:11:31 -07:00
this . _reportError (
2016-06-08 16:38:52 -07:00
` Components on an embedded template: ${ componentTypeNames . join ( ',' ) } ` , sourceSpan ) ;
2015-08-27 16:29:02 -07:00
}
elementProps . forEach ( prop = > {
2016-10-24 11:11:31 -07:00
this . _reportError (
2016-12-07 21:38:52 -08:00
` Property binding ${ prop . name } not used by any directive on an embedded template. Make sure that the property name is spelled correctly and all directives are listed in the "@NgModule.declarations". ` ,
2015-10-07 09:34:21 -07:00
sourceSpan ) ;
2015-08-27 16:29:02 -07:00
} ) ;
2015-10-13 17:43:15 -07:00
}
2016-06-08 16:38:52 -07:00
private _assertAllEventsPublishedByDirectives (
directives : DirectiveAst [ ] , events : BoundEventAst [ ] ) {
2016-07-08 10:05:27 -07:00
const allDirectiveEvents = new Set < string > ( ) ;
2016-08-23 10:52:40 -07:00
2015-10-13 17:43:15 -07:00
directives . forEach ( directive = > {
2016-10-03 16:46:05 -07:00
Object . keys ( directive . directive . outputs ) . forEach ( k = > {
const eventName = directive . directive . outputs [ k ] ;
2016-06-22 17:43:27 +02:00
allDirectiveEvents . add ( eventName ) ;
} ) ;
2015-10-13 17:43:15 -07:00
} ) ;
2016-08-23 10:52:40 -07:00
2015-08-27 16:29:02 -07:00
events . forEach ( event = > {
2017-03-02 09:37:01 -08:00
if ( event . target != null || ! allDirectiveEvents . has ( event . name ) ) {
2016-10-24 11:11:31 -07:00
this . _reportError (
2016-12-07 21:38:52 -08:00
` Event binding ${ event . fullName } not emitted by any directive on an embedded template. Make sure that the event name is spelled correctly and all directives are listed in the "@NgModule.declarations". ` ,
2015-10-07 09:34:21 -07:00
event . sourceSpan ) ;
2015-10-13 17:43:15 -07:00
}
2015-08-27 16:29:02 -07:00
} ) ;
}
2016-10-24 11:11:31 -07:00
2017-02-09 14:59:57 -08:00
private _checkPropertiesInSchema ( elementName : string , boundProps : BoundElementPropertyAst [ ] ) :
BoundElementPropertyAst [ ] {
// Note: We can't filter out empty expressions before this method,
// as we still want to validate them!
return boundProps . filter ( ( boundProp ) = > {
2016-10-24 09:58:52 -07:00
if ( boundProp . type === PropertyBindingType . Property &&
! this . _schemaRegistry . hasProperty ( elementName , boundProp . name , this . _schemas ) ) {
let errorMsg =
` Can't bind to ' ${ boundProp . name } ' since it isn't a known property of ' ${ elementName } '. ` ;
2017-02-09 21:22:44 +03:00
if ( elementName . startsWith ( 'ng-' ) ) {
errorMsg +=
` \ n1. If ' ${ boundProp . name } ' is an Angular directive, then add 'CommonModule' to the '@NgModule.imports' of this component. ` +
` \ n2. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component. ` ;
} else if ( elementName . indexOf ( '-' ) > - 1 ) {
2016-10-24 09:58:52 -07:00
errorMsg +=
` \ n1. If ' ${ elementName } ' is an Angular component and it has ' ${ boundProp . name } ' input, then verify that it is part of this module. ` +
2017-02-09 12:29:14 +03:00
` \ n2. If ' ${ elementName } ' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message. ` +
` \ n3. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component. ` ;
2016-10-24 09:58:52 -07:00
}
this . _reportError ( errorMsg , boundProp . sourceSpan ) ;
}
2017-02-09 14:59:57 -08:00
return ! isEmptyExpression ( boundProp . value ) ;
2016-10-24 09:58:52 -07:00
} ) ;
}
2016-10-24 11:11:31 -07:00
private _reportError (
message : string , sourceSpan : ParseSourceSpan ,
2017-03-14 17:12:44 -07:00
level : ParseErrorLevel = ParseErrorLevel . ERROR ) {
2016-10-24 11:11:31 -07:00
this . _targetErrors . push ( new ParseError ( sourceSpan , message , level ) ) ;
}
2015-08-27 16:29:02 -07:00
}
2016-07-21 13:56:58 -07:00
class NonBindableVisitor implements html . Visitor {
2017-03-24 09:59:58 -07:00
visitElement ( ast : html.Element , parent : ElementContext ) : ElementAst | null {
2016-07-08 10:05:27 -07:00
const preparsedElement = preparseElement ( ast ) ;
2015-09-18 10:33:23 -07:00
if ( preparsedElement . type === PreparsedElementType . SCRIPT ||
preparsedElement . type === PreparsedElementType . STYLE ||
preparsedElement . type === PreparsedElementType . STYLESHEET ) {
// Skipping <script> for security reasons
// Skipping <style> and stylesheets as we already processed them
// in the StyleCompiler
return null ;
}
2016-12-27 15:23:49 -08:00
const attrNameAndValues = ast . attrs . map ( ( attr ) : [ string , string ] = > [ attr . name , attr . value ] ) ;
2016-07-08 10:05:27 -07:00
const selector = createElementCssSelector ( ast . name , attrNameAndValues ) ;
const ngContentIndex = parent . findNgContentIndex ( selector ) ;
2017-02-15 08:36:49 -08:00
const children : TemplateAst [ ] = html . visitAll ( this , ast . children , EMPTY_ELEMENT_CONTEXT ) ;
2016-06-08 16:38:52 -07:00
return new ElementAst (
2017-02-02 15:01:35 -08:00
ast . name , html . visitAll ( this , ast . attrs ) , [ ] , [ ] , [ ] , [ ] , [ ] , false , [ ] , children ,
2016-10-10 09:12:05 -07:00
ngContentIndex , ast . sourceSpan , ast . endSourceSpan ) ;
2015-09-18 10:33:23 -07:00
}
2016-07-21 13:56:58 -07:00
visitComment ( comment : html.Comment , context : any ) : any { return null ; }
visitAttribute ( attribute : html.Attribute , context : any ) : AttrAst {
return new AttrAst ( attribute . name , attribute . value , attribute . sourceSpan ) ;
2015-09-18 10:33:23 -07:00
}
2016-07-21 13:56:58 -07:00
visitText ( text : html.Text , parent : ElementContext ) : TextAst {
2017-03-24 09:59:58 -07:00
const ngContentIndex = parent . findNgContentIndex ( TEXT_CSS_SELECTOR ) ! ;
return new TextAst ( text . value , ngContentIndex , text . sourceSpan ! ) ;
2015-09-18 10:33:23 -07:00
}
2016-07-21 13:56:58 -07:00
visitExpansion ( expansion : html.Expansion , context : any ) : any { return expansion ; }
visitExpansionCase ( expansionCase : html.ExpansionCase , context : any ) : any { return expansionCase ; }
2015-09-18 10:33:23 -07:00
}
2016-04-25 19:52:24 -07:00
class ElementOrDirectiveRef {
constructor ( public name : string , public value : string , public sourceSpan : ParseSourceSpan ) { }
}
2015-08-25 15:36:02 -07:00
export function splitClasses ( classAttrValue : string ) : string [ ] {
2016-07-21 13:56:58 -07:00
return classAttrValue . trim ( ) . split ( /\s+/g ) ;
2015-08-25 15:36:02 -07:00
}
2015-08-27 16:29:02 -07:00
2016-01-06 14:13:44 -08:00
class ElementContext {
2016-06-08 16:38:52 -07:00
static create (
isTemplateElement : boolean , directives : DirectiveAst [ ] ,
providerContext : ProviderElementContext ) : ElementContext {
2016-07-08 10:05:27 -07:00
const matcher = new SelectorMatcher ( ) ;
2017-03-24 09:59:58 -07:00
let wildcardNgContentIndex : number = null ! ;
2016-07-08 10:05:27 -07:00
const component = directives . find ( directive = > directive . directive . isComponent ) ;
2016-08-23 10:52:40 -07:00
if ( component ) {
2017-03-24 09:59:58 -07:00
const ngContentSelectors = component . directive . template ! . ngContentSelectors ;
2016-07-08 10:05:27 -07:00
for ( let i = 0 ; i < ngContentSelectors . length ; i ++ ) {
const selector = ngContentSelectors [ i ] ;
2016-07-21 13:56:58 -07:00
if ( selector === '*' ) {
2016-01-06 14:13:44 -08:00
wildcardNgContentIndex = i ;
} else {
matcher . addSelectables ( CssSelector . parse ( ngContentSelectors [ i ] ) , i ) ;
}
2015-09-11 13:37:05 -07:00
}
}
2016-01-06 14:13:44 -08:00
return new ElementContext ( isTemplateElement , matcher , wildcardNgContentIndex , providerContext ) ;
2015-09-11 13:37:05 -07:00
}
2016-06-08 16:38:52 -07:00
constructor (
public isTemplateElement : boolean , private _ngContentIndexMatcher : SelectorMatcher ,
2017-03-24 09:59:58 -07:00
private _wildcardNgContentIndex : number | null ,
public providerContext : ProviderElementContext | null ) { }
2015-09-11 13:37:05 -07:00
2017-03-24 09:59:58 -07:00
findNgContentIndex ( selector : CssSelector ) : number | null {
2016-07-08 10:05:27 -07:00
const ngContentIndices : number [ ] = [ ] ;
2016-01-06 14:13:44 -08:00
this . _ngContentIndexMatcher . match (
2015-09-11 13:37:05 -07:00
selector , ( selector , ngContentIndex ) = > { ngContentIndices . push ( ngContentIndex ) ; } ) ;
2016-08-23 10:52:40 -07:00
ngContentIndices . sort ( ) ;
2017-03-02 09:37:01 -08:00
if ( this . _wildcardNgContentIndex != null ) {
2016-01-06 14:13:44 -08:00
ngContentIndices . push ( this . _wildcardNgContentIndex ) ;
2015-10-29 16:23:13 -07:00
}
2015-09-11 13:37:05 -07:00
return ngContentIndices . length > 0 ? ngContentIndices [ 0 ] : null ;
}
}
2016-11-02 14:49:07 +00:00
export function createElementCssSelector (
2016-12-27 15:23:49 -08:00
elementName : string , attributes : [ string , string ] [ ] ) : CssSelector {
2016-07-08 10:05:27 -07:00
const cssSelector = new CssSelector ( ) ;
2016-08-23 10:52:40 -07:00
const elNameNoNs = splitNsName ( elementName ) [ 1 ] ;
2015-12-09 09:32:15 -08:00
cssSelector . setElement ( elNameNoNs ) ;
2015-09-18 10:33:23 -07:00
2016-12-27 15:23:49 -08:00
for ( let i = 0 ; i < attributes . length ; i ++ ) {
const attrName = attributes [ i ] [ 0 ] ;
2016-11-12 14:08:58 +01:00
const attrNameNoNs = splitNsName ( attrName ) [ 1 ] ;
2016-12-27 15:23:49 -08:00
const attrValue = attributes [ i ] [ 1 ] ;
2015-12-09 09:32:15 -08:00
cssSelector . addAttribute ( attrNameNoNs , attrValue ) ;
if ( attrName . toLowerCase ( ) == CLASS_ATTR ) {
2016-07-08 10:05:27 -07:00
const classes = splitClasses ( attrValue ) ;
2015-09-18 10:33:23 -07:00
classes . forEach ( className = > cssSelector . addClassName ( className ) ) ;
}
}
return cssSelector ;
}
2016-07-08 10:05:27 -07:00
const EMPTY_ELEMENT_CONTEXT = new ElementContext ( true , new SelectorMatcher ( ) , null , null ) ;
const NON_BINDABLE_VISITOR = new NonBindableVisitor ( ) ;
2015-12-02 10:35:51 -08:00
2016-10-12 07:58:06 -07:00
function _isEmptyTextNode ( node : html.Node ) : boolean {
return node instanceof html . Text && node . value . trim ( ) . length == 0 ;
2016-10-28 20:59:00 -07:00
}
2016-11-10 16:27:53 -08:00
export function removeSummaryDuplicates < T extends { type : CompileTypeMetadata } > ( items : T [ ] ) : T [ ] {
const map = new Map < any , T > ( ) ;
items . forEach ( ( item ) = > {
if ( ! map . get ( item . type . reference ) ) {
map . set ( item . type . reference , item ) ;
}
} ) ;
return Array . from ( map . values ( ) ) ;
}
2017-02-09 14:59:57 -08:00
function isEmptyExpression ( ast : AST ) : boolean {
if ( ast instanceof ASTWithSource ) {
ast = ast . ast ;
}
return ast instanceof EmptyExpr ;
2017-02-17 12:55:55 -08:00
}
2017-01-09 13:16:46 -08:00
// `template` is deprecated in 4.x
function isTemplate (
2017-01-10 19:07:03 -08:00
el : html.Element , enableLegacyTemplate : boolean ,
reportDeprecation : ( m : string , span : ParseSourceSpan ) = > void ) : boolean {
2017-04-13 10:37:45 -07:00
if ( isNgTemplate ( el . name ) ) return true ;
2017-01-09 13:16:46 -08:00
const tagNoNs = splitNsName ( el . name ) [ 1 ] ;
// `<template>` is HTML and case insensitive
if ( tagNoNs . toLowerCase ( ) === TEMPLATE_ELEMENT ) {
2017-01-10 19:07:03 -08:00
if ( enableLegacyTemplate && tagNoNs . toLowerCase ( ) === TEMPLATE_ELEMENT ) {
2017-03-24 09:59:58 -07:00
reportDeprecation ( TEMPLATE_ELEMENT_DEPRECATION_WARNING , el . sourceSpan ! ) ;
2017-01-10 19:07:03 -08:00
return true ;
}
}
2017-03-24 09:59:58 -07:00
return false ;
2017-03-02 09:37:01 -08:00
}