2016-03-14 10:51:23 -07:00
|
|
|
import {HtmlParser} from 'angular2/src/compiler/html_parser';
|
|
|
|
import {ParseSourceSpan, ParseError} from 'angular2/src/compiler/parse_util';
|
2016-04-12 09:40:37 -07:00
|
|
|
import {
|
|
|
|
HtmlAst,
|
|
|
|
HtmlAstVisitor,
|
|
|
|
HtmlElementAst,
|
|
|
|
HtmlAttrAst,
|
|
|
|
HtmlTextAst,
|
|
|
|
HtmlCommentAst,
|
|
|
|
htmlVisitAll
|
|
|
|
} from 'angular2/src/compiler/html_ast';
|
2016-03-14 10:51:23 -07:00
|
|
|
import {isPresent, isBlank} from 'angular2/src/facade/lang';
|
|
|
|
import {StringMapWrapper} from 'angular2/src/facade/collection';
|
2016-01-06 14:13:44 -08:00
|
|
|
import {Parser} from 'angular2/src/compiler/expression_parser/parser';
|
2016-03-14 11:45:14 -07:00
|
|
|
import {Message, id} from './message';
|
2016-04-13 16:01:25 -07:00
|
|
|
import {expandNodes} from './expander';
|
2016-04-12 09:40:37 -07:00
|
|
|
import {
|
|
|
|
I18nError,
|
|
|
|
Part,
|
|
|
|
I18N_ATTR_PREFIX,
|
|
|
|
partition,
|
|
|
|
meaning,
|
|
|
|
description,
|
|
|
|
stringifyNodes,
|
|
|
|
messageFromAttribute
|
|
|
|
} from './shared';
|
2016-03-14 10:51:23 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* All messages extracted from a template.
|
|
|
|
*/
|
|
|
|
export class ExtractionResult {
|
|
|
|
constructor(public messages: Message[], public errors: ParseError[]) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes duplicate messages.
|
|
|
|
*
|
|
|
|
* E.g.
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* var m = [new Message("message", "meaning", "desc1"), new Message("message", "meaning",
|
|
|
|
* "desc2")];
|
|
|
|
* expect(removeDuplicates(m)).toEqual([new Message("message", "meaning", "desc1")]);
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
export function removeDuplicates(messages: Message[]): Message[] {
|
|
|
|
let uniq: {[key: string]: Message} = {};
|
|
|
|
messages.forEach(m => {
|
2016-03-14 11:45:14 -07:00
|
|
|
if (!StringMapWrapper.contains(uniq, id(m))) {
|
|
|
|
uniq[id(m)] = m;
|
2016-03-14 10:51:23 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return StringMapWrapper.values(uniq);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extracts all messages from a template.
|
|
|
|
*
|
2016-03-23 13:44:45 -07:00
|
|
|
* Algorithm:
|
|
|
|
*
|
|
|
|
* To understand the algorithm, you need to know how partitioning works.
|
|
|
|
* Partitioning is required as we can use two i18n comments to group node siblings together.
|
|
|
|
* That is why we cannot just use nodes.
|
|
|
|
*
|
|
|
|
* Partitioning transforms an array of HtmlAst into an array of Part.
|
|
|
|
* A part can optionally contain a root element or a root text node. And it can also contain
|
|
|
|
* children.
|
|
|
|
* A part can contain i18n property, in which case it needs to be extracted.
|
|
|
|
*
|
|
|
|
* Example:
|
2016-03-14 10:51:23 -07:00
|
|
|
*
|
2016-03-23 13:44:45 -07:00
|
|
|
* The following array of nodes will be split into four parts:
|
2016-03-14 10:51:23 -07:00
|
|
|
*
|
2016-03-23 13:44:45 -07:00
|
|
|
* ```
|
|
|
|
* <a>A</a>
|
|
|
|
* <b i18n>B</b>
|
|
|
|
* <!-- i18n -->
|
|
|
|
* <c>C</c>
|
|
|
|
* D
|
|
|
|
* <!-- /i18n -->
|
|
|
|
* E
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Part 1 containing the a tag. It should not be translated.
|
|
|
|
* Part 2 containing the b tag. It should be translated.
|
|
|
|
* Part 3 containing the c tag and the D text node. It should be translated.
|
|
|
|
* Part 4 containing the E text node. It should not be translated..
|
|
|
|
*
|
|
|
|
* It is also important to understand how we stringify nodes to create a message.
|
|
|
|
*
|
|
|
|
* We walk the tree and replace every element node with a placeholder. We also replace
|
|
|
|
* all expressions in interpolation with placeholders. We also insert a placeholder element
|
|
|
|
* to wrap a text node containing interpolation.
|
2016-03-14 10:51:23 -07:00
|
|
|
*
|
2016-03-23 13:44:45 -07:00
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* The following tree:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <a>A{{I}}</a><b>B</b>
|
|
|
|
* ```
|
2016-03-14 10:51:23 -07:00
|
|
|
*
|
2016-03-23 13:44:45 -07:00
|
|
|
* will be stringified into:
|
|
|
|
* ```
|
|
|
|
* <ph name="e0"><ph name="t1">A<ph name="0"/></ph></ph><ph name="e2">B</ph>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* This is what the algorithm does:
|
|
|
|
*
|
|
|
|
* 1. Use the provided html parser to get the html AST of the template.
|
|
|
|
* 2. Partition the root nodes, and process each part separately.
|
|
|
|
* 3. If a part does not have the i18n attribute, recurse to process children and attributes.
|
|
|
|
* 4. If a part has the i18n attribute, stringify the nodes to create a Message.
|
2016-03-14 10:51:23 -07:00
|
|
|
*/
|
|
|
|
export class MessageExtractor {
|
|
|
|
messages: Message[];
|
|
|
|
errors: ParseError[];
|
|
|
|
|
|
|
|
constructor(private _htmlParser: HtmlParser, private _parser: Parser) {}
|
|
|
|
|
|
|
|
extract(template: string, sourceUrl: string): ExtractionResult {
|
|
|
|
this.messages = [];
|
|
|
|
this.errors = [];
|
|
|
|
|
2016-04-12 11:46:49 -07:00
|
|
|
let res = this._htmlParser.parse(template, sourceUrl, true);
|
2016-03-14 10:51:23 -07:00
|
|
|
if (res.errors.length > 0) {
|
|
|
|
return new ExtractionResult([], res.errors);
|
|
|
|
} else {
|
2016-04-13 16:01:25 -07:00
|
|
|
this._recurse(expandNodes(res.rootNodes).nodes);
|
2016-03-14 10:51:23 -07:00
|
|
|
return new ExtractionResult(this.messages, this.errors);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-23 13:44:45 -07:00
|
|
|
private _extractMessagesFromPart(p: Part): void {
|
2016-03-14 10:51:23 -07:00
|
|
|
if (p.hasI18n) {
|
2016-03-23 13:44:45 -07:00
|
|
|
this.messages.push(p.createMessage(this._parser));
|
2016-03-14 10:51:23 -07:00
|
|
|
this._recurseToExtractMessagesFromAttributes(p.children);
|
|
|
|
} else {
|
|
|
|
this._recurse(p.children);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isPresent(p.rootElement)) {
|
|
|
|
this._extractMessagesFromAttributes(p.rootElement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private _recurse(nodes: HtmlAst[]): void {
|
2016-03-23 13:44:45 -07:00
|
|
|
if (isPresent(nodes)) {
|
|
|
|
let ps = partition(nodes, this.errors);
|
|
|
|
ps.forEach(p => this._extractMessagesFromPart(p));
|
|
|
|
}
|
2016-03-14 10:51:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _recurseToExtractMessagesFromAttributes(nodes: HtmlAst[]): void {
|
|
|
|
nodes.forEach(n => {
|
|
|
|
if (n instanceof HtmlElementAst) {
|
|
|
|
this._extractMessagesFromAttributes(n);
|
|
|
|
this._recurseToExtractMessagesFromAttributes(n.children);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _extractMessagesFromAttributes(p: HtmlElementAst): void {
|
|
|
|
p.attrs.forEach(attr => {
|
2016-03-27 18:31:02 -07:00
|
|
|
if (attr.name.startsWith(I18N_ATTR_PREFIX)) {
|
2016-03-23 13:44:45 -07:00
|
|
|
try {
|
|
|
|
this.messages.push(messageFromAttribute(this._parser, p, attr));
|
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof I18nError) {
|
|
|
|
this.errors.push(e);
|
|
|
|
} else {
|
|
|
|
throw e;
|
2016-03-14 10:51:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-03-23 13:44:45 -07:00
|
|
|
});
|
2016-03-14 10:51:23 -07:00
|
|
|
}
|
|
|
|
}
|