4 var grammarTests, bcp47Tests;
6 QUnit.module( 'mediawiki.language', QUnit.newMwEnvironment( {
8 this.liveLangData = mw.language.data;
11 teardown: function () {
12 mw.language.data = this.liveLangData;
15 // mw.language.listToText test
17 'comma-separator': ', ',
22 QUnit.test( 'mw.language getData and setData', function ( assert ) {
23 mw.language.setData( 'en', 'testkey', 'testvalue' );
24 assert.equal( mw.language.getData( 'en', 'testkey' ), 'testvalue', 'Getter setter test for mw.language' );
25 assert.equal( mw.language.getData( 'en', 'invalidkey' ), undefined, 'Getter setter test for mw.language with invalid key' );
26 mw.language.setData( 'en-us', 'testkey', 'testvalue' );
27 assert.equal( mw.language.getData( 'en-US', 'testkey' ), 'testvalue', 'Case insensitive test for mw.language' );
30 QUnit.test( 'mw.language.commafy test', function ( assert ) {
31 mw.language.setData( 'en', 'digitGroupingPattern', null );
32 mw.language.setData( 'en', 'digitTransformTable', null );
33 mw.language.setData( 'en', 'separatorTransformTable', null );
35 mw.config.set( 'wgUserLanguage', 'en' );
36 // Number grouping patterns are as per http://cldr.unicode.org/translation/number-patterns
37 assert.equal( mw.language.commafy( 1234.567, '###0.#####' ), '1234.567', 'Pattern with no digit grouping separator defined' );
38 assert.equal( mw.language.commafy( 123456789.567, '###0.#####' ), '123456789.567', 'Pattern with no digit grouping separator defined, bigger decimal part' );
39 assert.equal( mw.language.commafy( 0.567, '###0.#####' ), '0.567', 'Decimal part 0' );
40 assert.equal( mw.language.commafy( '.567', '###0.#####' ), '0.567', 'Decimal part missing. replace with zero' );
41 assert.equal( mw.language.commafy( 1234, '##,#0.#####' ), '12,34', 'Pattern with no fractional part' );
42 assert.equal( mw.language.commafy( -1234.567, '###0.#####' ), '-1234.567', 'Negative number' );
43 assert.equal( mw.language.commafy( -1234.567, '#,###.00' ), '-1,234.56', 'Fractional part bigger than pattern.' );
44 assert.equal( mw.language.commafy( 123456789.567, '###,##0.00' ), '123,456,789.56', 'Decimal part as group of 3' );
45 assert.equal( mw.language.commafy( 123456789.567, '###,###,#0.00' ), '1,234,567,89.56', 'Decimal part as group of 3 and last one 2' );
48 QUnit.test( 'mw.language.convertNumber', function ( assert ) {
49 mw.language.setData( 'en', 'digitGroupingPattern', null );
50 mw.language.setData( 'en', 'digitTransformTable', null );
51 mw.language.setData( 'en', 'separatorTransformTable', { ',': '.', '.': ',' } );
52 mw.config.set( 'wgUserLanguage', 'en' );
53 mw.config.set( 'wgTranslateNumerals', true );
55 assert.equal( mw.language.convertNumber( 1800 ), '1.800', 'formatting' );
56 assert.equal( mw.language.convertNumber( '1.800', true ), '1800', 'unformatting' );
59 QUnit.test( 'mw.language.convertNumber - digitTransformTable', function ( assert ) {
60 mw.config.set( 'wgUserLanguage', 'hi' );
61 mw.config.set( 'wgTranslateNumerals', true );
62 mw.language.setData( 'hi', 'digitGroupingPattern', null );
63 mw.language.setData( 'hi', 'separatorTransformTable', { ',': '.', '.': ',' } );
65 // Example from Hindi (MessagesHi.php)
66 mw.language.setData( 'hi', 'digitTransformTable', {
72 assert.equal( mw.language.convertNumber( 1200 ), '१.२००', 'format' );
73 assert.equal( mw.language.convertNumber( '१.२००', true ), '1200', 'unformat from digit transform' );
74 assert.equal( mw.language.convertNumber( '1.200', true ), '1200', 'unformat plain' );
76 mw.config.set( 'wgTranslateNumerals', false );
78 assert.equal( mw.language.convertNumber( 1200 ), '1.200', 'format (digit transform disabled)' );
79 assert.equal( mw.language.convertNumber( '१.२००', true ), '1200', 'unformat from digit transform (when disabled)' );
80 assert.equal( mw.language.convertNumber( '1.200', true ), '1200', 'unformat plain (digit transform disabled)' );
83 function grammarTest( langCode, test ) {
84 // The test works only if the content language is opt.language
85 // because it requires [lang].js to be loaded.
86 QUnit.test( 'Grammar test for lang=' + langCode, function ( assert ) {
88 for ( i = 0; i < test.length; i++ ) {
90 mw.language.convertGrammar( test[ i ].word, test[ i ].grammarForm ),
98 // These tests run only for the current UI language.
103 grammarForm: 'instrumental',
105 description: 'Grammar test for instrumental case'
109 grammarForm: 'lokativ',
111 description: 'Grammar test for lokativ case'
118 grammarForm: 'prefixed',
119 expected: 'וויקיפדיה',
120 description: 'Duplicate the "Waw" if prefixed'
124 grammarForm: 'prefixed',
126 description: 'Duplicate the "Waw" if prefixed, but not if it is already duplicated.'
130 grammarForm: 'prefixed',
132 description: 'Remove the "He" if prefixed'
136 grammarForm: 'תחילית',
137 expected: '־Wikipedia',
138 description: 'Add a hyphen (maqaf) before non-Hebrew letters'
142 grammarForm: 'תחילית',
144 description: 'Add a hyphen (maqaf) before numbers'
151 grammarForm: 'instrumental',
153 description: 'Grammar test for instrumental case'
157 grammarForm: 'lokatiw',
159 description: 'Grammar test for lokatiw case'
166 grammarForm: 'instrumental',
168 description: 'Grammar test for instrumental case'
172 grammarForm: 'lokatiw',
174 description: 'Grammar test for lokatiw case'
181 grammarForm: 'genitive',
182 expected: 'Մաունայի',
183 description: 'Grammar test for genitive case'
187 grammarForm: 'genitive',
189 description: 'Grammar test for genitive case'
193 grammarForm: 'genitive',
195 description: 'Grammar test for genitive case'
199 grammarForm: 'genitive',
200 expected: 'ժամանակիի',
201 description: 'Grammar test for genitive case'
208 grammarForm: 'genitive',
210 description: 'Grammar test for genitive case'
214 grammarForm: 'genitive',
216 description: 'Grammar test for genitive case'
220 grammarForm: 'elative',
222 description: 'Grammar test for elative case'
226 grammarForm: 'partitive',
227 expected: 'pastöroitua',
228 description: 'Grammar test for partitive case'
232 grammarForm: 'partitive',
234 description: 'Grammar test for partitive case'
238 grammarForm: 'illative',
240 description: 'Grammar test for illative case'
244 grammarForm: 'inessive',
245 expected: 'linuxissa',
246 description: 'Grammar test for inessive case'
253 grammarForm: 'genitive',
255 description: 'Grammar test for genitive case, тесть -> тестя'
259 grammarForm: 'genitive',
260 expected: 'привилегии',
261 description: 'Grammar test for genitive case, привилегия -> привилегии'
265 grammarForm: 'genitive',
266 expected: 'установки',
267 description: 'Grammar test for genitive case, установка -> установки'
271 grammarForm: 'genitive',
273 description: 'Grammar test for genitive case, похоти -> похотей'
277 grammarForm: 'genitive',
279 description: 'Grammar test for genitive case, доводы -> доводов'
283 grammarForm: 'genitive',
284 expected: 'песчаника',
285 description: 'Grammar test for genitive case, песчаник -> песчаника'
289 grammarForm: 'genitive',
291 description: 'Grammar test for genitive case, данные -> данных'
295 grammarForm: 'prepositional',
297 description: 'Grammar test for prepositional case, тесть -> тесте'
301 grammarForm: 'prepositional',
302 expected: 'привилегии',
303 description: 'Grammar test for prepositional case, привилегия -> привилегии'
307 grammarForm: 'prepositional',
308 expected: 'установке',
309 description: 'Grammar test for prepositional case, установка -> установке'
313 grammarForm: 'prepositional',
315 description: 'Grammar test for prepositional case, похоти -> похотях'
319 grammarForm: 'prepositional',
321 description: 'Grammar test for prepositional case, доводы -> доводах'
325 grammarForm: 'prepositional',
326 expected: 'Викискладе',
327 description: 'Grammar test for prepositional case, Викисклад -> Викискладе'
331 grammarForm: 'genitive',
332 expected: 'Викисклада',
333 description: 'Grammar test for genitive case, Викисклад -> Викисклада'
337 grammarForm: 'prepositional',
338 expected: 'песчанике',
339 description: 'Grammar test for prepositional case, песчаник -> песчанике'
343 grammarForm: 'prepositional',
345 description: 'Grammar test for prepositional case, данные -> данных'
349 grammarForm: 'languagegen',
350 expected: 'русского',
351 description: 'Grammar test for languagegen case, русский -> русского'
355 grammarForm: 'languagegen',
356 expected: 'немецкого',
357 description: 'Grammar test for languagegen case, немецкий -> немецкого'
361 grammarForm: 'languagegen',
363 description: 'Grammar test for languagegen case, иврит -> иврита'
367 grammarForm: 'languagegen',
368 expected: 'эсперанто',
369 description: 'Grammar test for languagegen case, эсперанто -> эсперанто'
373 grammarForm: 'languageprep',
375 description: 'Grammar test for languageprep case, русский -> русском'
379 grammarForm: 'languageprep',
380 expected: 'немецком',
381 description: 'Grammar test for languageprep case, немецкий -> немецком'
385 grammarForm: 'languageprep',
387 description: 'Grammar test for languageprep case, идиш -> идише'
391 grammarForm: 'languageprep',
392 expected: 'эсперанто',
393 description: 'Grammar test for languageprep case, эсперанто -> эсперанто'
397 grammarForm: 'languageadverb',
398 expected: 'по-русски',
399 description: 'Grammar test for languageadverb case, русский -> по-русски'
403 grammarForm: 'languageadverb',
404 expected: 'по-немецки',
405 description: 'Grammar test for languageadverb case, немецкий -> по-немецки'
409 grammarForm: 'languageadverb',
410 expected: 'на иврите',
411 description: 'Grammar test for languageadverb case, иврит -> на иврите'
415 grammarForm: 'languageadverb',
416 expected: 'на эсперанто',
417 description: 'Grammar test for languageadverb case, эсперанто -> на эсперанто'
421 grammarForm: 'languageadverb',
422 expected: 'на языке гуарани',
423 description: 'Grammar test for languageadverb case, гуарани -> на языке гуарани'
431 expected: 'Wikipédiáról',
432 description: 'Grammar test for rol case'
437 expected: 'Wikipédiába',
438 description: 'Grammar test for ba case'
443 expected: 'Wikipédiák',
444 description: 'Grammar test for k case'
451 grammarForm: 'ainmlae',
452 expected: 'Dé Domhnaigh',
453 description: 'Grammar test for ainmlae case'
457 grammarForm: 'ainmlae',
458 expected: 'Dé Luain',
459 description: 'Grammar test for ainmlae case'
463 grammarForm: 'ainmlae',
464 expected: 'Dé Sathairn',
465 description: 'Grammar test for ainmlae case'
472 grammarForm: 'genitive',
473 expected: 'Вікіпедії',
474 description: 'Grammar test for genitive case'
478 grammarForm: 'genitive',
479 expected: 'Віківидів',
480 description: 'Grammar test for genitive case'
484 grammarForm: 'genitive',
485 expected: 'Вікіцитат',
486 description: 'Grammar test for genitive case'
489 word: 'Вікіпідручник',
490 grammarForm: 'genitive',
491 expected: 'Вікіпідручника',
492 description: 'Grammar test for genitive case'
496 grammarForm: 'accusative',
497 expected: 'Вікіпедію',
498 description: 'Grammar test for accusative case'
505 grammarForm: 'orodnik',
507 description: 'Grammar test for orodnik case'
511 grammarForm: 'mestnik',
513 description: 'Grammar test for mestnik case'
520 grammarForm: 'genitive',
522 description: 'Grammar test for genitive case'
526 grammarForm: 'allative',
528 description: 'Grammar test for allative case'
532 grammarForm: 'dative',
534 description: 'Grammar test for dative case'
538 grammarForm: 'dative',
539 expected: 'цъитийæн',
540 description: 'Grammar test for dative case'
544 grammarForm: 'genitive',
546 description: 'Grammar test for genitive case'
550 grammarForm: 'equative',
552 description: 'Grammar test for equative case'
559 grammarForm: 'genitive',
560 expected: 'Translationis',
561 description: 'Grammar test for genitive case'
565 grammarForm: 'accusative',
566 expected: 'Translationem',
567 description: 'Grammar test for accusative case'
571 grammarForm: 'ablative',
572 expected: 'Translatione',
573 description: 'Grammar test for ablative case'
578 $.each( grammarTests, function ( langCode, test ) {
579 if ( langCode === mw.config.get( 'wgUserLanguage' ) ) {
580 grammarTest( langCode, test );
584 QUnit.test( 'List to text test', function ( assert ) {
585 assert.equal( mw.language.listToText( [] ), '', 'Blank list' );
586 assert.equal( mw.language.listToText( [ 'a' ] ), 'a', 'Single item' );
587 assert.equal( mw.language.listToText( [ 'a', 'b' ] ), 'a and b', 'Two items' );
588 assert.equal( mw.language.listToText( [ 'a', 'b', 'c' ] ), 'a, b and c', 'More than two items' );
592 // Extracted from BCP 47 (list not exhaustive)
594 [ 'en-ca-x-ca', 'en-CA-x-ca' ],
595 [ 'sgn-be-fr', 'sgn-BE-FR' ],
596 [ 'az-latn-x-latn', 'az-Latn-x-latn' ],
598 [ 'sr-Latn-RS', 'sr-Latn-RS' ],
599 [ 'az-arab-ir', 'az-Arab-IR' ],
602 [ 'sl-nedis', 'sl-nedis' ],
603 [ 'de-ch-1996', 'de-CH-1996' ],
607 'en-latn-gb-boont-r-extended-sequence-x-private',
608 'en-Latn-GB-boont-r-extended-sequence-x-private'
611 // Examples from BCP 47 Appendix A
612 // # Simple language subtag:
617 // # Language subtag plus script subtag:
618 [ 'zh-hans', 'zh-Hans' ],
619 [ 'sr-cyrl', 'sr-Cyrl' ],
620 [ 'sr-latn', 'sr-Latn' ],
622 // # Extended language subtags and their primary language subtag
624 [ 'zh-cmn-hans-cn', 'zh-cmn-Hans-CN' ],
625 [ 'cmn-hans-cn', 'cmn-Hans-CN' ],
626 [ 'zh-yue-hk', 'zh-yue-HK' ],
627 [ 'yue-hk', 'yue-HK' ],
629 // # Language-Script-Region:
630 [ 'zh-hans-cn', 'zh-Hans-CN' ],
631 [ 'sr-latn-RS', 'sr-Latn-RS' ],
633 // # Language-Variant:
634 [ 'sl-rozaj', 'sl-rozaj' ],
635 [ 'sl-rozaj-biske', 'sl-rozaj-biske' ],
636 [ 'sl-nedis', 'sl-nedis' ],
638 // # Language-Region-Variant:
639 [ 'de-ch-1901', 'de-CH-1901' ],
640 [ 'sl-it-nedis', 'sl-IT-nedis' ],
642 // # Language-Script-Region-Variant:
643 [ 'hy-latn-it-arevela', 'hy-Latn-IT-arevela' ],
645 // # Language-Region:
646 [ 'de-de', 'de-DE' ],
647 [ 'en-us', 'en-US' ],
648 [ 'es-419', 'es-419' ],
650 // # Private use subtags:
651 [ 'de-ch-x-phonebk', 'de-CH-x-phonebk' ],
652 [ 'az-arab-x-aze-derbend', 'az-Arab-x-aze-derbend' ],
654 * Previous test does not reflect the BCP 47 which states:
655 * az-Arab-x-AZE-derbend
656 * AZE being private, it should be lower case, hence the test above
657 * should probably be:
658 * [ 'az-arab-x-aze-derbend', 'az-Arab-x-AZE-derbend' ],
661 // # Private use registry values:
662 [ 'x-whatever', 'x-whatever' ],
663 [ 'qaa-qaaa-qm-x-southern', 'qaa-Qaaa-QM-x-southern' ],
664 [ 'de-qaaa', 'de-Qaaa' ],
665 [ 'sr-latn-qm', 'sr-Latn-QM' ],
666 [ 'sr-qaaa-rs', 'sr-Qaaa-RS' ],
668 // # Tags that use extensions
669 [ 'en-us-u-islamcal', 'en-US-u-islamcal' ],
670 [ 'zh-cn-a-myext-x-private', 'zh-CN-a-myext-x-private' ],
671 [ 'en-a-myext-b-another', 'en-a-myext-b-another' ]
676 // ar-a-aaa-b-bbb-a-ccc
679 QUnit.test( 'mw.language.bcp47', function ( assert ) {
680 bcp47Tests.forEach( function ( data ) {
681 var input = data[ 0 ],
682 expected = data[ 1 ];
683 assert.equal( mw.language.bcp47( input ), expected );
686 }( mediaWiki, jQuery ) );