Prior to this commit, the absolute spans (relative to template source file rather than the start of an expression) of expressions in a template attribute like `*ngIf` were generated incorrectly, equating to the relative spans. This fixes the bug by passing an `absoluteOffset` parameter when parsing template bindings. Through some levels of indirection, this is required for the Language Service to support text replacement in https://github.com/angular/angular/pull/33091. PR Close #33189
		
			
				
	
	
		
			137 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
		
			4.1 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);
 | 
						|
    t.visitAll(this, ast.templateAttrs);
 | 
						|
  }
 | 
						|
  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;
 | 
						|
}
 |