@applicaster/zapp-react-dom-app 15.0.0-rc.11 → 15.0.0-rc.110

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.
@@ -524,8 +524,16 @@ class InteractionManagerClass extends React.Component<Props, State> {
524
524
  }
525
525
 
526
526
  onKeyDown(event) {
527
- const { displayState, setDisplayState, navigator } = this.props;
527
+ const { displayState, setDisplayState, navigator, xray } = this.props;
528
528
  const { isDialogVisible } = confirmationDialogStore.getState();
529
+ const { isKeyboardVisible } = this.state;
530
+
531
+ if (
532
+ xray?.isPromptEnabled ||
533
+ (isKeyboardVisible && shouldUseOnScreenKeyboard())
534
+ ) {
535
+ return;
536
+ }
529
537
 
530
538
  if (keyCode(event).matches("Backspace")) {
531
539
  this.onBackspacePress(event);
@@ -259,6 +259,11 @@ export const getDeviceData = async () => {
259
259
  try {
260
260
  let deviceData = {};
261
261
 
262
+ const deviceDimensions = {
263
+ deviceWidth: window?.screen?.width,
264
+ deviceHeight: window?.screen?.height,
265
+ };
266
+
262
267
  if (isLgPlatform()) {
263
268
  const [webOSInfo, webOSConnectionInfo] = await Promise.all([
264
269
  getWebOSInfo(),
@@ -270,6 +275,7 @@ export const getDeviceData = async () => {
270
275
  ...webOSInfo,
271
276
  ...webOSConnectionInfo,
272
277
  device_type: getDeviceType(),
278
+ ...deviceDimensions,
273
279
  };
274
280
  } else if (isSamsungPlatform()) {
275
281
  const [tizenInfo, tizenConnectionInfo] = await Promise.all([
@@ -282,6 +288,7 @@ export const getDeviceData = async () => {
282
288
  ...tizenInfo,
283
289
  ...tizenConnectionInfo,
284
290
  device_type: getDeviceType(),
291
+ ...deviceDimensions,
285
292
  };
286
293
  } else if (isVizioPlatform()) {
287
294
  deviceData = {
@@ -289,8 +296,7 @@ export const getDeviceData = async () => {
289
296
  platform: "vizio",
290
297
  deviceMake: "Vizio",
291
298
  deviceType: "tv",
292
- deviceWidth: window.innerWidth,
293
- deviceHeight: window.innerHeight,
299
+ ...deviceDimensions,
294
300
  };
295
301
 
296
302
  if (hasVizioAPIs()) {
@@ -122,6 +122,10 @@ export const AnalyticsBridge = (function () {
122
122
  return;
123
123
  }
124
124
 
125
+ log_debug(`AnalyticsBridge postEvent called: ${eventName}`, {
126
+ payload,
127
+ });
128
+
125
129
  const updatedPayload = {
126
130
  ...prepareEventPayload(),
127
131
  ...payload,
@@ -109,7 +109,7 @@ describe("QuickBrickCommunicationModule Polyfill", () => {
109
109
  });
110
110
  });
111
111
 
112
- it("should fallback to default language if browser / device locale (en-UK) does not match list of supported languages", () => {
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("Missing required 'languages' from the build time data.", {
126
- buildTimeData,
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 = window.localStorage.getItem(
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 [browserLanguageCode, browserCountryLocale] = browserLocale.split("-");
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: browserLanguageCode,
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, while preserving the country locale from the browser.
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": "15.0.0-rc.11",
3
+ "version": "15.0.0-rc.110",
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": "15.0.0-rc.11",
26
- "@applicaster/zapp-react-native-bridge": "15.0.0-rc.11",
27
- "@applicaster/zapp-react-native-redux": "15.0.0-rc.11",
28
- "@applicaster/zapp-react-native-ui-components": "15.0.0-rc.11",
29
- "@applicaster/zapp-react-native-utils": "15.0.0-rc.11",
25
+ "@applicaster/zapp-react-dom-ui-components": "15.0.0-rc.110",
26
+ "@applicaster/zapp-react-native-bridge": "15.0.0-rc.110",
27
+ "@applicaster/zapp-react-native-redux": "15.0.0-rc.110",
28
+ "@applicaster/zapp-react-native-ui-components": "15.0.0-rc.110",
29
+ "@applicaster/zapp-react-native-utils": "15.0.0-rc.110",
30
30
  "abortcontroller-polyfill": "^1.7.5",
31
31
  "typeface-montserrat": "^0.0.54",
32
32
  "video.js": "7.14.3",