feat(compiler): add "original" placeholder value on extracted XMB (#25079)
Update XMB placeholders(<ph>) to include the original value on top of an
example. Placeholders can by definition have one example(<ex>) tag and a
text node. The text node is used by TC as the "original" value from the
placeholder, while the example should represent a dummy value.
For example: <ph name="PET"><ex>Gopher</ex>{{ petName }}</ph>.
This change makes sure that we have the original text, but it *DOES NOT*
make sure that the example is correct. The example has the same wrong
behavior of showing the interpolation text rather than a useful
example.
No breaking changes, but tools that depend on the previous behavior and
don't consider the full XMB definition may fail to parse the XMB.
Fixes b/72565847
PR Close #25079
			
			
This commit is contained in:
		
							parent
							
								
									24789e9ad9
								
							
						
					
					
						commit
						e99d860393
					
				| @ -43,6 +43,7 @@ jasmine_node_test( | |||||||
|     ], |     ], | ||||||
|     deps = [ |     deps = [ | ||||||
|         ":extract_i18n_lib", |         ":extract_i18n_lib", | ||||||
|  |         "//packages/common:npm_package", | ||||||
|         "//packages/core", |         "//packages/core", | ||||||
|         "//tools/testing:node", |         "//tools/testing:node", | ||||||
|     ], |     ], | ||||||
|  | |||||||
| @ -46,6 +46,15 @@ const EXPECTED_XMB = `<?xml version="1.0" encoding="UTF-8" ?> | |||||||
|   <msg id="8136548302122759730" desc="desc" meaning="meaning"><source>src/basic.html:1</source><source>src/comp2.ts:1</source><source>src/basic.html:1</source>translate me</msg> |   <msg id="8136548302122759730" desc="desc" meaning="meaning"><source>src/basic.html:1</source><source>src/comp2.ts:1</source><source>src/basic.html:1</source>translate me</msg> | ||||||
|   <msg id="9038505069473852515"><source>src/basic.html:3,4</source><source>src/comp2.ts:3,4</source><source>src/comp2.ts:2,3</source><source>src/basic.html:3,4</source> |   <msg id="9038505069473852515"><source>src/basic.html:3,4</source><source>src/comp2.ts:3,4</source><source>src/comp2.ts:2,3</source><source>src/basic.html:3,4</source> | ||||||
|     Welcome</msg> |     Welcome</msg> | ||||||
|  |   <msg id="5611534349548281834" desc="with ICU"><source>src/icu.html:1,3</source><source>src/icu.html:5</source>{VAR_PLURAL, plural, =1 {book} other {books} }</msg> | ||||||
|  |   <msg id="5811701742971715242" desc="with ICU and other things"><source>src/icu.html:4,6</source> | ||||||
|  |      foo <ph name="ICU"><ex>{ count, plural, =1 {...} other {...}}</ex>{ count, plural, =1 {...} other {...}}</ph> | ||||||
|  |     </msg> | ||||||
|  |   <msg id="7254052530614200029" desc="with placeholders"><source>src/placeholders.html:1</source>Name: <ph name="START_BOLD_TEXT"><ex><b></ex><b></ph><ph name="NAME"><ex>{{ | ||||||
|  |       name // i18n(ph="name")
 | ||||||
|  |     }}</ex>{{ | ||||||
|  |       name // i18n(ph="name")
 | ||||||
|  |     }}</ph><ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph></msg> | ||||||
| </messagebundle> | </messagebundle> | ||||||
| `;
 | `;
 | ||||||
| 
 | 
 | ||||||
| @ -90,6 +99,41 @@ const EXPECTED_XLIFF = `<?xml version="1.0" encoding="UTF-8" ?> | |||||||
|           <context context-type="linenumber">3</context> |           <context context-type="linenumber">3</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|  |       <trans-unit id="83937c05b1216e7f4c02a85454260e28fd72d1e3" datatype="html"> | ||||||
|  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | ||||||
|  |         <context-group purpose="location"> | ||||||
|  |           <context context-type="sourcefile">src/icu.html</context> | ||||||
|  |           <context context-type="linenumber">1</context> | ||||||
|  |         </context-group> | ||||||
|  |         <note priority="1" from="description">with ICU</note> | ||||||
|  |       </trans-unit> | ||||||
|  |       <trans-unit id="540c5f481129419ef21017f396b6c2d0869ca4d2" datatype="html"> | ||||||
|  |         <source> | ||||||
|  |      foo <x id="ICU" equiv-text="{ count, plural, =1 {...} other {...}}"/> | ||||||
|  |     </source> | ||||||
|  |         <context-group purpose="location"> | ||||||
|  |           <context context-type="sourcefile">src/icu.html</context> | ||||||
|  |           <context context-type="linenumber">4</context> | ||||||
|  |         </context-group> | ||||||
|  |         <note priority="1" from="description">with ICU and other things</note> | ||||||
|  |       </trans-unit> | ||||||
|  |       <trans-unit id="ca7678090fddd04441d63b1218177af65f23342d" datatype="html"> | ||||||
|  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | ||||||
|  |         <context-group purpose="location"> | ||||||
|  |           <context context-type="sourcefile">src/icu.html</context> | ||||||
|  |           <context context-type="linenumber">5</context> | ||||||
|  |         </context-group> | ||||||
|  |       </trans-unit> | ||||||
|  |       <trans-unit id="9311399c1ca7c75f771d77acb129e50581c6ec1f" datatype="html"> | ||||||
|  |         <source>Name: <x id="START_BOLD_TEXT" ctype="x-b" equiv-text="<b>"/><x id="NAME" equiv-text="{{ | ||||||
|  |       name // i18n(ph="name")
 | ||||||
