Merge "mediawiki.language: Implement non-digit-grouping of four-digit numbers"
[lhc/web/wiklou.git] / tests / qunit / suites / resources / mediawiki / mediawiki.language.test.js
1 ( function ( mw, $ ) {
2 'use strict';
3
4 var grammarTests, bcp47Tests;
5
6 QUnit.module( 'mediawiki.language', QUnit.newMwEnvironment( {
7 setup: function () {
8 this.liveLangData = mw.language.data;
9 mw.language.data = {};
10 },
11 teardown: function () {
12 mw.language.data = this.liveLangData;
13 },
14 messages: {
15 // mw.language.listToText test
16 and: ' and',
17 'comma-separator': ', ',
18 'word-separator': ' '
19 }
20 } ) );
21
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' );
28 } );
29
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 );
34
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' );
46 } );
47
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.language.setData( 'en', 'minimumGroupingDigits', null );
53 mw.config.set( 'wgUserLanguage', 'en' );
54 mw.config.set( 'wgTranslateNumerals', true );
55
56 assert.equal( mw.language.convertNumber( 180 ), '180', 'formatting 3-digit' );
57 assert.equal( mw.language.convertNumber( 1800 ), '1.800', 'formatting 4-digit' );
58 assert.equal( mw.language.convertNumber( 18000 ), '18.000', 'formatting 5-digit' );
59
60 assert.equal( mw.language.convertNumber( '1.800', true ), '1800', 'unformatting' );
61
62 mw.language.setData( 'en', 'minimumGroupingDigits', 2 );
63 assert.equal( mw.language.convertNumber( 180 ), '180', 'formatting 3-digit with minimumGroupingDigits=2' );
64 assert.equal( mw.language.convertNumber( 1800 ), '1800', 'formatting 4-digit with minimumGroupingDigits=2' );
65 assert.equal( mw.language.convertNumber( 18000 ), '18.000', 'formatting 5-digit with minimumGroupingDigits=2' );
66 } );
67
68 QUnit.test( 'mw.language.convertNumber - digitTransformTable', function ( assert ) {
69 mw.config.set( 'wgUserLanguage', 'hi' );
70 mw.config.set( 'wgTranslateNumerals', true );
71 mw.language.setData( 'hi', 'digitGroupingPattern', null );
72 mw.language.setData( 'hi', 'separatorTransformTable', { ',': '.', '.': ',' } );
73 mw.language.setData( 'hi', 'minimumGroupingDigits', null );
74
75 // Example from Hindi (MessagesHi.php)
76 mw.language.setData( 'hi', 'digitTransformTable', {
77 0: '०',
78 1: '१',
79 2: '२'
80 } );
81
82 assert.equal( mw.language.convertNumber( 1200 ), '१.२००', 'format' );
83 assert.equal( mw.language.convertNumber( '१.२००', true ), '1200', 'unformat from digit transform' );
84 assert.equal( mw.language.convertNumber( '1.200', true ), '1200', 'unformat plain' );
85
86 mw.config.set( 'wgTranslateNumerals', false );
87
88 assert.equal( mw.language.convertNumber( 1200 ), '1.200', 'format (digit transform disabled)' );
89 assert.equal( mw.language.convertNumber( '१.२००', true ), '1200', 'unformat from digit transform (when disabled)' );
90 assert.equal( mw.language.convertNumber( '1.200', true ), '1200', 'unformat plain (digit transform disabled)' );
91 } );
92
93 function grammarTest( langCode, test ) {
94 // The test works only if the content language is opt.language
95 // because it requires [lang].js to be loaded.
96 QUnit.test( 'Grammar test for lang=' + langCode, function ( assert ) {
97 var i;
98 for ( i = 0; i < test.length; i++ ) {
99 assert.equal(
100 mw.language.convertGrammar( test[ i ].word, test[ i ].grammarForm ),
101 test[ i ].expected,
102 test[ i ].description
103 );
104 }
105 } );
106 }
107
108 // These tests run only for the current UI language.
109 grammarTests = {
110 bs: [
111 {
112 word: 'word',
113 grammarForm: 'instrumental',
114 expected: 's word',
115 description: 'Grammar test for instrumental case'
116 },
117 {
118 word: 'word',
119 grammarForm: 'lokativ',
120 expected: 'o word',
121 description: 'Grammar test for lokativ case'
122 }
123 ],
124
125 he: [
126 {
127 word: 'ויקיפדיה',
128 grammarForm: 'prefixed',
129 expected: 'וויקיפדיה',
130 description: 'Duplicate the "Waw" if prefixed'
131 },
132 {
133 word: 'וולפגנג',
134 grammarForm: 'prefixed',
135 expected: 'וולפגנג',
136 description: 'Duplicate the "Waw" if prefixed, but not if it is already duplicated.'
137 },
138 {
139 word: 'הקובץ',
140 grammarForm: 'prefixed',
141 expected: 'קובץ',
142 description: 'Remove the "He" if prefixed'
143 },
144 {
145 word: 'Wikipedia',
146 grammarForm: 'תחילית',
147 expected: '־Wikipedia',
148 description: 'Add a hyphen (maqaf) before non-Hebrew letters'
149 },
150 {
151 word: '1995',
152 grammarForm: 'תחילית',
153 expected: '־1995',
154 description: 'Add a hyphen (maqaf) before numbers'
155 }
156 ],
157
158 hsb: [
159 {
160 word: 'word',
161 grammarForm: 'instrumental',
162 expected: 'z word',
163 description: 'Grammar test for instrumental case'
164 },
165 {
166 word: 'word',
167 grammarForm: 'lokatiw',
168 expected: 'wo word',
169 description: 'Grammar test for lokatiw case'
170 }
171 ],
172
173 dsb: [
174 {
175 word: 'word',
176 grammarForm: 'instrumental',
177 expected: 'z word',
178 description: 'Grammar test for instrumental case'
179 },
180 {
181 word: 'word',
182 grammarForm: 'lokatiw',
183 expected: 'wo word',
184 description: 'Grammar test for lokatiw case'
185 }
186 ],
187
188 hy: [
189 {
190 word: 'Մաունա',
191 grammarForm: 'genitive',
192 expected: 'Մաունայի',
193 description: 'Grammar test for genitive case'
194 },
195 {
196 word: 'հետո',
197 grammarForm: 'genitive',
198 expected: 'հետոյի',
199 description: 'Grammar test for genitive case'
200 },
201 {
202 word: 'գիրք',
203 grammarForm: 'genitive',
204 expected: 'գրքի',
205 description: 'Grammar test for genitive case'
206 },
207 {
208 word: 'ժամանակի',
209 grammarForm: 'genitive',
210 expected: 'ժամանակիի',
211 description: 'Grammar test for genitive case'
212 }
213 ],
214
215 fi: [
216 {
217 word: 'talo',
218 grammarForm: 'genitive',
219 expected: 'talon',
220 description: 'Grammar test for genitive case'
221 },
222 {
223 word: 'linux',
224 grammarForm: 'genitive',
225 expected: 'linuxin',
226 description: 'Grammar test for genitive case'
227 },
228 {
229 word: 'talo',
230 grammarForm: 'elative',
231 expected: 'talosta',
232 description: 'Grammar test for elative case'
233 },
234 {
235 word: 'pastöroitu',
236 grammarForm: 'partitive',
237 expected: 'pastöroitua',
238 description: 'Grammar test for partitive case'
239 },
240 {
241 word: 'talo',
242 grammarForm: 'partitive',
243 expected: 'taloa',
244 description: 'Grammar test for partitive case'
245 },
246 {
247 word: 'talo',
248 grammarForm: 'illative',
249 expected: 'taloon',
250 description: 'Grammar test for illative case'
251 },
252 {
253 word: 'linux',
254 grammarForm: 'inessive',
255 expected: 'linuxissa',
256 description: 'Grammar test for inessive case'
257 }
258 ],
259
260 ru: [
261 {
262 word: 'тесть',
263 grammarForm: 'genitive',
264 expected: 'тестя',
265 description: 'Grammar test for genitive case, тесть -> тестя'
266 },
267 {
268 word: 'привилегия',
269 grammarForm: 'genitive',
270 expected: 'привилегии',
271 description: 'Grammar test for genitive case, привилегия -> привилегии'
272 },
273 {
274 word: 'установка',
275 grammarForm: 'genitive',
276 expected: 'установки',
277 description: 'Grammar test for genitive case, установка -> установки'
278 },
279 {
280 word: 'похоти',
281 grammarForm: 'genitive',
282 expected: 'похотей',
283 description: 'Grammar test for genitive case, похоти -> похотей'
284 },
285 {
286 word: 'доводы',
287 grammarForm: 'genitive',
288 expected: 'доводов',
289 description: 'Grammar test for genitive case, доводы -> доводов'
290 },
291 {
292 word: 'песчаник',
293 grammarForm: 'genitive',
294 expected: 'песчаника',
295 description: 'Grammar test for genitive case, песчаник -> песчаника'
296 },
297 {
298 word: 'данные',
299 grammarForm: 'genitive',
300 expected: 'данных',
301 description: 'Grammar test for genitive case, данные -> данных'
302 },
303 {
304 word: 'тесть',
305 grammarForm: 'prepositional',
306 expected: 'тесте',
307 description: 'Grammar test for prepositional case, тесть -> тесте'
308 },
309 {
310 word: 'привилегия',
311 grammarForm: 'prepositional',
312 expected: 'привилегии',
313 description: 'Grammar test for prepositional case, привилегия -> привилегии'
314 },
315 {
316 word: 'университет',
317 grammarForm: 'prepositional',
318 expected: 'университете',
319 description: 'Grammar test for prepositional case, университет -> университете'
320 },
321 {
322 word: 'университет',
323 grammarForm: 'genitive',
324 expected: 'университета',
325 description: 'Grammar test for prepositional case, университет -> университете'
326 },
327 {
328 word: 'установка',
329 grammarForm: 'prepositional',
330 expected: 'установке',
331 description: 'Grammar test for prepositional case, установка -> установке'
332 },
333 {
334 word: 'похоти',
335 grammarForm: 'prepositional',
336 expected: 'похотях',
337 description: 'Grammar test for prepositional case, похоти -> похотях'
338 },
339 {
340 word: 'доводы',
341 grammarForm: 'prepositional',
342 expected: 'доводах',
343 description: 'Grammar test for prepositional case, доводы -> доводах'
344 },
345 {
346 word: 'Викисклад',
347 grammarForm: 'prepositional',
348 expected: 'Викискладе',
349 description: 'Grammar test for prepositional case, Викисклад -> Викискладе'
350 },
351 {
352 word: 'Викисклад',
353 grammarForm: 'genitive',
354 expected: 'Викисклада',
355 description: 'Grammar test for genitive case, Викисклад -> Викисклада'
356 },
357 {
358 word: 'песчаник',
359 grammarForm: 'prepositional',
360 expected: 'песчанике',
361 description: 'Grammar test for prepositional case, песчаник -> песчанике'
362 },
363 {
364 word: 'данные',
365 grammarForm: 'prepositional',
366 expected: 'данных',
367 description: 'Grammar test for prepositional case, данные -> данных'
368 },
369 {
370 word: 'русский',
371 grammarForm: 'languagegen',
372 expected: 'русского',
373 description: 'Grammar test for languagegen case, русский -> русского'
374 },
375 {
376 word: 'немецкий',
377 grammarForm: 'languagegen',
378 expected: 'немецкого',
379 description: 'Grammar test for languagegen case, немецкий -> немецкого'
380 },
381 {
382 word: 'иврит',
383 grammarForm: 'languagegen',
384 expected: 'иврита',
385 description: 'Grammar test for languagegen case, иврит -> иврита'
386 },
387 {
388 word: 'эсперанто',
389 grammarForm: 'languagegen',
390 expected: 'эсперанто',
391 description: 'Grammar test for languagegen case, эсперанто -> эсперанто'
392 },
393 {
394 word: 'русский',
395 grammarForm: 'languageprep',
396 expected: 'русском',
397 description: 'Grammar test for languageprep case, русский -> русском'
398 },
399 {
400 word: 'немецкий',
401 grammarForm: 'languageprep',
402 expected: 'немецком',
403 description: 'Grammar test for languageprep case, немецкий -> немецком'
404 },
405 {
406 word: 'идиш',
407 grammarForm: 'languageprep',
408 expected: 'идише',
409 description: 'Grammar test for languageprep case, идиш -> идише'
410 },
411 {
412 word: 'эсперанто',
413 grammarForm: 'languageprep',
414 expected: 'эсперанто',
415 description: 'Grammar test for languageprep case, эсперанто -> эсперанто'
416 },
417 {
418 word: 'русский',
419 grammarForm: 'languageadverb',
420 expected: 'по-русски',
421 description: 'Grammar test for languageadverb case, русский -> по-русски'
422 },
423 {
424 word: 'немецкий',
425 grammarForm: 'languageadverb',
426 expected: 'по-немецки',
427 description: 'Grammar test for languageadverb case, немецкий -> по-немецки'
428 },
429 {
430 word: 'иврит',
431 grammarForm: 'languageadverb',
432 expected: 'на иврите',
433 description: 'Grammar test for languageadverb case, иврит -> на иврите'
434 },
435 {
436 word: 'эсперанто',
437 grammarForm: 'languageadverb',
438 expected: 'на эсперанто',
439 description: 'Grammar test for languageadverb case, эсперанто -> на эсперанто'
440 },
441 {
442 word: 'гуарани',
443 grammarForm: 'languageadverb',
444 expected: 'на языке гуарани',
445 description: 'Grammar test for languageadverb case, гуарани -> на языке гуарани'
446 }
447 ],
448
449 hu: [
450 {
451 word: 'Wikipédiá',
452 grammarForm: 'rol',
453 expected: 'Wikipédiáról',
454 description: 'Grammar test for rol case'
455 },
456 {
457 word: 'Wikipédiá',
458 grammarForm: 'ba',
459 expected: 'Wikipédiába',
460 description: 'Grammar test for ba case'
461 },
462 {
463 word: 'Wikipédiá',
464 grammarForm: 'k',
465 expected: 'Wikipédiák',
466 description: 'Grammar test for k case'
467 }
468 ],
469
470 ga: [
471 {
472 word: 'an Domhnach',
473 grammarForm: 'ainmlae',
474 expected: 'Dé Domhnaigh',
475 description: 'Grammar test for ainmlae case'
476 },
477 {
478 word: 'an Luan',
479 grammarForm: 'ainmlae',
480 expected: 'Dé Luain',
481 description: 'Grammar test for ainmlae case'
482 },
483 {
484 word: 'an Satharn',
485 grammarForm: 'ainmlae',
486 expected: 'Dé Sathairn',
487 description: 'Grammar test for ainmlae case'
488 }
489 ],
490
491 uk: [
492 {
493 word: 'Вікіпедія',
494 grammarForm: 'genitive',
495 expected: 'Вікіпедії',
496 description: 'Grammar test for genitive case'
497 },
498 {
499 word: 'Віківиди',
500 grammarForm: 'genitive',
501 expected: 'Віківидів',
502 description: 'Grammar test for genitive case'
503 },
504 {
505 word: 'Вікіцитати',
506 grammarForm: 'genitive',
507 expected: 'Вікіцитат',
508 description: 'Grammar test for genitive case'
509 },
510 {
511 word: 'Вікіпідручник',
512 grammarForm: 'genitive',
513 expected: 'Вікіпідручника',
514 description: 'Grammar test for genitive case'
515 },
516 {
517 word: 'Вікіпедія',
518 grammarForm: 'accusative',
519 expected: 'Вікіпедію',
520 description: 'Grammar test for accusative case'
521 }
522 ],
523
524 sl: [
525 {
526 word: 'word',
527 grammarForm: 'orodnik',
528 expected: 'z word',
529 description: 'Grammar test for orodnik case'
530 },
531 {
532 word: 'word',
533 grammarForm: 'mestnik',
534 expected: 'o word',
535 description: 'Grammar test for mestnik case'
536 }
537 ],
538
539 os: [
540 {
541 word: 'бæстæ',
542 grammarForm: 'genitive',
543 expected: 'бæсты',
544 description: 'Grammar test for genitive case'
545 },
546 {
547 word: 'бæстæ',
548 grammarForm: 'allative',
549 expected: 'бæстæм',
550 description: 'Grammar test for allative case'
551 },
552 {
553 word: 'Тигр',
554 grammarForm: 'dative',
555 expected: 'Тигрæн',
556 description: 'Grammar test for dative case'
557 },
558 {
559 word: 'цъити',
560 grammarForm: 'dative',
561 expected: 'цъитийæн',
562 description: 'Grammar test for dative case'
563 },
564 {
565 word: 'лæппу',
566 grammarForm: 'genitive',
567 expected: 'лæппуйы',
568 description: 'Grammar test for genitive case'
569 },
570 {
571 word: '2011',
572 grammarForm: 'equative',
573 expected: '2011-ау',
574 description: 'Grammar test for equative case'
575 }
576 ],
577
578 la: [
579 {
580 word: 'Translatio',
581 grammarForm: 'genitive',
582 expected: 'Translationis',
583 description: 'Grammar test for genitive case'
584 },
585 {
586 word: 'Translatio',
587 grammarForm: 'accusative',
588 expected: 'Translationem',
589 description: 'Grammar test for accusative case'
590 },
591 {
592 word: 'Translatio',
593 grammarForm: 'ablative',
594 expected: 'Translatione',
595 description: 'Grammar test for ablative case'
596 }
597 ]
598 };
599
600 $.each( grammarTests, function ( langCode, test ) {
601 if ( langCode === mw.config.get( 'wgUserLanguage' ) ) {
602 grammarTest( langCode, test );
603 }
604 } );
605
606 QUnit.test( 'List to text test', function ( assert ) {
607 assert.equal( mw.language.listToText( [] ), '', 'Blank list' );
608 assert.equal( mw.language.listToText( [ 'a' ] ), 'a', 'Single item' );
609 assert.equal( mw.language.listToText( [ 'a', 'b' ] ), 'a and b', 'Two items' );
610 assert.equal( mw.language.listToText( [ 'a', 'b', 'c' ] ), 'a, b and c', 'More than two items' );
611 } );
612
613 bcp47Tests = [
614 // Extracted from BCP 47 (list not exhaustive)
615 // # 2.1.1
616 [ 'en-ca-x-ca', 'en-CA-x-ca' ],
617 [ 'sgn-be-fr', 'sgn-BE-FR' ],
618 [ 'az-latn-x-latn', 'az-Latn-x-latn' ],
619 // # 2.2
620 [ 'sr-Latn-RS', 'sr-Latn-RS' ],
621 [ 'az-arab-ir', 'az-Arab-IR' ],
622
623 // # 2.2.5
624 [ 'sl-nedis', 'sl-nedis' ],
625 [ 'de-ch-1996', 'de-CH-1996' ],
626
627 // # 2.2.6
628 [
629 'en-latn-gb-boont-r-extended-sequence-x-private',
630 'en-Latn-GB-boont-r-extended-sequence-x-private'
631 ],
632
633 // Examples from BCP 47 Appendix A
634 // # Simple language subtag:
635 [ 'DE', 'de' ],
636 [ 'fR', 'fr' ],
637 [ 'ja', 'ja' ],
638
639 // # Language subtag plus script subtag:
640 [ 'zh-hans', 'zh-Hans' ],
641 [ 'sr-cyrl', 'sr-Cyrl' ],
642 [ 'sr-latn', 'sr-Latn' ],
643
644 // # Extended language subtags and their primary language subtag
645 // # counterparts:
646 [ 'zh-cmn-hans-cn', 'zh-cmn-Hans-CN' ],
647 [ 'cmn-hans-cn', 'cmn-Hans-CN' ],
648 [ 'zh-yue-hk', 'zh-yue-HK' ],
649 [ 'yue-hk', 'yue-HK' ],
650
651 // # Language-Script-Region:
652 [ 'zh-hans-cn', 'zh-Hans-CN' ],
653 [ 'sr-latn-RS', 'sr-Latn-RS' ],
654
655 // # Language-Variant:
656 [ 'sl-rozaj', 'sl-rozaj' ],
657 [ 'sl-rozaj-biske', 'sl-rozaj-biske' ],
658 [ 'sl-nedis', 'sl-nedis' ],
659
660 // # Language-Region-Variant:
661 [ 'de-ch-1901', 'de-CH-1901' ],
662 [ 'sl-it-nedis', 'sl-IT-nedis' ],
663
664 // # Language-Script-Region-Variant:
665 [ 'hy-latn-it-arevela', 'hy-Latn-IT-arevela' ],
666
667 // # Language-Region:
668 [ 'de-de', 'de-DE' ],
669 [ 'en-us', 'en-US' ],
670 [ 'es-419', 'es-419' ],
671
672 // # Private use subtags:
673 [ 'de-ch-x-phonebk', 'de-CH-x-phonebk' ],
674 [ 'az-arab-x-aze-derbend', 'az-Arab-x-aze-derbend' ],
675 /**
676 * Previous test does not reflect the BCP 47 which states:
677 * az-Arab-x-AZE-derbend
678 * AZE being private, it should be lower case, hence the test above
679 * should probably be:
680 * [ 'az-arab-x-aze-derbend', 'az-Arab-x-AZE-derbend' ],
681 */
682
683 // # Private use registry values:
684 [ 'x-whatever', 'x-whatever' ],
685 [ 'qaa-qaaa-qm-x-southern', 'qaa-Qaaa-QM-x-southern' ],
686 [ 'de-qaaa', 'de-Qaaa' ],
687 [ 'sr-latn-qm', 'sr-Latn-QM' ],
688 [ 'sr-qaaa-rs', 'sr-Qaaa-RS' ],
689
690 // # Tags that use extensions
691 [ 'en-us-u-islamcal', 'en-US-u-islamcal' ],
692 [ 'zh-cn-a-myext-x-private', 'zh-CN-a-myext-x-private' ],
693 [ 'en-a-myext-b-another', 'en-a-myext-b-another' ]
694
695 // # Invalid:
696 // de-419-DE
697 // a-DE
698 // ar-a-aaa-b-bbb-a-ccc
699 ];
700
701 QUnit.test( 'mw.language.bcp47', function ( assert ) {
702 bcp47Tests.forEach( function ( data ) {
703 var input = data[ 0 ],
704 expected = data[ 1 ];
705 assert.equal( mw.language.bcp47( input ), expected );
706 } );
707 } );
708 }( mediaWiki, jQuery ) );