2016-07-21 16:56:58 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-07-21 16:56:58 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-08-01 15:19:09 -04:00
|
|
|
import * as html from '../../src/ml_parser/ast';
|
|
|
|
import {ParseTreeResult} from '../../src/ml_parser/html_parser';
|
2016-07-21 16:56:58 -04:00
|
|
|
import {ParseLocation} from '../../src/parse_util';
|
|
|
|
|
|
|
|
export function humanizeDom(parseResult: ParseTreeResult, addSourceSpan: boolean = false): any[] {
|
|
|
|
if (parseResult.errors.length > 0) {
|
2016-11-12 08:08:58 -05:00
|
|
|
const errorString = parseResult.errors.join('\n');
|
2016-08-25 03:50:16 -04:00
|
|
|
throw new Error(`Unexpected parse errors:\n${errorString}`);
|
2016-07-21 16:56:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return humanizeNodes(parseResult.rootNodes, addSourceSpan);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function humanizeDomSourceSpans(parseResult: ParseTreeResult): any[] {
|
|
|
|
return humanizeDom(parseResult, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function humanizeNodes(nodes: html.Node[], addSourceSpan: boolean = false): any[] {
|
2016-11-12 08:08:58 -05:00
|
|
|
const humanizer = new _Humanizer(addSourceSpan);
|
2016-07-21 16:56:58 -04:00
|
|
|
html.visitAll(humanizer, nodes);
|
|
|
|
return humanizer.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function humanizeLineColumn(location: ParseLocation): string {
|
|
|
|
return `${location.line}:${location.col}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
class _Humanizer implements html.Visitor {
|
|
|
|
result: any[] = [];
|
|
|
|
elDepth: number = 0;
|
|
|
|
|
2017-09-22 13:51:03 -04:00
|
|
|
constructor(private includeSourceSpan: boolean) {}
|
2016-07-21 16:56:58 -04:00
|
|
|
|
|
|
|
visitElement(element: html.Element, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res = this._appendContext(element, [html.Element, element.name, this.elDepth++]);
|
fix(compiler): properly associate source spans for implicitly closed elements (#38126)
HTML is very lenient when it comes to closing elements, so Angular's parser has
rules that specify which elements are implicitly closed when closing a tag.
The parser keeps track of the nesting of tag names using a stack and parsing
a closing tag will pop as many elements off the stack as possible, provided
that the elements can be implicitly closed.
For example, consider the following templates:
- `<div><br></div>`, the `<br>` is implicitly closed when parsing `</div>`,
because `<br>` is a void element.
- `<div><p></div>`, the `<p>` is implicitly closed when parsing `</div>`,
as `<p>` is allowed to be closed by the closing of its parent element.
- `<ul><li>A <li>B</ul>`, the first `<li>` is implicitly closed when parsing
the second `<li>`, whereas the second `<li>` would be implicitly closed when
parsing the `</ul>`.
In all the cases above the parsed structure would be correct, however the source
span of the closing `</div>` would incorrectly be assigned to the element that
is implicitly closed. The problem was that closing an element would associate
the source span with the element at the top of the stack, however this may not
be the element that is actually being closed if some elements would be
implicitly closed.
This commit fixes the issue by assigning the end source span with the element
on the stack that is actually being closed. Any implicitly closed elements that
are popped off the stack will not be assigned an end source span, as the
implicit closing implies that no ending element is present.
Note that there is a difference between self-closed elements such as `<input/>`
and implicitly closed elements such as `<input>`. The former does have an end
source span (identical to its start source span) whereas the latter does not.
Fixes #36118
Resolves FW-2004
PR Close #38126
2020-07-17 18:05:11 -04:00
|
|
|
if (this.includeSourceSpan) {
|
2020-08-26 07:21:29 -04:00
|
|
|
res.push(element.startSourceSpan.toString() ?? null);
|
fix(compiler): properly associate source spans for implicitly closed elements (#38126)
HTML is very lenient when it comes to closing elements, so Angular's parser has
rules that specify which elements are implicitly closed when closing a tag.
The parser keeps track of the nesting of tag names using a stack and parsing
a closing tag will pop as many elements off the stack as possible, provided
that the elements can be implicitly closed.
For example, consider the following templates:
- `<div><br></div>`, the `<br>` is implicitly closed when parsing `</div>`,
because `<br>` is a void element.
- `<div><p></div>`, the `<p>` is implicitly closed when parsing `</div>`,
as `<p>` is allowed to be closed by the closing of its parent element.
- `<ul><li>A <li>B</ul>`, the first `<li>` is implicitly closed when parsing
the second `<li>`, whereas the second `<li>` would be implicitly closed when
parsing the `</ul>`.
In all the cases above the parsed structure would be correct, however the source
span of the closing `</div>` would incorrectly be assigned to the element that
is implicitly closed. The problem was that closing an element would associate
the source span with the element at the top of the stack, however this may not
be the element that is actually being closed if some elements would be
implicitly closed.
This commit fixes the issue by assigning the end source span with the element
on the stack that is actually being closed. Any implicitly closed elements that
are popped off the stack will not be assigned an end source span, as the
implicit closing implies that no ending element is present.
Note that there is a difference between self-closed elements such as `<input/>`
and implicitly closed elements such as `<input>`. The former does have an end
source span (identical to its start source span) whereas the latter does not.
Fixes #36118
Resolves FW-2004
PR Close #38126
2020-07-17 18:05:11 -04:00
|
|
|
res.push(element.endSourceSpan?.toString() ?? null);
|
|
|
|
}
|
2016-07-21 16:56:58 -04:00
|
|
|
this.result.push(res);
|
|
|
|
html.visitAll(this, element.attrs);
|
|
|
|
html.visitAll(this, element.children);
|
|
|
|
this.elDepth--;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitAttribute(attribute: html.Attribute, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res = this._appendContext(attribute, [html.Attribute, attribute.name, attribute.value]);
|
2016-07-21 16:56:58 -04:00
|
|
|
this.result.push(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
visitText(text: html.Text, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res = this._appendContext(text, [html.Text, text.value, this.elDepth]);
|
2016-07-21 16:56:58 -04:00
|
|
|
this.result.push(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
visitComment(comment: html.Comment, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res = this._appendContext(comment, [html.Comment, comment.value, this.elDepth]);
|
2016-07-21 16:56:58 -04:00
|
|
|
this.result.push(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
visitExpansion(expansion: html.Expansion, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res = this._appendContext(
|
2016-07-21 16:56:58 -04:00
|
|
|
expansion, [html.Expansion, expansion.switchValue, expansion.type, this.elDepth++]);
|
|
|
|
this.result.push(res);
|
|
|
|
html.visitAll(this, expansion.cases);
|
|
|
|
this.elDepth--;
|
|
|
|
}
|
|
|
|
|
|
|
|
visitExpansionCase(expansionCase: html.ExpansionCase, context: any): any {
|
2016-11-12 08:08:58 -05:00
|
|
|
const res =
|
2016-07-21 16:56:58 -04:00
|
|
|
this._appendContext(expansionCase, [html.ExpansionCase, expansionCase.value, this.elDepth]);
|
|
|
|
this.result.push(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _appendContext(ast: html.Node, input: any[]): any[] {
|
|
|
|
if (!this.includeSourceSpan) return input;
|
2020-08-26 07:21:29 -04:00
|
|
|
input.push(ast.sourceSpan.toString());
|
2016-07-21 16:56:58 -04:00
|
|
|
return input;
|
|
|
|
}
|
|
|
|
}
|