1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { ASTWithSource, EmptyExpr } from '../expression_parser/ast';
|
---|
9 | import { createTokenForExternalReference, createTokenForReference, Identifiers } from '../identifiers';
|
---|
10 | import * as html from '../ml_parser/ast';
|
---|
11 | import { ParseTreeResult } from '../ml_parser/html_parser';
|
---|
12 | import { removeWhitespaces, replaceNgsp } from '../ml_parser/html_whitespaces';
|
---|
13 | import { expandNodes } from '../ml_parser/icu_ast_expander';
|
---|
14 | import { InterpolationConfig } from '../ml_parser/interpolation_config';
|
---|
15 | import { isNgTemplate, splitNsName } from '../ml_parser/tags';
|
---|
16 | import { identifierName, ParseError, ParseErrorLevel, ParseSourceSpan, syntaxError } from '../parse_util';
|
---|
17 | import { ProviderElementContext, ProviderViewContext } from '../provider_analyzer';
|
---|
18 | import { CssSelector, SelectorMatcher } from '../selector';
|
---|
19 | import { isStyleUrlResolvable } from '../style_url_resolver';
|
---|
20 | import { newArray } from '../util';
|
---|
21 | import { BindingParser } from './binding_parser';
|
---|
22 | import * as t from './template_ast';
|
---|
23 | import { PreparsedElementType, preparseElement } from './template_preparser';
|
---|
24 | const BIND_NAME_REGEXP = /^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.*))|\[\(([^\)]+)\)\]|\[([^\]]+)\]|\(([^\)]+)\))$/;
|
---|
25 | // Group 1 = "bind-"
|
---|
26 | const KW_BIND_IDX = 1;
|
---|
27 | // Group 2 = "let-"
|
---|
28 | const KW_LET_IDX = 2;
|
---|
29 | // Group 3 = "ref-/#"
|
---|
30 | const KW_REF_IDX = 3;
|
---|
31 | // Group 4 = "on-"
|
---|
32 | const KW_ON_IDX = 4;
|
---|
33 | // Group 5 = "bindon-"
|
---|
34 | const KW_BINDON_IDX = 5;
|
---|
35 | // Group 6 = "@"
|
---|
36 | const KW_AT_IDX = 6;
|
---|
37 | // Group 7 = the identifier after "bind-", "let-", "ref-/#", "on-", "bindon-" or "@"
|
---|
38 | const IDENT_KW_IDX = 7;
|
---|
39 | // Group 8 = identifier inside [()]
|
---|
40 | const IDENT_BANANA_BOX_IDX = 8;
|
---|
41 | // Group 9 = identifier inside []
|
---|
42 | const IDENT_PROPERTY_IDX = 9;
|
---|
43 | // Group 10 = identifier inside ()
|
---|
44 | const IDENT_EVENT_IDX = 10;
|
---|
45 | const TEMPLATE_ATTR_PREFIX = '*';
|
---|
46 | const CLASS_ATTR = 'class';
|
---|
47 | let _TEXT_CSS_SELECTOR;
|
---|
48 | function TEXT_CSS_SELECTOR() {
|
---|
49 | if (!_TEXT_CSS_SELECTOR) {
|
---|
50 | _TEXT_CSS_SELECTOR = CssSelector.parse('*')[0];
|
---|
51 | }
|
---|
52 | return _TEXT_CSS_SELECTOR;
|
---|
53 | }
|
---|
54 | export class TemplateParseError extends ParseError {
|
---|
55 | constructor(message, span, level) {
|
---|
56 | super(span, message, level);
|
---|
57 | }
|
---|
58 | }
|
---|
59 | export class TemplateParseResult {
|
---|
60 | constructor(templateAst, usedPipes, errors) {
|
---|
61 | this.templateAst = templateAst;
|
---|
62 | this.usedPipes = usedPipes;
|
---|
63 | this.errors = errors;
|
---|
64 | }
|
---|
65 | }
|
---|
66 | export class TemplateParser {
|
---|
67 | constructor(_config, _reflector, _exprParser, _schemaRegistry, _htmlParser, _console, transforms) {
|
---|
68 | this._config = _config;
|
---|
69 | this._reflector = _reflector;
|
---|
70 | this._exprParser = _exprParser;
|
---|
71 | this._schemaRegistry = _schemaRegistry;
|
---|
72 | this._htmlParser = _htmlParser;
|
---|
73 | this._console = _console;
|
---|
74 | this.transforms = transforms;
|
---|
75 | }
|
---|
76 | get expressionParser() {
|
---|
77 | return this._exprParser;
|
---|
78 | }
|
---|
79 | parse(component, template, directives, pipes, schemas, templateUrl, preserveWhitespaces) {
|
---|
80 | var _a;
|
---|
81 | const result = this.tryParse(component, template, directives, pipes, schemas, templateUrl, preserveWhitespaces);
|
---|
82 | const warnings = result.errors.filter(error => error.level === ParseErrorLevel.WARNING);
|
---|
83 | const errors = result.errors.filter(error => error.level === ParseErrorLevel.ERROR);
|
---|
84 | if (warnings.length > 0) {
|
---|
85 | (_a = this._console) === null || _a === void 0 ? void 0 : _a.warn(`Template parse warnings:\n${warnings.join('\n')}`);
|
---|
86 | }
|
---|
87 | if (errors.length > 0) {
|
---|
88 | const errorString = errors.join('\n');
|
---|
89 | throw syntaxError(`Template parse errors:\n${errorString}`, errors);
|
---|
90 | }
|
---|
91 | return { template: result.templateAst, pipes: result.usedPipes };
|
---|
92 | }
|
---|
93 | tryParse(component, template, directives, pipes, schemas, templateUrl, preserveWhitespaces) {
|
---|
94 | let htmlParseResult = typeof template === 'string' ?
|
---|
95 | this._htmlParser.parse(template, templateUrl, {
|
---|
96 | tokenizeExpansionForms: true,
|
---|
97 | interpolationConfig: this.getInterpolationConfig(component)
|
---|
98 | }) :
|
---|
99 | template;
|
---|
100 | if (!preserveWhitespaces) {
|
---|
101 | htmlParseResult = removeWhitespaces(htmlParseResult);
|
---|
102 | }
|
---|
103 | return this.tryParseHtml(this.expandHtml(htmlParseResult), component, directives, pipes, schemas);
|
---|
104 | }
|
---|
105 | tryParseHtml(htmlAstWithErrors, component, directives, pipes, schemas) {
|
---|
106 | let result;
|
---|
107 | const errors = htmlAstWithErrors.errors;
|
---|
108 | const usedPipes = [];
|
---|
109 | if (htmlAstWithErrors.rootNodes.length > 0) {
|
---|
110 | const uniqDirectives = removeSummaryDuplicates(directives);
|
---|
111 | const uniqPipes = removeSummaryDuplicates(pipes);
|
---|
112 | const providerViewContext = new ProviderViewContext(this._reflector, component);
|
---|
113 | let interpolationConfig = undefined;
|
---|
114 | if (component.template && component.template.interpolation) {
|
---|
115 | interpolationConfig = {
|
---|
116 | start: component.template.interpolation[0],
|
---|
117 | end: component.template.interpolation[1]
|
---|
118 | };
|
---|
119 | }
|
---|
120 | const bindingParser = new BindingParser(this._exprParser, interpolationConfig, this._schemaRegistry, uniqPipes, errors);
|
---|
121 | const parseVisitor = new TemplateParseVisitor(this._reflector, this._config, providerViewContext, uniqDirectives, bindingParser, this._schemaRegistry, schemas, errors);
|
---|
122 | result = html.visitAll(parseVisitor, htmlAstWithErrors.rootNodes, EMPTY_ELEMENT_CONTEXT);
|
---|
123 | errors.push(...providerViewContext.errors);
|
---|
124 | usedPipes.push(...bindingParser.getUsedPipes());
|
---|
125 | }
|
---|
126 | else {
|
---|
127 | result = [];
|
---|
128 | }
|
---|
129 | this._assertNoReferenceDuplicationOnTemplate(result, errors);
|
---|
130 | if (errors.length > 0) {
|
---|
131 | return new TemplateParseResult(result, usedPipes, errors);
|
---|
132 | }
|
---|
133 | if (this.transforms) {
|
---|
134 | this.transforms.forEach((transform) => {
|
---|
135 | result = t.templateVisitAll(transform, result);
|
---|
136 | });
|
---|
137 | }
|
---|
138 | return new TemplateParseResult(result, usedPipes, errors);
|
---|
139 | }
|
---|
140 | expandHtml(htmlAstWithErrors, forced = false) {
|
---|
141 | const errors = htmlAstWithErrors.errors;
|
---|
142 | if (errors.length == 0 || forced) {
|
---|
143 | // Transform ICU messages to angular directives
|
---|
144 | const expandedHtmlAst = expandNodes(htmlAstWithErrors.rootNodes);
|
---|
145 | errors.push(...expandedHtmlAst.errors);
|
---|
146 | htmlAstWithErrors = new ParseTreeResult(expandedHtmlAst.nodes, errors);
|
---|
147 | }
|
---|
148 | return htmlAstWithErrors;
|
---|
149 | }
|
---|
150 | getInterpolationConfig(component) {
|
---|
151 | if (component.template) {
|
---|
152 | return InterpolationConfig.fromArray(component.template.interpolation);
|
---|
153 | }
|
---|
154 | return undefined;
|
---|
155 | }
|
---|
156 | /** @internal */
|
---|
157 | _assertNoReferenceDuplicationOnTemplate(result, errors) {
|
---|
158 | const existingReferences = [];
|
---|
159 | result.filter(element => !!element.references)
|
---|
160 | .forEach(element => element.references.forEach((reference) => {
|
---|
161 | const name = reference.name;
|
---|
162 | if (existingReferences.indexOf(name) < 0) {
|
---|
163 | existingReferences.push(name);
|
---|
164 | }
|
---|
165 | else {
|
---|
166 | const error = new TemplateParseError(`Reference "#${name}" is defined several times`, reference.sourceSpan, ParseErrorLevel.ERROR);
|
---|
167 | errors.push(error);
|
---|
168 | }
|
---|
169 | }));
|
---|
170 | }
|
---|
171 | }
|
---|
172 | class TemplateParseVisitor {
|
---|
173 | constructor(reflector, config, providerViewContext, directives, _bindingParser, _schemaRegistry, _schemas, _targetErrors) {
|
---|
174 | this.reflector = reflector;
|
---|
175 | this.config = config;
|
---|
176 | this.providerViewContext = providerViewContext;
|
---|
177 | this._bindingParser = _bindingParser;
|
---|
178 | this._schemaRegistry = _schemaRegistry;
|
---|
179 | this._schemas = _schemas;
|
---|
180 | this._targetErrors = _targetErrors;
|
---|
181 | this.selectorMatcher = new SelectorMatcher();
|
---|
182 | this.directivesIndex = new Map();
|
---|
183 | this.ngContentCount = 0;
|
---|
184 | // Note: queries start with id 1 so we can use the number in a Bloom filter!
|
---|
185 | this.contentQueryStartId = providerViewContext.component.viewQueries.length + 1;
|
---|
186 | directives.forEach((directive, index) => {
|
---|
187 | const selector = CssSelector.parse(directive.selector);
|
---|
188 | this.selectorMatcher.addSelectables(selector, directive);
|
---|
189 | this.directivesIndex.set(directive, index);
|
---|
190 | });
|
---|
191 | }
|
---|
192 | visitExpansion(expansion, context) {
|
---|
193 | return null;
|
---|
194 | }
|
---|
195 | visitExpansionCase(expansionCase, context) {
|
---|
196 | return null;
|
---|
197 | }
|
---|
198 | visitText(text, parent) {
|
---|
199 | const ngContentIndex = parent.findNgContentIndex(TEXT_CSS_SELECTOR());
|
---|
200 | const valueNoNgsp = replaceNgsp(text.value);
|
---|
201 | const expr = this._bindingParser.parseInterpolation(valueNoNgsp, text.sourceSpan);
|
---|
202 | return expr ? new t.BoundTextAst(expr, ngContentIndex, text.sourceSpan) :
|
---|
203 | new t.TextAst(valueNoNgsp, ngContentIndex, text.sourceSpan);
|
---|
204 | }
|
---|
205 | visitAttribute(attribute, context) {
|
---|
206 | return new t.AttrAst(attribute.name, attribute.value, attribute.sourceSpan);
|
---|
207 | }
|
---|
208 | visitComment(comment, context) {
|
---|
209 | return null;
|
---|
210 | }
|
---|
211 | visitElement(element, parent) {
|
---|
212 | const queryStartIndex = this.contentQueryStartId;
|
---|
213 | const elName = element.name;
|
---|
214 | const preparsedElement = preparseElement(element);
|
---|
215 | if (preparsedElement.type === PreparsedElementType.SCRIPT ||
|
---|
216 | preparsedElement.type === PreparsedElementType.STYLE) {
|
---|
217 | // Skipping <script> for security reasons
|
---|
218 | // Skipping <style> as we already processed them
|
---|
219 | // in the StyleCompiler
|
---|
220 | return null;
|
---|
221 | }
|
---|
222 | if (preparsedElement.type === PreparsedElementType.STYLESHEET &&
|
---|
223 | isStyleUrlResolvable(preparsedElement.hrefAttr)) {
|
---|
224 | // Skipping stylesheets with either relative urls or package scheme as we already processed
|
---|
225 | // them in the StyleCompiler
|
---|
226 | return null;
|
---|
227 | }
|
---|
228 | const matchableAttrs = [];
|
---|
229 | const elementOrDirectiveProps = [];
|
---|
230 | const elementOrDirectiveRefs = [];
|
---|
231 | const elementVars = [];
|
---|
232 | const events = [];
|
---|
233 | const templateElementOrDirectiveProps = [];
|
---|
234 | const templateMatchableAttrs = [];
|
---|
235 | const templateElementVars = [];
|
---|
236 | let hasInlineTemplates = false;
|
---|
237 | const attrs = [];
|
---|
238 | const isTemplateElement = isNgTemplate(element.name);
|
---|
239 | element.attrs.forEach(attr => {
|
---|
240 | const parsedVariables = [];
|
---|
241 | const hasBinding = this._parseAttr(isTemplateElement, attr, matchableAttrs, elementOrDirectiveProps, events, elementOrDirectiveRefs, elementVars);
|
---|
242 | elementVars.push(...parsedVariables.map(v => t.VariableAst.fromParsedVariable(v)));
|
---|
243 | let templateValue;
|
---|
244 | let templateKey;
|
---|
245 | const normalizedName = this._normalizeAttributeName(attr.name);
|
---|
246 | if (normalizedName.startsWith(TEMPLATE_ATTR_PREFIX)) {
|
---|
247 | templateValue = attr.value;
|
---|
248 | templateKey = normalizedName.substring(TEMPLATE_ATTR_PREFIX.length);
|
---|
249 | }
|
---|
250 | const hasTemplateBinding = templateValue != null;
|
---|
251 | if (hasTemplateBinding) {
|
---|
252 | if (hasInlineTemplates) {
|
---|
253 | this._reportError(`Can't have multiple template bindings on one element. Use only one attribute prefixed with *`, attr.sourceSpan);
|
---|
254 | }
|
---|
255 | hasInlineTemplates = true;
|
---|
256 | const parsedVariables = [];
|
---|
257 | const absoluteOffset = (attr.valueSpan || attr.sourceSpan).start.offset;
|
---|
258 | this._bindingParser.parseInlineTemplateBinding(templateKey, templateValue, attr.sourceSpan, absoluteOffset, templateMatchableAttrs, templateElementOrDirectiveProps, parsedVariables, false /* isIvyAst */);
|
---|
259 | templateElementVars.push(...parsedVariables.map(v => t.VariableAst.fromParsedVariable(v)));
|
---|
260 | }
|
---|
261 | if (!hasBinding && !hasTemplateBinding) {
|
---|
262 | // don't include the bindings as attributes as well in the AST
|
---|
263 | attrs.push(this.visitAttribute(attr, null));
|
---|
264 | matchableAttrs.push([attr.name, attr.value]);
|
---|
265 | }
|
---|
266 | });
|
---|
267 | const elementCssSelector = createElementCssSelector(elName, matchableAttrs);
|
---|
268 | const { directives: directiveMetas, matchElement } = this._parseDirectives(this.selectorMatcher, elementCssSelector);
|
---|
269 | const references = [];
|
---|
270 | const boundDirectivePropNames = new Set();
|
---|
271 | const directiveAsts = this._createDirectiveAsts(isTemplateElement, element.name, directiveMetas, elementOrDirectiveProps, elementOrDirectiveRefs, element.sourceSpan, references, boundDirectivePropNames);
|
---|
272 | const elementProps = this._createElementPropertyAsts(element.name, elementOrDirectiveProps, boundDirectivePropNames);
|
---|
273 | const isViewRoot = parent.isTemplateElement || hasInlineTemplates;
|
---|
274 | const providerContext = new ProviderElementContext(this.providerViewContext, parent.providerContext, isViewRoot, directiveAsts, attrs, references, isTemplateElement, queryStartIndex, element.sourceSpan);
|
---|
275 | const children = html.visitAll(preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this, element.children, ElementContext.create(isTemplateElement, directiveAsts, isTemplateElement ? parent.providerContext : providerContext));
|
---|
276 | providerContext.afterElement();
|
---|
277 | // Override the actual selector when the `ngProjectAs` attribute is provided
|
---|
278 | const projectionSelector = preparsedElement.projectAs != '' ?
|
---|
279 | CssSelector.parse(preparsedElement.projectAs)[0] :
|
---|
280 | elementCssSelector;
|
---|
281 | const ngContentIndex = parent.findNgContentIndex(projectionSelector);
|
---|
282 | let parsedElement;
|
---|
283 | if (preparsedElement.type === PreparsedElementType.NG_CONTENT) {
|
---|
284 | // `<ng-content>` element
|
---|
285 | if (element.children && !element.children.every(_isEmptyTextNode)) {
|
---|
286 | this._reportError(`<ng-content> element cannot have content.`, element.sourceSpan);
|
---|
287 | }
|
---|
288 | parsedElement = new t.NgContentAst(this.ngContentCount++, hasInlineTemplates ? null : ngContentIndex, element.sourceSpan);
|
---|
289 | }
|
---|
290 | else if (isTemplateElement) {
|
---|
291 | // `<ng-template>` element
|
---|
292 | this._assertAllEventsPublishedByDirectives(directiveAsts, events);
|
---|
293 | this._assertNoComponentsNorElementBindingsOnTemplate(directiveAsts, elementProps, element.sourceSpan);
|
---|
294 | parsedElement = new t.EmbeddedTemplateAst(attrs, events, references, elementVars, providerContext.transformedDirectiveAsts, providerContext.transformProviders, providerContext.transformedHasViewContainer, providerContext.queryMatches, children, hasInlineTemplates ? null : ngContentIndex, element.sourceSpan);
|
---|
295 | }
|
---|
296 | else {
|
---|
297 | // element other than `<ng-content>` and `<ng-template>`
|
---|
298 | this._assertElementExists(matchElement, element);
|
---|
299 | this._assertOnlyOneComponent(directiveAsts, element.sourceSpan);
|
---|
300 | const ngContentIndex = hasInlineTemplates ? null : parent.findNgContentIndex(projectionSelector);
|
---|
301 | parsedElement = new t.ElementAst(elName, attrs, elementProps, events, references, providerContext.transformedDirectiveAsts, providerContext.transformProviders, providerContext.transformedHasViewContainer, providerContext.queryMatches, children, hasInlineTemplates ? null : ngContentIndex, element.sourceSpan, element.endSourceSpan || null);
|
---|
302 | }
|
---|
303 | if (hasInlineTemplates) {
|
---|
304 | // The element as a *-attribute
|
---|
305 | const templateQueryStartIndex = this.contentQueryStartId;
|
---|
306 | const templateSelector = createElementCssSelector('ng-template', templateMatchableAttrs);
|
---|
307 | const { directives } = this._parseDirectives(this.selectorMatcher, templateSelector);
|
---|
308 | const templateBoundDirectivePropNames = new Set();
|
---|
309 | const templateDirectiveAsts = this._createDirectiveAsts(true, elName, directives, templateElementOrDirectiveProps, [], element.sourceSpan, [], templateBoundDirectivePropNames);
|
---|
310 | const templateElementProps = this._createElementPropertyAsts(elName, templateElementOrDirectiveProps, templateBoundDirectivePropNames);
|
---|
311 | this._assertNoComponentsNorElementBindingsOnTemplate(templateDirectiveAsts, templateElementProps, element.sourceSpan);
|
---|
312 | const templateProviderContext = new ProviderElementContext(this.providerViewContext, parent.providerContext, parent.isTemplateElement, templateDirectiveAsts, [], [], true, templateQueryStartIndex, element.sourceSpan);
|
---|
313 | templateProviderContext.afterElement();
|
---|
314 | parsedElement = new t.EmbeddedTemplateAst([], [], [], templateElementVars, templateProviderContext.transformedDirectiveAsts, templateProviderContext.transformProviders, templateProviderContext.transformedHasViewContainer, templateProviderContext.queryMatches, [parsedElement], ngContentIndex, element.sourceSpan);
|
---|
315 | }
|
---|
316 | return parsedElement;
|
---|
317 | }
|
---|
318 | _parseAttr(isTemplateElement, attr, targetMatchableAttrs, targetProps, targetEvents, targetRefs, targetVars) {
|
---|
319 | const name = this._normalizeAttributeName(attr.name);
|
---|
320 | const value = attr.value;
|
---|
321 | const srcSpan = attr.sourceSpan;
|
---|
322 | const absoluteOffset = attr.valueSpan ? attr.valueSpan.start.offset : srcSpan.start.offset;
|
---|
323 | const boundEvents = [];
|
---|
324 | const bindParts = name.match(BIND_NAME_REGEXP);
|
---|
325 | let hasBinding = false;
|
---|
326 | if (bindParts !== null) {
|
---|
327 | hasBinding = true;
|
---|
328 | if (bindParts[KW_BIND_IDX] != null) {
|
---|
329 | this._bindingParser.parsePropertyBinding(bindParts[IDENT_KW_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
330 | }
|
---|
331 | else if (bindParts[KW_LET_IDX]) {
|
---|
332 | if (isTemplateElement) {
|
---|
333 | const identifier = bindParts[IDENT_KW_IDX];
|
---|
334 | this._parseVariable(identifier, value, srcSpan, targetVars);
|
---|
335 | }
|
---|
336 | else {
|
---|
337 | this._reportError(`"let-" is only supported on ng-template elements.`, srcSpan);
|
---|
338 | }
|
---|
339 | }
|
---|
340 | else if (bindParts[KW_REF_IDX]) {
|
---|
341 | const identifier = bindParts[IDENT_KW_IDX];
|
---|
342 | this._parseReference(identifier, value, srcSpan, targetRefs);
|
---|
343 | }
|
---|
344 | else if (bindParts[KW_ON_IDX]) {
|
---|
345 | this._bindingParser.parseEvent(bindParts[IDENT_KW_IDX], value, srcSpan, attr.valueSpan || srcSpan, targetMatchableAttrs, boundEvents);
|
---|
346 | }
|
---|
347 | else if (bindParts[KW_BINDON_IDX]) {
|
---|
348 | this._bindingParser.parsePropertyBinding(bindParts[IDENT_KW_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
349 | this._parseAssignmentEvent(bindParts[IDENT_KW_IDX], value, srcSpan, attr.valueSpan || srcSpan, targetMatchableAttrs, boundEvents);
|
---|
350 | }
|
---|
351 | else if (bindParts[KW_AT_IDX]) {
|
---|
352 | this._bindingParser.parseLiteralAttr(name, value, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
353 | }
|
---|
354 | else if (bindParts[IDENT_BANANA_BOX_IDX]) {
|
---|
355 | this._bindingParser.parsePropertyBinding(bindParts[IDENT_BANANA_BOX_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
356 | this._parseAssignmentEvent(bindParts[IDENT_BANANA_BOX_IDX], value, srcSpan, attr.valueSpan || srcSpan, targetMatchableAttrs, boundEvents);
|
---|
357 | }
|
---|
358 | else if (bindParts[IDENT_PROPERTY_IDX]) {
|
---|
359 | this._bindingParser.parsePropertyBinding(bindParts[IDENT_PROPERTY_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
360 | }
|
---|
361 | else if (bindParts[IDENT_EVENT_IDX]) {
|
---|
362 | this._bindingParser.parseEvent(bindParts[IDENT_EVENT_IDX], value, srcSpan, attr.valueSpan || srcSpan, targetMatchableAttrs, boundEvents);
|
---|
363 | }
|
---|
364 | }
|
---|
365 | else {
|
---|
366 | hasBinding = this._bindingParser.parsePropertyInterpolation(name, value, srcSpan, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
367 | }
|
---|
368 | if (!hasBinding) {
|
---|
369 | this._bindingParser.parseLiteralAttr(name, value, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);
|
---|
370 | }
|
---|
371 | targetEvents.push(...boundEvents.map(e => t.BoundEventAst.fromParsedEvent(e)));
|
---|
372 | return hasBinding;
|
---|
373 | }
|
---|
374 | _normalizeAttributeName(attrName) {
|
---|
375 | return /^data-/i.test(attrName) ? attrName.substring(5) : attrName;
|
---|
376 | }
|
---|
377 | _parseVariable(identifier, value, sourceSpan, targetVars) {
|
---|
378 | if (identifier.indexOf('-') > -1) {
|
---|
379 | this._reportError(`"-" is not allowed in variable names`, sourceSpan);
|
---|
380 | }
|
---|
381 | else if (identifier.length === 0) {
|
---|
382 | this._reportError(`Variable does not have a name`, sourceSpan);
|
---|
383 | }
|
---|
384 | targetVars.push(new t.VariableAst(identifier, value, sourceSpan));
|
---|
385 | }
|
---|
386 | _parseReference(identifier, value, sourceSpan, targetRefs) {
|
---|
387 | if (identifier.indexOf('-') > -1) {
|
---|
388 | this._reportError(`"-" is not allowed in reference names`, sourceSpan);
|
---|
389 | }
|
---|
390 | else if (identifier.length === 0) {
|
---|
391 | this._reportError(`Reference does not have a name`, sourceSpan);
|
---|
392 | }
|
---|
393 | targetRefs.push(new ElementOrDirectiveRef(identifier, value, sourceSpan));
|
---|
394 | }
|
---|
395 | _parseAssignmentEvent(name, expression, sourceSpan, valueSpan, targetMatchableAttrs, targetEvents) {
|
---|
396 | this._bindingParser.parseEvent(`${name}Change`, `${expression}=$event`, sourceSpan, valueSpan, targetMatchableAttrs, targetEvents);
|
---|
397 | }
|
---|
398 | _parseDirectives(selectorMatcher, elementCssSelector) {
|
---|
399 | // Need to sort the directives so that we get consistent results throughout,
|
---|
400 | // as selectorMatcher uses Maps inside.
|
---|
401 | // Also deduplicate directives as they might match more than one time!
|
---|
402 | const directives = newArray(this.directivesIndex.size);
|
---|
403 | // Whether any directive selector matches on the element name
|
---|
404 | let matchElement = false;
|
---|
405 | selectorMatcher.match(elementCssSelector, (selector, directive) => {
|
---|
406 | directives[this.directivesIndex.get(directive)] = directive;
|
---|
407 | matchElement = matchElement || selector.hasElementSelector();
|
---|
408 | });
|
---|
409 | return {
|
---|
410 | directives: directives.filter(dir => !!dir),
|
---|
411 | matchElement,
|
---|
412 | };
|
---|
413 | }
|
---|
414 | _createDirectiveAsts(isTemplateElement, elementName, directives, props, elementOrDirectiveRefs, elementSourceSpan, targetReferences, targetBoundDirectivePropNames) {
|
---|
415 | const matchedReferences = new Set();
|
---|
416 | let component = null;
|
---|
417 | const directiveAsts = directives.map((directive) => {
|
---|
418 | const sourceSpan = new ParseSourceSpan(elementSourceSpan.start, elementSourceSpan.end, elementSourceSpan.fullStart, `Directive ${identifierName(directive.type)}`);
|
---|
419 | if (directive.isComponent) {
|
---|
420 | component = directive;
|
---|
421 | }
|
---|
422 | const directiveProperties = [];
|
---|
423 | const boundProperties = this._bindingParser.createDirectiveHostPropertyAsts(directive, elementName, sourceSpan);
|
---|
424 | let hostProperties = boundProperties.map(prop => t.BoundElementPropertyAst.fromBoundProperty(prop));
|
---|
425 | // Note: We need to check the host properties here as well,
|
---|
426 | // as we don't know the element name in the DirectiveWrapperCompiler yet.
|
---|
427 | hostProperties = this._checkPropertiesInSchema(elementName, hostProperties);
|
---|
428 | const parsedEvents = this._bindingParser.createDirectiveHostEventAsts(directive, sourceSpan);
|
---|
429 | this._createDirectivePropertyAsts(directive.inputs, props, directiveProperties, targetBoundDirectivePropNames);
|
---|
430 | elementOrDirectiveRefs.forEach((elOrDirRef) => {
|
---|
431 | if ((elOrDirRef.value.length === 0 && directive.isComponent) ||
|
---|
432 | (elOrDirRef.isReferenceToDirective(directive))) {
|
---|
433 | targetReferences.push(new t.ReferenceAst(elOrDirRef.name, createTokenForReference(directive.type.reference), elOrDirRef.value, elOrDirRef.sourceSpan));
|
---|
434 | matchedReferences.add(elOrDirRef.name);
|
---|
435 | }
|
---|
436 | });
|
---|
437 | const hostEvents = parsedEvents.map(e => t.BoundEventAst.fromParsedEvent(e));
|
---|
438 | const contentQueryStartId = this.contentQueryStartId;
|
---|
439 | this.contentQueryStartId += directive.queries.length;
|
---|
440 | return new t.DirectiveAst(directive, directiveProperties, hostProperties, hostEvents, contentQueryStartId, sourceSpan);
|
---|
441 | });
|
---|
442 | elementOrDirectiveRefs.forEach((elOrDirRef) => {
|
---|
443 | if (elOrDirRef.value.length > 0) {
|
---|
444 | if (!matchedReferences.has(elOrDirRef.name)) {
|
---|
445 | this._reportError(`There is no directive with "exportAs" set to "${elOrDirRef.value}"`, elOrDirRef.sourceSpan);
|
---|
446 | }
|
---|
447 | }
|
---|
448 | else if (!component) {
|
---|
449 | let refToken = null;
|
---|
450 | if (isTemplateElement) {
|
---|
451 | refToken = createTokenForExternalReference(this.reflector, Identifiers.TemplateRef);
|
---|
452 | }
|
---|
453 | targetReferences.push(new t.ReferenceAst(elOrDirRef.name, refToken, elOrDirRef.value, elOrDirRef.sourceSpan));
|
---|
454 | }
|
---|
455 | });
|
---|
456 | return directiveAsts;
|
---|
457 | }
|
---|
458 | _createDirectivePropertyAsts(directiveProperties, boundProps, targetBoundDirectiveProps, targetBoundDirectivePropNames) {
|
---|
459 | if (directiveProperties) {
|
---|
460 | const boundPropsByName = new Map();
|
---|
461 | boundProps.forEach(boundProp => {
|
---|
462 | const prevValue = boundPropsByName.get(boundProp.name);
|
---|
463 | if (!prevValue || prevValue.isLiteral) {
|
---|
464 | // give [a]="b" a higher precedence than a="b" on the same element
|
---|
465 | boundPropsByName.set(boundProp.name, boundProp);
|
---|
466 | }
|
---|
467 | });
|
---|
468 | Object.keys(directiveProperties).forEach(dirProp => {
|
---|
469 | const elProp = directiveProperties[dirProp];
|
---|
470 | const boundProp = boundPropsByName.get(elProp);
|
---|
471 | // Bindings are optional, so this binding only needs to be set up if an expression is given.
|
---|
472 | if (boundProp) {
|
---|
473 | targetBoundDirectivePropNames.add(boundProp.name);
|
---|
474 | if (!isEmptyExpression(boundProp.expression)) {
|
---|
475 | targetBoundDirectiveProps.push(new t.BoundDirectivePropertyAst(dirProp, boundProp.name, boundProp.expression, boundProp.sourceSpan));
|
---|
476 | }
|
---|
477 | }
|
---|
478 | });
|
---|
479 | }
|
---|
480 | }
|
---|
481 | _createElementPropertyAsts(elementName, props, boundDirectivePropNames) {
|
---|
482 | const boundElementProps = [];
|
---|
483 | props.forEach((prop) => {
|
---|
484 | if (!prop.isLiteral && !boundDirectivePropNames.has(prop.name)) {
|
---|
485 | const boundProp = this._bindingParser.createBoundElementProperty(elementName, prop);
|
---|
486 | boundElementProps.push(t.BoundElementPropertyAst.fromBoundProperty(boundProp));
|
---|
487 | }
|
---|
488 | });
|
---|
489 | return this._checkPropertiesInSchema(elementName, boundElementProps);
|
---|
490 | }
|
---|
491 | _findComponentDirectives(directives) {
|
---|
492 | return directives.filter(directive => directive.directive.isComponent);
|
---|
493 | }
|
---|
494 | _findComponentDirectiveNames(directives) {
|
---|
495 | return this._findComponentDirectives(directives)
|
---|
496 | .map(directive => identifierName(directive.directive.type));
|
---|
497 | }
|
---|
498 | _assertOnlyOneComponent(directives, sourceSpan) {
|
---|
499 | const componentTypeNames = this._findComponentDirectiveNames(directives);
|
---|
500 | if (componentTypeNames.length > 1) {
|
---|
501 | this._reportError(`More than one component matched on this element.\n` +
|
---|
502 | `Make sure that only one component's selector can match a given element.\n` +
|
---|
503 | `Conflicting components: ${componentTypeNames.join(',')}`, sourceSpan);
|
---|
504 | }
|
---|
505 | }
|
---|
506 | /**
|
---|
507 | * Make sure that non-angular tags conform to the schemas.
|
---|
508 | *
|
---|
509 | * Note: An element is considered an angular tag when at least one directive selector matches the
|
---|
510 | * tag name.
|
---|
511 | *
|
---|
512 | * @param matchElement Whether any directive has matched on the tag name
|
---|
513 | * @param element the html element
|
---|
514 | */
|
---|
515 | _assertElementExists(matchElement, element) {
|
---|
516 | const elName = element.name.replace(/^:xhtml:/, '');
|
---|
517 | if (!matchElement && !this._schemaRegistry.hasElement(elName, this._schemas)) {
|
---|
518 | let errorMsg = `'${elName}' is not a known element:\n`;
|
---|
519 | errorMsg += `1. If '${elName}' is an Angular component, then verify that it is part of this module.\n`;
|
---|
520 | if (elName.indexOf('-') > -1) {
|
---|
521 | errorMsg += `2. If '${elName}' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.`;
|
---|
522 | }
|
---|
523 | else {
|
---|
524 | errorMsg +=
|
---|
525 | `2. To allow any element add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;
|
---|
526 | }
|
---|
527 | this._reportError(errorMsg, element.sourceSpan);
|
---|
528 | }
|
---|
529 | }
|
---|
530 | _assertNoComponentsNorElementBindingsOnTemplate(directives, elementProps, sourceSpan) {
|
---|
531 | const componentTypeNames = this._findComponentDirectiveNames(directives);
|
---|
532 | if (componentTypeNames.length > 0) {
|
---|
533 | this._reportError(`Components on an embedded template: ${componentTypeNames.join(',')}`, sourceSpan);
|
---|
534 | }
|
---|
535 | elementProps.forEach(prop => {
|
---|
536 | this._reportError(`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".`, sourceSpan);
|
---|
537 | });
|
---|
538 | }
|
---|
539 | _assertAllEventsPublishedByDirectives(directives, events) {
|
---|
540 | const allDirectiveEvents = new Set();
|
---|
541 | directives.forEach(directive => {
|
---|
542 | Object.keys(directive.directive.outputs).forEach(k => {
|
---|
543 | const eventName = directive.directive.outputs[k];
|
---|
544 | allDirectiveEvents.add(eventName);
|
---|
545 | });
|
---|
546 | });
|
---|
547 | events.forEach(event => {
|
---|
548 | if (event.target != null || !allDirectiveEvents.has(event.name)) {
|
---|
549 | this._reportError(`Event binding ${event
|
---|
550 | .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".`, event.sourceSpan);
|
---|
551 | }
|
---|
552 | });
|
---|
553 | }
|
---|
554 | _checkPropertiesInSchema(elementName, boundProps) {
|
---|
555 | // Note: We can't filter out empty expressions before this method,
|
---|
556 | // as we still want to validate them!
|
---|
557 | return boundProps.filter((boundProp) => {
|
---|
558 | if (boundProp.type === 0 /* Property */ &&
|
---|
559 | !this._schemaRegistry.hasProperty(elementName, boundProp.name, this._schemas)) {
|
---|
560 | let errorMsg = `Can't bind to '${boundProp.name}' since it isn't a known property of '${elementName}'.`;
|
---|
561 | if (elementName.startsWith('ng-')) {
|
---|
562 | errorMsg +=
|
---|
563 | `\n1. If '${boundProp
|
---|
564 | .name}' is an Angular directive, then add 'CommonModule' to the '@NgModule.imports' of this component.` +
|
---|
565 | `\n2. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;
|
---|
566 | }
|
---|
567 | else if (elementName.indexOf('-') > -1) {
|
---|
568 | errorMsg +=
|
---|
569 | `\n1. If '${elementName}' is an Angular component and it has '${boundProp.name}' input, then verify that it is part of this module.` +
|
---|
570 | `\n2. If '${elementName}' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.` +
|
---|
571 | `\n3. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;
|
---|
572 | }
|
---|
573 | this._reportError(errorMsg, boundProp.sourceSpan);
|
---|
574 | }
|
---|
575 | return !isEmptyExpression(boundProp.value);
|
---|
576 | });
|
---|
577 | }
|
---|
578 | _reportError(message, sourceSpan, level = ParseErrorLevel.ERROR) {
|
---|
579 | this._targetErrors.push(new ParseError(sourceSpan, message, level));
|
---|
580 | }
|
---|
581 | }
|
---|
582 | class NonBindableVisitor {
|
---|
583 | visitElement(ast, parent) {
|
---|
584 | const preparsedElement = preparseElement(ast);
|
---|
585 | if (preparsedElement.type === PreparsedElementType.SCRIPT ||
|
---|
586 | preparsedElement.type === PreparsedElementType.STYLE ||
|
---|
587 | preparsedElement.type === PreparsedElementType.STYLESHEET) {
|
---|
588 | // Skipping <script> for security reasons
|
---|
589 | // Skipping <style> and stylesheets as we already processed them
|
---|
590 | // in the StyleCompiler
|
---|
591 | return null;
|
---|
592 | }
|
---|
593 | const attrNameAndValues = ast.attrs.map((attr) => [attr.name, attr.value]);
|
---|
594 | const selector = createElementCssSelector(ast.name, attrNameAndValues);
|
---|
595 | const ngContentIndex = parent.findNgContentIndex(selector);
|
---|
596 | const children = html.visitAll(this, ast.children, EMPTY_ELEMENT_CONTEXT);
|
---|
597 | return new t.ElementAst(ast.name, html.visitAll(this, ast.attrs), [], [], [], [], [], false, [], children, ngContentIndex, ast.sourceSpan, ast.endSourceSpan);
|
---|
598 | }
|
---|
599 | visitComment(comment, context) {
|
---|
600 | return null;
|
---|
601 | }
|
---|
602 | visitAttribute(attribute, context) {
|
---|
603 | return new t.AttrAst(attribute.name, attribute.value, attribute.sourceSpan);
|
---|
604 | }
|
---|
605 | visitText(text, parent) {
|
---|
606 | const ngContentIndex = parent.findNgContentIndex(TEXT_CSS_SELECTOR());
|
---|
607 | return new t.TextAst(text.value, ngContentIndex, text.sourceSpan);
|
---|
608 | }
|
---|
609 | visitExpansion(expansion, context) {
|
---|
610 | return expansion;
|
---|
611 | }
|
---|
612 | visitExpansionCase(expansionCase, context) {
|
---|
613 | return expansionCase;
|
---|
614 | }
|
---|
615 | }
|
---|
616 | /**
|
---|
617 | * A reference to an element or directive in a template. E.g., the reference in this template:
|
---|
618 | *
|
---|
619 | * <div #myMenu="coolMenu">
|
---|
620 | *
|
---|
621 | * would be {name: 'myMenu', value: 'coolMenu', sourceSpan: ...}
|
---|
622 | */
|
---|
623 | class ElementOrDirectiveRef {
|
---|
624 | constructor(name, value, sourceSpan) {
|
---|
625 | this.name = name;
|
---|
626 | this.value = value;
|
---|
627 | this.sourceSpan = sourceSpan;
|
---|
628 | }
|
---|
629 | /** Gets whether this is a reference to the given directive. */
|
---|
630 | isReferenceToDirective(directive) {
|
---|
631 | return splitExportAs(directive.exportAs).indexOf(this.value) !== -1;
|
---|
632 | }
|
---|
633 | }
|
---|
634 | /** Splits a raw, potentially comma-delimited `exportAs` value into an array of names. */
|
---|
635 | function splitExportAs(exportAs) {
|
---|
636 | return exportAs ? exportAs.split(',').map(e => e.trim()) : [];
|
---|
637 | }
|
---|
638 | export function splitClasses(classAttrValue) {
|
---|
639 | return classAttrValue.trim().split(/\s+/g);
|
---|
640 | }
|
---|
641 | class ElementContext {
|
---|
642 | constructor(isTemplateElement, _ngContentIndexMatcher, _wildcardNgContentIndex, providerContext) {
|
---|
643 | this.isTemplateElement = isTemplateElement;
|
---|
644 | this._ngContentIndexMatcher = _ngContentIndexMatcher;
|
---|
645 | this._wildcardNgContentIndex = _wildcardNgContentIndex;
|
---|
646 | this.providerContext = providerContext;
|
---|
647 | }
|
---|
648 | static create(isTemplateElement, directives, providerContext) {
|
---|
649 | const matcher = new SelectorMatcher();
|
---|
650 | let wildcardNgContentIndex = null;
|
---|
651 | const component = directives.find(directive => directive.directive.isComponent);
|
---|
652 | if (component) {
|
---|
653 | const ngContentSelectors = component.directive.template.ngContentSelectors;
|
---|
654 | for (let i = 0; i < ngContentSelectors.length; i++) {
|
---|
655 | const selector = ngContentSelectors[i];
|
---|
656 | if (selector === '*') {
|
---|
657 | wildcardNgContentIndex = i;
|
---|
658 | }
|
---|
659 | else {
|
---|
660 | matcher.addSelectables(CssSelector.parse(ngContentSelectors[i]), i);
|
---|
661 | }
|
---|
662 | }
|
---|
663 | }
|
---|
664 | return new ElementContext(isTemplateElement, matcher, wildcardNgContentIndex, providerContext);
|
---|
665 | }
|
---|
666 | findNgContentIndex(selector) {
|
---|
667 | const ngContentIndices = [];
|
---|
668 | this._ngContentIndexMatcher.match(selector, (selector, ngContentIndex) => {
|
---|
669 | ngContentIndices.push(ngContentIndex);
|
---|
670 | });
|
---|
671 | ngContentIndices.sort();
|
---|
672 | if (this._wildcardNgContentIndex != null) {
|
---|
673 | ngContentIndices.push(this._wildcardNgContentIndex);
|
---|
674 | }
|
---|
675 | return ngContentIndices.length > 0 ? ngContentIndices[0] : null;
|
---|
676 | }
|
---|
677 | }
|
---|
678 | export function createElementCssSelector(elementName, attributes) {
|
---|
679 | const cssSelector = new CssSelector();
|
---|
680 | const elNameNoNs = splitNsName(elementName)[1];
|
---|
681 | cssSelector.setElement(elNameNoNs);
|
---|
682 | for (let i = 0; i < attributes.length; i++) {
|
---|
683 | const attrName = attributes[i][0];
|
---|
684 | const attrNameNoNs = splitNsName(attrName)[1];
|
---|
685 | const attrValue = attributes[i][1];
|
---|
686 | cssSelector.addAttribute(attrNameNoNs, attrValue);
|
---|
687 | if (attrName.toLowerCase() == CLASS_ATTR) {
|
---|
688 | const classes = splitClasses(attrValue);
|
---|
689 | classes.forEach(className => cssSelector.addClassName(className));
|
---|
690 | }
|
---|
691 | }
|
---|
692 | return cssSelector;
|
---|
693 | }
|
---|
694 | const EMPTY_ELEMENT_CONTEXT = new ElementContext(true, new SelectorMatcher(), null, null);
|
---|
695 | const NON_BINDABLE_VISITOR = new NonBindableVisitor();
|
---|
696 | function _isEmptyTextNode(node) {
|
---|
697 | return node instanceof html.Text && node.value.trim().length == 0;
|
---|
698 | }
|
---|
699 | export function removeSummaryDuplicates(items) {
|
---|
700 | const map = new Map();
|
---|
701 | items.forEach((item) => {
|
---|
702 | if (!map.get(item.type.reference)) {
|
---|
703 | map.set(item.type.reference, item);
|
---|
704 | }
|
---|
705 | });
|
---|
706 | return Array.from(map.values());
|
---|
707 | }
|
---|
708 | export function isEmptyExpression(ast) {
|
---|
709 | if (ast instanceof ASTWithSource) {
|
---|
710 | ast = ast.ast;
|
---|
711 | }
|
---|
712 | return ast instanceof EmptyExpr;
|
---|
713 | }
|
---|
714 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"template_parser.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/template_parser/template_parser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {CompileDirectiveMetadata, CompileDirectiveSummary, CompilePipeSummary, CompileTokenMetadata, CompileTypeMetadata} from '../compile_metadata';\nimport {CompileReflector} from '../compile_reflector';\nimport {CompilerConfig} from '../config';\nimport {SchemaMetadata} from '../core';\nimport {AST, ASTWithSource, EmptyExpr, ParsedEvent, ParsedProperty, ParsedVariable} from '../expression_parser/ast';\nimport {Parser} from '../expression_parser/parser';\nimport {createTokenForExternalReference, createTokenForReference, Identifiers} from '../identifiers';\nimport * as html from '../ml_parser/ast';\nimport {HtmlParser, ParseTreeResult} from '../ml_parser/html_parser';\nimport {removeWhitespaces, replaceNgsp} from '../ml_parser/html_whitespaces';\nimport {expandNodes} from '../ml_parser/icu_ast_expander';\nimport {InterpolationConfig} from '../ml_parser/interpolation_config';\nimport {isNgTemplate, splitNsName} from '../ml_parser/tags';\nimport {identifierName, ParseError, ParseErrorLevel, ParseSourceSpan, syntaxError} from '../parse_util';\nimport {ProviderElementContext, ProviderViewContext} from '../provider_analyzer';\nimport {ElementSchemaRegistry} from '../schema/element_schema_registry';\nimport {CssSelector, SelectorMatcher} from '../selector';\nimport {isStyleUrlResolvable} from '../style_url_resolver';\nimport {Console, newArray} from '../util';\n\nimport {BindingParser} from './binding_parser';\nimport * as t from './template_ast';\nimport {PreparsedElementType, preparseElement} from './template_preparser';\n\nconst BIND_NAME_REGEXP =\n    /^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.*))|\\[\\(([^\\)]+)\\)\\]|\\[([^\\]]+)\\]|\\(([^\\)]+)\\))$/;\n\n// Group 1 = \"bind-\"\nconst KW_BIND_IDX = 1;\n// Group 2 = \"let-\"\nconst KW_LET_IDX = 2;\n// Group 3 = \"ref-/#\"\nconst KW_REF_IDX = 3;\n// Group 4 = \"on-\"\nconst KW_ON_IDX = 4;\n// Group 5 = \"bindon-\"\nconst KW_BINDON_IDX = 5;\n// Group 6 = \"@\"\nconst KW_AT_IDX = 6;\n// Group 7 = the identifier after \"bind-\", \"let-\", \"ref-/#\", \"on-\", \"bindon-\" or \"@\"\nconst IDENT_KW_IDX = 7;\n// Group 8 = identifier inside [()]\nconst IDENT_BANANA_BOX_IDX = 8;\n// Group 9 = identifier inside []\nconst IDENT_PROPERTY_IDX = 9;\n// Group 10 = identifier inside ()\nconst IDENT_EVENT_IDX = 10;\n\nconst TEMPLATE_ATTR_PREFIX = '*';\nconst CLASS_ATTR = 'class';\n\nlet _TEXT_CSS_SELECTOR!: CssSelector;\nfunction TEXT_CSS_SELECTOR(): CssSelector {\n  if (!_TEXT_CSS_SELECTOR) {\n    _TEXT_CSS_SELECTOR = CssSelector.parse('*')[0];\n  }\n  return _TEXT_CSS_SELECTOR;\n}\n\nexport class TemplateParseError extends ParseError {\n  constructor(message: string, span: ParseSourceSpan, level: ParseErrorLevel) {\n    super(span, message, level);\n  }\n}\n\nexport class TemplateParseResult {\n  constructor(\n      public templateAst?: t.TemplateAst[], public usedPipes?: CompilePipeSummary[],\n      public errors?: ParseError[]) {}\n}\n\nexport class TemplateParser {\n  constructor(\n      private _config: CompilerConfig, private _reflector: CompileReflector,\n      private _exprParser: Parser, private _schemaRegistry: ElementSchemaRegistry,\n      private _htmlParser: HtmlParser, private _console: Console|null,\n      public transforms: t.TemplateAstVisitor[]) {}\n\n  public get expressionParser() {\n    return this._exprParser;\n  }\n\n  parse(\n      component: CompileDirectiveMetadata, template: string|ParseTreeResult,\n      directives: CompileDirectiveSummary[], pipes: CompilePipeSummary[], schemas: SchemaMetadata[],\n      templateUrl: string,\n      preserveWhitespaces: boolean): {template: t.TemplateAst[], pipes: CompilePipeSummary[]} {\n    const result = this.tryParse(\n        component, template, directives, pipes, schemas, templateUrl, preserveWhitespaces);\n    const warnings = result.errors!.filter(error => error.level === ParseErrorLevel.WARNING);\n\n    const errors = result.errors!.filter(error => error.level === ParseErrorLevel.ERROR);\n\n    if (warnings.length > 0) {\n      this._console?.warn(`Template parse warnings:\\n${warnings.join('\\n')}`);\n    }\n\n    if (errors.length > 0) {\n      const errorString = errors.join('\\n');\n      throw syntaxError(`Template parse errors:\\n${errorString}`, errors);\n    }\n\n    return {template: result.templateAst!, pipes: result.usedPipes!};\n  }\n\n  tryParse(\n      component: CompileDirectiveMetadata, template: string|ParseTreeResult,\n      directives: CompileDirectiveSummary[], pipes: CompilePipeSummary[], schemas: SchemaMetadata[],\n      templateUrl: string, preserveWhitespaces: boolean): TemplateParseResult {\n    let htmlParseResult = typeof template === 'string' ?\n        this._htmlParser!.parse(template, templateUrl, {\n          tokenizeExpansionForms: true,\n          interpolationConfig: this.getInterpolationConfig(component)\n        }) :\n        template;\n\n    if (!preserveWhitespaces) {\n      htmlParseResult = removeWhitespaces(htmlParseResult);\n    }\n\n    return this.tryParseHtml(\n        this.expandHtml(htmlParseResult), component, directives, pipes, schemas);\n  }\n\n  tryParseHtml(\n      htmlAstWithErrors: ParseTreeResult, component: CompileDirectiveMetadata,\n      directives: CompileDirectiveSummary[], pipes: CompilePipeSummary[],\n      schemas: SchemaMetadata[]): TemplateParseResult {\n    let result: t.TemplateAst[];\n    const errors = htmlAstWithErrors.errors;\n    const usedPipes: CompilePipeSummary[] = [];\n    if (htmlAstWithErrors.rootNodes.length > 0) {\n      const uniqDirectives = removeSummaryDuplicates(directives);\n      const uniqPipes = removeSummaryDuplicates(pipes);\n      const providerViewContext = new ProviderViewContext(this._reflector, component);\n      let interpolationConfig: InterpolationConfig = undefined!;\n      if (component.template && component.template.interpolation) {\n        interpolationConfig = {\n          start: component.template.interpolation[0],\n          end: component.template.interpolation[1]\n        };\n      }\n      const bindingParser = new BindingParser(\n          this._exprParser, interpolationConfig!, this._schemaRegistry, uniqPipes, errors);\n      const parseVisitor = new TemplateParseVisitor(\n          this._reflector, this._config, providerViewContext, uniqDirectives, bindingParser,\n          this._schemaRegistry, schemas, errors);\n      result = html.visitAll(parseVisitor, htmlAstWithErrors.rootNodes, EMPTY_ELEMENT_CONTEXT);\n      errors.push(...providerViewContext.errors);\n      usedPipes.push(...bindingParser.getUsedPipes());\n    } else {\n      result = [];\n    }\n    this._assertNoReferenceDuplicationOnTemplate(result, errors);\n\n    if (errors.length > 0) {\n      return new TemplateParseResult(result, usedPipes, errors);\n    }\n\n    if (this.transforms) {\n      this.transforms.forEach((transform: t.TemplateAstVisitor) => {\n        result = t.templateVisitAll(transform, result);\n      });\n    }\n\n    return new TemplateParseResult(result, usedPipes, errors);\n  }\n\n  expandHtml(htmlAstWithErrors: ParseTreeResult, forced: boolean = false): ParseTreeResult {\n    const errors: ParseError[] = htmlAstWithErrors.errors;\n\n    if (errors.length == 0 || forced) {\n      // Transform ICU messages to angular directives\n      const expandedHtmlAst = expandNodes(htmlAstWithErrors.rootNodes);\n      errors.push(...expandedHtmlAst.errors);\n      htmlAstWithErrors = new ParseTreeResult(expandedHtmlAst.nodes, errors);\n    }\n    return htmlAstWithErrors;\n  }\n\n  getInterpolationConfig(component: CompileDirectiveMetadata): InterpolationConfig|undefined {\n    if (component.template) {\n      return InterpolationConfig.fromArray(component.template.interpolation);\n    }\n    return undefined;\n  }\n\n  /** @internal */\n  _assertNoReferenceDuplicationOnTemplate(result: t.TemplateAst[], errors: TemplateParseError[]):\n      void {\n    const existingReferences: string[] = [];\n\n    result.filter(element => !!(<any>element).references)\n        .forEach(element => (<any>element).references.forEach((reference: t.ReferenceAst) => {\n          const name = reference.name;\n          if (existingReferences.indexOf(name) < 0) {\n            existingReferences.push(name);\n          } else {\n            const error = new TemplateParseError(\n                `Reference \"#${name}\" is defined several times`, reference.sourceSpan,\n                ParseErrorLevel.ERROR);\n            errors.push(error);\n          }\n        }));\n  }\n}\n\nclass TemplateParseVisitor implements html.Visitor {\n  selectorMatcher = new SelectorMatcher();\n  directivesIndex = new Map<CompileDirectiveSummary, number>();\n  ngContentCount = 0;\n  contentQueryStartId: number;\n\n  constructor(\n      private reflector: CompileReflector, private config: CompilerConfig,\n      public providerViewContext: ProviderViewContext, directives: CompileDirectiveSummary[],\n      private _bindingParser: BindingParser, private _schemaRegistry: ElementSchemaRegistry,\n      private _schemas: SchemaMetadata[], private _targetErrors: TemplateParseError[]) {\n    // Note: queries start with id 1 so we can use the number in a Bloom filter!\n    this.contentQueryStartId = providerViewContext.component.viewQueries.length + 1;\n    directives.forEach((directive, index) => {\n      const selector = CssSelector.parse(directive.selector!);\n      this.selectorMatcher.addSelectables(selector, directive);\n      this.directivesIndex.set(directive, index);\n    });\n  }\n\n  visitExpansion(expansion: html.Expansion, context: any): any {\n    return null;\n  }\n\n  visitExpansionCase(expansionCase: html.ExpansionCase, context: any): any {\n    return null;\n  }\n\n  visitText(text: html.Text, parent: ElementContext): any {\n    const ngContentIndex = parent.findNgContentIndex(TEXT_CSS_SELECTOR())!;\n    const valueNoNgsp = replaceNgsp(text.value);\n    const expr = this._bindingParser.parseInterpolation(valueNoNgsp, text.sourceSpan);\n    return expr ? new t.BoundTextAst(expr, ngContentIndex, text.sourceSpan) :\n                  new t.TextAst(valueNoNgsp, ngContentIndex, text.sourceSpan);\n  }\n\n  visitAttribute(attribute: html.Attribute, context: any): any {\n    return new t.AttrAst(attribute.name, attribute.value, attribute.sourceSpan);\n  }\n\n  visitComment(comment: html.Comment, context: any): any {\n    return null;\n  }\n\n  visitElement(element: html.Element, parent: ElementContext): any {\n    const queryStartIndex = this.contentQueryStartId;\n    const elName = element.name;\n    const preparsedElement = preparseElement(element);\n    if (preparsedElement.type === PreparsedElementType.SCRIPT ||\n        preparsedElement.type === PreparsedElementType.STYLE) {\n      // Skipping <script> for security reasons\n      // Skipping <style> as we already processed them\n      // in the StyleCompiler\n      return null;\n    }\n    if (preparsedElement.type === PreparsedElementType.STYLESHEET &&\n        isStyleUrlResolvable(preparsedElement.hrefAttr)) {\n      // Skipping stylesheets with either relative urls or package scheme as we already processed\n      // them in the StyleCompiler\n      return null;\n    }\n\n    const matchableAttrs: [string, string][] = [];\n    const elementOrDirectiveProps: ParsedProperty[] = [];\n    const elementOrDirectiveRefs: ElementOrDirectiveRef[] = [];\n    const elementVars: t.VariableAst[] = [];\n    const events: t.BoundEventAst[] = [];\n\n    const templateElementOrDirectiveProps: ParsedProperty[] = [];\n    const templateMatchableAttrs: [string, string][] = [];\n    const templateElementVars: t.VariableAst[] = [];\n\n    let hasInlineTemplates = false;\n    const attrs: t.AttrAst[] = [];\n    const isTemplateElement = isNgTemplate(element.name);\n\n    element.attrs.forEach(attr => {\n      const parsedVariables: ParsedVariable[] = [];\n      const hasBinding = this._parseAttr(\n          isTemplateElement, attr, matchableAttrs, elementOrDirectiveProps, events,\n          elementOrDirectiveRefs, elementVars);\n      elementVars.push(...parsedVariables.map(v => t.VariableAst.fromParsedVariable(v)));\n\n      let templateValue: string|undefined;\n      let templateKey: string|undefined;\n      const normalizedName = this._normalizeAttributeName(attr.name);\n\n      if (normalizedName.startsWith(TEMPLATE_ATTR_PREFIX)) {\n        templateValue = attr.value;\n        templateKey = normalizedName.substring(TEMPLATE_ATTR_PREFIX.length);\n      }\n\n      const hasTemplateBinding = templateValue != null;\n      if (hasTemplateBinding) {\n        if (hasInlineTemplates) {\n          this._reportError(\n              `Can't have multiple template bindings on one element. Use only one attribute prefixed with *`,\n              attr.sourceSpan);\n        }\n        hasInlineTemplates = true;\n        const parsedVariables: ParsedVariable[] = [];\n        const absoluteOffset = (attr.valueSpan || attr.sourceSpan).start.offset;\n        this._bindingParser.parseInlineTemplateBinding(\n            templateKey!, templateValue!, attr.sourceSpan, absoluteOffset, templateMatchableAttrs,\n            templateElementOrDirectiveProps, parsedVariables, false /* isIvyAst */);\n        templateElementVars.push(...parsedVariables.map(v => t.VariableAst.fromParsedVariable(v)));\n      }\n\n      if (!hasBinding && !hasTemplateBinding) {\n        // don't include the bindings as attributes as well in the AST\n        attrs.push(this.visitAttribute(attr, null));\n        matchableAttrs.push([attr.name, attr.value]);\n      }\n    });\n\n    const elementCssSelector = createElementCssSelector(elName, matchableAttrs);\n    const {directives: directiveMetas, matchElement} =\n        this._parseDirectives(this.selectorMatcher, elementCssSelector);\n    const references: t.ReferenceAst[] = [];\n    const boundDirectivePropNames = new Set<string>();\n    const directiveAsts = this._createDirectiveAsts(\n        isTemplateElement, element.name, directiveMetas, elementOrDirectiveProps,\n        elementOrDirectiveRefs, element.sourceSpan, references, boundDirectivePropNames);\n    const elementProps: t.BoundElementPropertyAst[] = this._createElementPropertyAsts(\n        element.name, elementOrDirectiveProps, boundDirectivePropNames);\n    const isViewRoot = parent.isTemplateElement || hasInlineTemplates;\n\n    const providerContext = new ProviderElementContext(\n        this.providerViewContext, parent.providerContext!, isViewRoot, directiveAsts, attrs,\n        references, isTemplateElement, queryStartIndex, element.sourceSpan);\n\n    const children: t.TemplateAst[] = html.visitAll(\n        preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this, element.children,\n        ElementContext.create(\n            isTemplateElement, directiveAsts,\n            isTemplateElement ? parent.providerContext! : providerContext));\n    providerContext.afterElement();\n    // Override the actual selector when the `ngProjectAs` attribute is provided\n    const projectionSelector = preparsedElement.projectAs != '' ?\n        CssSelector.parse(preparsedElement.projectAs)[0] :\n        elementCssSelector;\n    const ngContentIndex = parent.findNgContentIndex(projectionSelector)!;\n    let parsedElement: t.TemplateAst;\n\n    if (preparsedElement.type === PreparsedElementType.NG_CONTENT) {\n      // `<ng-content>` element\n      if (element.children && !element.children.every(_isEmptyTextNode)) {\n        this._reportError(`<ng-content> element cannot have content.`, element.sourceSpan);\n      }\n\n      parsedElement = new t.NgContentAst(\n          this.ngContentCount++, hasInlineTemplates ? null! : ngContentIndex, element.sourceSpan);\n    } else if (isTemplateElement) {\n      // `<ng-template>` element\n      this._assertAllEventsPublishedByDirectives(directiveAsts, events);\n      this._assertNoComponentsNorElementBindingsOnTemplate(\n          directiveAsts, elementProps, element.sourceSpan);\n\n      parsedElement = new t.EmbeddedTemplateAst(\n          attrs, events, references, elementVars, providerContext.transformedDirectiveAsts,\n          providerContext.transformProviders, providerContext.transformedHasViewContainer,\n          providerContext.queryMatches, children, hasInlineTemplates ? null! : ngContentIndex,\n          element.sourceSpan);\n    } else {\n      // element other than `<ng-content>` and `<ng-template>`\n      this._assertElementExists(matchElement, element);\n      this._assertOnlyOneComponent(directiveAsts, element.sourceSpan);\n\n      const ngContentIndex =\n          hasInlineTemplates ? null : parent.findNgContentIndex(projectionSelector);\n      parsedElement = new t.ElementAst(\n          elName, attrs, elementProps, events, references, providerContext.transformedDirectiveAsts,\n          providerContext.transformProviders, providerContext.transformedHasViewContainer,\n          providerContext.queryMatches, children, hasInlineTemplates ? null : ngContentIndex,\n          element.sourceSpan, element.endSourceSpan || null);\n    }\n\n    if (hasInlineTemplates) {\n      // The element as a *-attribute\n      const templateQueryStartIndex = this.contentQueryStartId;\n      const templateSelector = createElementCssSelector('ng-template', templateMatchableAttrs);\n      const {directives} = this._parseDirectives(this.selectorMatcher, templateSelector);\n      const templateBoundDirectivePropNames = new Set<string>();\n      const templateDirectiveAsts = this._createDirectiveAsts(\n          true, elName, directives, templateElementOrDirectiveProps, [], element.sourceSpan, [],\n          templateBoundDirectivePropNames);\n      const templateElementProps: t.BoundElementPropertyAst[] = this._createElementPropertyAsts(\n          elName, templateElementOrDirectiveProps, templateBoundDirectivePropNames);\n      this._assertNoComponentsNorElementBindingsOnTemplate(\n          templateDirectiveAsts, templateElementProps, element.sourceSpan);\n      const templateProviderContext = new ProviderElementContext(\n          this.providerViewContext, parent.providerContext!, parent.isTemplateElement,\n          templateDirectiveAsts, [], [], true, templateQueryStartIndex, element.sourceSpan);\n      templateProviderContext.afterElement();\n\n      parsedElement = new t.EmbeddedTemplateAst(\n          [], [], [], templateElementVars, templateProviderContext.transformedDirectiveAsts,\n          templateProviderContext.transformProviders,\n          templateProviderContext.transformedHasViewContainer, templateProviderContext.queryMatches,\n          [parsedElement], ngContentIndex, element.sourceSpan);\n    }\n\n    return parsedElement;\n  }\n\n  private _parseAttr(\n      isTemplateElement: boolean, attr: html.Attribute, targetMatchableAttrs: string[][],\n      targetProps: ParsedProperty[], targetEvents: t.BoundEventAst[],\n      targetRefs: ElementOrDirectiveRef[], targetVars: t.VariableAst[]): boolean {\n    const name = this._normalizeAttributeName(attr.name);\n    const value = attr.value;\n    const srcSpan = attr.sourceSpan;\n    const absoluteOffset = attr.valueSpan ? attr.valueSpan.start.offset : srcSpan.start.offset;\n\n    const boundEvents: ParsedEvent[] = [];\n    const bindParts = name.match(BIND_NAME_REGEXP);\n    let hasBinding = false;\n\n    if (bindParts !== null) {\n      hasBinding = true;\n      if (bindParts[KW_BIND_IDX] != null) {\n        this._bindingParser.parsePropertyBinding(\n            bindParts[IDENT_KW_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan,\n            targetMatchableAttrs, targetProps);\n\n      } else if (bindParts[KW_LET_IDX]) {\n        if (isTemplateElement) {\n          const identifier = bindParts[IDENT_KW_IDX];\n          this._parseVariable(identifier, value, srcSpan, targetVars);\n        } else {\n          this._reportError(`\"let-\" is only supported on ng-template elements.`, srcSpan);\n        }\n\n      } else if (bindParts[KW_REF_IDX]) {\n        const identifier = bindParts[IDENT_KW_IDX];\n        this._parseReference(identifier, value, srcSpan, targetRefs);\n\n      } else if (bindParts[KW_ON_IDX]) {\n        this._bindingParser.parseEvent(\n            bindParts[IDENT_KW_IDX], value, srcSpan, attr.valueSpan || srcSpan,\n            targetMatchableAttrs, boundEvents);\n\n      } else if (bindParts[KW_BINDON_IDX]) {\n        this._bindingParser.parsePropertyBinding(\n            bindParts[IDENT_KW_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan,\n            targetMatchableAttrs, targetProps);\n        this._parseAssignmentEvent(\n            bindParts[IDENT_KW_IDX], value, srcSpan, attr.valueSpan || srcSpan,\n            targetMatchableAttrs, boundEvents);\n\n      } else if (bindParts[KW_AT_IDX]) {\n        this._bindingParser.parseLiteralAttr(\n            name, value, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs,\n            targetProps);\n\n      } else if (bindParts[IDENT_BANANA_BOX_IDX]) {\n        this._bindingParser.parsePropertyBinding(\n            bindParts[IDENT_BANANA_BOX_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan,\n            targetMatchableAttrs, targetProps);\n        this._parseAssignmentEvent(\n            bindParts[IDENT_BANANA_BOX_IDX], value, srcSpan, attr.valueSpan || srcSpan,\n            targetMatchableAttrs, boundEvents);\n\n      } else if (bindParts[IDENT_PROPERTY_IDX]) {\n        this._bindingParser.parsePropertyBinding(\n            bindParts[IDENT_PROPERTY_IDX], value, false, srcSpan, absoluteOffset, attr.valueSpan,\n            targetMatchableAttrs, targetProps);\n\n      } else if (bindParts[IDENT_EVENT_IDX]) {\n        this._bindingParser.parseEvent(\n            bindParts[IDENT_EVENT_IDX], value, srcSpan, attr.valueSpan || srcSpan,\n            targetMatchableAttrs, boundEvents);\n      }\n    } else {\n      hasBinding = this._bindingParser.parsePropertyInterpolation(\n          name, value, srcSpan, attr.valueSpan, targetMatchableAttrs, targetProps);\n    }\n\n    if (!hasBinding) {\n      this._bindingParser.parseLiteralAttr(\n          name, value, srcSpan, absoluteOffset, attr.valueSpan, targetMatchableAttrs, targetProps);\n    }\n\n    targetEvents.push(...boundEvents.map(e => t.BoundEventAst.fromParsedEvent(e)));\n\n    return hasBinding;\n  }\n\n  private _normalizeAttributeName(attrName: string): string {\n    return /^data-/i.test(attrName) ? attrName.substring(5) : attrName;\n  }\n\n  private _parseVariable(\n      identifier: string, value: string, sourceSpan: ParseSourceSpan, targetVars: t.VariableAst[]) {\n    if (identifier.indexOf('-') > -1) {\n      this._reportError(`\"-\" is not allowed in variable names`, sourceSpan);\n    } else if (identifier.length === 0) {\n      this._reportError(`Variable does not have a name`, sourceSpan);\n    }\n\n    targetVars.push(new t.VariableAst(identifier, value, sourceSpan));\n  }\n\n  private _parseReference(\n      identifier: string, value: string, sourceSpan: ParseSourceSpan,\n      targetRefs: ElementOrDirectiveRef[]) {\n    if (identifier.indexOf('-') > -1) {\n      this._reportError(`\"-\" is not allowed in reference names`, sourceSpan);\n    } else if (identifier.length === 0) {\n      this._reportError(`Reference does not have a name`, sourceSpan);\n    }\n\n    targetRefs.push(new ElementOrDirectiveRef(identifier, value, sourceSpan));\n  }\n\n  private _parseAssignmentEvent(\n      name: string, expression: string, sourceSpan: ParseSourceSpan, valueSpan: ParseSourceSpan,\n      targetMatchableAttrs: string[][], targetEvents: ParsedEvent[]) {\n    this._bindingParser.parseEvent(\n        `${name}Change`, `${expression}=$event`, sourceSpan, valueSpan, targetMatchableAttrs,\n        targetEvents);\n  }\n\n  private _parseDirectives(selectorMatcher: SelectorMatcher, elementCssSelector: CssSelector):\n      {directives: CompileDirectiveSummary[], matchElement: boolean} {\n    // Need to sort the directives so that we get consistent results throughout,\n    // as selectorMatcher uses Maps inside.\n    // Also deduplicate directives as they might match more than one time!\n    const directives = newArray(this.directivesIndex.size);\n    // Whether any directive selector matches on the element name\n    let matchElement = false;\n\n    selectorMatcher.match(elementCssSelector, (selector, directive) => {\n      directives[this.directivesIndex.get(directive)!] = directive;\n      matchElement = matchElement || selector.hasElementSelector();\n    });\n\n    return {\n      directives: directives.filter(dir => !!dir),\n      matchElement,\n    };\n  }\n\n  private _createDirectiveAsts(\n      isTemplateElement: boolean, elementName: string, directives: CompileDirectiveSummary[],\n      props: ParsedProperty[], elementOrDirectiveRefs: ElementOrDirectiveRef[],\n      elementSourceSpan: ParseSourceSpan, targetReferences: t.ReferenceAst[],\n      targetBoundDirectivePropNames: Set<string>): t.DirectiveAst[] {\n    const matchedReferences = new Set<string>();\n    let component: CompileDirectiveSummary = null!;\n\n    const directiveAsts = directives.map((directive) => {\n      const sourceSpan = new ParseSourceSpan(\n          elementSourceSpan.start, elementSourceSpan.end, elementSourceSpan.fullStart,\n          `Directive ${identifierName(directive.type)}`);\n\n      if (directive.isComponent) {\n        component = directive;\n      }\n      const directiveProperties: t.BoundDirectivePropertyAst[] = [];\n      const boundProperties =\n          this._bindingParser.createDirectiveHostPropertyAsts(directive, elementName, sourceSpan)!;\n\n      let hostProperties =\n          boundProperties.map(prop => t.BoundElementPropertyAst.fromBoundProperty(prop));\n      // Note: We need to check the host properties here as well,\n      // as we don't know the element name in the DirectiveWrapperCompiler yet.\n      hostProperties = this._checkPropertiesInSchema(elementName, hostProperties);\n      const parsedEvents = this._bindingParser.createDirectiveHostEventAsts(directive, sourceSpan)!;\n      this._createDirectivePropertyAsts(\n          directive.inputs, props, directiveProperties, targetBoundDirectivePropNames);\n      elementOrDirectiveRefs.forEach((elOrDirRef) => {\n        if ((elOrDirRef.value.length === 0 && directive.isComponent) ||\n            (elOrDirRef.isReferenceToDirective(directive))) {\n          targetReferences.push(new t.ReferenceAst(\n              elOrDirRef.name, createTokenForReference(directive.type.reference), elOrDirRef.value,\n              elOrDirRef.sourceSpan));\n          matchedReferences.add(elOrDirRef.name);\n        }\n      });\n      const hostEvents = parsedEvents.map(e => t.BoundEventAst.fromParsedEvent(e));\n      const contentQueryStartId = this.contentQueryStartId;\n      this.contentQueryStartId += directive.queries.length;\n      return new t.DirectiveAst(\n          directive, directiveProperties, hostProperties, hostEvents, contentQueryStartId,\n          sourceSpan);\n    });\n\n    elementOrDirectiveRefs.forEach((elOrDirRef) => {\n      if (elOrDirRef.value.length > 0) {\n        if (!matchedReferences.has(elOrDirRef.name)) {\n          this._reportError(\n              `There is no directive with \"exportAs\" set to \"${elOrDirRef.value}\"`,\n              elOrDirRef.sourceSpan);\n        }\n      } else if (!component) {\n        let refToken: CompileTokenMetadata = null!;\n        if (isTemplateElement) {\n          refToken = createTokenForExternalReference(this.reflector, Identifiers.TemplateRef);\n        }\n        targetReferences.push(\n            new t.ReferenceAst(elOrDirRef.name, refToken, elOrDirRef.value, elOrDirRef.sourceSpan));\n      }\n    });\n    return directiveAsts;\n  }\n\n  private _createDirectivePropertyAsts(\n      directiveProperties: {[key: string]: string}, boundProps: ParsedProperty[],\n      targetBoundDirectiveProps: t.BoundDirectivePropertyAst[],\n      targetBoundDirectivePropNames: Set<string>) {\n    if (directiveProperties) {\n      const boundPropsByName = new Map<string, ParsedProperty>();\n      boundProps.forEach(boundProp => {\n        const prevValue = boundPropsByName.get(boundProp.name);\n        if (!prevValue || prevValue.isLiteral) {\n          // give [a]=\"b\" a higher precedence than a=\"b\" on the same element\n          boundPropsByName.set(boundProp.name, boundProp);\n        }\n      });\n\n      Object.keys(directiveProperties).forEach(dirProp => {\n        const elProp = directiveProperties[dirProp];\n        const boundProp = boundPropsByName.get(elProp);\n\n        // Bindings are optional, so this binding only needs to be set up if an expression is given.\n        if (boundProp) {\n          targetBoundDirectivePropNames.add(boundProp.name);\n          if (!isEmptyExpression(boundProp.expression)) {\n            targetBoundDirectiveProps.push(new t.BoundDirectivePropertyAst(\n                dirProp, boundProp.name, boundProp.expression, boundProp.sourceSpan));\n          }\n        }\n      });\n    }\n  }\n\n  private _createElementPropertyAsts(\n      elementName: string, props: ParsedProperty[],\n      boundDirectivePropNames: Set<string>): t.BoundElementPropertyAst[] {\n    const boundElementProps: t.BoundElementPropertyAst[] = [];\n\n    props.forEach((prop: ParsedProperty) => {\n      if (!prop.isLiteral && !boundDirectivePropNames.has(prop.name)) {\n        const boundProp = this._bindingParser.createBoundElementProperty(elementName, prop);\n        boundElementProps.push(t.BoundElementPropertyAst.fromBoundProperty(boundProp));\n      }\n    });\n    return this._checkPropertiesInSchema(elementName, boundElementProps);\n  }\n\n  private _findComponentDirectives(directives: t.DirectiveAst[]): t.DirectiveAst[] {\n    return directives.filter(directive => directive.directive.isComponent);\n  }\n\n  private _findComponentDirectiveNames(directives: t.DirectiveAst[]): string[] {\n    return this._findComponentDirectives(directives)\n        .map(directive => identifierName(directive.directive.type)!);\n  }\n\n  private _assertOnlyOneComponent(directives: t.DirectiveAst[], sourceSpan: ParseSourceSpan) {\n    const componentTypeNames = this._findComponentDirectiveNames(directives);\n    if (componentTypeNames.length > 1) {\n      this._reportError(\n          `More than one component matched on this element.\\n` +\n              `Make sure that only one component's selector can match a given element.\\n` +\n              `Conflicting components: ${componentTypeNames.join(',')}`,\n          sourceSpan);\n    }\n  }\n\n  /**\n   * Make sure that non-angular tags conform to the schemas.\n   *\n   * Note: An element is considered an angular tag when at least one directive selector matches the\n   * tag name.\n   *\n   * @param matchElement Whether any directive has matched on the tag name\n   * @param element the html element\n   */\n  private _assertElementExists(matchElement: boolean, element: html.Element) {\n    const elName = element.name.replace(/^:xhtml:/, '');\n\n    if (!matchElement && !this._schemaRegistry.hasElement(elName, this._schemas)) {\n      let errorMsg = `'${elName}' is not a known element:\\n`;\n      errorMsg += `1. If '${\n          elName}' is an Angular component, then verify that it is part of this module.\\n`;\n      if (elName.indexOf('-') > -1) {\n        errorMsg += `2. If '${\n            elName}' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.`;\n      } else {\n        errorMsg +=\n            `2. To allow any element add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;\n      }\n      this._reportError(errorMsg, element.sourceSpan);\n    }\n  }\n\n  private _assertNoComponentsNorElementBindingsOnTemplate(\n      directives: t.DirectiveAst[], elementProps: t.BoundElementPropertyAst[],\n      sourceSpan: ParseSourceSpan) {\n    const componentTypeNames: string[] = this._findComponentDirectiveNames(directives);\n    if (componentTypeNames.length > 0) {\n      this._reportError(\n          `Components on an embedded template: ${componentTypeNames.join(',')}`, sourceSpan);\n    }\n    elementProps.forEach(prop => {\n      this._reportError(\n          `Property binding ${\n              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\".`,\n          sourceSpan);\n    });\n  }\n\n  private _assertAllEventsPublishedByDirectives(\n      directives: t.DirectiveAst[], events: t.BoundEventAst[]) {\n    const allDirectiveEvents = new Set<string>();\n\n    directives.forEach(directive => {\n      Object.keys(directive.directive.outputs).forEach(k => {\n        const eventName = directive.directive.outputs[k];\n        allDirectiveEvents.add(eventName);\n      });\n    });\n\n    events.forEach(event => {\n      if (event.target != null || !allDirectiveEvents.has(event.name)) {\n        this._reportError(\n            `Event binding ${\n                event\n                    .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\".`,\n            event.sourceSpan);\n      }\n    });\n  }\n\n  private _checkPropertiesInSchema(elementName: string, boundProps: t.BoundElementPropertyAst[]):\n      t.BoundElementPropertyAst[] {\n    // Note: We can't filter out empty expressions before this method,\n    // as we still want to validate them!\n    return boundProps.filter((boundProp) => {\n      if (boundProp.type === t.PropertyBindingType.Property &&\n          !this._schemaRegistry.hasProperty(elementName, boundProp.name, this._schemas)) {\n        let errorMsg = `Can't bind to '${boundProp.name}' since it isn't a known property of '${\n            elementName}'.`;\n        if (elementName.startsWith('ng-')) {\n          errorMsg +=\n              `\\n1. If '${\n                  boundProp\n                      .name}' is an Angular directive, then add 'CommonModule' to the '@NgModule.imports' of this component.` +\n              `\\n2. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;\n        } else if (elementName.indexOf('-') > -1) {\n          errorMsg +=\n              `\\n1. If '${elementName}' is an Angular component and it has '${\n                  boundProp.name}' input, then verify that it is part of this module.` +\n              `\\n2. If '${\n                  elementName}' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.` +\n              `\\n3. To allow any property add 'NO_ERRORS_SCHEMA' to the '@NgModule.schemas' of this component.`;\n        }\n        this._reportError(errorMsg, boundProp.sourceSpan);\n      }\n      return !isEmptyExpression(boundProp.value);\n    });\n  }\n\n  private _reportError(\n      message: string, sourceSpan: ParseSourceSpan,\n      level: ParseErrorLevel = ParseErrorLevel.ERROR) {\n    this._targetErrors.push(new ParseError(sourceSpan, message, level));\n  }\n}\n\nclass NonBindableVisitor implements html.Visitor {\n  visitElement(ast: html.Element, parent: ElementContext): t.ElementAst|null {\n    const preparsedElement = preparseElement(ast);\n    if (preparsedElement.type === PreparsedElementType.SCRIPT ||\n        preparsedElement.type === PreparsedElementType.STYLE ||\n        preparsedElement.type === PreparsedElementType.STYLESHEET) {\n      // Skipping <script> for security reasons\n      // Skipping <style> and stylesheets as we already processed them\n      // in the StyleCompiler\n      return null;\n    }\n\n    const attrNameAndValues = ast.attrs.map((attr): [string, string] => [attr.name, attr.value]);\n    const selector = createElementCssSelector(ast.name, attrNameAndValues);\n    const ngContentIndex = parent.findNgContentIndex(selector);\n    const children: t.TemplateAst[] = html.visitAll(this, ast.children, EMPTY_ELEMENT_CONTEXT);\n    return new t.ElementAst(\n        ast.name, html.visitAll(this, ast.attrs), [], [], [], [], [], false, [], children,\n        ngContentIndex, ast.sourceSpan, ast.endSourceSpan);\n  }\n  visitComment(comment: html.Comment, context: any): any {\n    return null;\n  }\n\n  visitAttribute(attribute: html.Attribute, context: any): t.AttrAst {\n    return new t.AttrAst(attribute.name, attribute.value, attribute.sourceSpan);\n  }\n\n  visitText(text: html.Text, parent: ElementContext): t.TextAst {\n    const ngContentIndex = parent.findNgContentIndex(TEXT_CSS_SELECTOR())!;\n    return new t.TextAst(text.value, ngContentIndex, text.sourceSpan);\n  }\n\n  visitExpansion(expansion: html.Expansion, context: any): any {\n    return expansion;\n  }\n\n  visitExpansionCase(expansionCase: html.ExpansionCase, context: any): any {\n    return expansionCase;\n  }\n}\n\n/**\n * A reference to an element or directive in a template. E.g., the reference in this template:\n *\n * <div #myMenu=\"coolMenu\">\n *\n * would be {name: 'myMenu', value: 'coolMenu', sourceSpan: ...}\n */\nclass ElementOrDirectiveRef {\n  constructor(public name: string, public value: string, public sourceSpan: ParseSourceSpan) {}\n\n  /** Gets whether this is a reference to the given directive. */\n  isReferenceToDirective(directive: CompileDirectiveSummary) {\n    return splitExportAs(directive.exportAs).indexOf(this.value) !== -1;\n  }\n}\n\n/** Splits a raw, potentially comma-delimited `exportAs` value into an array of names. */\nfunction splitExportAs(exportAs: string|null): string[] {\n  return exportAs ? exportAs.split(',').map(e => e.trim()) : [];\n}\n\nexport function splitClasses(classAttrValue: string): string[] {\n  return classAttrValue.trim().split(/\\s+/g);\n}\n\nclass ElementContext {\n  static create(\n      isTemplateElement: boolean, directives: t.DirectiveAst[],\n      providerContext: ProviderElementContext): ElementContext {\n    const matcher = new SelectorMatcher();\n    let wildcardNgContentIndex: number = null!;\n    const component = directives.find(directive => directive.directive.isComponent);\n    if (component) {\n      const ngContentSelectors = component.directive.template !.ngContentSelectors;\n      for (let i = 0; i < ngContentSelectors.length; i++) {\n        const selector = ngContentSelectors[i];\n        if (selector === '*') {\n          wildcardNgContentIndex = i;\n        } else {\n          matcher.addSelectables(CssSelector.parse(ngContentSelectors[i]), i);\n        }\n      }\n    }\n    return new ElementContext(isTemplateElement, matcher, wildcardNgContentIndex, providerContext);\n  }\n  constructor(\n      public isTemplateElement: boolean, private _ngContentIndexMatcher: SelectorMatcher,\n      private _wildcardNgContentIndex: number|null,\n      public providerContext: ProviderElementContext|null) {}\n\n  findNgContentIndex(selector: CssSelector): number|null {\n    const ngContentIndices: number[] = [];\n    this._ngContentIndexMatcher.match(selector, (selector, ngContentIndex) => {\n      ngContentIndices.push(ngContentIndex);\n    });\n    ngContentIndices.sort();\n    if (this._wildcardNgContentIndex != null) {\n      ngContentIndices.push(this._wildcardNgContentIndex);\n    }\n    return ngContentIndices.length > 0 ? ngContentIndices[0] : null;\n  }\n}\n\nexport function createElementCssSelector(\n    elementName: string, attributes: [string, string][]): CssSelector {\n  const cssSelector = new CssSelector();\n  const elNameNoNs = splitNsName(elementName)[1];\n\n  cssSelector.setElement(elNameNoNs);\n\n  for (let i = 0; i < attributes.length; i++) {\n    const attrName = attributes[i][0];\n    const attrNameNoNs = splitNsName(attrName)[1];\n    const attrValue = attributes[i][1];\n\n    cssSelector.addAttribute(attrNameNoNs, attrValue);\n    if (attrName.toLowerCase() == CLASS_ATTR) {\n      const classes = splitClasses(attrValue);\n      classes.forEach(className => cssSelector.addClassName(className));\n    }\n  }\n  return cssSelector;\n}\n\nconst EMPTY_ELEMENT_CONTEXT = new ElementContext(true, new SelectorMatcher(), null, null);\nconst NON_BINDABLE_VISITOR = new NonBindableVisitor();\n\nfunction _isEmptyTextNode(node: html.Node): boolean {\n  return node instanceof html.Text && node.value.trim().length == 0;\n}\n\nexport function removeSummaryDuplicates<T extends {type: CompileTypeMetadata}>(items: T[]): T[] {\n  const map = new Map<any, T>();\n\n  items.forEach((item) => {\n    if (!map.get(item.type.reference)) {\n      map.set(item.type.reference, item);\n    }\n  });\n\n  return Array.from(map.values());\n}\n\nexport function isEmptyExpression(ast: AST): boolean {\n  if (ast instanceof ASTWithSource) {\n    ast = ast.ast;\n  }\n  return ast instanceof EmptyExpr;\n}\n"]} |
---|