|  |     }}"/><x id="CLOSE_BOLD_TEXT" ctype="x-b" equiv-text="</b>"/></source> | ||||||
|  |         <context-group purpose="location"> | ||||||
|  |           <context context-type="sourcefile">src/placeholders.html</context> | ||||||
|  |           <context context-type="linenumber">1</context> | ||||||
|  |         </context-group> | ||||||
|  |         <note priority="1" from="description">with placeholders</note> | ||||||
|  |       </trans-unit> | ||||||
|     </body> |     </body> | ||||||
|   </file> |   </file> | ||||||
| </xliff> | </xliff> | ||||||
| @ -122,6 +166,38 @@ const EXPECTED_XLIFF2 = `<?xml version="1.0" encoding="UTF-8" ?> | |||||||
|     Welcome</source> |     Welcome</source> | ||||||
|       </segment> |       </segment> | ||||||
|     </unit> |     </unit> | ||||||
|  |     <unit id="5611534349548281834"> | ||||||
|  |       <notes> | ||||||
|  |         <note category="description">with ICU</note> | ||||||
|  |         <note category="location">src/icu.html:1,3</note> | ||||||
|  |         <note category="location">src/icu.html:5</note> | ||||||
|  |       </notes> | ||||||
|  |       <segment> | ||||||
|  |         <source>{VAR_PLURAL, plural, =1 {book} other {books} }</source> | ||||||
|  |       </segment> | ||||||
|  |     </unit> | ||||||
|  |     <unit id="5811701742971715242"> | ||||||
|  |       <notes> | ||||||
|  |         <note category="description">with ICU and other things</note> | ||||||
|  |         <note category="location">src/icu.html:4,6</note> | ||||||
|  |       </notes> | ||||||
|  |       <segment> | ||||||
|  |         <source> | ||||||
|  |      foo <ph id="0" equiv="ICU" disp="{ count, plural, =1 {...} other {...}}"/> | ||||||
|  |     </source> | ||||||
|  |       </segment> | ||||||
|  |     </unit> | ||||||
|  |     <unit id="7254052530614200029"> | ||||||
|  |       <notes> | ||||||
|  |         <note category="description">with placeholders</note> | ||||||
|  |         <note category="location">src/placeholders.html:1</note> | ||||||
|  |       </notes> | ||||||
|  |       <segment> | ||||||
|  |         <source>Name: <pc id="0" equivStart="START_BOLD_TEXT" equivEnd="CLOSE_BOLD_TEXT" type="fmt" dispStart="<b>" dispEnd="</b>"><ph id="1" equiv="NAME" disp="{{ | ||||||
|  |       name // i18n(ph="name")
 | ||||||
|  |     }}"/></pc></source> | ||||||
|  |       </segment> | ||||||
|  |     </unit> | ||||||
|   </file> |   </file> | ||||||
| </xliff> | </xliff> | ||||||
| `;
 | `;
 | ||||||
| @ -132,7 +208,7 @@ describe('extract_i18n command line', () => { | |||||||
|   let write: (fileName: string, content: string) => void; |   let write: (fileName: string, content: string) => void; | ||||||
|   let errorSpy: jasmine.Spy&((s: string) => void); |   let errorSpy: jasmine.Spy&((s: string) => void); | ||||||
| 
 | 
 | ||||||
|   function writeConfig(tsconfig: string = '{"extends": "./tsconfig-base.json"}') { |   function writeConfig(tsconfig = '{"extends": "./tsconfig-base.json"}') { | ||||||
|     write('tsconfig.json', tsconfig); |     write('tsconfig.json', tsconfig); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -147,7 +223,7 @@ describe('extract_i18n command line', () => { | |||||||
|       basePath = makeTempDir(); |       basePath = makeTempDir(); | ||||||
|       write = (fileName: string, content: string) => { |       write = (fileName: string, content: string) => { | ||||||
|         const dir = path.dirname(fileName); |         const dir = path.dirname(fileName); | ||||||
|         if (dir != '.') { |         if (dir !== '.') { | ||||||
|           const newDir = path.join(basePath, dir); |           const newDir = path.join(basePath, dir); | ||||||
|           if (!fs.existsSync(newDir)) fs.mkdirSync(newDir); |           if (!fs.existsSync(newDir)) fs.mkdirSync(newDir); | ||||||
|         } |         } | ||||||
| @ -223,14 +299,54 @@ describe('extract_i18n command line', () => { | |||||||
|     }) |     }) | ||||||
|     export class BasicCmp3 {}`);
 |     export class BasicCmp3 {}`);
 | ||||||
| 
 | 
 | ||||||
|  |     write('src/placeholders.html', `<div i18n="with placeholders">Name: <b>{{
 | ||||||
|  |       name // i18n(ph="name")
 | ||||||
|  |     }}</b></div>`);
 | ||||||
|  | 
 | ||||||
|  |     write('src/placeholder_cmp.ts', ` | ||||||
|  |     import {Component} from '@angular/core'; | ||||||
|  | 
 | ||||||
