source: trip-planner-front/node_modules/@angular/compiler/esm2015/src/render3/r3_template_transform.js

Last change on this file was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 78.0 KB
Line 
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 */
8import * as html from '../ml_parser/ast';
9import { replaceNgsp } from '../ml_parser/html_whitespaces';
10import { isNgTemplate } from '../ml_parser/tags';
11import { ParseError, ParseErrorLevel, ParseSourceSpan } from '../parse_util';
12import { isStyleUrlResolvable } from '../style_url_resolver';
13import { PreparsedElementType, preparseElement } from '../template_parser/template_preparser';
14import * as t from './r3_ast';
15import { I18N_ICU_VAR_PREFIX, isI18nRootNode } from './view/i18n/util';
16const BIND_NAME_REGEXP = /^(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.*)$/;
17// Group 1 = "bind-"
18const KW_BIND_IDX = 1;
19// Group 2 = "let-"
20const KW_LET_IDX = 2;
21// Group 3 = "ref-/#"
22const KW_REF_IDX = 3;
23// Group 4 = "on-"
24const KW_ON_IDX = 4;
25// Group 5 = "bindon-"
26const KW_BINDON_IDX = 5;
27// Group 6 = "@"
28const KW_AT_IDX = 6;
29// Group 7 = the identifier after "bind-", "let-", "ref-/#", "on-", "bindon-" or "@"
30const IDENT_KW_IDX = 7;
31const BINDING_DELIMS = {
32 BANANA_BOX: { start: '[(', end: ')]' },
33 PROPERTY: { start: '[', end: ']' },
34 EVENT: { start: '(', end: ')' },
35};
36const TEMPLATE_ATTR_PREFIX = '*';
37export function htmlAstToRender3Ast(htmlNodes, bindingParser, options) {
38 const transformer = new HtmlAstToIvyAst(bindingParser, options);
39 const ivyNodes = html.visitAll(transformer, htmlNodes);
40 // Errors might originate in either the binding parser or the html to ivy transformer
41 const allErrors = bindingParser.errors.concat(transformer.errors);
42 const result = {
43 nodes: ivyNodes,
44 errors: allErrors,
45 styleUrls: transformer.styleUrls,
46 styles: transformer.styles,
47 ngContentSelectors: transformer.ngContentSelectors
48 };
49 if (options.collectCommentNodes) {
50 result.commentNodes = transformer.commentNodes;
51 }
52 return result;
53}
54class HtmlAstToIvyAst {
55 constructor(bindingParser, options) {
56 this.bindingParser = bindingParser;
57 this.options = options;
58 this.errors = [];
59 this.styles = [];
60 this.styleUrls = [];
61 this.ngContentSelectors = [];
62 // This array will be populated if `Render3ParseOptions['collectCommentNodes']` is true
63 this.commentNodes = [];
64 this.inI18nBlock = false;
65 }
66 // HTML visitor
67 visitElement(element) {
68 const isI18nRootElement = isI18nRootNode(element.i18n);
69 if (isI18nRootElement) {
70 if (this.inI18nBlock) {
71 this.reportError('Cannot mark an element as translatable inside of a translatable section. Please remove the nested i18n marker.', element.sourceSpan);
72 }
73 this.inI18nBlock = true;
74 }
75 const preparsedElement = preparseElement(element);
76 if (preparsedElement.type === PreparsedElementType.SCRIPT) {
77 return null;
78 }
79 else if (preparsedElement.type === PreparsedElementType.STYLE) {
80 const contents = textContents(element);
81 if (contents !== null) {
82 this.styles.push(contents);
83 }
84 return null;
85 }
86 else if (preparsedElement.type === PreparsedElementType.STYLESHEET &&
87 isStyleUrlResolvable(preparsedElement.hrefAttr)) {
88 this.styleUrls.push(preparsedElement.hrefAttr);
89 return null;
90 }
91 // Whether the element is a `<ng-template>`
92 const isTemplateElement = isNgTemplate(element.name);
93 const parsedProperties = [];
94 const boundEvents = [];
95 const variables = [];
96 const references = [];
97 const attributes = [];
98 const i18nAttrsMeta = {};
99 const templateParsedProperties = [];
100 const templateVariables = [];
101 // Whether the element has any *-attribute
102 let elementHasInlineTemplate = false;
103 for (const attribute of element.attrs) {
104 let hasBinding = false;
105 const normalizedName = normalizeAttributeName(attribute.name);
106 // `*attr` defines template bindings
107 let isTemplateBinding = false;
108 if (attribute.i18n) {
109 i18nAttrsMeta[attribute.name] = attribute.i18n;
110 }
111 if (normalizedName.startsWith(TEMPLATE_ATTR_PREFIX)) {
112 // *-attributes
113 if (elementHasInlineTemplate) {
114 this.reportError(`Can't have multiple template bindings on one element. Use only one attribute prefixed with *`, attribute.sourceSpan);
115 }
116 isTemplateBinding = true;
117 elementHasInlineTemplate = true;
118 const templateValue = attribute.value;
119 const templateKey = normalizedName.substring(TEMPLATE_ATTR_PREFIX.length);
120 const parsedVariables = [];
121 const absoluteValueOffset = attribute.valueSpan ?
122 attribute.valueSpan.start.offset :
123 // If there is no value span the attribute does not have a value, like `attr` in
124 //`<div attr></div>`. In this case, point to one character beyond the last character of
125 // the attribute name.
126 attribute.sourceSpan.start.offset + attribute.name.length;
127 this.bindingParser.parseInlineTemplateBinding(templateKey, templateValue, attribute.sourceSpan, absoluteValueOffset, [], templateParsedProperties, parsedVariables, true /* isIvyAst */);
128 templateVariables.push(...parsedVariables.map(v => new t.Variable(v.name, v.value, v.sourceSpan, v.keySpan, v.valueSpan)));
129 }
130 else {
131 // Check for variables, events, property bindings, interpolation
132 hasBinding = this.parseAttribute(isTemplateElement, attribute, [], parsedProperties, boundEvents, variables, references);
133 }
134 if (!hasBinding && !isTemplateBinding) {
135 // don't include the bindings as attributes as well in the AST
136 attributes.push(this.visitAttribute(attribute));
137 }
138 }
139 const children = html.visitAll(preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this, element.children);
140 let parsedElement;
141 if (preparsedElement.type === PreparsedElementType.NG_CONTENT) {
142 // `<ng-content>`
143 if (element.children &&
144 !element.children.every((node) => isEmptyTextNode(node) || isCommentNode(node))) {
145 this.reportError(`<ng-content> element cannot have content.`, element.sourceSpan);
146 }
147 const selector = preparsedElement.selectAttr;
148 const attrs = element.attrs.map(attr => this.visitAttribute(attr));
149 parsedElement = new t.Content(selector, attrs, element.sourceSpan, element.i18n);
150 this.ngContentSelectors.push(selector);
151 }
152 else if (isTemplateElement) {
153 // `<ng-template>`
154 const attrs = this.extractAttributes(element.name, parsedProperties, i18nAttrsMeta);
155 parsedElement = new t.Template(element.name, attributes, attrs.bound, boundEvents, [ /* no template attributes */], children, references, variables, element.sourceSpan, element.startSourceSpan, element.endSourceSpan, element.i18n);
156 }
157 else {
158 const attrs = this.extractAttributes(element.name, parsedProperties, i18nAttrsMeta);
159 parsedElement = new t.Element(element.name, attributes, attrs.bound, boundEvents, children, references, element.sourceSpan, element.startSourceSpan, element.endSourceSpan, element.i18n);
160 }
161 if (elementHasInlineTemplate) {
162 // If this node is an inline-template (e.g. has *ngFor) then we need to create a template
163 // node that contains this node.
164 // Moreover, if the node is an element, then we need to hoist its attributes to the template
165 // node for matching against content projection selectors.
166 const attrs = this.extractAttributes('ng-template', templateParsedProperties, i18nAttrsMeta);
167 const templateAttrs = [];
168 attrs.literal.forEach(attr => templateAttrs.push(attr));
169 attrs.bound.forEach(attr => templateAttrs.push(attr));
170 const hoistedAttrs = parsedElement instanceof t.Element ?
171 {
172 attributes: parsedElement.attributes,
173 inputs: parsedElement.inputs,
174 outputs: parsedElement.outputs,
175 } :
176 { attributes: [], inputs: [], outputs: [] };
177 // For <ng-template>s with structural directives on them, avoid passing i18n information to
178 // the wrapping template to prevent unnecessary i18n instructions from being generated. The
179 // necessary i18n meta information will be extracted from child elements.
180 const i18n = isTemplateElement && isI18nRootElement ? undefined : element.i18n;
181 // TODO(pk): test for this case
182 parsedElement = new t.Template(parsedElement.name, hoistedAttrs.attributes, hoistedAttrs.inputs, hoistedAttrs.outputs, templateAttrs, [parsedElement], [ /* no references */], templateVariables, element.sourceSpan, element.startSourceSpan, element.endSourceSpan, i18n);
183 }
184 if (isI18nRootElement) {
185 this.inI18nBlock = false;
186 }
187 return parsedElement;
188 }
189 visitAttribute(attribute) {
190 return new t.TextAttribute(attribute.name, attribute.value, attribute.sourceSpan, attribute.keySpan, attribute.valueSpan, attribute.i18n);
191 }
192 visitText(text) {
193 return this._visitTextWithInterpolation(text.value, text.sourceSpan, text.i18n);
194 }
195 visitExpansion(expansion) {
196 if (!expansion.i18n) {
197 // do not generate Icu in case it was created
198 // outside of i18n block in a template
199 return null;
200 }
201 if (!isI18nRootNode(expansion.i18n)) {
202 throw new Error(`Invalid type "${expansion.i18n.constructor}" for "i18n" property of ${expansion.sourceSpan.toString()}. Expected a "Message"`);
203 }
204 const message = expansion.i18n;
205 const vars = {};
206 const placeholders = {};
207 // extract VARs from ICUs - we process them separately while
208 // assembling resulting message via goog.getMsg function, since
209 // we need to pass them to top-level goog.getMsg call
210 Object.keys(message.placeholders).forEach(key => {
211 const value = message.placeholders[key];
212 if (key.startsWith(I18N_ICU_VAR_PREFIX)) {
213 // Currently when the `plural` or `select` keywords in an ICU contain trailing spaces (e.g.
214 // `{count, select , ...}`), these spaces are also included into the key names in ICU vars
215 // (e.g. "VAR_SELECT "). These trailing spaces are not desirable, since they will later be
216 // converted into `_` symbols while normalizing placeholder names, which might lead to
217 // mismatches at runtime (i.e. placeholder will not be replaced with the correct value).
218 const formattedKey = key.trim();
219 const ast = this.bindingParser.parseInterpolationExpression(value.text, value.sourceSpan);
220 vars[formattedKey] = new t.BoundText(ast, value.sourceSpan);
221 }
222 else {
223 placeholders[key] = this._visitTextWithInterpolation(value.text, value.sourceSpan);
224 }
225 });
226 return new t.Icu(vars, placeholders, expansion.sourceSpan, message);
227 }
228 visitExpansionCase(expansionCase) {
229 return null;
230 }
231 visitComment(comment) {
232 if (this.options.collectCommentNodes) {
233 this.commentNodes.push(new t.Comment(comment.value || '', comment.sourceSpan));
234 }
235 return null;
236 }
237 // convert view engine `ParsedProperty` to a format suitable for IVY
238 extractAttributes(elementName, properties, i18nPropsMeta) {
239 const bound = [];
240 const literal = [];
241 properties.forEach(prop => {
242 const i18n = i18nPropsMeta[prop.name];
243 if (prop.isLiteral) {
244 literal.push(new t.TextAttribute(prop.name, prop.expression.source || '', prop.sourceSpan, prop.keySpan, prop.valueSpan, i18n));
245 }
246 else {
247 // Note that validation is skipped and property mapping is disabled
248 // due to the fact that we need to make sure a given prop is not an
249 // input of a directive and directive matching happens at runtime.
250 const bep = this.bindingParser.createBoundElementProperty(elementName, prop, /* skipValidation */ true, /* mapPropertyName */ false);
251 bound.push(t.BoundAttribute.fromBoundElementProperty(bep, i18n));
252 }
253 });
254 return { bound, literal };
255 }
256 parseAttribute(isTemplateElement, attribute, matchableAttributes, parsedProperties, boundEvents, variables, references) {
257 const name = normalizeAttributeName(attribute.name);
258 const value = attribute.value;
259 const srcSpan = attribute.sourceSpan;
260 const absoluteOffset = attribute.valueSpan ? attribute.valueSpan.start.offset : srcSpan.start.offset;
261 function createKeySpan(srcSpan, prefix, identifier) {
262 // We need to adjust the start location for the keySpan to account for the removed 'data-'
263 // prefix from `normalizeAttributeName`.
264 const normalizationAdjustment = attribute.name.length - name.length;
265 const keySpanStart = srcSpan.start.moveBy(prefix.length + normalizationAdjustment);
266 const keySpanEnd = keySpanStart.moveBy(identifier.length);
267 return new ParseSourceSpan(keySpanStart, keySpanEnd, keySpanStart, identifier);
268 }
269 const bindParts = name.match(BIND_NAME_REGEXP);
270 if (bindParts) {
271 if (bindParts[KW_BIND_IDX] != null) {
272 const identifier = bindParts[IDENT_KW_IDX];
273 const keySpan = createKeySpan(srcSpan, bindParts[KW_BIND_IDX], identifier);
274 this.bindingParser.parsePropertyBinding(identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
275 }
276 else if (bindParts[KW_LET_IDX]) {
277 if (isTemplateElement) {
278 const identifier = bindParts[IDENT_KW_IDX];
279 const keySpan = createKeySpan(srcSpan, bindParts[KW_LET_IDX], identifier);
280 this.parseVariable(identifier, value, srcSpan, keySpan, attribute.valueSpan, variables);
281 }
282 else {
283 this.reportError(`"let-" is only supported on ng-template elements.`, srcSpan);
284 }
285 }
286 else if (bindParts[KW_REF_IDX]) {
287 const identifier = bindParts[IDENT_KW_IDX];
288 const keySpan = createKeySpan(srcSpan, bindParts[KW_REF_IDX], identifier);
289 this.parseReference(identifier, value, srcSpan, keySpan, attribute.valueSpan, references);
290 }
291 else if (bindParts[KW_ON_IDX]) {
292 const events = [];
293 const identifier = bindParts[IDENT_KW_IDX];
294 const keySpan = createKeySpan(srcSpan, bindParts[KW_ON_IDX], identifier);
295 this.bindingParser.parseEvent(identifier, value, srcSpan, attribute.valueSpan || srcSpan, matchableAttributes, events, keySpan);
296 addEvents(events, boundEvents);
297 }
298 else if (bindParts[KW_BINDON_IDX]) {
299 const identifier = bindParts[IDENT_KW_IDX];
300 const keySpan = createKeySpan(srcSpan, bindParts[KW_BINDON_IDX], identifier);
301 this.bindingParser.parsePropertyBinding(identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
302 this.parseAssignmentEvent(identifier, value, srcSpan, attribute.valueSpan, matchableAttributes, boundEvents, keySpan);
303 }
304 else if (bindParts[KW_AT_IDX]) {
305 const keySpan = createKeySpan(srcSpan, '', name);
306 this.bindingParser.parseLiteralAttr(name, value, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
307 }
308 return true;
309 }
310 // We didn't see a kw-prefixed property binding, but we have not yet checked
311 // for the []/()/[()] syntax.
312 let delims = null;
313 if (name.startsWith(BINDING_DELIMS.BANANA_BOX.start)) {
314 delims = BINDING_DELIMS.BANANA_BOX;
315 }
316 else if (name.startsWith(BINDING_DELIMS.PROPERTY.start)) {
317 delims = BINDING_DELIMS.PROPERTY;
318 }
319 else if (name.startsWith(BINDING_DELIMS.EVENT.start)) {
320 delims = BINDING_DELIMS.EVENT;
321 }
322 if (delims !== null &&
323 // NOTE: older versions of the parser would match a start/end delimited
324 // binding iff the property name was terminated by the ending delimiter
325 // and the identifier in the binding was non-empty.
326 // TODO(ayazhafiz): update this to handle malformed bindings.
327 name.endsWith(delims.end) && name.length > delims.start.length + delims.end.length) {
328 const identifier = name.substring(delims.start.length, name.length - delims.end.length);
329 const keySpan = createKeySpan(srcSpan, delims.start, identifier);
330 if (delims.start === BINDING_DELIMS.BANANA_BOX.start) {
331 this.bindingParser.parsePropertyBinding(identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
332 this.parseAssignmentEvent(identifier, value, srcSpan, attribute.valueSpan, matchableAttributes, boundEvents, keySpan);
333 }
334 else if (delims.start === BINDING_DELIMS.PROPERTY.start) {
335 this.bindingParser.parsePropertyBinding(identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
336 }
337 else {
338 const events = [];
339 this.bindingParser.parseEvent(identifier, value, srcSpan, attribute.valueSpan || srcSpan, matchableAttributes, events, keySpan);
340 addEvents(events, boundEvents);
341 }
342 return true;
343 }
344 // No explicit binding found.
345 const keySpan = createKeySpan(srcSpan, '' /* prefix */, name);
346 const hasBinding = this.bindingParser.parsePropertyInterpolation(name, value, srcSpan, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);
347 return hasBinding;
348 }
349 _visitTextWithInterpolation(value, sourceSpan, i18n) {
350 const valueNoNgsp = replaceNgsp(value);
351 const expr = this.bindingParser.parseInterpolation(valueNoNgsp, sourceSpan);
352 return expr ? new t.BoundText(expr, sourceSpan, i18n) : new t.Text(valueNoNgsp, sourceSpan);
353 }
354 parseVariable(identifier, value, sourceSpan, keySpan, valueSpan, variables) {
355 if (identifier.indexOf('-') > -1) {
356 this.reportError(`"-" is not allowed in variable names`, sourceSpan);
357 }
358 else if (identifier.length === 0) {
359 this.reportError(`Variable does not have a name`, sourceSpan);
360 }
361 variables.push(new t.Variable(identifier, value, sourceSpan, keySpan, valueSpan));
362 }
363 parseReference(identifier, value, sourceSpan, keySpan, valueSpan, references) {
364 if (identifier.indexOf('-') > -1) {
365 this.reportError(`"-" is not allowed in reference names`, sourceSpan);
366 }
367 else if (identifier.length === 0) {
368 this.reportError(`Reference does not have a name`, sourceSpan);
369 }
370 else if (references.some(reference => reference.name === identifier)) {
371 this.reportError(`Reference "#${identifier}" is defined more than once`, sourceSpan);
372 }
373 references.push(new t.Reference(identifier, value, sourceSpan, keySpan, valueSpan));
374 }
375 parseAssignmentEvent(name, expression, sourceSpan, valueSpan, targetMatchableAttrs, boundEvents, keySpan) {
376 const events = [];
377 this.bindingParser.parseEvent(`${name}Change`, `${expression}=$event`, sourceSpan, valueSpan || sourceSpan, targetMatchableAttrs, events, keySpan);
378 addEvents(events, boundEvents);
379 }
380 reportError(message, sourceSpan, level = ParseErrorLevel.ERROR) {
381 this.errors.push(new ParseError(sourceSpan, message, level));
382 }
383}
384class NonBindableVisitor {
385 visitElement(ast) {
386 const preparsedElement = preparseElement(ast);
387 if (preparsedElement.type === PreparsedElementType.SCRIPT ||
388 preparsedElement.type === PreparsedElementType.STYLE ||
389 preparsedElement.type === PreparsedElementType.STYLESHEET) {
390 // Skipping <script> for security reasons
391 // Skipping <style> and stylesheets as we already processed them
392 // in the StyleCompiler
393 return null;
394 }
395 const children = html.visitAll(this, ast.children, null);
396 return new t.Element(ast.name, html.visitAll(this, ast.attrs),
397 /* inputs */ [], /* outputs */ [], children, /* references */ [], ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);
398 }
399 visitComment(comment) {
400 return null;
401 }
402 visitAttribute(attribute) {
403 return new t.TextAttribute(attribute.name, attribute.value, attribute.sourceSpan, attribute.keySpan, attribute.valueSpan, attribute.i18n);
404 }
405 visitText(text) {
406 return new t.Text(text.value, text.sourceSpan);
407 }
408 visitExpansion(expansion) {
409 return null;
410 }
411 visitExpansionCase(expansionCase) {
412 return null;
413 }
414}
415const NON_BINDABLE_VISITOR = new NonBindableVisitor();
416function normalizeAttributeName(attrName) {
417 return /^data-/i.test(attrName) ? attrName.substring(5) : attrName;
418}
419function addEvents(events, boundEvents) {
420 boundEvents.push(...events.map(e => t.BoundEvent.fromParsedEvent(e)));
421}
422function isEmptyTextNode(node) {
423 return node instanceof html.Text && node.value.trim().length == 0;
424}
425function isCommentNode(node) {
426 return node instanceof html.Comment;
427}
428function textContents(node) {
429 if (node.children.length !== 1 || !(node.children[0] instanceof html.Text)) {
430 return null;
431 }
432 else {
433 return node.children[0].value;
434 }
435}
436//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_template_transform.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_template_transform.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 {ParsedEvent, ParsedProperty, ParsedVariable} from '../expression_parser/ast';\nimport * as i18n from '../i18n/i18n_ast';\nimport * as html from '../ml_parser/ast';\nimport {replaceNgsp} from '../ml_parser/html_whitespaces';\nimport {isNgTemplate} from '../ml_parser/tags';\nimport {ParseError, ParseErrorLevel, ParseSourceSpan} from '../parse_util';\nimport {isStyleUrlResolvable} from '../style_url_resolver';\nimport {BindingParser} from '../template_parser/binding_parser';\nimport {PreparsedElementType, preparseElement} from '../template_parser/template_preparser';\n\nimport * as t from './r3_ast';\nimport {I18N_ICU_VAR_PREFIX, isI18nRootNode} from './view/i18n/util';\n\nconst BIND_NAME_REGEXP = /^(?:(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\nconst BINDING_DELIMS = {\n  BANANA_BOX: {start: '[(', end: ')]'},\n  PROPERTY: {start: '[', end: ']'},\n  EVENT: {start: '(', end: ')'},\n};\n\nconst TEMPLATE_ATTR_PREFIX = '*';\n\n// Result of the html AST to Ivy AST transformation\nexport interface Render3ParseResult {\n  nodes: t.Node[];\n  errors: ParseError[];\n  styles: string[];\n  styleUrls: string[];\n  ngContentSelectors: string[];\n  // Will be defined if `Render3ParseOptions['collectCommentNodes']` is true\n  commentNodes?: t.Comment[];\n}\n\ninterface Render3ParseOptions {\n  collectCommentNodes: boolean;\n}\n\nexport function htmlAstToRender3Ast(\n    htmlNodes: html.Node[], bindingParser: BindingParser,\n    options: Render3ParseOptions): Render3ParseResult {\n  const transformer = new HtmlAstToIvyAst(bindingParser, options);\n  const ivyNodes = html.visitAll(transformer, htmlNodes);\n\n  // Errors might originate in either the binding parser or the html to ivy transformer\n  const allErrors = bindingParser.errors.concat(transformer.errors);\n\n  const result: Render3ParseResult = {\n    nodes: ivyNodes,\n    errors: allErrors,\n    styleUrls: transformer.styleUrls,\n    styles: transformer.styles,\n    ngContentSelectors: transformer.ngContentSelectors\n  };\n  if (options.collectCommentNodes) {\n    result.commentNodes = transformer.commentNodes;\n  }\n  return result;\n}\n\nclass HtmlAstToIvyAst implements html.Visitor {\n  errors: ParseError[] = [];\n  styles: string[] = [];\n  styleUrls: string[] = [];\n  ngContentSelectors: string[] = [];\n  // This array will be populated if `Render3ParseOptions['collectCommentNodes']` is true\n  commentNodes: t.Comment[] = [];\n  private inI18nBlock: boolean = false;\n\n  constructor(private bindingParser: BindingParser, private options: Render3ParseOptions) {}\n\n  // HTML visitor\n  visitElement(element: html.Element): t.Node|null {\n    const isI18nRootElement = isI18nRootNode(element.i18n);\n    if (isI18nRootElement) {\n      if (this.inI18nBlock) {\n        this.reportError(\n            'Cannot mark an element as translatable inside of a translatable section. Please remove the nested i18n marker.',\n            element.sourceSpan);\n      }\n      this.inI18nBlock = true;\n    }\n    const preparsedElement = preparseElement(element);\n    if (preparsedElement.type === PreparsedElementType.SCRIPT) {\n      return null;\n    } else if (preparsedElement.type === PreparsedElementType.STYLE) {\n      const contents = textContents(element);\n      if (contents !== null) {\n        this.styles.push(contents);\n      }\n      return null;\n    } else if (\n        preparsedElement.type === PreparsedElementType.STYLESHEET &&\n        isStyleUrlResolvable(preparsedElement.hrefAttr)) {\n      this.styleUrls.push(preparsedElement.hrefAttr);\n      return null;\n    }\n\n    // Whether the element is a `<ng-template>`\n    const isTemplateElement = isNgTemplate(element.name);\n\n    const parsedProperties: ParsedProperty[] = [];\n    const boundEvents: t.BoundEvent[] = [];\n    const variables: t.Variable[] = [];\n    const references: t.Reference[] = [];\n    const attributes: t.TextAttribute[] = [];\n    const i18nAttrsMeta: {[key: string]: i18n.I18nMeta} = {};\n\n    const templateParsedProperties: ParsedProperty[] = [];\n    const templateVariables: t.Variable[] = [];\n\n    // Whether the element has any *-attribute\n    let elementHasInlineTemplate = false;\n\n    for (const attribute of element.attrs) {\n      let hasBinding = false;\n      const normalizedName = normalizeAttributeName(attribute.name);\n\n      // `*attr` defines template bindings\n      let isTemplateBinding = false;\n\n      if (attribute.i18n) {\n        i18nAttrsMeta[attribute.name] = attribute.i18n;\n      }\n\n      if (normalizedName.startsWith(TEMPLATE_ATTR_PREFIX)) {\n        // *-attributes\n        if (elementHasInlineTemplate) {\n          this.reportError(\n              `Can't have multiple template bindings on one element. Use only one attribute prefixed with *`,\n              attribute.sourceSpan);\n        }\n        isTemplateBinding = true;\n        elementHasInlineTemplate = true;\n        const templateValue = attribute.value;\n        const templateKey = normalizedName.substring(TEMPLATE_ATTR_PREFIX.length);\n\n        const parsedVariables: ParsedVariable[] = [];\n        const absoluteValueOffset = attribute.valueSpan ?\n            attribute.valueSpan.start.offset :\n            // If there is no value span the attribute does not have a value, like `attr` in\n            //`<div attr></div>`. In this case, point to one character beyond the last character of\n            // the attribute name.\n            attribute.sourceSpan.start.offset + attribute.name.length;\n\n        this.bindingParser.parseInlineTemplateBinding(\n            templateKey, templateValue, attribute.sourceSpan, absoluteValueOffset, [],\n            templateParsedProperties, parsedVariables, true /* isIvyAst */);\n        templateVariables.push(...parsedVariables.map(\n            v => new t.Variable(v.name, v.value, v.sourceSpan, v.keySpan, v.valueSpan)));\n      } else {\n        // Check for variables, events, property bindings, interpolation\n        hasBinding = this.parseAttribute(\n            isTemplateElement, attribute, [], parsedProperties, boundEvents, variables, references);\n      }\n\n      if (!hasBinding && !isTemplateBinding) {\n        // don't include the bindings as attributes as well in the AST\n        attributes.push(this.visitAttribute(attribute));\n      }\n    }\n\n    const children: t.Node[] =\n        html.visitAll(preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this, element.children);\n\n    let parsedElement: t.Node|undefined;\n    if (preparsedElement.type === PreparsedElementType.NG_CONTENT) {\n      // `<ng-content>`\n      if (element.children &&\n          !element.children.every(\n              (node: html.Node) => isEmptyTextNode(node) || isCommentNode(node))) {\n        this.reportError(`<ng-content> element cannot have content.`, element.sourceSpan);\n      }\n      const selector = preparsedElement.selectAttr;\n      const attrs: t.TextAttribute[] = element.attrs.map(attr => this.visitAttribute(attr));\n      parsedElement = new t.Content(selector, attrs, element.sourceSpan, element.i18n);\n\n      this.ngContentSelectors.push(selector);\n    } else if (isTemplateElement) {\n      // `<ng-template>`\n      const attrs = this.extractAttributes(element.name, parsedProperties, i18nAttrsMeta);\n\n      parsedElement = new t.Template(\n          element.name, attributes, attrs.bound, boundEvents, [/* no template attributes */],\n          children, references, variables, element.sourceSpan, element.startSourceSpan,\n          element.endSourceSpan, element.i18n);\n    } else {\n      const attrs = this.extractAttributes(element.name, parsedProperties, i18nAttrsMeta);\n      parsedElement = new t.Element(\n          element.name, attributes, attrs.bound, boundEvents, children, references,\n          element.sourceSpan, element.startSourceSpan, element.endSourceSpan, element.i18n);\n    }\n\n    if (elementHasInlineTemplate) {\n      // If this node is an inline-template (e.g. has *ngFor) then we need to create a template\n      // node that contains this node.\n      // Moreover, if the node is an element, then we need to hoist its attributes to the template\n      // node for matching against content projection selectors.\n      const attrs = this.extractAttributes('ng-template', templateParsedProperties, i18nAttrsMeta);\n      const templateAttrs: (t.TextAttribute|t.BoundAttribute)[] = [];\n      attrs.literal.forEach(attr => templateAttrs.push(attr));\n      attrs.bound.forEach(attr => templateAttrs.push(attr));\n      const hoistedAttrs = parsedElement instanceof t.Element ?\n          {\n            attributes: parsedElement.attributes,\n            inputs: parsedElement.inputs,\n            outputs: parsedElement.outputs,\n          } :\n          {attributes: [], inputs: [], outputs: []};\n\n      // For <ng-template>s with structural directives on them, avoid passing i18n information to\n      // the wrapping template to prevent unnecessary i18n instructions from being generated. The\n      // necessary i18n meta information will be extracted from child elements.\n      const i18n = isTemplateElement && isI18nRootElement ? undefined : element.i18n;\n\n      // TODO(pk): test for this case\n      parsedElement = new t.Template(\n          (parsedElement as t.Element | t.Content).name, hoistedAttrs.attributes,\n          hoistedAttrs.inputs, hoistedAttrs.outputs, templateAttrs, [parsedElement],\n          [/* no references */], templateVariables, element.sourceSpan, element.startSourceSpan,\n          element.endSourceSpan, i18n);\n    }\n    if (isI18nRootElement) {\n      this.inI18nBlock = false;\n    }\n    return parsedElement;\n  }\n\n  visitAttribute(attribute: html.Attribute): t.TextAttribute {\n    return new t.TextAttribute(\n        attribute.name, attribute.value, attribute.sourceSpan, attribute.keySpan,\n        attribute.valueSpan, attribute.i18n);\n  }\n\n  visitText(text: html.Text): t.Node {\n    return this._visitTextWithInterpolation(text.value, text.sourceSpan, text.i18n);\n  }\n\n  visitExpansion(expansion: html.Expansion): t.Icu|null {\n    if (!expansion.i18n) {\n      // do not generate Icu in case it was created\n      // outside of i18n block in a template\n      return null;\n    }\n    if (!isI18nRootNode(expansion.i18n)) {\n      throw new Error(`Invalid type \"${expansion.i18n.constructor}\" for \"i18n\" property of ${\n          expansion.sourceSpan.toString()}. Expected a \"Message\"`);\n    }\n    const message = expansion.i18n;\n    const vars: {[name: string]: t.BoundText} = {};\n    const placeholders: {[name: string]: t.Text|t.BoundText} = {};\n    // extract VARs from ICUs - we process them separately while\n    // assembling resulting message via goog.getMsg function, since\n    // we need to pass them to top-level goog.getMsg call\n    Object.keys(message.placeholders).forEach(key => {\n      const value = message.placeholders[key];\n      if (key.startsWith(I18N_ICU_VAR_PREFIX)) {\n        // Currently when the `plural` or `select` keywords in an ICU contain trailing spaces (e.g.\n        // `{count, select , ...}`), these spaces are also included into the key names in ICU vars\n        // (e.g. \"VAR_SELECT \"). These trailing spaces are not desirable, since they will later be\n        // converted into `_` symbols while normalizing placeholder names, which might lead to\n        // mismatches at runtime (i.e. placeholder will not be replaced with the correct value).\n        const formattedKey = key.trim();\n\n        const ast = this.bindingParser.parseInterpolationExpression(value.text, value.sourceSpan);\n\n        vars[formattedKey] = new t.BoundText(ast, value.sourceSpan);\n      } else {\n        placeholders[key] = this._visitTextWithInterpolation(value.text, value.sourceSpan);\n      }\n    });\n    return new t.Icu(vars, placeholders, expansion.sourceSpan, message);\n  }\n\n  visitExpansionCase(expansionCase: html.ExpansionCase): null {\n    return null;\n  }\n\n  visitComment(comment: html.Comment): null {\n    if (this.options.collectCommentNodes) {\n      this.commentNodes.push(new t.Comment(comment.value || '', comment.sourceSpan));\n    }\n    return null;\n  }\n\n  // convert view engine `ParsedProperty` to a format suitable for IVY\n  private extractAttributes(\n      elementName: string, properties: ParsedProperty[],\n      i18nPropsMeta: {[key: string]: i18n.I18nMeta}):\n      {bound: t.BoundAttribute[], literal: t.TextAttribute[]} {\n    const bound: t.BoundAttribute[] = [];\n    const literal: t.TextAttribute[] = [];\n\n    properties.forEach(prop => {\n      const i18n = i18nPropsMeta[prop.name];\n      if (prop.isLiteral) {\n        literal.push(new t.TextAttribute(\n            prop.name, prop.expression.source || '', prop.sourceSpan, prop.keySpan, prop.valueSpan,\n            i18n));\n      } else {\n        // Note that validation is skipped and property mapping is disabled\n        // due to the fact that we need to make sure a given prop is not an\n        // input of a directive and directive matching happens at runtime.\n        const bep = this.bindingParser.createBoundElementProperty(\n            elementName, prop, /* skipValidation */ true, /* mapPropertyName */ false);\n        bound.push(t.BoundAttribute.fromBoundElementProperty(bep, i18n));\n      }\n    });\n\n    return {bound, literal};\n  }\n\n  private parseAttribute(\n      isTemplateElement: boolean, attribute: html.Attribute, matchableAttributes: string[][],\n      parsedProperties: ParsedProperty[], boundEvents: t.BoundEvent[], variables: t.Variable[],\n      references: t.Reference[]) {\n    const name = normalizeAttributeName(attribute.name);\n    const value = attribute.value;\n    const srcSpan = attribute.sourceSpan;\n    const absoluteOffset =\n        attribute.valueSpan ? attribute.valueSpan.start.offset : srcSpan.start.offset;\n\n    function createKeySpan(srcSpan: ParseSourceSpan, prefix: string, identifier: string) {\n      // We need to adjust the start location for the keySpan to account for the removed 'data-'\n      // prefix from `normalizeAttributeName`.\n      const normalizationAdjustment = attribute.name.length - name.length;\n      const keySpanStart = srcSpan.start.moveBy(prefix.length + normalizationAdjustment);\n      const keySpanEnd = keySpanStart.moveBy(identifier.length);\n      return new ParseSourceSpan(keySpanStart, keySpanEnd, keySpanStart, identifier);\n    }\n\n    const bindParts = name.match(BIND_NAME_REGEXP);\n\n    if (bindParts) {\n      if (bindParts[KW_BIND_IDX] != null) {\n        const identifier = bindParts[IDENT_KW_IDX];\n        const keySpan = createKeySpan(srcSpan, bindParts[KW_BIND_IDX], identifier);\n        this.bindingParser.parsePropertyBinding(\n            identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan,\n            matchableAttributes, parsedProperties, keySpan);\n\n      } else if (bindParts[KW_LET_IDX]) {\n        if (isTemplateElement) {\n          const identifier = bindParts[IDENT_KW_IDX];\n          const keySpan = createKeySpan(srcSpan, bindParts[KW_LET_IDX], identifier);\n          this.parseVariable(identifier, value, srcSpan, keySpan, attribute.valueSpan, variables);\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        const keySpan = createKeySpan(srcSpan, bindParts[KW_REF_IDX], identifier);\n        this.parseReference(identifier, value, srcSpan, keySpan, attribute.valueSpan, references);\n      } else if (bindParts[KW_ON_IDX]) {\n        const events: ParsedEvent[] = [];\n        const identifier = bindParts[IDENT_KW_IDX];\n        const keySpan = createKeySpan(srcSpan, bindParts[KW_ON_IDX], identifier);\n        this.bindingParser.parseEvent(\n            identifier, value, srcSpan, attribute.valueSpan || srcSpan, matchableAttributes, events,\n            keySpan);\n        addEvents(events, boundEvents);\n      } else if (bindParts[KW_BINDON_IDX]) {\n        const identifier = bindParts[IDENT_KW_IDX];\n        const keySpan = createKeySpan(srcSpan, bindParts[KW_BINDON_IDX], identifier);\n        this.bindingParser.parsePropertyBinding(\n            identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan,\n            matchableAttributes, parsedProperties, keySpan);\n        this.parseAssignmentEvent(\n            identifier, value, srcSpan, attribute.valueSpan, matchableAttributes, boundEvents,\n            keySpan);\n      } else if (bindParts[KW_AT_IDX]) {\n        const keySpan = createKeySpan(srcSpan, '', name);\n        this.bindingParser.parseLiteralAttr(\n            name, value, srcSpan, absoluteOffset, attribute.valueSpan, matchableAttributes,\n            parsedProperties, keySpan);\n      }\n      return true;\n    }\n\n    // We didn't see a kw-prefixed property binding, but we have not yet checked\n    // for the []/()/[()] syntax.\n    let delims: {start: string, end: string}|null = null;\n    if (name.startsWith(BINDING_DELIMS.BANANA_BOX.start)) {\n      delims = BINDING_DELIMS.BANANA_BOX;\n    } else if (name.startsWith(BINDING_DELIMS.PROPERTY.start)) {\n      delims = BINDING_DELIMS.PROPERTY;\n    } else if (name.startsWith(BINDING_DELIMS.EVENT.start)) {\n      delims = BINDING_DELIMS.EVENT;\n    }\n    if (delims !== null &&\n        // NOTE: older versions of the parser would match a start/end delimited\n        // binding iff the property name was terminated by the ending delimiter\n        // and the identifier in the binding was non-empty.\n        // TODO(ayazhafiz): update this to handle malformed bindings.\n        name.endsWith(delims.end) && name.length > delims.start.length + delims.end.length) {\n      const identifier = name.substring(delims.start.length, name.length - delims.end.length);\n      const keySpan = createKeySpan(srcSpan, delims.start, identifier);\n      if (delims.start === BINDING_DELIMS.BANANA_BOX.start) {\n        this.bindingParser.parsePropertyBinding(\n            identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan,\n            matchableAttributes, parsedProperties, keySpan);\n        this.parseAssignmentEvent(\n            identifier, value, srcSpan, attribute.valueSpan, matchableAttributes, boundEvents,\n            keySpan);\n      } else if (delims.start === BINDING_DELIMS.PROPERTY.start) {\n        this.bindingParser.parsePropertyBinding(\n            identifier, value, false, srcSpan, absoluteOffset, attribute.valueSpan,\n            matchableAttributes, parsedProperties, keySpan);\n      } else {\n        const events: ParsedEvent[] = [];\n        this.bindingParser.parseEvent(\n            identifier, value, srcSpan, attribute.valueSpan || srcSpan, matchableAttributes, events,\n            keySpan);\n        addEvents(events, boundEvents);\n      }\n\n      return true;\n    }\n\n    // No explicit binding found.\n    const keySpan = createKeySpan(srcSpan, '' /* prefix */, name);\n    const hasBinding = this.bindingParser.parsePropertyInterpolation(\n        name, value, srcSpan, attribute.valueSpan, matchableAttributes, parsedProperties, keySpan);\n    return hasBinding;\n  }\n\n  private _visitTextWithInterpolation(\n      value: string, sourceSpan: ParseSourceSpan, i18n?: i18n.I18nMeta): t.Text|t.BoundText {\n    const valueNoNgsp = replaceNgsp(value);\n    const expr = this.bindingParser.parseInterpolation(valueNoNgsp, sourceSpan);\n    return expr ? new t.BoundText(expr, sourceSpan, i18n) : new t.Text(valueNoNgsp, sourceSpan);\n  }\n\n  private parseVariable(\n      identifier: string, value: string, sourceSpan: ParseSourceSpan, keySpan: ParseSourceSpan,\n      valueSpan: ParseSourceSpan|undefined, variables: t.Variable[]) {\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    variables.push(new t.Variable(identifier, value, sourceSpan, keySpan, valueSpan));\n  }\n\n  private parseReference(\n      identifier: string, value: string, sourceSpan: ParseSourceSpan, keySpan: ParseSourceSpan,\n      valueSpan: ParseSourceSpan|undefined, references: t.Reference[]) {\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    } else if (references.some(reference => reference.name === identifier)) {\n      this.reportError(`Reference \"#${identifier}\" is defined more than once`, sourceSpan);\n    }\n\n    references.push(new t.Reference(identifier, value, sourceSpan, keySpan, valueSpan));\n  }\n\n  private parseAssignmentEvent(\n      name: string, expression: string, sourceSpan: ParseSourceSpan,\n      valueSpan: ParseSourceSpan|undefined, targetMatchableAttrs: string[][],\n      boundEvents: t.BoundEvent[], keySpan: ParseSourceSpan) {\n    const events: ParsedEvent[] = [];\n    this.bindingParser.parseEvent(\n        `${name}Change`, `${expression}=$event`, sourceSpan, valueSpan || sourceSpan,\n        targetMatchableAttrs, events, keySpan);\n    addEvents(events, boundEvents);\n  }\n\n  private reportError(\n      message: string, sourceSpan: ParseSourceSpan,\n      level: ParseErrorLevel = ParseErrorLevel.ERROR) {\n    this.errors.push(new ParseError(sourceSpan, message, level));\n  }\n}\n\nclass NonBindableVisitor implements html.Visitor {\n  visitElement(ast: html.Element): t.Element|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 children: t.Node[] = html.visitAll(this, ast.children, null);\n    return new t.Element(\n        ast.name, html.visitAll(this, ast.attrs) as t.TextAttribute[],\n        /* inputs */[], /* outputs */[], children,  /* references */[], ast.sourceSpan,\n        ast.startSourceSpan, ast.endSourceSpan);\n  }\n\n  visitComment(comment: html.Comment): any {\n    return null;\n  }\n\n  visitAttribute(attribute: html.Attribute): t.TextAttribute {\n    return new t.TextAttribute(\n        attribute.name, attribute.value, attribute.sourceSpan, attribute.keySpan,\n        attribute.valueSpan, attribute.i18n);\n  }\n\n  visitText(text: html.Text): t.Text {\n    return new t.Text(text.value, text.sourceSpan);\n  }\n\n  visitExpansion(expansion: html.Expansion): any {\n    return null;\n  }\n\n  visitExpansionCase(expansionCase: html.ExpansionCase): any {\n    return null;\n  }\n}\n\nconst NON_BINDABLE_VISITOR = new NonBindableVisitor();\n\nfunction normalizeAttributeName(attrName: string): string {\n  return /^data-/i.test(attrName) ? attrName.substring(5) : attrName;\n}\n\nfunction addEvents(events: ParsedEvent[], boundEvents: t.BoundEvent[]) {\n  boundEvents.push(...events.map(e => t.BoundEvent.fromParsedEvent(e)));\n}\n\nfunction isEmptyTextNode(node: html.Node): boolean {\n  return node instanceof html.Text && node.value.trim().length == 0;\n}\n\nfunction isCommentNode(node: html.Node): boolean {\n  return node instanceof html.Comment;\n}\n\nfunction textContents(node: html.Element): string|null {\n  if (node.children.length !== 1 || !(node.children[0] instanceof html.Text)) {\n    return null;\n  } else {\n    return (node.children[0] as html.Text).value;\n  }\n}\n"]}
Note: See TracBrowser for help on using the repository browser.