@applicaster/zapp-react-dom-app 14.0.9-rc.0 → 14.0.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -109,7 +109,7 @@ describe("QuickBrickCommunicationModule Polyfill", () => {
|
|
|
109
109
|
});
|
|
110
110
|
});
|
|
111
111
|
|
|
112
|
-
it("should
|
|
112
|
+
it("should use partial match en language when browser / device locale (en-UK) does not exactly match list of supported languages", () => {
|
|
113
113
|
// localizations: ["en", "en-US", "fr", "fr-FR"], device_language: en, device_region: UK -> ui: en, language: en, region: UK
|
|
114
114
|
|
|
115
115
|
global.window.navigator.language = "en-UK";
|
|
@@ -118,8 +118,22 @@ describe("QuickBrickCommunicationModule Polyfill", () => {
|
|
|
118
118
|
expect(result).toEqual({
|
|
119
119
|
uiLanguage: "en",
|
|
120
120
|
languageCode: "en",
|
|
121
|
-
countryLocale: "",
|
|
122
|
-
languageLocale: "en",
|
|
121
|
+
countryLocale: "UK",
|
|
122
|
+
languageLocale: "en-UK",
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
it("should partial match fr language when browser / device locale (fr-CA) does not exact match list of supported languages", () => {
|
|
127
|
+
// localizations: ["en", "en-US", "fr", "fr-FR"], device_language: fr, device_region: CA -> ui: fr, language: fr, region: CA
|
|
128
|
+
|
|
129
|
+
global.window.navigator.language = "fr-CA";
|
|
130
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
131
|
+
|
|
132
|
+
expect(result).toEqual({
|
|
133
|
+
uiLanguage: "fr",
|
|
134
|
+
languageCode: "fr",
|
|
135
|
+
countryLocale: "CA",
|
|
136
|
+
languageLocale: "fr-CA",
|
|
123
137
|
});
|
|
124
138
|
});
|
|
125
139
|
|
|
@@ -190,6 +204,312 @@ describe("QuickBrickCommunicationModule Polyfill", () => {
|
|
|
190
204
|
languageLocale: "fr-FR",
|
|
191
205
|
});
|
|
192
206
|
});
|
|
207
|
+
|
|
208
|
+
it("should not attempt to use user-selected language if it is improperly stored as undefined, null, or empty string", () => {
|
|
209
|
+
mockGetAppData.mockReturnValue(
|
|
210
|
+
getMockAppData({
|
|
211
|
+
languages: ["en", "es", "fr", "en-UK", "en-GB", "ar"],
|
|
212
|
+
})
|
|
213
|
+
);
|
|
214
|
+
|
|
215
|
+
global.window.navigator.language = "fr-CA";
|
|
216
|
+
|
|
217
|
+
// Test with undefined
|
|
218
|
+
window.localStorage.getItem.mockReturnValue("undefined");
|
|
219
|
+
let result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
220
|
+
|
|
221
|
+
expect(result).toEqual({
|
|
222
|
+
uiLanguage: "fr",
|
|
223
|
+
languageCode: "fr",
|
|
224
|
+
countryLocale: "CA",
|
|
225
|
+
languageLocale: "fr-CA",
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
// Test with null
|
|
229
|
+
window.localStorage.getItem.mockReturnValue("null");
|
|
230
|
+
result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
231
|
+
|
|
232
|
+
expect(result).toEqual({
|
|
233
|
+
uiLanguage: "fr",
|
|
234
|
+
languageCode: "fr",
|
|
235
|
+
countryLocale: "CA",
|
|
236
|
+
languageLocale: "fr-CA",
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
// Test with empty string
|
|
240
|
+
window.localStorage.getItem.mockReturnValue("");
|
|
241
|
+
result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
242
|
+
|
|
243
|
+
expect(result).toEqual({
|
|
244
|
+
uiLanguage: "fr",
|
|
245
|
+
languageCode: "fr",
|
|
246
|
+
countryLocale: "CA",
|
|
247
|
+
languageLocale: "fr-CA",
|
|
248
|
+
});
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
describe("user-selected language scenarios", () => {
|
|
252
|
+
it("should use exact match user-selected language when available, ignoring browser locale", () => {
|
|
253
|
+
window.localStorage.getItem.mockReturnValue("fr-FR");
|
|
254
|
+
global.window.navigator.language = "en-US";
|
|
255
|
+
|
|
256
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
257
|
+
|
|
258
|
+
expect(result).toEqual({
|
|
259
|
+
uiLanguage: "fr-FR",
|
|
260
|
+
languageCode: "fr",
|
|
261
|
+
countryLocale: "FR",
|
|
262
|
+
languageLocale: "fr-FR",
|
|
263
|
+
});
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
it("should use base language user-selected when available (single language code)", () => {
|
|
267
|
+
window.localStorage.getItem.mockReturnValue("en");
|
|
268
|
+
global.window.navigator.language = "fr-FR";
|
|
269
|
+
|
|
270
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
271
|
+
|
|
272
|
+
expect(result).toEqual({
|
|
273
|
+
uiLanguage: "en",
|
|
274
|
+
languageCode: "en",
|
|
275
|
+
countryLocale: "",
|
|
276
|
+
languageLocale: "en",
|
|
277
|
+
});
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
it("should use base language fr user-selected when available", () => {
|
|
281
|
+
window.localStorage.getItem.mockReturnValue("fr");
|
|
282
|
+
global.window.navigator.language = "en-US";
|
|
283
|
+
|
|
284
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
285
|
+
|
|
286
|
+
expect(result).toEqual({
|
|
287
|
+
uiLanguage: "fr",
|
|
288
|
+
languageCode: "fr",
|
|
289
|
+
countryLocale: "",
|
|
290
|
+
languageLocale: "fr",
|
|
291
|
+
});
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
it("should fallback to browser exact match when user-selected language not available", () => {
|
|
295
|
+
window.localStorage.getItem.mockReturnValue("de-DE");
|
|
296
|
+
global.window.navigator.language = "fr-FR";
|
|
297
|
+
|
|
298
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
299
|
+
|
|
300
|
+
expect(result).toEqual({
|
|
301
|
+
uiLanguage: "fr-FR",
|
|
302
|
+
languageCode: "fr",
|
|
303
|
+
countryLocale: "FR",
|
|
304
|
+
languageLocale: "fr-FR",
|
|
305
|
+
});
|
|
306
|
+
});
|
|
307
|
+
|
|
308
|
+
it("should fallback to browser partial match when user-selected language not available", () => {
|
|
309
|
+
window.localStorage.getItem.mockReturnValue("de-DE");
|
|
310
|
+
global.window.navigator.language = "fr-CA";
|
|
311
|
+
|
|
312
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
313
|
+
|
|
314
|
+
expect(result).toEqual({
|
|
315
|
+
uiLanguage: "fr",
|
|
316
|
+
languageCode: "fr",
|
|
317
|
+
countryLocale: "CA",
|
|
318
|
+
languageLocale: "fr-CA",
|
|
319
|
+
});
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
it("should fallback to default language when both user-selected and browser locales not available", () => {
|
|
323
|
+
window.localStorage.getItem.mockReturnValue("de-DE");
|
|
324
|
+
global.window.navigator.language = "ja-JP";
|
|
325
|
+
|
|
326
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
327
|
+
|
|
328
|
+
expect(result).toEqual({
|
|
329
|
+
uiLanguage: "en",
|
|
330
|
+
languageCode: "en",
|
|
331
|
+
countryLocale: "",
|
|
332
|
+
languageLocale: "en",
|
|
333
|
+
});
|
|
334
|
+
});
|
|
335
|
+
});
|
|
336
|
+
|
|
337
|
+
describe("browser/device locale scenarios", () => {
|
|
338
|
+
beforeEach(() => {
|
|
339
|
+
window.localStorage.getItem.mockReturnValue(null);
|
|
340
|
+
});
|
|
341
|
+
|
|
342
|
+
it("should use exact match browser locale when no user selection", () => {
|
|
343
|
+
global.window.navigator.language = "en-US";
|
|
344
|
+
|
|
345
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
346
|
+
|
|
347
|
+
expect(result).toEqual({
|
|
348
|
+
uiLanguage: "en-US",
|
|
349
|
+
languageCode: "en",
|
|
350
|
+
countryLocale: "US",
|
|
351
|
+
languageLocale: "en-US",
|
|
352
|
+
});
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
it("should use partial match browser locale when exact match not available", () => {
|
|
356
|
+
global.window.navigator.language = "en-AU";
|
|
357
|
+
|
|
358
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
359
|
+
|
|
360
|
+
expect(result).toEqual({
|
|
361
|
+
uiLanguage: "en",
|
|
362
|
+
languageCode: "en",
|
|
363
|
+
countryLocale: "AU",
|
|
364
|
+
languageLocale: "en-AU",
|
|
365
|
+
});
|
|
366
|
+
});
|
|
367
|
+
|
|
368
|
+
it("should fallback to default language when browser locale not available at all", () => {
|
|
369
|
+
global.window.navigator.language = "de-DE";
|
|
370
|
+
|
|
371
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
372
|
+
|
|
373
|
+
expect(result).toEqual({
|
|
374
|
+
uiLanguage: "en",
|
|
375
|
+
languageCode: "en",
|
|
376
|
+
countryLocale: "",
|
|
377
|
+
languageLocale: "en",
|
|
378
|
+
});
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
it("should handle browser locale without country code (base language only)", () => {
|
|
382
|
+
global.window.navigator.language = "fr";
|
|
383
|
+
|
|
384
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
385
|
+
|
|
386
|
+
expect(result).toEqual({
|
|
387
|
+
uiLanguage: "fr",
|
|
388
|
+
languageCode: "fr",
|
|
389
|
+
countryLocale: "",
|
|
390
|
+
languageLocale: "fr",
|
|
391
|
+
});
|
|
392
|
+
});
|
|
393
|
+
|
|
394
|
+
it("should handle browser locale without country code when only regional variants available", () => {
|
|
395
|
+
mockGetAppData.mockReturnValue(
|
|
396
|
+
getMockAppData({ languages: ["en-US", "en-GB", "fr-FR", "fr-CA"] })
|
|
397
|
+
);
|
|
398
|
+
|
|
399
|
+
jest.resetModules();
|
|
400
|
+
QuickBrickModule = require("../index");
|
|
401
|
+
|
|
402
|
+
global.window.navigator.language = "fr";
|
|
403
|
+
|
|
404
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
405
|
+
|
|
406
|
+
expect(result).toEqual({
|
|
407
|
+
uiLanguage: "fr-FR",
|
|
408
|
+
languageCode: "fr",
|
|
409
|
+
countryLocale: "",
|
|
410
|
+
languageLocale: "fr",
|
|
411
|
+
});
|
|
412
|
+
});
|
|
413
|
+
});
|
|
414
|
+
|
|
415
|
+
describe("edge cases", () => {
|
|
416
|
+
beforeEach(() => {
|
|
417
|
+
window.localStorage.getItem.mockReturnValue(null);
|
|
418
|
+
});
|
|
419
|
+
|
|
420
|
+
it("should fallback to locale when browser language is undefined", () => {
|
|
421
|
+
global.window.navigator.language = undefined;
|
|
422
|
+
|
|
423
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
424
|
+
|
|
425
|
+
expect(result).toEqual({
|
|
426
|
+
uiLanguage: "en",
|
|
427
|
+
languageCode: "en",
|
|
428
|
+
countryLocale: "",
|
|
429
|
+
languageLocale: "en",
|
|
430
|
+
});
|
|
431
|
+
});
|
|
432
|
+
|
|
433
|
+
it("should prefer exact match over partial match when both are available", () => {
|
|
434
|
+
mockGetAppData.mockReturnValue(
|
|
435
|
+
getMockAppData({ languages: ["en", "en-US", "en-GB", "fr"] })
|
|
436
|
+
);
|
|
437
|
+
|
|
438
|
+
jest.resetModules();
|
|
439
|
+
QuickBrickModule = require("../index");
|
|
440
|
+
|
|
441
|
+
global.window.navigator.language = "en-GB";
|
|
442
|
+
|
|
443
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
444
|
+
|
|
445
|
+
expect(result).toEqual({
|
|
446
|
+
uiLanguage: "en-GB",
|
|
447
|
+
languageCode: "en",
|
|
448
|
+
countryLocale: "GB",
|
|
449
|
+
languageLocale: "en-GB",
|
|
450
|
+
});
|
|
451
|
+
});
|
|
452
|
+
|
|
453
|
+
it("should use first partial match when multiple base languages could match", () => {
|
|
454
|
+
mockGetAppData.mockReturnValue(
|
|
455
|
+
getMockAppData({ languages: ["en-US", "en-GB", "fr-FR", "fr-CA"] })
|
|
456
|
+
);
|
|
457
|
+
|
|
458
|
+
jest.resetModules();
|
|
459
|
+
QuickBrickModule = require("../index");
|
|
460
|
+
|
|
461
|
+
global.window.navigator.language = "en-AU";
|
|
462
|
+
|
|
463
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
464
|
+
|
|
465
|
+
expect(result).toEqual({
|
|
466
|
+
uiLanguage: "en-US",
|
|
467
|
+
languageCode: "en",
|
|
468
|
+
countryLocale: "AU",
|
|
469
|
+
languageLocale: "en-AU",
|
|
470
|
+
});
|
|
471
|
+
});
|
|
472
|
+
|
|
473
|
+
it("should handle complex regional variants like zh-Hans-CN", () => {
|
|
474
|
+
mockGetAppData.mockReturnValue(
|
|
475
|
+
getMockAppData({ languages: ["en", "zh-CN", "zh-TW"] })
|
|
476
|
+
);
|
|
477
|
+
|
|
478
|
+
jest.resetModules();
|
|
479
|
+
QuickBrickModule = require("../index");
|
|
480
|
+
|
|
481
|
+
global.window.navigator.language = "zh-CN";
|
|
482
|
+
|
|
483
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
484
|
+
|
|
485
|
+
expect(result).toEqual({
|
|
486
|
+
uiLanguage: "zh-CN",
|
|
487
|
+
languageCode: "zh",
|
|
488
|
+
countryLocale: "CN",
|
|
489
|
+
languageLocale: "zh-CN",
|
|
490
|
+
});
|
|
491
|
+
});
|
|
492
|
+
|
|
493
|
+
it("should fallback to first language when supported languages list is populated but browser locale cannot be parsed", () => {
|
|
494
|
+
mockGetAppData.mockReturnValue(
|
|
495
|
+
getMockAppData({ languages: ["ar", "he", "fa"] })
|
|
496
|
+
);
|
|
497
|
+
|
|
498
|
+
jest.resetModules();
|
|
499
|
+
QuickBrickModule = require("../index");
|
|
500
|
+
|
|
501
|
+
global.window.navigator.language = "en-US";
|
|
502
|
+
|
|
503
|
+
const result = QuickBrickModule.getMatchingLanguageFromBuildData();
|
|
504
|
+
|
|
505
|
+
expect(result).toEqual({
|
|
506
|
+
uiLanguage: "ar",
|
|
507
|
+
languageCode: "ar",
|
|
508
|
+
countryLocale: "",
|
|
509
|
+
languageLocale: "ar",
|
|
510
|
+
});
|
|
511
|
+
});
|
|
512
|
+
});
|
|
193
513
|
});
|
|
194
514
|
|
|
195
515
|
describe("setAppLanguage", () => {
|
|
@@ -13,6 +13,27 @@ const { log_error, log_debug, log_warning } = createLogger({
|
|
|
13
13
|
|
|
14
14
|
const QuickBrickEventEmitter = {};
|
|
15
15
|
|
|
16
|
+
const getUserSelectedLanguage = () => {
|
|
17
|
+
const userSelectedLanguage = window.localStorage.getItem(
|
|
18
|
+
"applicaster.v2_::_uiLanguage"
|
|
19
|
+
);
|
|
20
|
+
|
|
21
|
+
if (
|
|
22
|
+
userSelectedLanguage === "null" ||
|
|
23
|
+
userSelectedLanguage === "undefined" ||
|
|
24
|
+
userSelectedLanguage === ""
|
|
25
|
+
) {
|
|
26
|
+
log_warning(
|
|
27
|
+
"Invalid user selected language found in localStorage:",
|
|
28
|
+
userSelectedLanguage
|
|
29
|
+
);
|
|
30
|
+
|
|
31
|
+
return null;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
return userSelectedLanguage;
|
|
35
|
+
};
|
|
36
|
+
|
|
16
37
|
export const QuickBrickEvents = (function (eventEmitter) {
|
|
17
38
|
let listeners = [];
|
|
18
39
|
|
|
@@ -122,9 +143,12 @@ export const getMatchingLanguageFromBuildData = () => {
|
|
|
122
143
|
const browserLocale = getBrowserLocale();
|
|
123
144
|
|
|
124
145
|
if (!Array.isArray(languages)) {
|
|
125
|
-
log_error(
|
|
126
|
-
|
|
127
|
-
|
|
146
|
+
log_error(
|
|
147
|
+
"Missing required 'languages' in build time data; using locale as UI language.",
|
|
148
|
+
{
|
|
149
|
+
buildTimeData,
|
|
150
|
+
}
|
|
151
|
+
);
|
|
128
152
|
|
|
129
153
|
return {
|
|
130
154
|
uiLanguage: locale,
|
|
@@ -134,9 +158,7 @@ export const getMatchingLanguageFromBuildData = () => {
|
|
|
134
158
|
};
|
|
135
159
|
}
|
|
136
160
|
|
|
137
|
-
const userSelectedLanguage =
|
|
138
|
-
"applicaster.v2_::_uiLanguage"
|
|
139
|
-
);
|
|
161
|
+
const userSelectedLanguage = getUserSelectedLanguage();
|
|
140
162
|
|
|
141
163
|
if (userSelectedLanguage) {
|
|
142
164
|
if (languages.find((lang) => lang === userSelectedLanguage)) {
|
|
@@ -152,7 +174,6 @@ export const getMatchingLanguageFromBuildData = () => {
|
|
|
152
174
|
|
|
153
175
|
log_debug(
|
|
154
176
|
"User selected language is not present in available localizations, selecting language using browser or device locale",
|
|
155
|
-
|
|
156
177
|
{ userSelectedLanguage, languages }
|
|
157
178
|
);
|
|
158
179
|
}
|
|
@@ -172,21 +193,35 @@ export const getMatchingLanguageFromBuildData = () => {
|
|
|
172
193
|
}
|
|
173
194
|
|
|
174
195
|
// Exact match
|
|
175
|
-
const [
|
|
196
|
+
const [browserBaseLanguage, browserCountryLocale] = browserLocale.split("-");
|
|
176
197
|
const browserLocaleMatch = languages.find((lang) => lang === browserLocale);
|
|
177
198
|
|
|
178
199
|
if (browserLocaleMatch) {
|
|
179
200
|
return {
|
|
180
201
|
uiLanguage: browserLocaleMatch,
|
|
181
|
-
languageCode:
|
|
202
|
+
languageCode: browserBaseLanguage,
|
|
182
203
|
countryLocale: browserCountryLocale || "",
|
|
183
204
|
languageLocale: browserLocaleMatch,
|
|
184
205
|
};
|
|
185
206
|
}
|
|
186
207
|
|
|
208
|
+
// Partial match: try to match the base language (e.g., "fr" from "fr-CA")
|
|
209
|
+
const partialMatch = languages.find((language) =>
|
|
210
|
+
language.startsWith(browserBaseLanguage)
|
|
211
|
+
);
|
|
212
|
+
|
|
213
|
+
if (partialMatch) {
|
|
214
|
+
return {
|
|
215
|
+
uiLanguage: partialMatch,
|
|
216
|
+
languageCode: browserBaseLanguage,
|
|
217
|
+
countryLocale: browserCountryLocale || "",
|
|
218
|
+
languageLocale: browserLocale,
|
|
219
|
+
};
|
|
220
|
+
}
|
|
221
|
+
|
|
187
222
|
// If there is no user-selected language, or the user-selected language is not found in the list of supported languages,
|
|
188
223
|
// and the browser locale is also not found in the list of supported languages,
|
|
189
|
-
// then fallback to the default language from the list of supported languages
|
|
224
|
+
// then fallback to the first (default) language from the list of supported languages.
|
|
190
225
|
const defaultLanguage = languages[0];
|
|
191
226
|
|
|
192
227
|
if (defaultLanguage) {
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@applicaster/zapp-react-dom-app",
|
|
3
|
-
"version": "14.0.9
|
|
3
|
+
"version": "14.0.9",
|
|
4
4
|
"description": "Zapp App Component for Applicaster's Quick Brick React Native App",
|
|
5
5
|
"main": "index.js",
|
|
6
6
|
"scripts": {
|
|
@@ -22,11 +22,11 @@
|
|
|
22
22
|
},
|
|
23
23
|
"homepage": "https://github.com/applicaster/zapp-react-dom-app#readme",
|
|
24
24
|
"dependencies": {
|
|
25
|
-
"@applicaster/zapp-react-dom-ui-components": "14.0.9
|
|
26
|
-
"@applicaster/zapp-react-native-bridge": "14.0.9
|
|
27
|
-
"@applicaster/zapp-react-native-redux": "14.0.9
|
|
28
|
-
"@applicaster/zapp-react-native-ui-components": "14.0.9
|
|
29
|
-
"@applicaster/zapp-react-native-utils": "14.0.9
|
|
25
|
+
"@applicaster/zapp-react-dom-ui-components": "14.0.9",
|
|
26
|
+
"@applicaster/zapp-react-native-bridge": "14.0.9",
|
|
27
|
+
"@applicaster/zapp-react-native-redux": "14.0.9",
|
|
28
|
+
"@applicaster/zapp-react-native-ui-components": "14.0.9",
|
|
29
|
+
"@applicaster/zapp-react-native-utils": "14.0.9",
|
|
30
30
|
"abortcontroller-polyfill": "^1.7.5",
|
|
31
31
|
"typeface-montserrat": "^0.0.54",
|
|
32
32
|
"video.js": "7.14.3",
|