|  |     @Component({ | ||||||
|  |       selector: 'placeholders', | ||||||
|  |       templateUrl: './placeholders.html', | ||||||
|  |     }) | ||||||
|  |     export class PlaceholderCmp { name = 'whatever'; }`);
 | ||||||
|  | 
 | ||||||
|  |     write('src/icu.html', `<div i18n="with ICU">{
 | ||||||
|  |       count, plural, =1 {book} other {books} | ||||||
|  |     }</div> | ||||||
|  |     <div i18n="with ICU and other things"> | ||||||
|  |      foo { count, plural, =1 {book} other {books} } | ||||||
|  |     </div>`);
 | ||||||
|  | 
 | ||||||
|  |     write('src/icu_cmp.ts', ` | ||||||
|  |     import {Component} from '@angular/core'; | ||||||
|  | 
 | ||||||
|  |     @Component({ | ||||||
|  |       selector: 'icu', | ||||||
|  |       templateUrl: './icu.html', | ||||||
|  |     }) | ||||||
|  |     export class IcuCmp { count = 3; }`);
 | ||||||
|  | 
 | ||||||
|     write('src/module.ts', ` |     write('src/module.ts', ` | ||||||
|     import {NgModule} from '@angular/core'; |     import {NgModule} from '@angular/core'; | ||||||
|  |     import {CommonModule} from '@angular/common'; | ||||||
|     import {BasicCmp1} from './comp1'; |     import {BasicCmp1} from './comp1'; | ||||||
|     import {BasicCmp2, BasicCmp4} from './comp2'; |     import {BasicCmp2, BasicCmp4} from './comp2'; | ||||||
|     import {BasicCmp3} from './comp3'; |     import {BasicCmp3} from './comp3'; | ||||||
|  |     import {PlaceholderCmp} from './placeholder_cmp'; | ||||||
|  |     import {IcuCmp} from './icu_cmp'; | ||||||
| 
 | 
 | ||||||
|     @NgModule({ |     @NgModule({ | ||||||
|       declarations: [BasicCmp1, BasicCmp2, BasicCmp3, BasicCmp4] |       declarations: [ | ||||||
|  |         BasicCmp1, | ||||||
|  |         BasicCmp2, | ||||||
|  |         BasicCmp3, | ||||||
|  |         BasicCmp4, | ||||||
|  |         PlaceholderCmp, | ||||||
|  |         IcuCmp, | ||||||
|  |       ], | ||||||
|  |       imports: [CommonModule], | ||||||
|     }) |     }) | ||||||
|     export class I18nModule {} |     export class I18nModule {} | ||||||
|     `);
 |     `);
 | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ import * as xml from './xml_helper'; | |||||||
| const _MESSAGES_TAG = 'messagebundle'; | const _MESSAGES_TAG = 'messagebundle'; | ||||||
| const _MESSAGE_TAG = 'msg'; | const _MESSAGE_TAG = 'msg'; | ||||||
| const _PLACEHOLDER_TAG = 'ph'; | const _PLACEHOLDER_TAG = 'ph'; | ||||||
| const _EXEMPLE_TAG = 'ex'; | const _EXAMPLE_TAG = 'ex'; | ||||||
| const _SOURCE_TAG = 'source'; | const _SOURCE_TAG = 'source'; | ||||||
| 
 | 
 | ||||||
| const _DOCTYPE = `<!ELEMENT messagebundle (msg)*>
 | const _DOCTYPE = `<!ELEMENT messagebundle (msg)*>
 | ||||||
| @ -115,30 +115,45 @@ class _Visitor implements i18n.Visitor { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   visitTagPlaceholder(ph: i18n.TagPlaceholder, context?: any): xml.Node[] { |   visitTagPlaceholder(ph: i18n.TagPlaceholder, context?: any): xml.Node[] { | ||||||
|     const startEx = new xml.Tag(_EXEMPLE_TAG, {}, [new xml.Text(`<${ph.tag}>`)]); |     const startTagAsText = new xml.Text(`<${ph.tag}>`); | ||||||
|     const startTagPh = new xml.Tag(_PLACEHOLDER_TAG, {name: ph.startName}, [startEx]); |     const startEx = new xml.Tag(_EXAMPLE_TAG, {}, [startTagAsText]); | ||||||
|  |     // TC requires PH to have a non empty EX, and uses the text node to show the "original" value.
 | ||||||
|  |     const startTagPh = | ||||||
|  |         new xml.Tag(_PLACEHOLDER_TAG, {name: ph.startName}, [startEx, startTagAsText]); | ||||||
|     if (ph.isVoid) { |     if (ph.isVoid) { | ||||||
|       // void tags have no children nor closing tags
 |       // void tags have no children nor closing tags
 | ||||||
|       return [startTagPh]; |       return [startTagPh]; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const closeEx = new xml.Tag(_EXEMPLE_TAG, {}, [new xml.Text(`</${ph.tag}>`)]); |     const closeTagAsText = new xml.Text(`</${ph.tag}>`); | ||||||
|     const closeTagPh = new xml.Tag(_PLACEHOLDER_TAG, {name: ph.closeName}, [closeEx]); |     const closeEx = new xml.Tag(_EXAMPLE_TAG, {}, [closeTagAsText]); | ||||||
|  |     // TC requires PH to have a non empty EX, and uses the text node to show the "original" value.
 | ||||||
|  |     const closeTagPh = | ||||||
|  |         new xml.Tag(_PLACEHOLDER_TAG, {name: ph.closeName}, [closeEx, closeTagAsText]); | ||||||
| 
 | 
 | ||||||
|     return [startTagPh, ...this.serialize(ph.children), closeTagPh]; |     return [startTagPh, ...this.serialize(ph.children), closeTagPh]; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   visitPlaceholder(ph: i18n.Placeholder, context?: any): xml.Node[] { |   visitPlaceholder(ph: i18n.Placeholder, context?: any): xml.Node[] { | ||||||
|     const exTag = new xml.Tag(_EXEMPLE_TAG, {}, [new xml.Text(`{{${ph.value}}}`)]); |     const interpolationAsText = new xml.Text(`{{${ph.value}}}`); | ||||||
|     return [new xml.Tag(_PLACEHOLDER_TAG, {name: ph.name}, [exTag])]; |     // Example tag needs to be not-empty for TC.
 | ||||||
|  |     const exTag = new xml.Tag(_EXAMPLE_TAG, {}, [interpolationAsText]); | ||||||
|  |     return [ | ||||||
|  |       // TC requires PH to have a non empty EX, and uses the text node to show the "original" value.
 | ||||||
|  |       new xml.Tag(_PLACEHOLDER_TAG, {name: ph.name}, [exTag, interpolationAsText]) | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   visitIcuPlaceholder(ph: i18n.IcuPlaceholder, context?: any): xml.Node[] { |   visitIcuPlaceholder(ph: i18n.IcuPlaceholder, context?: any): xml.Node[] { | ||||||
|     const exTag = new xml.Tag(_EXEMPLE_TAG, {}, [ |     const icuExpression = ph.value.expression; | ||||||
|       new xml.Text( |     const icuType = ph.value.type; | ||||||
|           `{${ph.value.expression}, ${ph.value.type}, ${Object.keys(ph.value.cases).map((value: string) => value + ' {...}').join(' ')}}`) |     const icuCases = Object.keys(ph.value.cases).map((value: string) => value + ' {...}').join(' '); | ||||||
|     ]); |     const icuAsText = new xml.Text(`{${icuExpression}, ${icuType}, ${icuCases}}`); | ||||||
|     return [new xml.Tag(_PLACEHOLDER_TAG, {name: ph.name}, [exTag])]; |     const exTag = new xml.Tag(_EXAMPLE_TAG, {}, [icuAsText]); | ||||||
|  |     return [ | ||||||
|  |       // TC requires PH to have a non empty EX, and uses the text node to show the "original" value.
 | ||||||
|  |       new xml.Tag(_PLACEHOLDER_TAG, {name: ph.name}, [exTag, icuAsText]) | ||||||
|  |     ]; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   serialize(nodes: i18n.Node[]): xml.Node[] { |   serialize(nodes: i18n.Node[]): xml.Node[] { | ||||||
| @ -161,7 +176,7 @@ class ExampleVisitor implements xml.IVisitor { | |||||||
|     if (tag.name === _PLACEHOLDER_TAG) { |     if (tag.name === _PLACEHOLDER_TAG) { | ||||||
|       if (!tag.children || tag.children.length == 0) { |       if (!tag.children || tag.children.length == 0) { | ||||||
|         const exText = new xml.Text(tag.attrs['name'] || '...'); |         const exText = new xml.Text(tag.attrs['name'] || '...'); | ||||||
|         tag.children = [new xml.Tag(_EXEMPLE_TAG, {}, [exText])]; |         tag.children = [new xml.Tag(_EXAMPLE_TAG, {}, [exText])]; | ||||||
|       } |       } | ||||||
|     } else if (tag.children) { |     } else if (tag.children) { | ||||||
|       tag.children.forEach(node => node.visit(this)); |       tag.children.forEach(node => node.visit(this)); | ||||||
|  | |||||||
| @ -91,32 +91,32 @@ const XTB = ` | |||||||
| const XMB = `<msg id="615790887472569365"><source>file.ts:3</source>i18n attribute on tags</msg>
 | const XMB = `<msg id="615790887472569365"><source>file.ts:3</source>i18n attribute on tags</msg>
 | ||||||
