Parse Pipes with Angular Compiler AST, enable ternary operator parsing (#159)
(feature) Use AST-based approach to translate pipe parsing. Also enables parsing translate pipes from any position in a pipe chain. Fixes #111, Fixes #154. (Thanks @TekSiDoT)
This commit is contained in:
@@ -1,6 +1,7 @@
|
||||
import { ParserInterface } from './parser.interface';
|
||||
import { TranslationCollection } from '../utils/translation.collection';
|
||||
import { isPathAngularComponent, extractComponentInlineTemplate } from '../utils/utils';
|
||||
import { TmplAstNode, parseTemplate, BindingPipe, LiteralPrimitive, Conditional, TmplAstTextAttribute } from '@angular/compiler';
|
||||
|
||||
export class PipeParser implements ParserInterface {
|
||||
public extract(source: string, filePath: string): TranslationCollection | null {
|
||||
@@ -8,18 +9,89 @@ export class PipeParser implements ParserInterface {
|
||||
source = extractComponentInlineTemplate(source);
|
||||
}
|
||||
|
||||
return this.parseTemplate(source);
|
||||
}
|
||||
|
||||
protected parseTemplate(template: string): TranslationCollection {
|
||||
let collection: TranslationCollection = new TranslationCollection();
|
||||
|
||||
const regExp: RegExp = /(['"`])((?:(?!\1).|\\\1)+)\1\s*\|\s*translate/g;
|
||||
let matches: RegExpExecArray;
|
||||
while ((matches = regExp.exec(template))) {
|
||||
collection = collection.add(matches[2].split("\\'").join("'"));
|
||||
}
|
||||
|
||||
const nodes: TmplAstNode[] = this.parseTemplate(source, filePath);
|
||||
const pipes: BindingPipe[] = nodes.map(node => this.findPipesInNode(node)).flat();
|
||||
pipes.forEach(pipe => {
|
||||
this.parseTranslationKeysFromPipe(pipe).forEach((key: string) => {
|
||||
collection = collection.add(key);
|
||||
});
|
||||
});
|
||||
return collection;
|
||||
}
|
||||
|
||||
protected findPipesInNode(node: any): BindingPipe[] {
|
||||
let ret: BindingPipe[] = [];
|
||||
|
||||
if (node.children) {
|
||||
ret = node.children.reduce(
|
||||
(result: BindingPipe[], childNode: TmplAstNode) => {
|
||||
const children = this.findPipesInNode(childNode);
|
||||
return result.concat(children);
|
||||
},
|
||||
[ret]
|
||||
);
|
||||
}
|
||||
|
||||
if (node.value && node.value.ast && node.value.ast.expressions) {
|
||||
const translateables = node.value.ast.expressions.filter((exp: any) => this.expressionIsOrHasBindingPipe(exp));
|
||||
ret.push(...translateables);
|
||||
}
|
||||
|
||||
if (node.attributes) {
|
||||
const translateableAttributes = node.attributes.filter((attr: TmplAstTextAttribute) => {
|
||||
return attr.name === 'translate';
|
||||
});
|
||||
ret = [...ret, ...translateableAttributes];
|
||||
}
|
||||
|
||||
if (node.inputs) {
|
||||
node.inputs.forEach((input: any) => {
|
||||
// <element [attrib]="'identifier' | translate">
|
||||
if (input.value && input.value.ast && this.expressionIsOrHasBindingPipe(input.value.ast)) {
|
||||
ret.push(input.value.ast);
|
||||
}
|
||||
|
||||
// <element attrib="{{'identifier' | translate}}>"
|
||||
if (input.value && input.value.ast && input.value.ast.expressions) {
|
||||
input.value.ast.expressions.forEach((exp: BindingPipe) => {
|
||||
if (this.expressionIsOrHasBindingPipe(exp)) {
|
||||
ret.push(exp);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
protected parseTranslationKeysFromPipe(pipeContent: BindingPipe | LiteralPrimitive | Conditional): string[] {
|
||||
const ret: string[] = [];
|
||||
if (pipeContent instanceof LiteralPrimitive) {
|
||||
ret.push(pipeContent.value);
|
||||
} else if (pipeContent instanceof Conditional) {
|
||||
const trueExp: LiteralPrimitive | Conditional = pipeContent.trueExp as any;
|
||||
ret.push(...this.parseTranslationKeysFromPipe(trueExp));
|
||||
const falseExp: LiteralPrimitive | Conditional = pipeContent.falseExp as any;
|
||||
ret.push(...this.parseTranslationKeysFromPipe(falseExp));
|
||||
} else if (pipeContent instanceof BindingPipe) {
|
||||
ret.push(...this.parseTranslationKeysFromPipe(pipeContent.exp as any));
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
protected expressionIsOrHasBindingPipe(exp: any): boolean {
|
||||
if (exp.name && exp.name === 'translate') {
|
||||
return true;
|
||||
}
|
||||
if (exp.exp && exp.exp instanceof BindingPipe) {
|
||||
return this.expressionIsOrHasBindingPipe(exp.exp);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
protected parseTemplate(template: string, path: string): TmplAstNode[] {
|
||||
return parseTemplate(template, path).nodes;
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user