UNPKG

@angular/compiler

Version:

Angular - the compiler library

181 lines • 29.1 kB
/** * @license * Copyright Google LLC 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 */ (function (factory) { if (typeof module === "object" && typeof module.exports === "object") { var v = factory(require, exports); if (v !== undefined) module.exports = v; } else if (typeof define === "function" && define.amd) { define("@angular/compiler/src/i18n/translation_bundle", ["require", "exports", "tslib", "@angular/compiler/src/core", "@angular/compiler/src/ml_parser/html_parser", "@angular/compiler/src/i18n/parse_util", "@angular/compiler/src/i18n/serializers/xml_helper"], factory); } })(function (require, exports) { "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.TranslationBundle = void 0; var tslib_1 = require("tslib"); var core_1 = require("@angular/compiler/src/core"); var html_parser_1 = require("@angular/compiler/src/ml_parser/html_parser"); var parse_util_1 = require("@angular/compiler/src/i18n/parse_util"); var xml_helper_1 = require("@angular/compiler/src/i18n/serializers/xml_helper"); /** * A container for translated messages */ var TranslationBundle = /** @class */ (function () { function TranslationBundle(_i18nNodesByMsgId, locale, digest, mapperFactory, missingTranslationStrategy, console) { if (_i18nNodesByMsgId === void 0) { _i18nNodesByMsgId = {}; } if (missingTranslationStrategy === void 0) { missingTranslationStrategy = core_1.MissingTranslationStrategy.Warning; } this._i18nNodesByMsgId = _i18nNodesByMsgId; this.digest = digest; this.mapperFactory = mapperFactory; this._i18nToHtml = new I18nToHtmlVisitor(_i18nNodesByMsgId, locale, digest, mapperFactory, missingTranslationStrategy, console); } // Creates a `TranslationBundle` by parsing the given `content` with the `serializer`. TranslationBundle.load = function (content, url, serializer, missingTranslationStrategy, console) { var _a = serializer.load(content, url), locale = _a.locale, i18nNodesByMsgId = _a.i18nNodesByMsgId; var digestFn = function (m) { return serializer.digest(m); }; var mapperFactory = function (m) { return serializer.createNameMapper(m); }; return new TranslationBundle(i18nNodesByMsgId, locale, digestFn, mapperFactory, missingTranslationStrategy, console); }; // Returns the translation as HTML nodes from the given source message. TranslationBundle.prototype.get = function (srcMsg) { var html = this._i18nToHtml.convert(srcMsg); if (html.errors.length) { throw new Error(html.errors.join('\n')); } return html.nodes; }; TranslationBundle.prototype.has = function (srcMsg) { return this.digest(srcMsg) in this._i18nNodesByMsgId; }; return TranslationBundle; }()); exports.TranslationBundle = TranslationBundle; var I18nToHtmlVisitor = /** @class */ (function () { function I18nToHtmlVisitor(_i18nNodesByMsgId, _locale, _digest, _mapperFactory, _missingTranslationStrategy, _console) { if (_i18nNodesByMsgId === void 0) { _i18nNodesByMsgId = {}; } this._i18nNodesByMsgId = _i18nNodesByMsgId; this._locale = _locale; this._digest = _digest; this._mapperFactory = _mapperFactory; this._missingTranslationStrategy = _missingTranslationStrategy; this._console = _console; this._contextStack = []; this._errors = []; } I18nToHtmlVisitor.prototype.convert = function (srcMsg) { this._contextStack.length = 0; this._errors.length = 0; // i18n to text var text = this._convertToText(srcMsg); // text to html var url = srcMsg.nodes[0].sourceSpan.start.file.url; var html = new html_parser_1.HtmlParser().parse(text, url, { tokenizeExpansionForms: true }); return { nodes: html.rootNodes, errors: tslib_1.__spread(this._errors, html.errors), }; }; I18nToHtmlVisitor.prototype.visitText = function (text, context) { // `convert()` uses an `HtmlParser` to return `html.Node`s // we should then make sure that any special characters are escaped return xml_helper_1.escapeXml(text.value); }; I18nToHtmlVisitor.prototype.visitContainer = function (container, context) { var _this = this; return container.children.map(function (n) { return n.visit(_this); }).join(''); }; I18nToHtmlVisitor.prototype.visitIcu = function (icu, context) { var _this = this; var cases = Object.keys(icu.cases).map(function (k) { return k + " {" + icu.cases[k].visit(_this) + "}"; }); // TODO(vicb): Once all format switch to using expression placeholders // we should throw when the placeholder is not in the source message var exp = this._srcMsg.placeholders.hasOwnProperty(icu.expression) ? this._srcMsg.placeholders[icu.expression].text : icu.expression; return "{" + exp + ", " + icu.type + ", " + cases.join(' ') + "}"; }; I18nToHtmlVisitor.prototype.visitPlaceholder = function (ph, context) { var phName = this._mapper(ph.name); if (this._srcMsg.placeholders.hasOwnProperty(phName)) { return this._srcMsg.placeholders[phName].text; } if (this._srcMsg.placeholderToMessage.hasOwnProperty(phName)) { return this._convertToText(this._srcMsg.placeholderToMessage[phName]); } this._addError(ph, "Unknown placeholder \"" + ph.name + "\""); return ''; }; // Loaded message contains only placeholders (vs tag and icu placeholders). // However when a translation can not be found, we need to serialize the source message // which can contain tag placeholders I18nToHtmlVisitor.prototype.visitTagPlaceholder = function (ph, context) { var _this = this; var tag = "" + ph.tag; var attrs = Object.keys(ph.attrs).map(function (name) { return name + "=\"" + ph.attrs[name] + "\""; }).join(' '); if (ph.isVoid) { return "<" + tag + " " + attrs + "/>"; } var children = ph.children.map(function (c) { return c.visit(_this); }).join(''); return "<" + tag + " " + attrs + ">" + children + "</" + tag + ">"; }; // Loaded message contains only placeholders (vs tag and icu placeholders). // However when a translation can not be found, we need to serialize the source message // which can contain tag placeholders I18nToHtmlVisitor.prototype.visitIcuPlaceholder = function (ph, context) { // An ICU placeholder references the source message to be serialized return this._convertToText(this._srcMsg.placeholderToMessage[ph.name]); }; /** * Convert a source message to a translated text string: * - text nodes are replaced with their translation, * - placeholders are replaced with their content, * - ICU nodes are converted to ICU expressions. */ I18nToHtmlVisitor.prototype._convertToText = function (srcMsg) { var _this = this; var id = this._digest(srcMsg); var mapper = this._mapperFactory ? this._mapperFactory(srcMsg) : null; var nodes; this._contextStack.push({ msg: this._srcMsg, mapper: this._mapper }); this._srcMsg = srcMsg; if (this._i18nNodesByMsgId.hasOwnProperty(id)) { // When there is a translation use its nodes as the source // And create a mapper to convert serialized placeholder names to internal names nodes = this._i18nNodesByMsgId[id]; this._mapper = function (name) { return mapper ? mapper.toInternalName(name) : name; }; } else { // When no translation has been found // - report an error / a warning / nothing, // - use the nodes from the original message // - placeholders are already internal and need no mapper if (this._missingTranslationStrategy === core_1.MissingTranslationStrategy.Error) { var ctx = this._locale ? " for locale \"" + this._locale + "\"" : ''; this._addError(srcMsg.nodes[0], "Missing translation for message \"" + id + "\"" + ctx); } else if (this._console && this._missingTranslationStrategy === core_1.MissingTranslationStrategy.Warning) { var ctx = this._locale ? " for locale \"" + this._locale + "\"" : ''; this._console.warn("Missing translation for message \"" + id + "\"" + ctx); } nodes = srcMsg.nodes; this._mapper = function (name) { return name; }; } var text = nodes.map(function (node) { return node.visit(_this); }).join(''); var context = this._contextStack.pop(); this._srcMsg = context.msg; this._mapper = context.mapper; return text; }; I18nToHtmlVisitor.prototype._addError = function (el, msg) { this._errors.push(new parse_util_1.I18nError(el.sourceSpan, msg)); }; return I18nToHtmlVisitor; }()); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"translation_bundle.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/i18n/translation_bundle.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {MissingTranslationStrategy} from '../core';\nimport * as html from '../ml_parser/ast';\nimport {HtmlParser} from '../ml_parser/html_parser';\nimport {Console} from '../util';\n\nimport * as i18n from './i18n_ast';\nimport {I18nError} from './parse_util';\nimport {PlaceholderMapper, Serializer} from './serializers/serializer';\nimport {escapeXml} from './serializers/xml_helper';\n\n\n/**\n * A container for translated messages\n */\nexport class TranslationBundle {\n  private _i18nToHtml: I18nToHtmlVisitor;\n\n  constructor(\n      private _i18nNodesByMsgId: {[msgId: string]: i18n.Node[]} = {}, locale: string|null,\n      public digest: (m: i18n.Message) => string,\n      public mapperFactory?: (m: i18n.Message) => PlaceholderMapper,\n      missingTranslationStrategy: MissingTranslationStrategy = MissingTranslationStrategy.Warning,\n      console?: Console) {\n    this._i18nToHtml = new I18nToHtmlVisitor(\n        _i18nNodesByMsgId, locale, digest, mapperFactory!, missingTranslationStrategy, console);\n  }\n\n  // Creates a `TranslationBundle` by parsing the given `content` with the `serializer`.\n  static load(\n      content: string, url: string, serializer: Serializer,\n      missingTranslationStrategy: MissingTranslationStrategy,\n      console?: Console): TranslationBundle {\n    const {locale, i18nNodesByMsgId} = serializer.load(content, url);\n    const digestFn = (m: i18n.Message) => serializer.digest(m);\n    const mapperFactory = (m: i18n.Message) => serializer.createNameMapper(m)!;\n    return new TranslationBundle(\n        i18nNodesByMsgId, locale, digestFn, mapperFactory, missingTranslationStrategy, console);\n  }\n\n  // Returns the translation as HTML nodes from the given source message.\n  get(srcMsg: i18n.Message): html.Node[] {\n    const html = this._i18nToHtml.convert(srcMsg);\n\n    if (html.errors.length) {\n      throw new Error(html.errors.join('\\n'));\n    }\n\n    return html.nodes;\n  }\n\n  has(srcMsg: i18n.Message): boolean {\n    return this.digest(srcMsg) in this._i18nNodesByMsgId;\n  }\n}\n\nclass I18nToHtmlVisitor implements i18n.Visitor {\n  // TODO(issue/24571): remove '!'.\n  private _srcMsg!: i18n.Message;\n  private _contextStack: {msg: i18n.Message, mapper: (name: string) => string}[] = [];\n  private _errors: I18nError[] = [];\n  // TODO(issue/24571): remove '!'.\n  private _mapper!: (name: string) => string;\n\n  constructor(\n      private _i18nNodesByMsgId: {[msgId: string]: i18n.Node[]} = {}, private _locale: string|null,\n      private _digest: (m: i18n.Message) => string,\n      private _mapperFactory: (m: i18n.Message) => PlaceholderMapper,\n      private _missingTranslationStrategy: MissingTranslationStrategy, private _console?: Console) {\n  }\n\n  convert(srcMsg: i18n.Message): {nodes: html.Node[], errors: I18nError[]} {\n    this._contextStack.length = 0;\n    this._errors.length = 0;\n\n    // i18n to text\n    const text = this._convertToText(srcMsg);\n\n    // text to html\n    const url = srcMsg.nodes[0].sourceSpan.start.file.url;\n    const html = new HtmlParser().parse(text, url, {tokenizeExpansionForms: true});\n\n    return {\n      nodes: html.rootNodes,\n      errors: [...this._errors, ...html.errors],\n    };\n  }\n\n  visitText(text: i18n.Text, context?: any): string {\n    // `convert()` uses an `HtmlParser` to return `html.Node`s\n    // we should then make sure that any special characters are escaped\n    return escapeXml(text.value);\n  }\n\n  visitContainer(container: i18n.Container, context?: any): any {\n    return container.children.map(n => n.visit(this)).join('');\n  }\n\n  visitIcu(icu: i18n.Icu, context?: any): any {\n    const cases = Object.keys(icu.cases).map(k => `${k} {${icu.cases[k].visit(this)}}`);\n\n    // TODO(vicb): Once all format switch to using expression placeholders\n    // we should throw when the placeholder is not in the source message\n    const exp = this._srcMsg.placeholders.hasOwnProperty(icu.expression) ?\n        this._srcMsg.placeholders[icu.expression].text :\n        icu.expression;\n\n    return `{${exp}, ${icu.type}, ${cases.join(' ')}}`;\n  }\n\n  visitPlaceholder(ph: i18n.Placeholder, context?: any): string {\n    const phName = this._mapper(ph.name);\n    if (this._srcMsg.placeholders.hasOwnProperty(phName)) {\n      return this._srcMsg.placeholders[phName].text;\n    }\n\n    if (this._srcMsg.placeholderToMessage.hasOwnProperty(phName)) {\n      return this._convertToText(this._srcMsg.placeholderToMessage[phName]);\n    }\n\n    this._addError(ph, `Unknown placeholder \"${ph.name}\"`);\n    return '';\n  }\n\n  // Loaded message contains only placeholders (vs tag and icu placeholders).\n  // However when a translation can not be found, we need to serialize the source message\n  // which can contain tag placeholders\n  visitTagPlaceholder(ph: i18n.TagPlaceholder, context?: any): string {\n    const tag = `${ph.tag}`;\n    const attrs = Object.keys(ph.attrs).map(name => `${name}=\"${ph.attrs[name]}\"`).join(' ');\n    if (ph.isVoid) {\n      return `<${tag} ${attrs}/>`;\n    }\n    const children = ph.children.map((c: i18n.Node) => c.visit(this)).join('');\n    return `<${tag} ${attrs}>${children}</${tag}>`;\n  }\n\n  // Loaded message contains only placeholders (vs tag and icu placeholders).\n  // However when a translation can not be found, we need to serialize the source message\n  // which can contain tag placeholders\n  visitIcuPlaceholder(ph: i18n.IcuPlaceholder, context?: any): string {\n    // An ICU placeholder references the source message to be serialized\n    return this._convertToText(this._srcMsg.placeholderToMessage[ph.name]);\n  }\n\n  /**\n   * Convert a source message to a translated text string:\n   * - text nodes are replaced with their translation,\n   * - placeholders are replaced with their content,\n   * - ICU nodes are converted to ICU expressions.\n   */\n  private _convertToText(srcMsg: i18n.Message): string {\n    const id = this._digest(srcMsg);\n    const mapper = this._mapperFactory ? this._mapperFactory(srcMsg) : null;\n    let nodes: i18n.Node[];\n\n    this._contextStack.push({msg: this._srcMsg, mapper: this._mapper});\n    this._srcMsg = srcMsg;\n\n    if (this._i18nNodesByMsgId.hasOwnProperty(id)) {\n      // When there is a translation use its nodes as the source\n      // And create a mapper to convert serialized placeholder names to internal names\n      nodes = this._i18nNodesByMsgId[id];\n      this._mapper = (name: string) => mapper ? mapper.toInternalName(name)! : name;\n    } else {\n      // When no translation has been found\n      // - report an error / a warning / nothing,\n      // - use the nodes from the original message\n      // - placeholders are already internal and need no mapper\n      if (this._missingTranslationStrategy === MissingTranslationStrategy.Error) {\n        const ctx = this._locale ? ` for locale \"${this._locale}\"` : '';\n        this._addError(srcMsg.nodes[0], `Missing translation for message \"${id}\"${ctx}`);\n      } else if (\n          this._console &&\n          this._missingTranslationStrategy === MissingTranslationStrategy.Warning) {\n        const ctx = this._locale ? ` for locale \"${this._locale}\"` : '';\n        this._console.warn(`Missing translation for message \"${id}\"${ctx}`);\n      }\n      nodes = srcMsg.nodes;\n      this._mapper = (name: string) => name;\n    }\n    const text = nodes.map(node => node.visit(this)).join('');\n    const context = this._contextStack.pop()!;\n    this._srcMsg = context.msg;\n    this._mapper = context.mapper;\n    return text;\n  }\n\n  private _addError(el: i18n.Node, msg: string) {\n    this._errors.push(new I18nError(el.sourceSpan, msg));\n  }\n}\n"]}