|   <msg id="3707494640264351337"><source>file.ts:5</source>nested</msg> |   <msg id="3707494640264351337"><source>file.ts:5</source>nested</msg> | ||||||
|   <msg id="5539162898278769904" meaning="different meaning"><source>file.ts:7</source>nested</msg> |   <msg id="5539162898278769904" meaning="different meaning"><source>file.ts:7</source>nested</msg> | ||||||
|   <msg id="3780349238193953556"><source>file.ts:9</source><source>file.ts:10</source><ph name="START_ITALIC_TEXT"><ex><i></ex></ph>with placeholders<ph name="CLOSE_ITALIC_TEXT"><ex></i></ex></ph></msg> |   <msg id="3780349238193953556"><source>file.ts:9</source><source>file.ts:10</source><ph name="START_ITALIC_TEXT"><ex><i></ex><i></ph>with placeholders<ph name="CLOSE_ITALIC_TEXT"><ex></i></ex></i></ph></msg> | ||||||
|   <msg id="5415448997399451992"><source>file.ts:11</source><ph name="START_TAG_DIV"><ex><div></ex></ph>with <ph name="START_TAG_DIV"><ex><div></ex></ph>nested<ph name="CLOSE_TAG_DIV"><ex></div></ex></ph> placeholders<ph name="CLOSE_TAG_DIV"><ex></div></ex></ph></msg> |   <msg id="5415448997399451992"><source>file.ts:11</source><ph name="START_TAG_DIV"><ex><div></ex><div></ph>with <ph name="START_TAG_DIV"><ex><div></ex><div></ph>nested<ph name="CLOSE_TAG_DIV"><ex></div></ex></div></ph> placeholders<ph name="CLOSE_TAG_DIV"><ex></div></ex></div></ph></msg> | ||||||
|   <msg id="5525133077318024839"><source>file.ts:14</source>on not translatable node</msg> |   <msg id="5525133077318024839"><source>file.ts:14</source>on not translatable node</msg> | ||||||
|   <msg id="2174788525135228764"><source>file.ts:14</source><b>bold</b></msg> |   <msg id="2174788525135228764"><source>file.ts:14</source><b>bold</b></msg> | ||||||
|   <msg id="8670732454866344690"><source>file.ts:15</source>on translatable node</msg> |   <msg id="8670732454866344690"><source>file.ts:15</source>on translatable node</msg> | ||||||
|   <msg id="4593805537723189714"><source>file.ts:20</source><source>file.ts:37</source>{VAR_PLURAL, plural, =0 {zero} =1 {one} =2 {two} other {<ph name="START_BOLD_TEXT"><ex><b></ex></ph>many<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph>} }</msg> |   <msg id="4593805537723189714"><source>file.ts:20</source><source>file.ts:37</source>{VAR_PLURAL, plural, =0 {zero} =1 {one} =2 {two} other {<ph name="START_BOLD_TEXT"><ex><b></ex><b></ph>many<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph>} }</msg> | ||||||
|   <msg id="703464324060964421"><source>file.ts:22,24</source> |   <msg id="703464324060964421"><source>file.ts:22,24</source> | ||||||
|         <ph name="ICU"><ex>{sex, select, male {...} female {...} other {...}}</ex></ph> |         <ph name="ICU"><ex>{sex, select, male {...} female {...} other {...}}</ex>{sex, select, male {...} female {...} other {...}}</ph> | ||||||
|     </msg> |     </msg> | ||||||
|   <msg id="5430374139308914421"><source>file.ts:23</source>{VAR_SELECT, select, male {m} female {f} other {other} }</msg> |   <msg id="5430374139308914421"><source>file.ts:23</source>{VAR_SELECT, select, male {m} female {f} other {other} }</msg> | ||||||
|   <msg id="1300564767229037107"><source>file.ts:25,27</source> |   <msg id="1300564767229037107"><source>file.ts:25,27</source> | ||||||
|         <ph name="ICU"><ex>{sexB, select, male {...} female {...}}</ex></ph> |         <ph name="ICU"><ex>{sexB, select, male {...} female {...}}</ex>{sexB, select, male {...} female {...}}</ph> | ||||||
|     </msg> |     </msg> | ||||||
|   <msg id="2500580913783245106"><source>file.ts:26</source>{VAR_SELECT, select, male {m} female {f} }</msg> |   <msg id="2500580913783245106"><source>file.ts:26</source>{VAR_SELECT, select, male {m} female {f} }</msg> | ||||||
|   <msg id="4851788426695310455"><source>file.ts:29</source><ph name="INTERPOLATION"><ex>{{ "count = " + count }}</ex></ph></msg> |   <msg id="4851788426695310455"><source>file.ts:29</source><ph name="INTERPOLATION"><ex>{{ "count = " + count }}</ex>{{ "count = " + count }}</ph></msg> | ||||||
|   <msg id="9013357158046221374"><source>file.ts:30</source>sex = <ph name="INTERPOLATION"><ex>{{ sex }}</ex></ph></msg> |   <msg id="9013357158046221374"><source>file.ts:30</source>sex = <ph name="INTERPOLATION"><ex>{{ sex }}</ex>{{ sex }}</ph></msg> | ||||||
|   <msg id="8324617391167353662"><source>file.ts:31</source><ph name="CUSTOM_NAME"><ex>{{ "custom name" //i18n(ph="CUSTOM_NAME") }}</ex></ph></msg>
 |   <msg id="8324617391167353662"><source>file.ts:31</source><ph name="CUSTOM_NAME"><ex>{{ "custom name" //i18n(ph="CUSTOM_NAME") }}</ex>{{ "custom name" //i18n(ph="CUSTOM_NAME") }}</ph></msg>
 | ||||||
|   <msg id="7685649297917455806"><source>file.ts:36</source><source>file.ts:54</source>in a translatable section</msg> |   <msg id="7685649297917455806"><source>file.ts:36</source><source>file.ts:54</source>in a translatable section</msg> | ||||||
|   <msg id="2329001734457059408"><source>file.ts:34,38</source> |   <msg id="2329001734457059408"><source>file.ts:34,38</source> | ||||||
|     <ph name="START_HEADING_LEVEL1"><ex><h1></ex></ph>Markers in html comments<ph name="CLOSE_HEADING_LEVEL1"><ex></h1></ex></ph> |     <ph name="START_HEADING_LEVEL1"><ex><h1></ex><h1></ph>Markers in html comments<ph name="CLOSE_HEADING_LEVEL1"><ex></h1></ex></h1></ph> | ||||||
|     <ph name="START_TAG_DIV"><ex><div></ex></ph><ph name="CLOSE_TAG_DIV"><ex></div></ex></ph> |     <ph name="START_TAG_DIV"><ex><div></ex><div></ph><ph name="CLOSE_TAG_DIV"><ex></div></ex></div></ph> | ||||||
|     <ph name="START_TAG_DIV_1"><ex><div></ex></ph><ph name="ICU"><ex>{count, plural, =0 {...} =1 {...} =2 {...} other {...}}</ex></ph><ph name="CLOSE_TAG_DIV"><ex></div></ex></ph> |     <ph name="START_TAG_DIV_1"><ex><div></ex><div></ph><ph name="ICU"><ex>{count, plural, =0 {...} =1 {...} =2 {...} other {...}}</ex>{count, plural, =0 {...} =1 {...} =2 {...} other {...}}</ph><ph name="CLOSE_TAG_DIV"><ex></div></ex></div></ph> | ||||||
| </msg> | </msg> | ||||||
|   <msg id="1491627405349178954"><source>file.ts:40</source>it <ph name="START_BOLD_TEXT"><ex><b></ex></ph>should<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph> work</msg> |   <msg id="1491627405349178954"><source>file.ts:40</source>it <ph name="START_BOLD_TEXT"><ex><b></ex><b></ph>should<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph> work</msg> | ||||||
|   <msg id="i18n16"><source>file.ts:42</source>with an explicit ID</msg> |   <msg id="i18n16"><source>file.ts:42</source>with an explicit ID</msg> | ||||||
|   <msg id="i18n17"><source>file.ts:43</source>{VAR_PLURAL, plural, =0 {zero} =1 {one} =2 {two} other {<ph name="START_BOLD_TEXT"><ex><b></ex></ph>many<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph>} }</msg> |   <msg id="i18n17"><source>file.ts:43</source>{VAR_PLURAL, plural, =0 {zero} =1 {one} =2 {two} other {<ph name="START_BOLD_TEXT"><ex><b></ex><b></ph>many<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph>} }</msg> | ||||||
|   <msg id="4085484936881858615" desc="desc"><source>file.ts:46,52</source>{VAR_PLURAL, plural, =0 {Found no results} =1 {Found one result} other {Found <ph name="INTERPOLATION"><ex>{{response.getItemsList().length}}</ex></ph> results} }</msg> |   <msg id="4085484936881858615" desc="desc"><source>file.ts:46,52</source>{VAR_PLURAL, plural, =0 {Found no results} =1 {Found one result} other {Found <ph name="INTERPOLATION"><ex>{{response.getItemsList().length}}</ex>{{response.getItemsList().length}}</ph> results} }</msg> | ||||||
|   <msg id="4035252431381981115"><source>file.ts:54</source>foo<ph name="START_LINK"><ex><a></ex></ph>bar<ph name="CLOSE_LINK"><ex></a></ex></ph></msg> |   <msg id="4035252431381981115"><source>file.ts:54</source>foo<ph name="START_LINK"><ex><a></ex><a></ph>bar<ph name="CLOSE_LINK"><ex></a></ex></a></ph></msg> | ||||||
|   <msg id="5339604010413301604"><source>file.ts:56</source><ph name="MAP_NAME"><ex>{{ 'test' //i18n(ph="map name") }}</ex></ph></msg>`;
 |   <msg id="5339604010413301604"><source>file.ts:56</source><ph name="MAP_NAME"><ex>{{ 'test' //i18n(ph="map name") }}</ex>{{ 'test' //i18n(ph="map name") }}</ph></msg>`;
 | ||||||
|  | |||||||
| @ -48,14 +48,14 @@ lines</p>`; | |||||||
| <!ELEMENT ex (#PCDATA)> | <!ELEMENT ex (#PCDATA)> | ||||||
| ]> | ]> | ||||||
| <messagebundle> | <messagebundle> | ||||||
|   <msg id="7056919470098446707"><source>file.ts:3</source>translatable element <ph name="START_BOLD_TEXT"><ex><b></ex></ph>with placeholders<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph> <ph name="INTERPOLATION"><ex>{{ interpolation}}</ex></ph></msg> |   <msg id="7056919470098446707"><source>file.ts:3</source>translatable element <ph name="START_BOLD_TEXT"><ex><b></ex><b></ph>with placeholders<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></b></ph> <ph name="INTERPOLATION"><ex>{{ interpolation}}</ex>{{ interpolation}}</ph></msg> | ||||||
|   <msg id="2981514368455622387"><source>file.ts:4</source>{VAR_PLURAL, plural, =0 {<ph name="START_PARAGRAPH"><ex><p></ex></ph>test<ph name="CLOSE_PARAGRAPH"><ex></p></ex></ph>} }</msg> |   <msg id="2981514368455622387"><source>file.ts:4</source>{VAR_PLURAL, plural, =0 {<ph name="START_PARAGRAPH"><ex><p></ex><p></ph>test<ph name="CLOSE_PARAGRAPH"><ex></p></ex></p></ph>} }</msg> | ||||||
|   <msg id="7999024498831672133" desc="d" meaning="m"><source>file.ts:5</source>foo</msg> |   <msg id="7999024498831672133" desc="d" meaning="m"><source>file.ts:5</source>foo</msg> | ||||||
|   <msg id="i" desc="d" meaning="m"><source>file.ts:6</source>foo</msg> |   <msg id="i" desc="d" meaning="m"><source>file.ts:6</source>foo</msg> | ||||||
|   <msg id="bar"><source>file.ts:7</source>foo</msg> |   <msg id="bar"><source>file.ts:7</source>foo</msg> | ||||||
|   <msg id="baz"><source>file.ts:8</source>{VAR_PLURAL, plural, =0 {{VAR_SELECT, select, other {<ph name="START_PARAGRAPH"><ex><p></ex></ph>deeply nested<ph name="CLOSE_PARAGRAPH"><ex></p></ex></ph>} } } }</msg> |   <msg id="baz"><source>file.ts:8</source>{VAR_PLURAL, plural, =0 {{VAR_SELECT, select, other {<ph name="START_PARAGRAPH"><ex><p></ex><p></ph>deeply nested<ph name="CLOSE_PARAGRAPH"><ex></p></ex></p></ph>} } } }</msg> | ||||||
|   <msg id="6997386649824869937"><source>file.ts:9</source>Test: <ph name="ICU"><ex>{ count, plural, =0 {...} =other {...}}</ex></ph></msg> |   <msg id="6997386649824869937"><source>file.ts:9</source>Test: <ph name="ICU"><ex>{ count, plural, =0 {...} =other {...}}</ex>{ count, plural, =0 {...} =other {...}}</ph></msg> | ||||||
|   <msg id="5229984852258993423"><source>file.ts:9</source>{VAR_PLURAL, plural, =0 {{VAR_SELECT, select, other {<ph name="START_PARAGRAPH"><ex><p></ex></ph>deeply nested<ph name="CLOSE_PARAGRAPH"><ex></p></ex></ph>} } } =other {a lot} }</msg> |   <msg id="5229984852258993423"><source>file.ts:9</source>{VAR_PLURAL, plural, =0 {{VAR_SELECT, select, other {<ph name="START_PARAGRAPH"><ex><p></ex><p></ph>deeply nested<ph name="CLOSE_PARAGRAPH"><ex></p></ex></p></ph>} } } =other {a lot} }</msg> | ||||||
|   <msg id="2340165783990709777"><source>file.ts:10,11</source>multi |   <msg id="2340165783990709777"><source>file.ts:10,11</source>multi | ||||||
| lines</msg> | lines</msg> | ||||||
| </messagebundle> | </messagebundle> | ||||||
|  | |||||||
| @ -1737,7 +1737,7 @@ | |||||||
|     "name": "_ESCAPED_CHARS" |     "name": "_ESCAPED_CHARS" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "name": "_EXEMPLE_TAG" |     "name": "_EXAMPLE_TAG" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     "name": "_EmittedLine" |     "name": "_EmittedLine" | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user