parent
562f7a2f8b
commit
56c361ff6a
|
@ -40,7 +40,8 @@ class _I18nVisitor implements html.Visitor {
|
||||||
private _expressionParser: ExpressionParser,
|
private _expressionParser: ExpressionParser,
|
||||||
private _interpolationConfig: InterpolationConfig) {}
|
private _interpolationConfig: InterpolationConfig) {}
|
||||||
|
|
||||||
public toI18nMessage(nodes: html.Node[], meaning: string, description: string, id: string): i18n.Message {
|
public toI18nMessage(nodes: html.Node[], meaning: string, description: string, id: string):
|
||||||
|
i18n.Message {
|
||||||
this._isIcu = nodes.length == 1 && nodes[0] instanceof html.Expansion;
|
this._isIcu = nodes.length == 1 && nodes[0] instanceof html.Expansion;
|
||||||
this._icuDepth = 0;
|
this._icuDepth = 0;
|
||||||
this._placeholderRegistry = new PlaceholderRegistry();
|
this._placeholderRegistry = new PlaceholderRegistry();
|
||||||
|
|
|
@ -6,10 +6,23 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {computeMsgId, sha1} from '../../src/i18n/digest';
|
import {computeMsgId, digest, sha1} from '../../src/i18n/digest';
|
||||||
|
|
||||||
export function main(): void {
|
export function main(): void {
|
||||||
describe('digest', () => {
|
describe('digest', () => {
|
||||||
|
describe('digest', () => {
|
||||||
|
it('must return the ID if it\'s explicit', () => {
|
||||||
|
expect(digest({
|
||||||
|
id: 'i',
|
||||||
|
nodes: [],
|
||||||
|
placeholders: {},
|
||||||
|
placeholderToMessage: {},
|
||||||
|
meaning: '',
|
||||||
|
description: '',
|
||||||
|
})).toEqual('i');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('sha1', () => {
|
describe('sha1', () => {
|
||||||
it('should work on empty strings',
|
it('should work on empty strings',
|
||||||
() => { expect(sha1('')).toEqual('da39a3ee5e6b4b0d3255bfef95601890afd80709'); });
|
() => { expect(sha1('')).toEqual('da39a3ee5e6b4b0d3255bfef95601890afd80709'); });
|
||||||
|
|
|
@ -20,7 +20,10 @@ export function main() {
|
||||||
describe('elements', () => {
|
describe('elements', () => {
|
||||||
it('should extract from elements', () => {
|
it('should extract from elements', () => {
|
||||||
expect(extract('<div i18n="m|d|e">text<span>nested</span></div>')).toEqual([
|
expect(extract('<div i18n="m|d|e">text<span>nested</span></div>')).toEqual([
|
||||||
[['text', '<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], 'm', 'd|e'],
|
[
|
||||||
|
['text', '<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], 'm', 'd|e',
|
||||||
|
''
|
||||||
|
],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -29,11 +32,45 @@ export function main() {
|
||||||
extract(
|
extract(
|
||||||
'<div i18n="m1|d1"><span i18n-title="m2|d2" title="single child">nested</span></div>'))
|
'<div i18n="m1|d1"><span i18n-title="m2|d2" title="single child">nested</span></div>'))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], 'm1', 'd1'],
|
[['<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], 'm1', 'd1', ''],
|
||||||
[['single child'], 'm2', 'd2'],
|
[['single child'], 'm2', 'd2', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should extract from attributes with id', () => {
|
||||||
|
expect(
|
||||||
|
extract(
|
||||||
|
'<div i18n="m1|d1@@i1"><span i18n-title="m2|d2@@i2" title="single child">nested</span></div>'))
|
||||||
|
.toEqual([
|
||||||
|
[
|
||||||
|
['<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], 'm1', 'd1',
|
||||||
|
'i1'
|
||||||
|
],
|
||||||
|
[['single child'], 'm2', 'd2', 'i2'],
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should extract from attributes without meaning and with id', () => {
|
||||||
|
expect(
|
||||||
|
extract(
|
||||||
|
'<div i18n="d1@@i1"><span i18n-title="d2@@i2" title="single child">nested</span></div>'))
|
||||||
|
.toEqual([
|
||||||
|
[['<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], '', 'd1', 'i1'],
|
||||||
|
[['single child'], '', 'd2', 'i2'],
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should extract from attributes with id only', () => {
|
||||||
|
expect(
|
||||||
|
extract(
|
||||||
|
'<div i18n="@@i1"><span i18n-title="@@i2" title="single child">nested</span></div>'))
|
||||||
|
.toEqual([
|
||||||
|
[['<ph tag name="START_TAG_SPAN">nested</ph name="CLOSE_TAG_SPAN">'], '', '', 'i1'],
|
||||||
|
[['single child'], '', '', 'i2'],
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
it('should extract from ICU messages', () => {
|
it('should extract from ICU messages', () => {
|
||||||
expect(
|
expect(
|
||||||
extract(
|
extract(
|
||||||
|
@ -43,10 +80,10 @@ export function main() {
|
||||||
[
|
[
|
||||||
'{count, plural, =0 {[<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">]}}'
|
'{count, plural, =0 {[<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">]}}'
|
||||||
],
|
],
|
||||||
'm', 'd'
|
'm', 'd', ''
|
||||||
],
|
],
|
||||||
[['title'], '', ''],
|
[['title'], '', '', ''],
|
||||||
[['desc'], '', ''],
|
[['desc'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -55,7 +92,7 @@ export function main() {
|
||||||
|
|
||||||
it('should ignore implicit elements in translatable elements', () => {
|
it('should ignore implicit elements in translatable elements', () => {
|
||||||
expect(extract('<div i18n="m|d"><p></p></div>', ['p'])).toEqual([
|
expect(extract('<div i18n="m|d"><p></p></div>', ['p'])).toEqual([
|
||||||
[['<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">'], 'm', 'd']
|
[['<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">'], 'm', 'd', '']
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -64,17 +101,19 @@ export function main() {
|
||||||
it('should extract from blocks', () => {
|
it('should extract from blocks', () => {
|
||||||
expect(extract(`<!-- i18n: meaning1|desc1 -->message1<!-- /i18n -->
|
expect(extract(`<!-- i18n: meaning1|desc1 -->message1<!-- /i18n -->
|
||||||
<!-- i18n: desc2 -->message2<!-- /i18n -->
|
<!-- i18n: desc2 -->message2<!-- /i18n -->
|
||||||
<!-- i18n -->message3<!-- /i18n -->`))
|
<!-- i18n -->message3<!-- /i18n -->
|
||||||
|
<!-- i18n: meaning4|desc4@@id4 -->message4<!-- /i18n -->
|
||||||
|
<!-- i18n: @@id5 -->message5<!-- /i18n -->`))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['message1'], 'meaning1', 'desc1'],
|
[['message1'], 'meaning1', 'desc1', ''], [['message2'], '', 'desc2', ''],
|
||||||
[['message2'], '', 'desc2'],
|
[['message3'], '', '', ''], [['message4'], 'meaning4', 'desc4', 'id4'],
|
||||||
[['message3'], '', ''],
|
[['message5'], '', '', 'id5']
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should ignore implicit elements in blocks', () => {
|
it('should ignore implicit elements in blocks', () => {
|
||||||
expect(extract('<!-- i18n:m|d --><p></p><!-- /i18n -->', ['p'])).toEqual([
|
expect(extract('<!-- i18n:m|d --><p></p><!-- /i18n -->', ['p'])).toEqual([
|
||||||
[['<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">'], 'm', 'd']
|
[['<ph tag name="START_PARAGRAPH"></ph name="CLOSE_PARAGRAPH">'], 'm', 'd', '']
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -88,7 +127,7 @@ export function main() {
|
||||||
[
|
[
|
||||||
'{count, plural, =0 {[<ph tag name="START_TAG_SPAN">html</ph name="CLOSE_TAG_SPAN">]}}'
|
'{count, plural, =0 {[<ph tag name="START_TAG_SPAN">html</ph name="CLOSE_TAG_SPAN">]}}'
|
||||||
],
|
],
|
||||||
'', ''
|
'', '', ''
|
||||||
],
|
],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
|
@ -98,15 +137,15 @@ export function main() {
|
||||||
' name="START_TAG_SPAN">html</ph name="CLOSE_TAG_SPAN">]}}</ph>',
|
' name="START_TAG_SPAN">html</ph name="CLOSE_TAG_SPAN">]}}</ph>',
|
||||||
'[<ph name="INTERPOLATION">interp</ph>]'
|
'[<ph name="INTERPOLATION">interp</ph>]'
|
||||||
],
|
],
|
||||||
'', ''
|
'', '', ''
|
||||||
],
|
],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should ignore other comments', () => {
|
it('should ignore other comments', () => {
|
||||||
expect(extract(`<!-- i18n: meaning1|desc1 --><!-- other -->message1<!-- /i18n -->`))
|
expect(extract(`<!-- i18n: meaning1|desc1@@id1 --><!-- other -->message1<!-- /i18n -->`))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['message1'], 'meaning1', 'desc1'],
|
[['message1'], 'meaning1', 'desc1', 'id1'],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -118,34 +157,37 @@ export function main() {
|
||||||
it('should extract ICU messages from translatable elements', () => {
|
it('should extract ICU messages from translatable elements', () => {
|
||||||
// single message when ICU is the only children
|
// single message when ICU is the only children
|
||||||
expect(extract('<div i18n="m|d">{count, plural, =0 {text}}</div>')).toEqual([
|
expect(extract('<div i18n="m|d">{count, plural, =0 {text}}</div>')).toEqual([
|
||||||
[['{count, plural, =0 {[text]}}'], 'm', 'd'],
|
[['{count, plural, =0 {[text]}}'], 'm', 'd', ''],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// single message when ICU is the only (implicit) children
|
// single message when ICU is the only (implicit) children
|
||||||
expect(extract('<div>{count, plural, =0 {text}}</div>', ['div'])).toEqual([
|
expect(extract('<div>{count, plural, =0 {text}}</div>', ['div'])).toEqual([
|
||||||
[['{count, plural, =0 {[text]}}'], '', ''],
|
[['{count, plural, =0 {[text]}}'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// one message for the element content and one message for the ICU
|
// one message for the element content and one message for the ICU
|
||||||
expect(extract('<div i18n="m|d">before{count, plural, =0 {text}}after</div>')).toEqual([
|
expect(extract('<div i18n="m|d@@i">before{count, plural, =0 {text}}after</div>')).toEqual([
|
||||||
[['before', '<ph icu name="ICU">{count, plural, =0 {[text]}}</ph>', 'after'], 'm', 'd'],
|
[
|
||||||
[['{count, plural, =0 {[text]}}'], '', ''],
|
['before', '<ph icu name="ICU">{count, plural, =0 {[text]}}</ph>', 'after'], 'm', 'd',
|
||||||
|
'i'
|
||||||
|
],
|
||||||
|
[['{count, plural, =0 {[text]}}'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should extract ICU messages from translatable block', () => {
|
it('should extract ICU messages from translatable block', () => {
|
||||||
// single message when ICU is the only children
|
// single message when ICU is the only children
|
||||||
expect(extract('<!-- i18n:m|d -->{count, plural, =0 {text}}<!-- /i18n -->')).toEqual([
|
expect(extract('<!-- i18n:m|d -->{count, plural, =0 {text}}<!-- /i18n -->')).toEqual([
|
||||||
[['{count, plural, =0 {[text]}}'], 'm', 'd'],
|
[['{count, plural, =0 {[text]}}'], 'm', 'd', ''],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// one message for the block content and one message for the ICU
|
// one message for the block content and one message for the ICU
|
||||||
expect(extract('<!-- i18n:m|d -->before{count, plural, =0 {text}}after<!-- /i18n -->'))
|
expect(extract('<!-- i18n:m|d -->before{count, plural, =0 {text}}after<!-- /i18n -->'))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['{count, plural, =0 {[text]}}'], '', ''],
|
[['{count, plural, =0 {[text]}}'], '', '', ''],
|
||||||
[
|
[
|
||||||
['before', '<ph icu name="ICU">{count, plural, =0 {[text]}}</ph>', 'after'], 'm',
|
['before', '<ph icu name="ICU">{count, plural, =0 {[text]}}</ph>', 'after'], 'm',
|
||||||
'd'
|
'd', ''
|
||||||
],
|
],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
@ -156,20 +198,20 @@ export function main() {
|
||||||
it('should ignore nested ICU messages', () => {
|
it('should ignore nested ICU messages', () => {
|
||||||
expect(extract('<div i18n="m|d">{count, plural, =0 { {sex, select, male {m}} }}</div>'))
|
expect(extract('<div i18n="m|d">{count, plural, =0 { {sex, select, male {m}} }}</div>'))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['{count, plural, =0 {[{sex, select, male {[m]}}, ]}}'], 'm', 'd'],
|
[['{count, plural, =0 {[{sex, select, male {[m]}}, ]}}'], 'm', 'd', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should ignore implicit elements in non translatable ICU messages', () => {
|
it('should ignore implicit elements in non translatable ICU messages', () => {
|
||||||
expect(
|
expect(extract(
|
||||||
extract(
|
'<div i18n="m|d@@i">{count, plural, =0 { {sex, select, male {<p>ignore</p>}}' +
|
||||||
'<div i18n="m|d">{count, plural, =0 { {sex, select, male {<p>ignore</p>}} }}</div>',
|
' }}</div>',
|
||||||
['p']))
|
['p']))
|
||||||
.toEqual([[
|
.toEqual([[
|
||||||
[
|
[
|
||||||
'{count, plural, =0 {[{sex, select, male {[<ph tag name="START_PARAGRAPH">ignore</ph name="CLOSE_PARAGRAPH">]}}, ]}}'
|
'{count, plural, =0 {[{sex, select, male {[<ph tag name="START_PARAGRAPH">ignore</ph name="CLOSE_PARAGRAPH">]}}, ]}}'
|
||||||
],
|
],
|
||||||
'm', 'd'
|
'm', 'd', 'i'
|
||||||
]]);
|
]]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -181,46 +223,45 @@ export function main() {
|
||||||
|
|
||||||
describe('attributes', () => {
|
describe('attributes', () => {
|
||||||
it('should extract from attributes outside of translatable sections', () => {
|
it('should extract from attributes outside of translatable sections', () => {
|
||||||
expect(extract('<div i18n-title="m|d" title="msg"></div>')).toEqual([
|
expect(extract('<div i18n-title="m|d@@i" title="msg"></div>')).toEqual([
|
||||||
[['msg'], 'm', 'd'],
|
[['msg'], 'm', 'd', 'i'],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should extract from attributes in translatable elements', () => {
|
it('should extract from attributes in translatable elements', () => {
|
||||||
expect(extract('<div i18n><p><b i18n-title="m|d" title="msg"></b></p></div>')).toEqual([
|
expect(extract('<div i18n><p><b i18n-title="m|d@@i" title="msg"></b></p></div>')).toEqual([
|
||||||
[
|
[
|
||||||
['<ph tag name="START_PARAGRAPH"><ph tag name="START_BOLD_TEXT"></ph' +
|
['<ph tag name="START_PARAGRAPH"><ph tag name="START_BOLD_TEXT"></ph' +
|
||||||
' name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">'],
|
' name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">'],
|
||||||
'', ''
|
'', '', ''
|
||||||
],
|
],
|
||||||
[['msg'], 'm', 'd'],
|
[['msg'], 'm', 'd', 'i'],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should extract from attributes in translatable blocks', () => {
|
it('should extract from attributes in translatable blocks', () => {
|
||||||
expect(extract('<!-- i18n --><p><b i18n-title="m|d" title="msg"></b></p><!-- /i18n -->'))
|
expect(extract('<!-- i18n --><p><b i18n-title="m|d" title="msg"></b></p><!-- /i18n -->'))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['msg'], 'm', 'd'],
|
[['msg'], 'm', 'd', ''],
|
||||||
[
|
[
|
||||||
['<ph tag name="START_PARAGRAPH"><ph tag name="START_BOLD_TEXT"></ph' +
|
['<ph tag name="START_PARAGRAPH"><ph tag name="START_BOLD_TEXT"></ph' +
|
||||||
' name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">'],
|
' name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">'],
|
||||||
'', ''
|
'', '', ''
|
||||||
],
|
],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should extract from attributes in translatable ICUs', () => {
|
it('should extract from attributes in translatable ICUs', () => {
|
||||||
expect(
|
expect(extract(`<!-- i18n -->{count, plural, =0 {<p><b i18n-title="m|d@@i"
|
||||||
extract(
|
title="msg"></b></p>}}<!-- /i18n -->`))
|
||||||
'<!-- i18n -->{count, plural, =0 {<p><b i18n-title="m|d" title="msg"></b></p>}}<!-- /i18n -->'))
|
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['msg'], 'm', 'd'],
|
[['msg'], 'm', 'd', 'i'],
|
||||||
[
|
[
|
||||||
[
|
[
|
||||||
'{count, plural, =0 {[<ph tag name="START_PARAGRAPH"><ph tag' +
|
'{count, plural, =0 {[<ph tag name="START_PARAGRAPH"><ph tag' +
|
||||||
' name="START_BOLD_TEXT"></ph name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">]}}'
|
' name="START_BOLD_TEXT"></ph name="CLOSE_BOLD_TEXT"></ph name="CLOSE_PARAGRAPH">]}}'
|
||||||
],
|
],
|
||||||
'', ''
|
'', '', ''
|
||||||
],
|
],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
@ -228,7 +269,7 @@ export function main() {
|
||||||
it('should extract from attributes in non translatable ICUs', () => {
|
it('should extract from attributes in non translatable ICUs', () => {
|
||||||
expect(extract('{count, plural, =0 {<p><b i18n-title="m|d" title="msg"></b></p>}}'))
|
expect(extract('{count, plural, =0 {<p><b i18n-title="m|d" title="msg"></b></p>}}'))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['msg'], 'm', 'd'],
|
[['msg'], 'm', 'd', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -239,7 +280,7 @@ export function main() {
|
||||||
describe('implicit elements', () => {
|
describe('implicit elements', () => {
|
||||||
it('should extract from implicit elements', () => {
|
it('should extract from implicit elements', () => {
|
||||||
expect(extract('<b>bold</b><i>italic</i>', ['b'])).toEqual([
|
expect(extract('<b>bold</b><i>italic</i>', ['b'])).toEqual([
|
||||||
[['bold'], '', ''],
|
[['bold'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -251,7 +292,7 @@ export function main() {
|
||||||
}).not.toThrow();
|
}).not.toThrow();
|
||||||
|
|
||||||
expect(result).toEqual([
|
expect(result).toEqual([
|
||||||
[['outer', '<ph tag name="START_TAG_DIV">inner</ph name="CLOSE_TAG_DIV">'], '', ''],
|
[['outer', '<ph tag name="START_TAG_DIV">inner</ph name="CLOSE_TAG_DIV">'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -261,7 +302,7 @@ export function main() {
|
||||||
it('should extract implicit attributes', () => {
|
it('should extract implicit attributes', () => {
|
||||||
expect(extract('<b title="bb">bold</b><i title="ii">italic</i>', [], {'b': ['title']}))
|
expect(extract('<b title="bb">bold</b><i title="ii">italic</i>', [], {'b': ['title']}))
|
||||||
.toEqual([
|
.toEqual([
|
||||||
[['bb'], '', ''],
|
[['bb'], '', '', ''],
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -433,7 +474,7 @@ function extract(
|
||||||
// clang-format off
|
// clang-format off
|
||||||
// https://github.com/angular/clang-format/issues/35
|
// https://github.com/angular/clang-format/issues/35
|
||||||
return result.messages.map(
|
return result.messages.map(
|
||||||
message => [serializeI18nNodes(message.nodes), message.meaning, message.description, ]) as [string[], string, string][];
|
message => [serializeI18nNodes(message.nodes), message.meaning, message.description, message.id]) as [string[], string, string][];
|
||||||
// clang-format on
|
// clang-format on
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -59,14 +59,17 @@ export function main() {
|
||||||
tb.detectChanges();
|
tb.detectChanges();
|
||||||
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('un');
|
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('un');
|
||||||
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('un');
|
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('un');
|
||||||
|
expect(el.query(By.css('#i18n-17')).nativeElement).toHaveText('un');
|
||||||
cmp.count = 2;
|
cmp.count = 2;
|
||||||
tb.detectChanges();
|
tb.detectChanges();
|
||||||
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('deux');
|
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('deux');
|
||||||
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('deux');
|
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('deux');
|
||||||
|
expect(el.query(By.css('#i18n-17')).nativeElement).toHaveText('deux');
|
||||||
cmp.count = 3;
|
cmp.count = 3;
|
||||||
tb.detectChanges();
|
tb.detectChanges();
|
||||||
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('beaucoup');
|
expect(el.query(By.css('#i18n-7')).nativeElement).toHaveText('beaucoup');
|
||||||
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('beaucoup');
|
expect(el.query(By.css('#i18n-14')).nativeElement).toHaveText('beaucoup');
|
||||||
|
expect(el.query(By.css('#i18n-17')).nativeElement).toHaveText('beaucoup');
|
||||||
|
|
||||||
cmp.sex = 'm';
|
cmp.sex = 'm';
|
||||||
cmp.sexB = 'f';
|
cmp.sexB = 'f';
|
||||||
|
@ -90,8 +93,8 @@ export function main() {
|
||||||
.toEqual('<h1 id="i18n-12">Balises dans les commentaires html</h1>');
|
.toEqual('<h1 id="i18n-12">Balises dans les commentaires html</h1>');
|
||||||
expectHtml(el, '#i18n-13')
|
expectHtml(el, '#i18n-13')
|
||||||
.toBe('<div id="i18n-13" title="dans une section traductible"></div>');
|
.toBe('<div id="i18n-13" title="dans une section traductible"></div>');
|
||||||
|
|
||||||
expectHtml(el, '#i18n-15').toMatch(/ca <b>devrait<\/b> marcher/);
|
expectHtml(el, '#i18n-15').toMatch(/ca <b>devrait<\/b> marcher/);
|
||||||
|
expectHtml(el, '#i18n-16').toMatch(/avec un ID explicite/);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -141,6 +144,8 @@ function expectHtml(el: DebugElement, cssSelector: string): any {
|
||||||
<!-- /i18n -->
|
<!-- /i18n -->
|
||||||
|
|
||||||
<div id="i18n-15"><ng-container i18n>it <b>should</b> work</ng-container></div>
|
<div id="i18n-15"><ng-container i18n>it <b>should</b> work</ng-container></div>
|
||||||
|
<div id="i18n-16" i18n="@@i18n16">with an explicit ID</div>
|
||||||
|
<div id="i18n-17" i18n="@@i18n17">{count, plural, =0 {zero} =1 {one} =2 {two} other {<b>many</b>}}</div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
class I18nComponent {
|
class I18nComponent {
|
||||||
|
@ -182,6 +187,9 @@ const XTB = `
|
||||||
<ph name="START_TAG_DIV_1"/><ph name="ICU"/><ph name="CLOSE_TAG_DIV"></ph>
|
<ph name="START_TAG_DIV_1"/><ph name="ICU"/><ph name="CLOSE_TAG_DIV"></ph>
|
||||||
</translation>
|
</translation>
|
||||||
<translation id="1491627405349178954">ca <ph name="START_BOLD_TEXT"/>devrait<ph name="CLOSE_BOLD_TEXT"/> marcher</translation>
|
<translation id="1491627405349178954">ca <ph name="START_BOLD_TEXT"/>devrait<ph name="CLOSE_BOLD_TEXT"/> marcher</translation>
|
||||||
|
<translation id="i18n16">avec un ID explicite</translation>
|
||||||
|
<translation id="i18n17">{VAR_PLURAL, plural, =0 {zero} =1 {un} =2 {deux} other {<ph
|
||||||
|
name="START_BOLD_TEXT"><ex><b></ex></ph>beaucoup<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph>} }</translation>
|
||||||
</translationbundle>`;
|
</translationbundle>`;
|
||||||
|
|
||||||
// unused, for reference only
|
// unused, for reference only
|
||||||
|
@ -210,5 +218,7 @@ const XMB = `
|
||||||
<ph name="START_TAG_DIV_1"><ex><div></ex></ph><ph name="ICU"/><ph name="CLOSE_TAG_DIV"><ex></div></ex></ph>
|
<ph name="START_TAG_DIV_1"><ex><div></ex></ph><ph name="ICU"/><ph name="CLOSE_TAG_DIV"><ex></div></ex></ph>
|
||||||
</msg>
|
</msg>
|
||||||
<msg id="1491627405349178954">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">it <ph name="START_BOLD_TEXT"><ex><b></ex></ph>should<ph name="CLOSE_BOLD_TEXT"><ex></b></ex></ph> work</msg>
|
||||||
|
<msg id="i18n16">with an explicit ID</msg>
|
||||||
|
<msg id="i18n17">{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>
|
||||||
</messagebundle>
|
</messagebundle>
|
||||||
`;
|
`;
|
||||||
|
|
|
@ -18,6 +18,8 @@ const HTML = `
|
||||||
<p i18n-title title="translatable attribute">not translatable</p>
|
<p i18n-title title="translatable attribute">not translatable</p>
|
||||||
<p i18n>translatable element <b>with placeholders</b> {{ interpolation}}</p>
|
<p i18n>translatable element <b>with placeholders</b> {{ interpolation}}</p>
|
||||||
<p i18n="m|d">foo</p>
|
<p i18n="m|d">foo</p>
|
||||||
|
<p i18n="m|d@@i">foo</p>
|
||||||
|
<p i18n="@@bar">foo</p>
|
||||||
<p i18n="ph names"><br><img><div></div></p>
|
<p i18n="ph names"><br><img><div></div></p>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
@ -39,6 +41,16 @@ const WRITE_XLIFF = `<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
<note priority="1" from="description">d</note>
|
<note priority="1" from="description">d</note>
|
||||||
<note priority="1" from="meaning">m</note>
|
<note priority="1" from="meaning">m</note>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="i" datatype="html">
|
||||||
|
<source>foo</source>
|
||||||
|
<target/>
|
||||||
|
<note priority="1" from="description">d</note>
|
||||||
|
<note priority="1" from="meaning">m</note>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="bar" datatype="html">
|
||||||
|
<source>foo</source>
|
||||||
|
<target/>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit id="d7fa2d59aaedcaa5309f13028c59af8c85b8c49d" datatype="html">
|
<trans-unit id="d7fa2d59aaedcaa5309f13028c59af8c85b8c49d" datatype="html">
|
||||||
<source><x id="LINE_BREAK" ctype="lb"/><x id="TAG_IMG" ctype="image"/><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/></source>
|
<source><x id="LINE_BREAK" ctype="lb"/><x id="TAG_IMG" ctype="image"/><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/></source>
|
||||||
<target/>
|
<target/>
|
||||||
|
@ -67,6 +79,16 @@ const LOAD_XLIFF = `<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
<note priority="1" from="description">d</note>
|
<note priority="1" from="description">d</note>
|
||||||
<note priority="1" from="meaning">m</note>
|
<note priority="1" from="meaning">m</note>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="i" datatype="html">
|
||||||
|
<source>foo</source>
|
||||||
|
<target>toto</target>
|
||||||
|
<note priority="1" from="description">d</note>
|
||||||
|
<note priority="1" from="meaning">m</note>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="bar" datatype="html">
|
||||||
|
<source>foo</source>
|
||||||
|
<target>tata</target>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit id="d7fa2d59aaedcaa5309f13028c59af8c85b8c49d" datatype="html">
|
<trans-unit id="d7fa2d59aaedcaa5309f13028c59af8c85b8c49d" datatype="html">
|
||||||
<source><x id="LINE_BREAK" ctype="lb"/><x id="TAG_IMG" ctype="image"/><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/></source>
|
<source><x id="LINE_BREAK" ctype="lb"/><x id="TAG_IMG" ctype="image"/><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/></source>
|
||||||
<target><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/><x id="TAG_IMG" ctype="image"/><x id="LINE_BREAK" ctype="lb"/></target>
|
<target><x id="START_TAG_DIV" ctype="x-div"/><x id="CLOSE_TAG_DIV" ctype="x-div"/><x id="TAG_IMG" ctype="image"/><x id="LINE_BREAK" ctype="lb"/></target>
|
||||||
|
@ -107,6 +129,8 @@ export function main(): void {
|
||||||
'ec1d033f2436133c14ab038286c4f5df4697484a':
|
'ec1d033f2436133c14ab038286c4f5df4697484a':
|
||||||
'<ph name="INTERPOLATION"/> footnemele elbatalsnart <ph name="START_BOLD_TEXT"/>sredlohecalp htiw<ph name="CLOSE_BOLD_TEXT"/>',
|
'<ph name="INTERPOLATION"/> footnemele elbatalsnart <ph name="START_BOLD_TEXT"/>sredlohecalp htiw<ph name="CLOSE_BOLD_TEXT"/>',
|
||||||
'db3e0a6a5a96481f60aec61d98c3eecddef5ac23': 'oof',
|
'db3e0a6a5a96481f60aec61d98c3eecddef5ac23': 'oof',
|
||||||
|
'i': 'toto',
|
||||||
|
'bar': 'tata',
|
||||||
'd7fa2d59aaedcaa5309f13028c59af8c85b8c49d':
|
'd7fa2d59aaedcaa5309f13028c59af8c85b8c49d':
|
||||||
'<ph name="START_TAG_DIV"/><ph name="CLOSE_TAG_DIV"/><ph name="TAG_IMG"/><ph name="LINE_BREAK"/>',
|
'<ph name="START_TAG_DIV"/><ph name="CLOSE_TAG_DIV"/><ph name="TAG_IMG"/><ph name="LINE_BREAK"/>',
|
||||||
});
|
});
|
||||||
|
|
|
@ -18,6 +18,9 @@ export function main(): void {
|
||||||
<p i18n>translatable element <b>with placeholders</b> {{ interpolation}}</p>
|
<p i18n>translatable element <b>with placeholders</b> {{ interpolation}}</p>
|
||||||
<!-- i18n -->{ count, plural, =0 {<p>test</p>}}<!-- /i18n -->
|
<!-- i18n -->{ count, plural, =0 {<p>test</p>}}<!-- /i18n -->
|
||||||
<p i18n="m|d">foo</p>
|
<p i18n="m|d">foo</p>
|
||||||
|
<p i18n="m|d@@i">foo</p>
|
||||||
|
<p i18n="@@bar">foo</p>
|
||||||
|
<p i18n="@@baz">{ count, plural, =0 { { sex, select, other {<p>deeply nested</p>}} }}</p>
|
||||||
<p i18n>{ count, plural, =0 { { sex, select, other {<p>deeply nested</p>}} }}</p>`;
|
<p i18n>{ count, plural, =0 { { sex, select, other {<p>deeply nested</p>}} }}</p>`;
|
||||||
|
|
||||||
const XMB = `<?xml version="1.0" encoding="UTF-8" ?>
|
const XMB = `<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
@ -46,6 +49,9 @@ export function main(): void {
|
||||||
<msg id="7056919470098446707">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"/></msg>
|
<msg id="7056919470098446707">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"/></msg>
|
||||||
<msg id="2981514368455622387">{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">{VAR_PLURAL, plural, =0 {<ph name="START_PARAGRAPH"><ex><p></ex></ph>test<ph name="CLOSE_PARAGRAPH"><ex></p></ex></ph>} }</msg>
|
||||||
<msg id="7999024498831672133" desc="d" meaning="m">foo</msg>
|
<msg id="7999024498831672133" desc="d" meaning="m">foo</msg>
|
||||||
|
<msg id="i" desc="d" meaning="m">foo</msg>
|
||||||
|
<msg id="bar">foo</msg>
|
||||||
|
<msg id="baz">{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="2015957479576096115">{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="2015957479576096115">{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>
|
||||||
</messagebundle>
|
</messagebundle>
|
||||||
`;
|
`;
|
||||||
|
|
|
@ -21,7 +21,7 @@ export function main(): void {
|
||||||
it('should translate a plain message', () => {
|
it('should translate a plain message', () => {
|
||||||
const msgMap = {foo: [new i18n.Text('bar', null)]};
|
const msgMap = {foo: [new i18n.Text('bar', null)]};
|
||||||
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
||||||
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd', 'i');
|
||||||
expect(serializeNodes(tb.get(msg))).toEqual(['bar']);
|
expect(serializeNodes(tb.get(msg))).toEqual(['bar']);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ export function main(): void {
|
||||||
ph1: '*phContent*',
|
ph1: '*phContent*',
|
||||||
};
|
};
|
||||||
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
||||||
const msg = new i18n.Message([srcNode], phMap, {}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], phMap, {}, 'm', 'd', 'i');
|
||||||
expect(serializeNodes(tb.get(msg))).toEqual(['bar*phContent*']);
|
expect(serializeNodes(tb.get(msg))).toEqual(['bar*phContent*']);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -51,8 +51,8 @@ export function main(): void {
|
||||||
new i18n.Text('*refMsg*', null),
|
new i18n.Text('*refMsg*', null),
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
const refMsg = new i18n.Message([srcNode], {}, {}, 'm', 'd');
|
const refMsg = new i18n.Message([srcNode], {}, {}, 'm', 'd', 'i');
|
||||||
const msg = new i18n.Message([srcNode], {}, {ph1: refMsg}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], {}, {ph1: refMsg}, 'm', 'd', 'i');
|
||||||
let count = 0;
|
let count = 0;
|
||||||
const digest = (_: any) => count++ ? 'ref' : 'foo';
|
const digest = (_: any) => count++ ? 'ref' : 'foo';
|
||||||
const tb = new TranslationBundle(msgMap, digest);
|
const tb = new TranslationBundle(msgMap, digest);
|
||||||
|
@ -69,13 +69,13 @@ export function main(): void {
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
||||||
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd', 'i');
|
||||||
expect(() => tb.get(msg)).toThrowError(/Unknown placeholder/);
|
expect(() => tb.get(msg)).toThrowError(/Unknown placeholder/);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should report missing translation', () => {
|
it('should report missing translation', () => {
|
||||||
const tb = new TranslationBundle({}, (_) => 'foo');
|
const tb = new TranslationBundle({}, (_) => 'foo');
|
||||||
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], {}, {}, 'm', 'd', 'i');
|
||||||
expect(() => tb.get(msg)).toThrowError(/Missing translation for message foo/);
|
expect(() => tb.get(msg)).toThrowError(/Missing translation for message foo/);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -83,8 +83,8 @@ export function main(): void {
|
||||||
const msgMap = {
|
const msgMap = {
|
||||||
foo: [new i18n.Placeholder('', 'ph1', span)],
|
foo: [new i18n.Placeholder('', 'ph1', span)],
|
||||||
};
|
};
|
||||||
const refMsg = new i18n.Message([srcNode], {}, {}, 'm', 'd');
|
const refMsg = new i18n.Message([srcNode], {}, {}, 'm', 'd', 'i');
|
||||||
const msg = new i18n.Message([srcNode], {}, {ph1: refMsg}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], {}, {ph1: refMsg}, 'm', 'd', 'i');
|
||||||
let count = 0;
|
let count = 0;
|
||||||
const digest = (_: any) => count++ ? 'ref' : 'foo';
|
const digest = (_: any) => count++ ? 'ref' : 'foo';
|
||||||
const tb = new TranslationBundle(msgMap, digest);
|
const tb = new TranslationBundle(msgMap, digest);
|
||||||
|
@ -102,7 +102,7 @@ export function main(): void {
|
||||||
ph1: '</b>',
|
ph1: '</b>',
|
||||||
};
|
};
|
||||||
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
const tb = new TranslationBundle(msgMap, (_) => 'foo');
|
||||||
const msg = new i18n.Message([srcNode], phMap, {}, 'm', 'd');
|
const msg = new i18n.Message([srcNode], phMap, {}, 'm', 'd', 'i');
|
||||||
expect(() => tb.get(msg)).toThrowError(/Unexpected closing tag "b"/);
|
expect(() => tb.get(msg)).toThrowError(/Unexpected closing tag "b"/);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue