UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

165 lines 17.1 kB
'use strict';"use strict"; var html_ast_1 = require('angular2/src/compiler/html_ast'); var lang_1 = require('angular2/src/facade/lang'); var collection_1 = require('angular2/src/facade/collection'); var message_1 = require('./message'); var expander_1 = require('./expander'); var shared_1 = require('./shared'); /** * All messages extracted from a template. */ var ExtractionResult = (function () { function ExtractionResult(messages, errors) { this.messages = messages; this.errors = errors; } return ExtractionResult; }()); exports.ExtractionResult = ExtractionResult; /** * 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")]); * ``` */ function removeDuplicates(messages) { var uniq = {}; messages.forEach(function (m) { if (!collection_1.StringMapWrapper.contains(uniq, message_1.id(m))) { uniq[message_1.id(m)] = m; } }); return collection_1.StringMapWrapper.values(uniq); } exports.removeDuplicates = removeDuplicates; /** * Extracts all messages from a template. * * 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: * * The following array of nodes will be split into four parts: * * ``` * <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. * * Example: * * The following tree: * * ``` * <a>A{{I}}</a><b>B</b> * ``` * * 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. */ var MessageExtractor = (function () { function MessageExtractor(_htmlParser, _parser) { this._htmlParser = _htmlParser; this._parser = _parser; } MessageExtractor.prototype.extract = function (template, sourceUrl) { this.messages = []; this.errors = []; var res = this._htmlParser.parse(template, sourceUrl, true); if (res.errors.length > 0) { return new ExtractionResult([], res.errors); } else { this._recurse(expander_1.expandNodes(res.rootNodes).nodes); return new ExtractionResult(this.messages, this.errors); } }; MessageExtractor.prototype._extractMessagesFromPart = function (p) { if (p.hasI18n) { this.messages.push(p.createMessage(this._parser)); this._recurseToExtractMessagesFromAttributes(p.children); } else { this._recurse(p.children); } if (lang_1.isPresent(p.rootElement)) { this._extractMessagesFromAttributes(p.rootElement); } }; MessageExtractor.prototype._recurse = function (nodes) { var _this = this; if (lang_1.isPresent(nodes)) { var ps = shared_1.partition(nodes, this.errors); ps.forEach(function (p) { return _this._extractMessagesFromPart(p); }); } }; MessageExtractor.prototype._recurseToExtractMessagesFromAttributes = function (nodes) { var _this = this; nodes.forEach(function (n) { if (n instanceof html_ast_1.HtmlElementAst) { _this._extractMessagesFromAttributes(n); _this._recurseToExtractMessagesFromAttributes(n.children); } }); }; MessageExtractor.prototype._extractMessagesFromAttributes = function (p) { var _this = this; p.attrs.forEach(function (attr) { if (attr.name.startsWith(shared_1.I18N_ATTR_PREFIX)) { try { _this.messages.push(shared_1.messageFromAttribute(_this._parser, p, attr)); } catch (e) { if (e instanceof shared_1.I18nError) { _this.errors.push(e); } else { throw e; } } } }); }; return MessageExtractor; }()); exports.MessageExtractor = MessageExtractor; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message_extractor.js","sourceRoot":"","sources":["diffing_plugin_wrapper-output_path-BRJer1J9.tmp/angular2/src/i18n/message_extractor.ts"],"names":[],"mappings":"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","sourcesContent":["import {HtmlParser} from 'angular2/src/compiler/html_parser';\nimport {ParseSourceSpan, ParseError} from 'angular2/src/compiler/parse_util';\nimport {\n  HtmlAst,\n  HtmlAstVisitor,\n  HtmlElementAst,\n  HtmlAttrAst,\n  HtmlTextAst,\n  HtmlCommentAst,\n  htmlVisitAll\n} from 'angular2/src/compiler/html_ast';\nimport {isPresent, isBlank} from 'angular2/src/facade/lang';\nimport {StringMapWrapper} from 'angular2/src/facade/collection';\nimport {Parser} from 'angular2/src/compiler/expression_parser/parser';\nimport {Message, id} from './message';\nimport {expandNodes} from './expander';\nimport {\n  I18nError,\n  Part,\n  I18N_ATTR_PREFIX,\n  partition,\n  meaning,\n  description,\n  stringifyNodes,\n  messageFromAttribute\n} from './shared';\n\n/**\n * All messages extracted from a template.\n */\nexport class ExtractionResult {\n  constructor(public messages: Message[], public errors: ParseError[]) {}\n}\n\n/**\n * Removes duplicate messages.\n *\n * E.g.\n *\n * ```\n *  var m = [new Message(\"message\", \"meaning\", \"desc1\"), new Message(\"message\", \"meaning\",\n * \"desc2\")];\n *  expect(removeDuplicates(m)).toEqual([new Message(\"message\", \"meaning\", \"desc1\")]);\n * ```\n */\nexport function removeDuplicates(messages: Message[]): Message[] {\n  let uniq: {[key: string]: Message} = {};\n  messages.forEach(m => {\n    if (!StringMapWrapper.contains(uniq, id(m))) {\n      uniq[id(m)] = m;\n    }\n  });\n  return StringMapWrapper.values(uniq);\n}\n\n/**\n * Extracts all messages from a template.\n *\n * Algorithm:\n *\n * To understand the algorithm, you need to know how partitioning works.\n * Partitioning is required as we can use two i18n comments to group node siblings together.\n * That is why we cannot just use nodes.\n *\n * Partitioning transforms an array of HtmlAst into an array of Part.\n * A part can optionally contain a root element or a root text node. And it can also contain\n * children.\n * A part can contain i18n property, in which case it needs to be extracted.\n *\n * Example:\n *\n * The following array of nodes will be split into four parts:\n *\n * ```\n * <a>A</a>\n * <b i18n>B</b>\n * <!-- i18n -->\n * <c>C</c>\n * D\n * <!-- /i18n -->\n * E\n * ```\n *\n * Part 1 containing the a tag. It should not be translated.\n * Part 2 containing the b tag. It should be translated.\n * Part 3 containing the c tag and the D text node. It should be translated.\n * Part 4 containing the E text node. It should not be translated..\n *\n * It is also important to understand how we stringify nodes to create a message.\n *\n * We walk the tree and replace every element node with a placeholder. We also replace\n * all expressions in interpolation with placeholders. We also insert a placeholder element\n * to wrap a text node containing interpolation.\n *\n * Example:\n *\n * The following tree:\n *\n * ```\n * <a>A{{I}}</a><b>B</b>\n * ```\n *\n * will be stringified into:\n * ```\n * <ph name=\"e0\"><ph name=\"t1\">A<ph name=\"0\"/></ph></ph><ph name=\"e2\">B</ph>\n * ```\n *\n * This is what the algorithm does:\n *\n * 1. Use the provided html parser to get the html AST of the template.\n * 2. Partition the root nodes, and process each part separately.\n * 3. If a part does not have the i18n attribute, recurse to process children and attributes.\n * 4. If a part has the i18n attribute, stringify the nodes to create a Message.\n */\nexport class MessageExtractor {\n  messages: Message[];\n  errors: ParseError[];\n\n  constructor(private _htmlParser: HtmlParser, private _parser: Parser) {}\n\n  extract(template: string, sourceUrl: string): ExtractionResult {\n    this.messages = [];\n    this.errors = [];\n\n    let res = this._htmlParser.parse(template, sourceUrl, true);\n    if (res.errors.length > 0) {\n      return new ExtractionResult([], res.errors);\n    } else {\n      this._recurse(expandNodes(res.rootNodes).nodes);\n      return new ExtractionResult(this.messages, this.errors);\n    }\n  }\n\n  private _extractMessagesFromPart(p: Part): void {\n    if (p.hasI18n) {\n      this.messages.push(p.createMessage(this._parser));\n      this._recurseToExtractMessagesFromAttributes(p.children);\n    } else {\n      this._recurse(p.children);\n    }\n\n    if (isPresent(p.rootElement)) {\n      this._extractMessagesFromAttributes(p.rootElement);\n    }\n  }\n\n  private _recurse(nodes: HtmlAst[]): void {\n    if (isPresent(nodes)) {\n      let ps = partition(nodes, this.errors);\n      ps.forEach(p => this._extractMessagesFromPart(p));\n    }\n  }\n\n  private _recurseToExtractMessagesFromAttributes(nodes: HtmlAst[]): void {\n    nodes.forEach(n => {\n      if (n instanceof HtmlElementAst) {\n        this._extractMessagesFromAttributes(n);\n        this._recurseToExtractMessagesFromAttributes(n.children);\n      }\n    });\n  }\n\n  private _extractMessagesFromAttributes(p: HtmlElementAst): void {\n    p.attrs.forEach(attr => {\n      if (attr.name.startsWith(I18N_ATTR_PREFIX)) {\n        try {\n          this.messages.push(messageFromAttribute(this._parser, p, attr));\n        } catch (e) {\n          if (e instanceof I18nError) {\n            this.errors.push(e);\n          } else {\n            throw e;\n          }\n        }\n      }\n    });\n  }\n}"]}