UNPKG

@angular/compiler

Version:

Angular - the compiler library

260 lines • 33.8 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/render3/view/i18n/meta", ["require", "exports", "tslib", "@angular/compiler/src/i18n/digest", "@angular/compiler/src/i18n/i18n_ast", "@angular/compiler/src/i18n/i18n_parser", "@angular/compiler/src/ml_parser/ast", "@angular/compiler/src/ml_parser/interpolation_config", "@angular/compiler/src/output/output_ast", "@angular/compiler/src/render3/view/i18n/util"], factory); } })(function (require, exports) { "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.i18nMetaToJSDoc = exports.parseI18nMeta = exports.I18nMetaVisitor = void 0; var tslib_1 = require("tslib"); var digest_1 = require("@angular/compiler/src/i18n/digest"); var i18n = require("@angular/compiler/src/i18n/i18n_ast"); var i18n_parser_1 = require("@angular/compiler/src/i18n/i18n_parser"); var html = require("@angular/compiler/src/ml_parser/ast"); var interpolation_config_1 = require("@angular/compiler/src/ml_parser/interpolation_config"); var o = require("@angular/compiler/src/output/output_ast"); var util_1 = require("@angular/compiler/src/render3/view/i18n/util"); var setI18nRefs = function (htmlNode, i18nNode) { if (htmlNode instanceof html.NodeWithI18n) { if (i18nNode instanceof i18n.IcuPlaceholder && htmlNode.i18n instanceof i18n.Message) { // This html node represents an ICU but this is a second processing pass, and the legacy id // was computed in the previous pass and stored in the `i18n` property as a message. // We are about to wipe out that property so capture the previous message to be reused when // generating the message for this ICU later. See `_generateI18nMessage()`. i18nNode.previousMessage = htmlNode.i18n; } htmlNode.i18n = i18nNode; } return i18nNode; }; /** * This visitor walks over HTML parse tree and converts information stored in * i18n-related attributes ("i18n" and "i18n-*") into i18n meta object that is * stored with other element's and attribute's information. */ var I18nMetaVisitor = /** @class */ (function () { function I18nMetaVisitor(interpolationConfig, keepI18nAttrs, enableI18nLegacyMessageIdFormat) { if (interpolationConfig === void 0) { interpolationConfig = interpolation_config_1.DEFAULT_INTERPOLATION_CONFIG; } if (keepI18nAttrs === void 0) { keepI18nAttrs = false; } if (enableI18nLegacyMessageIdFormat === void 0) { enableI18nLegacyMessageIdFormat = false; } this.interpolationConfig = interpolationConfig; this.keepI18nAttrs = keepI18nAttrs; this.enableI18nLegacyMessageIdFormat = enableI18nLegacyMessageIdFormat; // whether visited nodes contain i18n information this.hasI18nMeta = false; // i18n message generation factory this._createI18nMessage = i18n_parser_1.createI18nMessageFactory(this.interpolationConfig); } I18nMetaVisitor.prototype._generateI18nMessage = function (nodes, meta, visitNodeFn) { if (meta === void 0) { meta = ''; } var _a = this._parseMetadata(meta), meaning = _a.meaning, description = _a.description, customId = _a.customId; var message = this._createI18nMessage(nodes, meaning, description, customId, visitNodeFn); this._setMessageId(message, meta); this._setLegacyIds(message, meta); return message; }; I18nMetaVisitor.prototype.visitElement = function (element) { var e_1, _a, e_2, _b; if (util_1.hasI18nAttrs(element)) { this.hasI18nMeta = true; var attrs = []; var attrsMeta = {}; try { for (var _c = tslib_1.__values(element.attrs), _d = _c.next(); !_d.done; _d = _c.next()) { var attr = _d.value; if (attr.name === util_1.I18N_ATTR) { // root 'i18n' node attribute var i18n_1 = element.i18n || attr.value; var message = this._generateI18nMessage(element.children, i18n_1, setI18nRefs); // do not assign empty i18n meta if (message.nodes.length) { element.i18n = message; } } else if (attr.name.startsWith(util_1.I18N_ATTR_PREFIX)) { // 'i18n-*' attributes var key = attr.name.slice(util_1.I18N_ATTR_PREFIX.length); attrsMeta[key] = attr.value; } else { // non-i18n attributes attrs.push(attr); } } } catch (e_1_1) { e_1 = { error: e_1_1 }; } finally { try { if (_d && !_d.done && (_a = _c.return)) _a.call(_c); } finally { if (e_1) throw e_1.error; } } // set i18n meta for attributes if (Object.keys(attrsMeta).length) { try { for (var attrs_1 = tslib_1.__values(attrs), attrs_1_1 = attrs_1.next(); !attrs_1_1.done; attrs_1_1 = attrs_1.next()) { var attr = attrs_1_1.value; var meta = attrsMeta[attr.name]; // do not create translation for empty attributes if (meta !== undefined && attr.value) { attr.i18n = this._generateI18nMessage([attr], attr.i18n || meta); } } } catch (e_2_1) { e_2 = { error: e_2_1 }; } finally { try { if (attrs_1_1 && !attrs_1_1.done && (_b = attrs_1.return)) _b.call(attrs_1); } finally { if (e_2) throw e_2.error; } } } if (!this.keepI18nAttrs) { // update element's attributes, // keeping only non-i18n related ones element.attrs = attrs; } } html.visitAll(this, element.children, element.i18n); return element; }; I18nMetaVisitor.prototype.visitExpansion = function (expansion, currentMessage) { var message; var meta = expansion.i18n; this.hasI18nMeta = true; if (meta instanceof i18n.IcuPlaceholder) { // set ICU placeholder name (e.g. "ICU_1"), // generated while processing root element contents, // so we can reference it when we output translation var name_1 = meta.name; message = this._generateI18nMessage([expansion], meta); var icu = util_1.icuFromI18nMessage(message); icu.name = name_1; } else { // ICU is a top level message, try to use metadata from container element if provided via // `context` argument. Note: context may not be available for standalone ICUs (without // wrapping element), so fallback to ICU metadata in this case. message = this._generateI18nMessage([expansion], currentMessage || meta); } expansion.i18n = message; return expansion; }; I18nMetaVisitor.prototype.visitText = function (text) { return text; }; I18nMetaVisitor.prototype.visitAttribute = function (attribute) { return attribute; }; I18nMetaVisitor.prototype.visitComment = function (comment) { return comment; }; I18nMetaVisitor.prototype.visitExpansionCase = function (expansionCase) { return expansionCase; }; /** * Parse the general form `meta` passed into extract the explicit metadata needed to create a * `Message`. * * There are three possibilities for the `meta` variable * 1) a string from an `i18n` template attribute: parse it to extract the metadata values. * 2) a `Message` from a previous processing pass: reuse the metadata values in the message. * 4) other: ignore this and just process the message metadata as normal * * @param meta the bucket that holds information about the message * @returns the parsed metadata. */ I18nMetaVisitor.prototype._parseMetadata = function (meta) { return typeof meta === 'string' ? parseI18nMeta(meta) : meta instanceof i18n.Message ? meta : {}; }; /** * Generate (or restore) message id if not specified already. */ I18nMetaVisitor.prototype._setMessageId = function (message, meta) { if (!message.id) { message.id = meta instanceof i18n.Message && meta.id || digest_1.decimalDigest(message); } }; /** * Update the `message` with a `legacyId` if necessary. * * @param message the message whose legacy id should be set * @param meta information about the message being processed */ I18nMetaVisitor.prototype._setLegacyIds = function (message, meta) { if (this.enableI18nLegacyMessageIdFormat) { message.legacyIds = [digest_1.computeDigest(message), digest_1.computeDecimalDigest(message)]; } else if (typeof meta !== 'string') { // This occurs if we are doing the 2nd pass after whitespace removal (see `parseTemplate()` in // `packages/compiler/src/render3/view/template.ts`). // In that case we want to reuse the legacy message generated in the 1st pass (see // `setI18nRefs()`). var previousMessage = meta instanceof i18n.Message ? meta : meta instanceof i18n.IcuPlaceholder ? meta.previousMessage : undefined; message.legacyIds = previousMessage ? previousMessage.legacyIds : []; } }; return I18nMetaVisitor; }()); exports.I18nMetaVisitor = I18nMetaVisitor; /** I18n separators for metadata **/ var I18N_MEANING_SEPARATOR = '|'; var I18N_ID_SEPARATOR = '@@'; /** * Parses i18n metas like: * - "@@id", * - "description[@@id]", * - "meaning|description[@@id]" * and returns an object with parsed output. * * @param meta String that represents i18n meta * @returns Object with id, meaning and description fields */ function parseI18nMeta(meta) { var _a, _b; if (meta === void 0) { meta = ''; } var customId; var meaning; var description; meta = meta.trim(); if (meta) { var idIndex = meta.indexOf(I18N_ID_SEPARATOR); var descIndex = meta.indexOf(I18N_MEANING_SEPARATOR); var meaningAndDesc = void 0; _a = tslib_1.__read((idIndex > -1) ? [meta.slice(0, idIndex), meta.slice(idIndex + 2)] : [meta, ''], 2), meaningAndDesc = _a[0], customId = _a[1]; _b = tslib_1.__read((descIndex > -1) ? [meaningAndDesc.slice(0, descIndex), meaningAndDesc.slice(descIndex + 1)] : ['', meaningAndDesc], 2), meaning = _b[0], description = _b[1]; } return { customId: customId, meaning: meaning, description: description }; } exports.parseI18nMeta = parseI18nMeta; // Converts i18n meta information for a message (id, description, meaning) // to a JsDoc statement formatted as expected by the Closure compiler. function i18nMetaToJSDoc(meta) { var tags = []; if (meta.description) { tags.push({ tagName: "desc" /* Desc */, text: meta.description }); } if (meta.meaning) { tags.push({ tagName: "meaning" /* Meaning */, text: meta.meaning }); } return tags.length == 0 ? null : o.jsDocComment(tags); } exports.i18nMetaToJSDoc = i18nMetaToJSDoc; }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"meta.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler/src/render3/view/i18n/meta.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 {computeDecimalDigest, computeDigest, decimalDigest} from '../../../i18n/digest';\nimport * as i18n from '../../../i18n/i18n_ast';\nimport {createI18nMessageFactory, VisitNodeFn} from '../../../i18n/i18n_parser';\nimport * as html from '../../../ml_parser/ast';\nimport {DEFAULT_INTERPOLATION_CONFIG, InterpolationConfig} from '../../../ml_parser/interpolation_config';\nimport * as o from '../../../output/output_ast';\n\nimport {hasI18nAttrs, I18N_ATTR, I18N_ATTR_PREFIX, icuFromI18nMessage} from './util';\n\nexport type I18nMeta = {\n  id?: string,\n  customId?: string,\n  legacyIds?: string[],\n  description?: string,\n  meaning?: string\n};\n\n\nconst setI18nRefs: VisitNodeFn = (htmlNode, i18nNode) => {\n  if (htmlNode instanceof html.NodeWithI18n) {\n    if (i18nNode instanceof i18n.IcuPlaceholder && htmlNode.i18n instanceof i18n.Message) {\n      // This html node represents an ICU but this is a second processing pass, and the legacy id\n      // was computed in the previous pass and stored in the `i18n` property as a message.\n      // We are about to wipe out that property so capture the previous message to be reused when\n      // generating the message for this ICU later. See `_generateI18nMessage()`.\n      i18nNode.previousMessage = htmlNode.i18n;\n    }\n    htmlNode.i18n = i18nNode;\n  }\n  return i18nNode;\n};\n\n/**\n * This visitor walks over HTML parse tree and converts information stored in\n * i18n-related attributes (\"i18n\" and \"i18n-*\") into i18n meta object that is\n * stored with other element's and attribute's information.\n */\nexport class I18nMetaVisitor implements html.Visitor {\n  // whether visited nodes contain i18n information\n  public hasI18nMeta: boolean = false;\n\n  // i18n message generation factory\n  private _createI18nMessage = createI18nMessageFactory(this.interpolationConfig);\n\n  constructor(\n      private interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG,\n      private keepI18nAttrs = false, private enableI18nLegacyMessageIdFormat = false) {}\n\n  private _generateI18nMessage(\n      nodes: html.Node[], meta: string|i18n.I18nMeta = '',\n      visitNodeFn?: VisitNodeFn): i18n.Message {\n    const {meaning, description, customId} = this._parseMetadata(meta);\n    const message = this._createI18nMessage(nodes, meaning, description, customId, visitNodeFn);\n    this._setMessageId(message, meta);\n    this._setLegacyIds(message, meta);\n    return message;\n  }\n\n  visitElement(element: html.Element): any {\n    if (hasI18nAttrs(element)) {\n      this.hasI18nMeta = true;\n      const attrs: html.Attribute[] = [];\n      const attrsMeta: {[key: string]: string} = {};\n\n      for (const attr of element.attrs) {\n        if (attr.name === I18N_ATTR) {\n          // root 'i18n' node attribute\n          const i18n = element.i18n || attr.value;\n          const message = this._generateI18nMessage(element.children, i18n, setI18nRefs);\n          // do not assign empty i18n meta\n          if (message.nodes.length) {\n            element.i18n = message;\n          }\n\n        } else if (attr.name.startsWith(I18N_ATTR_PREFIX)) {\n          // 'i18n-*' attributes\n          const key = attr.name.slice(I18N_ATTR_PREFIX.length);\n          attrsMeta[key] = attr.value;\n\n        } else {\n          // non-i18n attributes\n          attrs.push(attr);\n        }\n      }\n\n      // set i18n meta for attributes\n      if (Object.keys(attrsMeta).length) {\n        for (const attr of attrs) {\n          const meta = attrsMeta[attr.name];\n          // do not create translation for empty attributes\n          if (meta !== undefined && attr.value) {\n            attr.i18n = this._generateI18nMessage([attr], attr.i18n || meta);\n          }\n        }\n      }\n\n      if (!this.keepI18nAttrs) {\n        // update element's attributes,\n        // keeping only non-i18n related ones\n        element.attrs = attrs;\n      }\n    }\n    html.visitAll(this, element.children, element.i18n);\n    return element;\n  }\n\n  visitExpansion(expansion: html.Expansion, currentMessage: i18n.Message|undefined): any {\n    let message;\n    const meta = expansion.i18n;\n    this.hasI18nMeta = true;\n    if (meta instanceof i18n.IcuPlaceholder) {\n      // set ICU placeholder name (e.g. \"ICU_1\"),\n      // generated while processing root element contents,\n      // so we can reference it when we output translation\n      const name = meta.name;\n      message = this._generateI18nMessage([expansion], meta);\n      const icu = icuFromI18nMessage(message);\n      icu.name = name;\n    } else {\n      // ICU is a top level message, try to use metadata from container element if provided via\n      // `context` argument. Note: context may not be available for standalone ICUs (without\n      // wrapping element), so fallback to ICU metadata in this case.\n      message = this._generateI18nMessage([expansion], currentMessage || meta);\n    }\n    expansion.i18n = message;\n    return expansion;\n  }\n\n  visitText(text: html.Text): any {\n    return text;\n  }\n  visitAttribute(attribute: html.Attribute): any {\n    return attribute;\n  }\n  visitComment(comment: html.Comment): any {\n    return comment;\n  }\n  visitExpansionCase(expansionCase: html.ExpansionCase): any {\n    return expansionCase;\n  }\n\n  /**\n   * Parse the general form `meta` passed into extract the explicit metadata needed to create a\n   * `Message`.\n   *\n   * There are three possibilities for the `meta` variable\n   * 1) a string from an `i18n` template attribute: parse it to extract the metadata values.\n   * 2) a `Message` from a previous processing pass: reuse the metadata values in the message.\n   * 4) other: ignore this and just process the message metadata as normal\n   *\n   * @param meta the bucket that holds information about the message\n   * @returns the parsed metadata.\n   */\n  private _parseMetadata(meta: string|i18n.I18nMeta): I18nMeta {\n    return typeof meta === 'string' ? parseI18nMeta(meta) :\n                                      meta instanceof i18n.Message ? meta : {};\n  }\n\n  /**\n   * Generate (or restore) message id if not specified already.\n   */\n  private _setMessageId(message: i18n.Message, meta: string|i18n.I18nMeta): void {\n    if (!message.id) {\n      message.id = meta instanceof i18n.Message && meta.id || decimalDigest(message);\n    }\n  }\n\n  /**\n   * Update the `message` with a `legacyId` if necessary.\n   *\n   * @param message the message whose legacy id should be set\n   * @param meta information about the message being processed\n   */\n  private _setLegacyIds(message: i18n.Message, meta: string|i18n.I18nMeta): void {\n    if (this.enableI18nLegacyMessageIdFormat) {\n      message.legacyIds = [computeDigest(message), computeDecimalDigest(message)];\n    } else if (typeof meta !== 'string') {\n      // This occurs if we are doing the 2nd pass after whitespace removal (see `parseTemplate()` in\n      // `packages/compiler/src/render3/view/template.ts`).\n      // In that case we want to reuse the legacy message generated in the 1st pass (see\n      // `setI18nRefs()`).\n      const previousMessage = meta instanceof i18n.Message ?\n          meta :\n          meta instanceof i18n.IcuPlaceholder ? meta.previousMessage : undefined;\n      message.legacyIds = previousMessage ? previousMessage.legacyIds : [];\n    }\n  }\n}\n\n/** I18n separators for metadata **/\nconst I18N_MEANING_SEPARATOR = '|';\nconst I18N_ID_SEPARATOR = '@@';\n\n/**\n * Parses i18n metas like:\n *  - \"@@id\",\n *  - \"description[@@id]\",\n *  - \"meaning|description[@@id]\"\n * and returns an object with parsed output.\n *\n * @param meta String that represents i18n meta\n * @returns Object with id, meaning and description fields\n */\nexport function parseI18nMeta(meta: string = ''): I18nMeta {\n  let customId: string|undefined;\n  let meaning: string|undefined;\n  let description: string|undefined;\n\n  meta = meta.trim();\n  if (meta) {\n    const idIndex = meta.indexOf(I18N_ID_SEPARATOR);\n    const descIndex = meta.indexOf(I18N_MEANING_SEPARATOR);\n    let meaningAndDesc: string;\n    [meaningAndDesc, customId] =\n        (idIndex > -1) ? [meta.slice(0, idIndex), meta.slice(idIndex + 2)] : [meta, ''];\n    [meaning, description] = (descIndex > -1) ?\n        [meaningAndDesc.slice(0, descIndex), meaningAndDesc.slice(descIndex + 1)] :\n        ['', meaningAndDesc];\n  }\n\n  return {customId, meaning, description};\n}\n\n// Converts i18n meta information for a message (id, description, meaning)\n// to a JsDoc statement formatted as expected by the Closure compiler.\nexport function i18nMetaToJSDoc(meta: I18nMeta): o.JSDocComment|null {\n  const tags: o.JSDocTag[] = [];\n  if (meta.description) {\n    tags.push({tagName: o.JSDocTagName.Desc, text: meta.description});\n  }\n  if (meta.meaning) {\n    tags.push({tagName: o.JSDocTagName.Meaning, text: meta.meaning});\n  }\n  return tags.length == 0 ? null : o.jsDocComment(tags);\n}\n"]}