Currently, the spans of expressions are recorded only relative to the template node that they reside in, not their source file. Introduce a `sourceSpan` property on expression ASTs that records the location of an expression relative to the entire source code file that it is in. This may allow for reducing duplication of effort in ngtsc/typecheck/src/diagnostics later on as well. Child of #31898 PR Close #31897
134 lines
4.0 KiB
TypeScript
134 lines
4.0 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
|
|
import {AbsoluteSourceSpan} from '@angular/compiler';
|
|
import * as e from '../../../src/expression_parser/ast';
|
|
import * as t from '../../../src/render3/r3_ast';
|
|
import {unparse} from '../../expression_parser/utils/unparser';
|
|
|
|
type HumanizedExpressionSource = [string, AbsoluteSourceSpan];
|
|
class ExpressionSourceHumanizer extends e.RecursiveAstVisitor implements t.Visitor {
|
|
result: HumanizedExpressionSource[] = [];
|
|
|
|
private recordAst(ast: e.AST) { this.result.push([unparse(ast), ast.sourceSpan]); }
|
|
|
|
visitASTWithSource(ast: e.ASTWithSource) {
|
|
this.recordAst(ast);
|
|
this.visitAll([ast.ast], null);
|
|
}
|
|
visitBinary(ast: e.Binary) {
|
|
this.recordAst(ast);
|
|
super.visitBinary(ast, null);
|
|
}
|
|
visitChain(ast: e.Chain) {
|
|
this.recordAst(ast);
|
|
super.visitChain(ast, null);
|
|
}
|
|
visitConditional(ast: e.Conditional) {
|
|
this.recordAst(ast);
|
|
super.visitConditional(ast, null);
|
|
}
|
|
visitFunctionCall(ast: e.FunctionCall) {
|
|
this.recordAst(ast);
|
|
super.visitFunctionCall(ast, null);
|
|
}
|
|
visitImplicitReceiver(ast: e.ImplicitReceiver) {
|
|
this.recordAst(ast);
|
|
super.visitImplicitReceiver(ast, null);
|
|
}
|
|
visitInterpolation(ast: e.Interpolation) {
|
|
this.recordAst(ast);
|
|
super.visitInterpolation(ast, null);
|
|
}
|
|
visitKeyedRead(ast: e.KeyedRead) {
|
|
this.recordAst(ast);
|
|
super.visitKeyedRead(ast, null);
|
|
}
|
|
visitKeyedWrite(ast: e.KeyedWrite) {
|
|
this.recordAst(ast);
|
|
super.visitKeyedWrite(ast, null);
|
|
}
|
|
visitLiteralPrimitive(ast: e.LiteralPrimitive) {
|
|
this.recordAst(ast);
|
|
super.visitLiteralPrimitive(ast, null);
|
|
}
|
|
visitLiteralArray(ast: e.LiteralArray) {
|
|
this.recordAst(ast);
|
|
super.visitLiteralArray(ast, null);
|
|
}
|
|
visitLiteralMap(ast: e.LiteralMap) {
|
|
this.recordAst(ast);
|
|
super.visitLiteralMap(ast, null);
|
|
}
|
|
visitMethodCall(ast: e.MethodCall) {
|
|
this.recordAst(ast);
|
|
super.visitMethodCall(ast, null);
|
|
}
|
|
visitNonNullAssert(ast: e.NonNullAssert) {
|
|
this.recordAst(ast);
|
|
super.visitNonNullAssert(ast, null);
|
|
}
|
|
visitPipe(ast: e.BindingPipe) {
|
|
this.recordAst(ast);
|
|
super.visitPipe(ast, null);
|
|
}
|
|
visitPrefixNot(ast: e.PrefixNot) {
|
|
this.recordAst(ast);
|
|
super.visitPrefixNot(ast, null);
|
|
}
|
|
visitPropertyRead(ast: e.PropertyRead) {
|
|
this.recordAst(ast);
|
|
super.visitPropertyRead(ast, null);
|
|
}
|
|
visitPropertyWrite(ast: e.PropertyWrite) {
|
|
this.recordAst(ast);
|
|
super.visitPropertyWrite(ast, null);
|
|
}
|
|
visitSafeMethodCall(ast: e.SafeMethodCall) {
|
|
this.recordAst(ast);
|
|
super.visitSafeMethodCall(ast, null);
|
|
}
|
|
visitSafePropertyRead(ast: e.SafePropertyRead) {
|
|
this.recordAst(ast);
|
|
super.visitSafePropertyRead(ast, null);
|
|
}
|
|
visitQuote(ast: e.Quote) {
|
|
this.recordAst(ast);
|
|
super.visitQuote(ast, null);
|
|
}
|
|
|
|
visitTemplate(ast: t.Template) { t.visitAll(this, ast.children); }
|
|
visitElement(ast: t.Element) {
|
|
t.visitAll(this, ast.children);
|
|
t.visitAll(this, ast.inputs);
|
|
t.visitAll(this, ast.outputs);
|
|
}
|
|
visitReference(ast: t.Reference) {}
|
|
visitVariable(ast: t.Variable) {}
|
|
visitEvent(ast: t.BoundEvent) { ast.handler.visit(this); }
|
|
visitTextAttribute(ast: t.TextAttribute) {}
|
|
visitBoundAttribute(ast: t.BoundAttribute) { ast.value.visit(this); }
|
|
visitBoundEvent(ast: t.BoundEvent) { ast.handler.visit(this); }
|
|
visitBoundText(ast: t.BoundText) { ast.value.visit(this); }
|
|
visitContent(ast: t.Content) {}
|
|
visitText(ast: t.Text) {}
|
|
visitIcu(ast: t.Icu) {}
|
|
}
|
|
|
|
/**
|
|
* Humanizes expression AST source spans in a template by returning an array of tuples
|
|
* [unparsed AST, AST source span]
|
|
* for each expression in the template.
|
|
* @param templateAsts template AST to humanize
|
|
*/
|
|
export function humanizeExpressionSource(templateAsts: t.Node[]): HumanizedExpressionSource[] {
|
|
const humanizer = new ExpressionSourceHumanizer();
|
|
t.visitAll(humanizer, templateAsts);
|
|
return humanizer.result;
|
|
}
|