[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | (function (factory) {
|
---|
| 9 | if (typeof module === "object" && typeof module.exports === "object") {
|
---|
| 10 | var v = factory(require, exports);
|
---|
| 11 | if (v !== undefined) module.exports = v;
|
---|
| 12 | }
|
---|
| 13 | else if (typeof define === "function" && define.amd) {
|
---|
| 14 | define("@angular/compiler/src/i18n/i18n_parser", ["require", "exports", "tslib", "@angular/compiler/src/expression_parser/lexer", "@angular/compiler/src/expression_parser/parser", "@angular/compiler/src/ml_parser/ast", "@angular/compiler/src/ml_parser/html_tags", "@angular/compiler/src/parse_util", "@angular/compiler/src/i18n/i18n_ast", "@angular/compiler/src/i18n/serializers/placeholder"], factory);
|
---|
| 15 | }
|
---|
| 16 | })(function (require, exports) {
|
---|
| 17 | "use strict";
|
---|
| 18 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
| 19 | exports.createI18nMessageFactory = void 0;
|
---|
| 20 | var tslib_1 = require("tslib");
|
---|
| 21 | var lexer_1 = require("@angular/compiler/src/expression_parser/lexer");
|
---|
| 22 | var parser_1 = require("@angular/compiler/src/expression_parser/parser");
|
---|
| 23 | var html = require("@angular/compiler/src/ml_parser/ast");
|
---|
| 24 | var html_tags_1 = require("@angular/compiler/src/ml_parser/html_tags");
|
---|
| 25 | var parse_util_1 = require("@angular/compiler/src/parse_util");
|
---|
| 26 | var i18n = require("@angular/compiler/src/i18n/i18n_ast");
|
---|
| 27 | var placeholder_1 = require("@angular/compiler/src/i18n/serializers/placeholder");
|
---|
| 28 | var _expParser = new parser_1.Parser(new lexer_1.Lexer());
|
---|
| 29 | /**
|
---|
| 30 | * Returns a function converting html nodes to an i18n Message given an interpolationConfig
|
---|
| 31 | */
|
---|
| 32 | function createI18nMessageFactory(interpolationConfig) {
|
---|
| 33 | var visitor = new _I18nVisitor(_expParser, interpolationConfig);
|
---|
| 34 | return function (nodes, meaning, description, customId, visitNodeFn) {
|
---|
| 35 | return visitor.toI18nMessage(nodes, meaning, description, customId, visitNodeFn);
|
---|
| 36 | };
|
---|
| 37 | }
|
---|
| 38 | exports.createI18nMessageFactory = createI18nMessageFactory;
|
---|
| 39 | function noopVisitNodeFn(_html, i18n) {
|
---|
| 40 | return i18n;
|
---|
| 41 | }
|
---|
| 42 | var _I18nVisitor = /** @class */ (function () {
|
---|
| 43 | function _I18nVisitor(_expressionParser, _interpolationConfig) {
|
---|
| 44 | this._expressionParser = _expressionParser;
|
---|
| 45 | this._interpolationConfig = _interpolationConfig;
|
---|
| 46 | }
|
---|
| 47 | _I18nVisitor.prototype.toI18nMessage = function (nodes, meaning, description, customId, visitNodeFn) {
|
---|
| 48 | if (meaning === void 0) { meaning = ''; }
|
---|
| 49 | if (description === void 0) { description = ''; }
|
---|
| 50 | if (customId === void 0) { customId = ''; }
|
---|
| 51 | var context = {
|
---|
| 52 | isIcu: nodes.length == 1 && nodes[0] instanceof html.Expansion,
|
---|
| 53 | icuDepth: 0,
|
---|
| 54 | placeholderRegistry: new placeholder_1.PlaceholderRegistry(),
|
---|
| 55 | placeholderToContent: {},
|
---|
| 56 | placeholderToMessage: {},
|
---|
| 57 | visitNodeFn: visitNodeFn || noopVisitNodeFn,
|
---|
| 58 | };
|
---|
| 59 | var i18nodes = html.visitAll(this, nodes, context);
|
---|
| 60 | return new i18n.Message(i18nodes, context.placeholderToContent, context.placeholderToMessage, meaning, description, customId);
|
---|
| 61 | };
|
---|
| 62 | _I18nVisitor.prototype.visitElement = function (el, context) {
|
---|
| 63 | var _a;
|
---|
| 64 | var children = html.visitAll(this, el.children, context);
|
---|
| 65 | var attrs = {};
|
---|
| 66 | el.attrs.forEach(function (attr) {
|
---|
| 67 | // Do not visit the attributes, translatable ones are top-level ASTs
|
---|
| 68 | attrs[attr.name] = attr.value;
|
---|
| 69 | });
|
---|
| 70 | var isVoid = html_tags_1.getHtmlTagDefinition(el.name).isVoid;
|
---|
| 71 | var startPhName = context.placeholderRegistry.getStartTagPlaceholderName(el.name, attrs, isVoid);
|
---|
| 72 | context.placeholderToContent[startPhName] = {
|
---|
| 73 | text: el.startSourceSpan.toString(),
|
---|
| 74 | sourceSpan: el.startSourceSpan,
|
---|
| 75 | };
|
---|
| 76 | var closePhName = '';
|
---|
| 77 | if (!isVoid) {
|
---|
| 78 | closePhName = context.placeholderRegistry.getCloseTagPlaceholderName(el.name);
|
---|
| 79 | context.placeholderToContent[closePhName] = {
|
---|
| 80 | text: "</" + el.name + ">",
|
---|
| 81 | sourceSpan: (_a = el.endSourceSpan) !== null && _a !== void 0 ? _a : el.sourceSpan,
|
---|
| 82 | };
|
---|
| 83 | }
|
---|
| 84 | var node = new i18n.TagPlaceholder(el.name, attrs, startPhName, closePhName, children, isVoid, el.sourceSpan, el.startSourceSpan, el.endSourceSpan);
|
---|
| 85 | return context.visitNodeFn(el, node);
|
---|
| 86 | };
|
---|
| 87 | _I18nVisitor.prototype.visitAttribute = function (attribute, context) {
|
---|
| 88 | var node = attribute.valueTokens === undefined || attribute.valueTokens.length === 1 ?
|
---|
| 89 | new i18n.Text(attribute.value, attribute.valueSpan || attribute.sourceSpan) :
|
---|
| 90 | this._visitTextWithInterpolation(attribute.valueTokens, attribute.valueSpan || attribute.sourceSpan, context, attribute.i18n);
|
---|
| 91 | return context.visitNodeFn(attribute, node);
|
---|
| 92 | };
|
---|
| 93 | _I18nVisitor.prototype.visitText = function (text, context) {
|
---|
| 94 | var node = text.tokens.length === 1 ?
|
---|
| 95 | new i18n.Text(text.value, text.sourceSpan) :
|
---|
| 96 | this._visitTextWithInterpolation(text.tokens, text.sourceSpan, context, text.i18n);
|
---|
| 97 | return context.visitNodeFn(text, node);
|
---|
| 98 | };
|
---|
| 99 | _I18nVisitor.prototype.visitComment = function (comment, context) {
|
---|
| 100 | return null;
|
---|
| 101 | };
|
---|
| 102 | _I18nVisitor.prototype.visitExpansion = function (icu, context) {
|
---|
| 103 | var _this = this;
|
---|
| 104 | context.icuDepth++;
|
---|
| 105 | var i18nIcuCases = {};
|
---|
| 106 | var i18nIcu = new i18n.Icu(icu.switchValue, icu.type, i18nIcuCases, icu.sourceSpan);
|
---|
| 107 | icu.cases.forEach(function (caze) {
|
---|
| 108 | i18nIcuCases[caze.value] = new i18n.Container(caze.expression.map(function (node) { return node.visit(_this, context); }), caze.expSourceSpan);
|
---|
| 109 | });
|
---|
| 110 | context.icuDepth--;
|
---|
| 111 | if (context.isIcu || context.icuDepth > 0) {
|
---|
| 112 | // Returns an ICU node when:
|
---|
| 113 | // - the message (vs a part of the message) is an ICU message, or
|
---|
| 114 | // - the ICU message is nested.
|
---|
| 115 | var expPh = context.placeholderRegistry.getUniquePlaceholder("VAR_" + icu.type);
|
---|
| 116 | i18nIcu.expressionPlaceholder = expPh;
|
---|
| 117 | context.placeholderToContent[expPh] = {
|
---|
| 118 | text: icu.switchValue,
|
---|
| 119 | sourceSpan: icu.switchValueSourceSpan,
|
---|
| 120 | };
|
---|
| 121 | return context.visitNodeFn(icu, i18nIcu);
|
---|
| 122 | }
|
---|
| 123 | // Else returns a placeholder
|
---|
| 124 | // ICU placeholders should not be replaced with their original content but with the their
|
---|
| 125 | // translations.
|
---|
| 126 | // TODO(vicb): add a html.Node -> i18n.Message cache to avoid having to re-create the msg
|
---|
| 127 | var phName = context.placeholderRegistry.getPlaceholderName('ICU', icu.sourceSpan.toString());
|
---|
| 128 | context.placeholderToMessage[phName] = this.toI18nMessage([icu], '', '', '', undefined);
|
---|
| 129 | var node = new i18n.IcuPlaceholder(i18nIcu, phName, icu.sourceSpan);
|
---|
| 130 | return context.visitNodeFn(icu, node);
|
---|
| 131 | };
|
---|
| 132 | _I18nVisitor.prototype.visitExpansionCase = function (_icuCase, _context) {
|
---|
| 133 | throw new Error('Unreachable code');
|
---|
| 134 | };
|
---|
| 135 | /**
|
---|
| 136 | * Convert, text and interpolated tokens up into text and placeholder pieces.
|
---|
| 137 | *
|
---|
| 138 | * @param tokens The text and interpolated tokens.
|
---|
| 139 | * @param sourceSpan The span of the whole of the `text` string.
|
---|
| 140 | * @param context The current context of the visitor, used to compute and store placeholders.
|
---|
| 141 | * @param previousI18n Any i18n metadata associated with this `text` from a previous pass.
|
---|
| 142 | */
|
---|
| 143 | _I18nVisitor.prototype._visitTextWithInterpolation = function (tokens, sourceSpan, context, previousI18n) {
|
---|
| 144 | var e_1, _a;
|
---|
| 145 | // Return a sequence of `Text` and `Placeholder` nodes grouped in a `Container`.
|
---|
| 146 | var nodes = [];
|
---|
| 147 | // We will only create a container if there are actually interpolations,
|
---|
| 148 | // so this flag tracks that.
|
---|
| 149 | var hasInterpolation = false;
|
---|
| 150 | try {
|
---|
| 151 | for (var tokens_1 = tslib_1.__values(tokens), tokens_1_1 = tokens_1.next(); !tokens_1_1.done; tokens_1_1 = tokens_1.next()) {
|
---|
| 152 | var token = tokens_1_1.value;
|
---|
| 153 | switch (token.type) {
|
---|
| 154 | case 8 /* INTERPOLATION */:
|
---|
| 155 | case 17 /* ATTR_VALUE_INTERPOLATION */:
|
---|
| 156 | hasInterpolation = true;
|
---|
| 157 | var expression = token.parts[1];
|
---|
| 158 | var baseName = extractPlaceholderName(expression) || 'INTERPOLATION';
|
---|
| 159 | var phName = context.placeholderRegistry.getPlaceholderName(baseName, expression);
|
---|
| 160 | context.placeholderToContent[phName] = {
|
---|
| 161 | text: token.parts.join(''),
|
---|
| 162 | sourceSpan: token.sourceSpan
|
---|
| 163 | };
|
---|
| 164 | nodes.push(new i18n.Placeholder(expression, phName, token.sourceSpan));
|
---|
| 165 | break;
|
---|
| 166 | default:
|
---|
| 167 | if (token.parts[0].length > 0) {
|
---|
| 168 | // This token is text or an encoded entity.
|
---|
| 169 | // If it is following on from a previous text node then merge it into that node
|
---|
| 170 | // Otherwise, if it is following an interpolation, then add a new node.
|
---|
| 171 | var previous = nodes[nodes.length - 1];
|
---|
| 172 | if (previous instanceof i18n.Text) {
|
---|
| 173 | previous.value += token.parts[0];
|
---|
| 174 | previous.sourceSpan = new parse_util_1.ParseSourceSpan(previous.sourceSpan.start, token.sourceSpan.end, previous.sourceSpan.fullStart, previous.sourceSpan.details);
|
---|
| 175 | }
|
---|
| 176 | else {
|
---|
| 177 | nodes.push(new i18n.Text(token.parts[0], token.sourceSpan));
|
---|
| 178 | }
|
---|
| 179 | }
|
---|
| 180 | break;
|
---|
| 181 | }
|
---|
| 182 | }
|
---|
| 183 | }
|
---|
| 184 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
---|
| 185 | finally {
|
---|
| 186 | try {
|
---|
| 187 | if (tokens_1_1 && !tokens_1_1.done && (_a = tokens_1.return)) _a.call(tokens_1);
|
---|
| 188 | }
|
---|
| 189 | finally { if (e_1) throw e_1.error; }
|
---|
| 190 | }
|
---|
| 191 | if (hasInterpolation) {
|
---|
| 192 | // Whitespace removal may have invalidated the interpolation source-spans.
|
---|
| 193 | reusePreviousSourceSpans(nodes, previousI18n);
|
---|
| 194 | return new i18n.Container(nodes, sourceSpan);
|
---|
| 195 | }
|
---|
| 196 | else {
|
---|
| 197 | return nodes[0];
|
---|
| 198 | }
|
---|
| 199 | };
|
---|
| 200 | return _I18nVisitor;
|
---|
| 201 | }());
|
---|
| 202 | /**
|
---|
| 203 | * Re-use the source-spans from `previousI18n` metadata for the `nodes`.
|
---|
| 204 | *
|
---|
| 205 | * Whitespace removal can invalidate the source-spans of interpolation nodes, so we
|
---|
| 206 | * reuse the source-span stored from a previous pass before the whitespace was removed.
|
---|
| 207 | *
|
---|
| 208 | * @param nodes The `Text` and `Placeholder` nodes to be processed.
|
---|
| 209 | * @param previousI18n Any i18n metadata for these `nodes` stored from a previous pass.
|
---|
| 210 | */
|
---|
| 211 | function reusePreviousSourceSpans(nodes, previousI18n) {
|
---|
| 212 | if (previousI18n instanceof i18n.Message) {
|
---|
| 213 | // The `previousI18n` is an i18n `Message`, so we are processing an `Attribute` with i18n
|
---|
| 214 | // metadata. The `Message` should consist only of a single `Container` that contains the
|
---|
| 215 | // parts (`Text` and `Placeholder`) to process.
|
---|
| 216 | assertSingleContainerMessage(previousI18n);
|
---|
| 217 | previousI18n = previousI18n.nodes[0];
|
---|
| 218 | }
|
---|
| 219 | if (previousI18n instanceof i18n.Container) {
|
---|
| 220 | // The `previousI18n` is a `Container`, which means that this is a second i18n extraction pass
|
---|
| 221 | // after whitespace has been removed from the AST nodes.
|
---|
| 222 | assertEquivalentNodes(previousI18n.children, nodes);
|
---|
| 223 | // Reuse the source-spans from the first pass.
|
---|
| 224 | for (var i = 0; i < nodes.length; i++) {
|
---|
| 225 | nodes[i].sourceSpan = previousI18n.children[i].sourceSpan;
|
---|
| 226 | }
|
---|
| 227 | }
|
---|
| 228 | }
|
---|
| 229 | /**
|
---|
| 230 | * Asserts that the `message` contains exactly one `Container` node.
|
---|
| 231 | */
|
---|
| 232 | function assertSingleContainerMessage(message) {
|
---|
| 233 | var nodes = message.nodes;
|
---|
| 234 | if (nodes.length !== 1 || !(nodes[0] instanceof i18n.Container)) {
|
---|
| 235 | throw new Error('Unexpected previous i18n message - expected it to consist of only a single `Container` node.');
|
---|
| 236 | }
|
---|
| 237 | }
|
---|
| 238 | /**
|
---|
| 239 | * Asserts that the `previousNodes` and `node` collections have the same number of elements and
|
---|
| 240 | * corresponding elements have the same node type.
|
---|
| 241 | */
|
---|
| 242 | function assertEquivalentNodes(previousNodes, nodes) {
|
---|
| 243 | if (previousNodes.length !== nodes.length) {
|
---|
| 244 | throw new Error('The number of i18n message children changed between first and second pass.');
|
---|
| 245 | }
|
---|
| 246 | if (previousNodes.some(function (node, i) { return nodes[i].constructor !== node.constructor; })) {
|
---|
| 247 | throw new Error('The types of the i18n message children changed between first and second pass.');
|
---|
| 248 | }
|
---|
| 249 | }
|
---|
| 250 | var _CUSTOM_PH_EXP = /\/\/[\s\S]*i18n[\s\S]*\([\s\S]*ph[\s\S]*=[\s\S]*("|')([\s\S]*?)\1[\s\S]*\)/g;
|
---|
| 251 | function extractPlaceholderName(input) {
|
---|
| 252 | return input.split(_CUSTOM_PH_EXP)[2];
|
---|
| 253 | }
|
---|
| 254 | });
|
---|
| 255 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n_parser.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/i18n/i18n_parser.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 {Lexer as ExpressionLexer} from '../expression_parser/lexer';\nimport {Parser as ExpressionParser} from '../expression_parser/parser';\nimport * as html from '../ml_parser/ast';\nimport {getHtmlTagDefinition} from '../ml_parser/html_tags';\nimport {InterpolationConfig} from '../ml_parser/interpolation_config';\nimport {InterpolatedAttributeToken, InterpolatedTextToken, TokenType} from '../ml_parser/tokens';\nimport {ParseSourceSpan} from '../parse_util';\n\nimport * as i18n from './i18n_ast';\nimport {PlaceholderRegistry} from './serializers/placeholder';\n\nconst _expParser = new ExpressionParser(new ExpressionLexer());\n\nexport type VisitNodeFn = (html: html.Node, i18n: i18n.Node) => i18n.Node;\n\nexport interface I18nMessageFactory {\n  (nodes: html.Node[], meaning: string|undefined, description: string|undefined,\n   customId: string|undefined, visitNodeFn?: VisitNodeFn): i18n.Message;\n}\n\n/**\n * Returns a function converting html nodes to an i18n Message given an interpolationConfig\n */\nexport function createI18nMessageFactory(interpolationConfig: InterpolationConfig):\n    I18nMessageFactory {\n  const visitor = new _I18nVisitor(_expParser, interpolationConfig);\n  return (nodes, meaning, description, customId, visitNodeFn) =>\n             visitor.toI18nMessage(nodes, meaning, description, customId, visitNodeFn);\n}\n\ninterface I18nMessageVisitorContext {\n  isIcu: boolean;\n  icuDepth: number;\n  placeholderRegistry: PlaceholderRegistry;\n  placeholderToContent: {[phName: string]: i18n.MessagePlaceholder};\n  placeholderToMessage: {[phName: string]: i18n.Message};\n  visitNodeFn: VisitNodeFn;\n}\n\nfunction noopVisitNodeFn(_html: html.Node, i18n: i18n.Node): i18n.Node {\n  return i18n;\n}\n\nclass _I18nVisitor implements html.Visitor {\n  constructor(\n      private _expressionParser: ExpressionParser,\n      private _interpolationConfig: InterpolationConfig) {}\n\n  public toI18nMessage(\n      nodes: html.Node[], meaning = '', description = '', customId = '',\n      visitNodeFn: VisitNodeFn|undefined): i18n.Message {\n    const context: I18nMessageVisitorContext = {\n      isIcu: nodes.length == 1 && nodes[0] instanceof html.Expansion,\n      icuDepth: 0,\n      placeholderRegistry: new PlaceholderRegistry(),\n      placeholderToContent: {},\n      placeholderToMessage: {},\n      visitNodeFn: visitNodeFn || noopVisitNodeFn,\n    };\n\n    const i18nodes: i18n.Node[] = html.visitAll(this, nodes, context);\n\n    return new i18n.Message(\n        i18nodes, context.placeholderToContent, context.placeholderToMessage, meaning, description,\n        customId);\n  }\n\n  visitElement(el: html.Element, context: I18nMessageVisitorContext): i18n.Node {\n    const children = html.visitAll(this, el.children, context);\n    const attrs: {[k: string]: string} = {};\n    el.attrs.forEach(attr => {\n      // Do not visit the attributes, translatable ones are top-level ASTs\n      attrs[attr.name] = attr.value;\n    });\n\n    const isVoid: boolean = getHtmlTagDefinition(el.name).isVoid;\n    const startPhName =\n        context.placeholderRegistry.getStartTagPlaceholderName(el.name, attrs, isVoid);\n    context.placeholderToContent[startPhName] = {\n      text: el.startSourceSpan.toString(),\n      sourceSpan: el.startSourceSpan,\n    };\n\n    let closePhName = '';\n\n    if (!isVoid) {\n      closePhName = context.placeholderRegistry.getCloseTagPlaceholderName(el.name);\n      context.placeholderToContent[closePhName] = {\n        text: `</${el.name}>`,\n        sourceSpan: el.endSourceSpan ?? el.sourceSpan,\n      };\n    }\n\n    const node = new i18n.TagPlaceholder(\n        el.name, attrs, startPhName, closePhName, children, isVoid, el.sourceSpan,\n        el.startSourceSpan, el.endSourceSpan);\n    return context.visitNodeFn(el, node);\n  }\n\n  visitAttribute(attribute: html.Attribute, context: I18nMessageVisitorContext): i18n.Node {\n    const node = attribute.valueTokens === undefined || attribute.valueTokens.length === 1 ?\n        new i18n.Text(attribute.value, attribute.valueSpan || attribute.sourceSpan) :\n        this._visitTextWithInterpolation(\n            attribute.valueTokens, attribute.valueSpan || attribute.sourceSpan, context,\n            attribute.i18n);\n    return context.visitNodeFn(attribute, node);\n  }\n\n  visitText(text: html.Text, context: I18nMessageVisitorContext): i18n.Node {\n    const node = text.tokens.length === 1 ?\n        new i18n.Text(text.value, text.sourceSpan) :\n        this._visitTextWithInterpolation(text.tokens, text.sourceSpan, context, text.i18n);\n    return context.visitNodeFn(text, node);\n  }\n\n  visitComment(comment: html.Comment, context: I18nMessageVisitorContext): i18n.Node|null {\n    return null;\n  }\n\n  visitExpansion(icu: html.Expansion, context: I18nMessageVisitorContext): i18n.Node {\n    context.icuDepth++;\n    const i18nIcuCases: {[k: string]: i18n.Node} = {};\n    const i18nIcu = new i18n.Icu(icu.switchValue, icu.type, i18nIcuCases, icu.sourceSpan);\n    icu.cases.forEach((caze): void => {\n      i18nIcuCases[caze.value] = new i18n.Container(\n          caze.expression.map((node) => node.visit(this, context)), caze.expSourceSpan);\n    });\n    context.icuDepth--;\n\n    if (context.isIcu || context.icuDepth > 0) {\n      // Returns an ICU node when:\n      // - the message (vs a part of the message) is an ICU message, or\n      // - the ICU message is nested.\n      const expPh = context.placeholderRegistry.getUniquePlaceholder(`VAR_${icu.type}`);\n      i18nIcu.expressionPlaceholder = expPh;\n      context.placeholderToContent[expPh] = {\n        text: icu.switchValue,\n        sourceSpan: icu.switchValueSourceSpan,\n      };\n      return context.visitNodeFn(icu, i18nIcu);\n    }\n\n    // Else returns a placeholder\n    // ICU placeholders should not be replaced with their original content but with the their\n    // translations.\n    // TODO(vicb): add a html.Node -> i18n.Message cache to avoid having to re-create the msg\n    const phName = context.placeholderRegistry.getPlaceholderName('ICU', icu.sourceSpan.toString());\n    context.placeholderToMessage[phName] = this.toI18nMessage([icu], '', '', '', undefined);\n    const node = new i18n.IcuPlaceholder(i18nIcu, phName, icu.sourceSpan);\n    return context.visitNodeFn(icu, node);\n  }\n\n  visitExpansionCase(_icuCase: html.ExpansionCase, _context: I18nMessageVisitorContext): i18n.Node {\n    throw new Error('Unreachable code');\n  }\n\n  /**\n   * Convert, text and interpolated tokens up into text and placeholder pieces.\n   *\n   * @param tokens The text and interpolated tokens.\n   * @param sourceSpan The span of the whole of the `text` string.\n   * @param context The current context of the visitor, used to compute and store placeholders.\n   * @param previousI18n Any i18n metadata associated with this `text` from a previous pass.\n   */\n  private _visitTextWithInterpolation(\n      tokens: (InterpolatedTextToken|InterpolatedAttributeToken)[], sourceSpan: ParseSourceSpan,\n      context: I18nMessageVisitorContext, previousI18n: i18n.I18nMeta|undefined): i18n.Node {\n    // Return a sequence of `Text` and `Placeholder` nodes grouped in a `Container`.\n    const nodes: i18n.Node[] = [];\n    // We will only create a container if there are actually interpolations,\n    // so this flag tracks that.\n    let hasInterpolation = false;\n    for (const token of tokens) {\n      switch (token.type) {\n        case TokenType.INTERPOLATION:\n        case TokenType.ATTR_VALUE_INTERPOLATION:\n          hasInterpolation = true;\n          const expression = token.parts[1];\n          const baseName = extractPlaceholderName(expression) || 'INTERPOLATION';\n          const phName = context.placeholderRegistry.getPlaceholderName(baseName, expression);\n          context.placeholderToContent[phName] = {\n            text: token.parts.join(''),\n            sourceSpan: token.sourceSpan\n          };\n          nodes.push(new i18n.Placeholder(expression, phName, token.sourceSpan));\n          break;\n        default:\n          if (token.parts[0].length > 0) {\n            // This token is text or an encoded entity.\n            // If it is following on from a previous text node then merge it into that node\n            // Otherwise, if it is following an interpolation, then add a new node.\n            const previous = nodes[nodes.length - 1];\n            if (previous instanceof i18n.Text) {\n              previous.value += token.parts[0];\n              previous.sourceSpan = new ParseSourceSpan(\n                  previous.sourceSpan.start, token.sourceSpan.end, previous.sourceSpan.fullStart,\n                  previous.sourceSpan.details);\n            } else {\n              nodes.push(new i18n.Text(token.parts[0], token.sourceSpan));\n            }\n          }\n          break;\n      }\n    }\n\n    if (hasInterpolation) {\n      // Whitespace removal may have invalidated the interpolation source-spans.\n      reusePreviousSourceSpans(nodes, previousI18n);\n      return new i18n.Container(nodes, sourceSpan);\n    } else {\n      return nodes[0];\n    }\n  }\n}\n\n/**\n * Re-use the source-spans from `previousI18n` metadata for the `nodes`.\n *\n * Whitespace removal can invalidate the source-spans of interpolation nodes, so we\n * reuse the source-span stored from a previous pass before the whitespace was removed.\n *\n * @param nodes The `Text` and `Placeholder` nodes to be processed.\n * @param previousI18n Any i18n metadata for these `nodes` stored from a previous pass.\n */\nfunction reusePreviousSourceSpans(nodes: i18n.Node[], previousI18n: i18n.I18nMeta|undefined): void {\n  if (previousI18n instanceof i18n.Message) {\n    // The `previousI18n` is an i18n `Message`, so we are processing an `Attribute` with i18n\n    // metadata. The `Message` should consist only of a single `Container` that contains the\n    // parts (`Text` and `Placeholder`) to process.\n    assertSingleContainerMessage(previousI18n);\n    previousI18n = previousI18n.nodes[0];\n  }\n\n  if (previousI18n instanceof i18n.Container) {\n    // The `previousI18n` is a `Container`, which means that this is a second i18n extraction pass\n    // after whitespace has been removed from the AST nodes.\n    assertEquivalentNodes(previousI18n.children, nodes);\n\n    // Reuse the source-spans from the first pass.\n    for (let i = 0; i < nodes.length; i++) {\n      nodes[i].sourceSpan = previousI18n.children[i].sourceSpan;\n    }\n  }\n}\n\n/**\n * Asserts that the `message` contains exactly one `Container` node.\n */\nfunction assertSingleContainerMessage(message: i18n.Message): void {\n  const nodes = message.nodes;\n  if (nodes.length !== 1 || !(nodes[0] instanceof i18n.Container)) {\n    throw new Error(\n        'Unexpected previous i18n message - expected it to consist of only a single `Container` node.');\n  }\n}\n\n/**\n * Asserts that the `previousNodes` and `node` collections have the same number of elements and\n * corresponding elements have the same node type.\n */\nfunction assertEquivalentNodes(previousNodes: i18n.Node[], nodes: i18n.Node[]): void {\n  if (previousNodes.length !== nodes.length) {\n    throw new Error('The number of i18n message children changed between first and second pass.');\n  }\n  if (previousNodes.some((node, i) => nodes[i].constructor !== node.constructor)) {\n    throw new Error(\n        'The types of the i18n message children changed between first and second pass.');\n  }\n}\n\nconst _CUSTOM_PH_EXP =\n    /\\/\\/[\\s\\S]*i18n[\\s\\S]*\\([\\s\\S]*ph[\\s\\S]*=[\\s\\S]*(\"|')([\\s\\S]*?)\\1[\\s\\S]*\\)/g;\n\nfunction extractPlaceholderName(input: string): string {\n  return input.split(_CUSTOM_PH_EXP)[2];\n}\n"]} |
---|