/** * @license * Copyright Google Inc. 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 */ import {setup} from '../aot/test_util'; import {compile, expectEmit} from './mock_compile'; const TRANSLATION_NAME_REGEXP = /^MSG_[A-Z0-9]+/; describe('i18n support in the view compiler', () => { const angularFiles = setup({ compileAngular: true, compileAnimations: false, compileCommon: true, }); describe('single text nodes', () => { it('should translate single text nodes with the i18n attribute', () => { const files = { app: { 'spec.ts': ` import {Component, NgModule} from '@angular/core'; @Component({ selector: 'my-component', template: \`
Hello world
&
farewell
farewell
\` }) export class MyComponent {} @NgModule({declarations: [MyComponent]}) export class MyModule {} ` } }; const template = ` const $msg_1$ = goog.getMsg('Hello world'); const $msg_2$ = goog.getMsg('farewell'); … template: function MyComponent_Template(rf: IDENT, ctx: IDENT) { if (rf & 1) { … $r3$.ɵT(1, $msg_1$); … $r3$.ɵT(3,'&'); … $r3$.ɵT(5, $msg_2$); … $r3$.ɵT(7, $msg_2$); … } } `; const result = compile(files, angularFiles); expectEmit(result.source, template, 'Incorrect template', { '$msg_1$': TRANSLATION_NAME_REGEXP, '$msg_2$': TRANSLATION_NAME_REGEXP, }); }); it('should add the meaning and description as JsDoc comments', () => { const files = { app: { 'spec.ts': ` import {Component, NgModule} from '@angular/core'; @Component({ selector: 'my-component', template: \`
Hello world
\` }) export class MyComponent {} @NgModule({declarations: [MyComponent]}) export class MyModule {} ` } }; const template = ` /** * @desc desc */ const $msg_1$ = goog.getMsg('introduction'); const $c1$ = ['title', $msg_1$]; … /** * @desc desc * @meaning meaning */ const $msg_2$ = goog.getMsg('Hello world'); … template: function MyComponent_Template(rf: IDENT, ctx: IDENT) { if (rf & 1) { $r3$.ɵE(0, 'div', $c1$); $r3$.ɵT(1, $msg_2$); $r3$.ɵe(); } } `; const result = compile(files, angularFiles); expectEmit(result.source, template, 'Incorrect template', { '$msg_1$': TRANSLATION_NAME_REGEXP, }); }); }); describe('static attributes', () => { it('should translate static attributes', () => { const files = { app: { 'spec.ts': ` import {Component, NgModule} from '@angular/core'; @Component({ selector: 'my-component', template: \`
\` }) export class MyComponent {} @NgModule({declarations: [MyComponent]}) export class MyModule {} ` } }; const template = ` /** * @desc d * @meaning m */ const $msg_1$ = goog.getMsg('introduction'); const $c1$ = ['id', 'static', 'title', $msg_1$]; … template: function MyComponent_Template(rf: IDENT, ctx: IDENT) { if (rf & 1) { $r3$.ɵEe(0, 'div', $c1$); } } `; const result = compile(files, angularFiles); expectEmit(result.source, template, 'Incorrect template', { '$msg_1$': TRANSLATION_NAME_REGEXP, }); }); }); // TODO(vicb): this feature is not supported yet xdescribe('nested nodes', () => { it('should generate the placeholders maps', () => { const files = { app: { 'spec.ts': ` import {Component, NgModule} from '@angular/core'; @Component({ selector: 'my-component', template: \`
Hello {{name}}!!
Other
2nd
3rd
\` }) export class MyComponent {} @NgModule({declarations: [MyComponent]}) export class MyModule {} ` } }; const template = ` const $r1$ = {'b':[2], 'i':[4, 6]}; const $r2$ = {'i':[13]}; `; const result = compile(files, angularFiles); expectEmit(result.source, template, 'Incorrect template'); }); }); describe('errors', () => { it('should throw on nested i18n sections', () => { const files = { app: { 'spec.ts': ` import {Component, NgModule} from '@angular/core'; @Component({ selector: 'my-component', template: \`
\` }) export class MyComponent {} @NgModule({declarations: [MyComponent]}) export class MyModule {} ` } }; expect(() => compile(files, angularFiles)) .toThrowError( 'Could not mark an element as translatable inside of a translatable section'); }); }); });