@applicaster/zapp-react-native-utils 14.0.0-alpha.6391068513 → 14.0.0-alpha.6893149866

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.
Files changed (45) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +0 -1
  2. package/actionsExecutor/ScreenActions.ts +20 -19
  3. package/analyticsUtils/__tests__/analyticsUtils.test.js +0 -11
  4. package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
  5. package/appUtils/accessibilityManager/const.ts +13 -0
  6. package/appUtils/accessibilityManager/hooks.ts +35 -1
  7. package/appUtils/accessibilityManager/index.ts +151 -30
  8. package/appUtils/accessibilityManager/utils.ts +24 -0
  9. package/appUtils/contextKeysManager/contextResolver.ts +29 -1
  10. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +5 -0
  11. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  12. package/appUtils/focusManager/index.ios.ts +10 -0
  13. package/appUtils/focusManager/index.ts +82 -11
  14. package/appUtils/focusManagerAux/utils/index.ts +106 -3
  15. package/appUtils/platform/platformUtils.ts +31 -1
  16. package/configurationUtils/__tests__/manifestKeyParser.test.ts +0 -1
  17. package/configurationUtils/index.ts +1 -1
  18. package/index.d.ts +1 -1
  19. package/manifestUtils/defaultManifestConfigurations/player.js +16 -2
  20. package/navigationUtils/index.ts +1 -1
  21. package/package.json +2 -3
  22. package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
  23. package/playerUtils/PlayerTTS/index.ts +1 -0
  24. package/playerUtils/usePlayerTTS.ts +21 -0
  25. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  26. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  27. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +8 -2
  28. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +57 -37
  29. package/reactHooks/feed/index.ts +2 -0
  30. package/reactHooks/feed/useBatchLoading.ts +14 -9
  31. package/reactHooks/feed/useFeedLoader.tsx +39 -50
  32. package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
  33. package/reactHooks/navigation/useScreenStateStore.ts +3 -3
  34. package/reactHooks/state/index.ts +1 -1
  35. package/reactHooks/state/useHomeRiver.ts +4 -2
  36. package/screenPickerUtils/index.ts +7 -0
  37. package/storage/ScreenSingleValueProvider.ts +25 -22
  38. package/storage/ScreenStateMultiSelectProvider.ts +26 -23
  39. package/utils/__tests__/find.test.ts +36 -0
  40. package/utils/__tests__/pathOr.test.ts +37 -0
  41. package/utils/__tests__/startsWith.test.ts +30 -0
  42. package/utils/find.ts +3 -0
  43. package/utils/index.ts +7 -0
  44. package/utils/pathOr.ts +5 -0
  45. package/utils/startsWith.ts +9 -0
@@ -14,6 +14,15 @@ import { subscriber } from "../../functionUtils";
14
14
  import { coreLogger } from "../../logger";
15
15
  import { ACTION } from "./utils/enums";
16
16
 
17
+ import {
18
+ findSelectedTabId,
19
+ findSelectedMenuId,
20
+ isTabsScreenContentFocused,
21
+ isCurrentFocusOnContent,
22
+ isCurrentFocusOnMenu,
23
+ isCurrentFocusOn,
24
+ } from "../focusManagerAux/utils";
25
+
17
26
  const logger = coreLogger.addSubsystem("focusManager");
18
27
 
19
28
  const isFocusEnabled = (focusableItem): boolean => {
@@ -100,7 +109,7 @@ export const focusManager = (function () {
100
109
  * @private
101
110
  * @param {Object} direction of the navigation which led to this action
102
111
  */
103
- function focus(direction) {
112
+ function focus(direction, context?: FocusManager.FocusContext) {
104
113
  const currentFocusable = getCurrentFocus();
105
114
 
106
115
  if (
@@ -108,7 +117,7 @@ export const focusManager = (function () {
108
117
  !currentFocusable.isGroup &&
109
118
  currentFocusable.isMounted()
110
119
  ) {
111
- currentFocusable.setFocus(direction);
120
+ currentFocusable.setFocus(direction, context);
112
121
  }
113
122
  }
114
123
 
@@ -205,7 +214,7 @@ export const focusManager = (function () {
205
214
  * @param {Array<string>} ids - An array of node IDs to update.
206
215
  * @param {boolean} setFocus - A flag indicating whether to set focus (true) or blur (false) on the nodes.
207
216
  */
208
- const updateNodeFocus = (ids, action) => {
217
+ const updateNodeFocus = (ids, action, context: FocusManager.FocusContext) => {
209
218
  if (!ids || ids.length === 0) {
210
219
  return; // Nothing to do
211
220
  }
@@ -222,11 +231,13 @@ export const focusManager = (function () {
222
231
 
223
232
  // Function to apply the action (focus or blur)
224
233
  const applyAction = (node) => {
234
+ const direction = undefined;
235
+
225
236
  if (node && node.component) {
226
237
  if (action === "focus") {
227
- node.component.setFocus();
238
+ node.component.setFocus(direction, context);
228
239
  } else if (action === "blur") {
229
- node.component.setBlur();
240
+ node.component.setBlur(direction, context);
230
241
  }
231
242
  }
232
243
  };
@@ -253,7 +264,11 @@ export const focusManager = (function () {
253
264
  * @param {Object} direction of the navigation, which led to this focus change
254
265
  * to another group or not. defaults to false
255
266
  */
256
- function setFocus(id: string, direction?: FocusManager.Web.Direction) {
267
+ function setFocus(
268
+ id: string,
269
+ direction?: FocusManager.Web.Direction,
270
+ context?: FocusManager.FocusContext
271
+ ) {
257
272
  if (focusDisabled) return false;
258
273
 
259
274
  // due to optimisiation it's recommanded to set currentFocusNode before setFocus
@@ -266,21 +281,65 @@ export const focusManager = (function () {
266
281
  );
267
282
 
268
283
  // Set focus on current node parents and blur on previous node parents
269
- updateNodeFocus(currentNodeParentsIDs, ACTION.FOCUS);
270
- updateNodeFocus(previousNodeParentsIDs, ACTION.BLUR);
284
+ updateNodeFocus(currentNodeParentsIDs, ACTION.FOCUS, context);
285
+ updateNodeFocus(previousNodeParentsIDs, ACTION.BLUR, context);
271
286
 
272
287
  currentFocusNode = focusableTree.findInTree(id);
273
288
  }
274
289
 
275
290
  setLastFocusOnParentNode(currentFocusNode);
276
291
 
277
- focus(direction);
292
+ focus(direction, context);
293
+ }
294
+
295
+ function isFocusOnContent() {
296
+ return isCurrentFocusOnContent(currentFocusNode);
297
+ }
298
+
299
+ function isFocusOnMenu() {
300
+ return isCurrentFocusOnMenu(currentFocusNode);
301
+ }
302
+
303
+ // Move focus to appropriate top navigation tab with context
304
+ function focusTopNavigation(isTabsScreen: boolean, item: ZappEntry) {
305
+ const landFocusTo = (id) => {
306
+ if (id) {
307
+ // set focus on selected menu item
308
+ const direction = undefined;
309
+
310
+ const context: FocusManager.FocusContext = {
311
+ source: "back",
312
+ preserveScroll: true,
313
+ };
314
+
315
+ logger.log({ message: "landFocusTo", data: { id } });
316
+
317
+ blur(direction);
318
+ setFocus(id, direction, context);
319
+ }
320
+ };
321
+
322
+ if (isTabsScreen && isTabsScreenContentFocused(currentFocusNode)) {
323
+ const selectedTabId = findSelectedTabId(item);
324
+
325
+ // Set focus with back button context to tabs-menu
326
+ landFocusTo(selectedTabId);
327
+
328
+ return;
329
+ }
330
+
331
+ const selectedMenuItemId = findSelectedMenuId(focusableTree);
332
+ // Set focus with back button context to top-menu
333
+ landFocusTo(selectedMenuItemId);
278
334
  }
279
335
 
280
336
  /**
281
337
  * sets the initial focus when the screen loads, or when focus is lost
282
338
  */
283
- function setInitialFocus(lastAddedParentNode?: any) {
339
+ function setInitialFocus(
340
+ lastAddedParentNode?: any,
341
+ context?: FocusManager.FocusContext
342
+ ) {
284
343
  const preferredFocus = findPriorityItem(
285
344
  lastAddedParentNode?.children || focusableTree.root.children
286
345
  );
@@ -326,7 +385,7 @@ export const focusManager = (function () {
326
385
  },
327
386
  });
328
387
 
329
- focusableItem && setFocus(focusCandidate.id, null);
388
+ focusableItem && setFocus(focusCandidate.id, null, context);
330
389
 
331
390
  return { success: true };
332
391
  }
@@ -546,6 +605,14 @@ export const focusManager = (function () {
546
605
  return preferredFocus[0];
547
606
  }
548
607
 
608
+ function isFocusOn(id): boolean {
609
+ return (
610
+ id &&
611
+ isCurrentFocusOnTheTopScreen() &&
612
+ isCurrentFocusOn(id, currentFocusNode)
613
+ );
614
+ }
615
+
549
616
  /**
550
617
  * this is the list of the functions available externally
551
618
  * when importing the focus manager
@@ -576,5 +643,9 @@ export const focusManager = (function () {
576
643
  recoverFocus,
577
644
  isCurrentFocusOnTheTopScreen,
578
645
  findPreferredFocusChild,
646
+ focusTopNavigation,
647
+ isFocusOnContent,
648
+ isFocusOnMenu,
649
+ isFocusOn,
579
650
  };
580
651
  })();
@@ -1,14 +1,31 @@
1
- import { isNotNil } from "@applicaster/zapp-react-native-utils/reactUtils/helpers";
2
- import { find, last, pathOr, startsWith } from "ramda";
1
+ import {
2
+ isNil,
3
+ last,
4
+ startsWith,
5
+ find,
6
+ pathOr,
7
+ } from "@applicaster/zapp-react-native-utils/utils";
8
+
3
9
  import {
4
10
  QUICK_BRICK_CONTENT,
5
11
  QUICK_BRICK_NAVBAR,
6
12
  } from "@applicaster/quick-brick-core/const";
7
13
 
14
+ import {
15
+ getFocusableId,
16
+ SCREEN_PICKER_CONTAINER,
17
+ } from "@applicaster/zapp-react-native-utils/screenPickerUtils";
18
+
8
19
  // run check each 300 ms
9
20
  // run this check too often could lead to performance penalty on low-end devices
10
21
  const HOW_OFTEN_TO_CHECK_CONDITION = 300; // ms
11
22
 
23
+ const isTopMenu = (node) => startsWith(QUICK_BRICK_NAVBAR, node?.id);
24
+ const isContent = (node) => startsWith(QUICK_BRICK_CONTENT, node?.id);
25
+ const isRoot = (node) => node?.id === "root";
26
+
27
+ const isScrenPicker = (node) => startsWith(SCREEN_PICKER_CONTAINER, node?.id);
28
+
12
29
  type Props = {
13
30
  maxTimeout: number;
14
31
  conditionFn: () => boolean;
@@ -49,7 +66,7 @@ export const waitForActiveScreen = (currentRoute: string, focusableTree) => {
49
66
 
50
67
  const route = find((route) => route.id === currentRoute, routes);
51
68
 
52
- return isNotNil(route);
69
+ return !isNil(route);
53
70
  };
54
71
 
55
72
  return waitUntil({
@@ -99,3 +116,89 @@ export const waitForContent = (focusableTree) => {
99
116
  conditionFn: contentHasAnyChildren,
100
117
  });
101
118
  };
119
+
120
+ export const findSelectedTabId = (item: ZappEntry): string => {
121
+ const selectedTabId = getFocusableId(item.id);
122
+
123
+ return selectedTabId;
124
+ };
125
+
126
+ export const findSelectedMenuId = (focusableTree) => {
127
+ // Set focus with back button context
128
+ const navbar = getNavbarNode(focusableTree);
129
+
130
+ const selectedMenuItemId = find(
131
+ (child) => child.component.props.selected,
132
+ navbar.children
133
+ )?.id;
134
+
135
+ return selectedMenuItemId;
136
+ };
137
+
138
+ export const isTabsScreenContentFocused = (node) => {
139
+ if (isRoot(node)) {
140
+ return false;
141
+ }
142
+
143
+ if (isTopMenu(node)) {
144
+ return false;
145
+ }
146
+
147
+ if (isContent(node)) {
148
+ return false;
149
+ }
150
+
151
+ if (isScrenPicker(node)) {
152
+ return true;
153
+ }
154
+
155
+ return isTabsScreenContentFocused(node.parent);
156
+ };
157
+
158
+ export const isCurrentFocusOnMenu = (node) => {
159
+ if (isRoot(node)) {
160
+ return false;
161
+ }
162
+
163
+ if (isTopMenu(node)) {
164
+ return true;
165
+ }
166
+
167
+ if (isContent(node)) {
168
+ return false;
169
+ }
170
+
171
+ return isCurrentFocusOnMenu(node.parent);
172
+ };
173
+
174
+ export const isCurrentFocusOnContent = (node) => {
175
+ if (isRoot(node)) {
176
+ return false;
177
+ }
178
+
179
+ if (isTopMenu(node)) {
180
+ return false;
181
+ }
182
+
183
+ if (isContent(node)) {
184
+ return true;
185
+ }
186
+
187
+ return isCurrentFocusOnContent(node.parent);
188
+ };
189
+
190
+ export const isCurrentFocusOn = (id, node) => {
191
+ if (!node) {
192
+ return false;
193
+ }
194
+
195
+ if (isRoot(node)) {
196
+ return false;
197
+ }
198
+
199
+ if (node?.id === id) {
200
+ return true;
201
+ }
202
+
203
+ return isCurrentFocusOn(id, node.parent);
204
+ };
@@ -10,6 +10,7 @@ import { PLATFORM_KEYS, PLATFORMS, ZappPlatform } from "./const";
10
10
  import { createLogger, utilsLogger } from "../../logger";
11
11
  import { getPlatform } from "../../reactUtils";
12
12
  import { appStore } from "@applicaster/zapp-react-native-redux/AppStore";
13
+ import { calculateReadingTime } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/utils";
13
14
 
14
15
  const { log_debug } = createLogger({
15
16
  category: "General",
@@ -212,9 +213,16 @@ export class TTSManager {
212
213
  }
213
214
 
214
215
  readText(text: string) {
216
+ this.ttsState$.next(true);
217
+
215
218
  if (isSamsungPlatform() && window.speechSynthesis) {
216
219
  const utterance = new SpeechSynthesisUtterance(text);
220
+
221
+ window.speechSynthesis.cancel(); // Cancel previous speech before speaking new text
217
222
  window.speechSynthesis.speak(utterance);
223
+
224
+ // Estimate reading time and set inactive when done
225
+ this.scheduleTTSComplete(text);
218
226
  }
219
227
 
220
228
  if (isLgPlatform() && window.webOS?.service) {
@@ -225,23 +233,45 @@ export class TTSManager {
225
233
  log_debug("There was a failure setting up webOS TTS service", {
226
234
  error,
227
235
  });
236
+
237
+ this.ttsState$.next(false);
228
238
  },
229
239
  onSuccess(response: any) {
230
240
  log_debug("webOS TTS service is configured successfully", {
231
241
  response,
232
242
  });
243
+
244
+ // Estimate reading time and set inactive when done
245
+ this.scheduleTTSComplete(text);
233
246
  },
234
247
  parameters: {
235
248
  text,
249
+ clear: true, // Clear any previous speech before speaking new text
236
250
  },
237
251
  });
238
252
  } catch (error) {
239
253
  log_debug("webOS TTS service error", { error });
254
+ this.ttsState$.next(false);
240
255
  }
241
256
  }
242
257
 
243
- if (!window.VIZIO?.Chromevox) return;
258
+ if (!window.VIZIO?.Chromevox) {
259
+ // For platforms without TTS, estimate reading time
260
+ this.scheduleTTSComplete(text);
261
+
262
+ return;
263
+ }
244
264
 
245
265
  window.VIZIO.Chromevox.play(text);
266
+ // Estimate reading time and set inactive when done
267
+ this.scheduleTTSComplete(text);
268
+ }
269
+
270
+ private scheduleTTSComplete(text: string) {
271
+ const readingTime = calculateReadingTime(text);
272
+
273
+ setTimeout(() => {
274
+ this.ttsState$.next(false);
275
+ }, readingTime);
246
276
  }
247
277
  }
@@ -1,6 +1,5 @@
1
1
  import { getAllSpecificStyles } from "../manifestKeyParser";
2
2
 
3
- // Mock the dependencies
4
3
  jest.mock("@applicaster/zapp-react-native-utils/reactUtils", () => ({
5
4
  platformSelect: jest.fn((platforms) => platforms.samsung_tv), // Default to samsung for tests
6
5
  }));
@@ -399,7 +399,7 @@ export const populateConfigurationValues =
399
399
  flattenAndPopulateFields(fields, configuration, skipDefaults)
400
400
  );
401
401
 
402
- export const getAccesabilityProps = (item: ZappEntry) => ({
402
+ export const getAccessibilityProps = (item: ZappEntry) => ({
403
403
  accessible: item?.extensions?.accessibility,
404
404
  accessibilityLabel: item?.extensions?.accessibility?.label || item?.title,
405
405
  accessibilityHint: item?.extensions?.accessibility?.hint,
package/index.d.ts CHANGED
@@ -69,7 +69,7 @@ declare type ExtraProps = ZappUIComponentProps & {
69
69
  };
70
70
 
71
71
  declare type WebConfirmationDialog = {
72
- message: string;
72
+ message?: string;
73
73
  confirmCompletion?: () => void;
74
74
  cancelCompletion?: () => void;
75
75
  };
@@ -208,7 +208,7 @@ function getPlayerConfiguration({ platform, version }) {
208
208
  {
209
209
  key: "accessibility_forward_label",
210
210
  label: "Accessibility forward label",
211
- initial_value: "Forward button",
211
+ initial_value: "Fast forward button",
212
212
  label_tooltip: "Label for forward button accessibility",
213
213
  type: "text_input",
214
214
  },
@@ -292,7 +292,7 @@ function getPlayerConfiguration({ platform, version }) {
292
292
  {
293
293
  key: "accessibility_back_label",
294
294
  label: "Accessibility back label",
295
- initial_value: "Back button",
295
+ initial_value: "Exit player button",
296
296
  label_tooltip: "Label for back button accessibility",
297
297
  type: "text_input",
298
298
  },
@@ -317,6 +317,20 @@ function getPlayerConfiguration({ platform, version }) {
317
317
  label_tooltip: "Hint for fullscreen button accessibility",
318
318
  type: "text_input",
319
319
  },
320
+ {
321
+ key: "accessibility_skip_intro_label",
322
+ label: "Accessibility skip intro label",
323
+ initial_value: "Skip intro - button",
324
+ label_tooltip: "Label for skip intro button accessibility",
325
+ type: "text_input",
326
+ },
327
+ {
328
+ key: "accessibility_skip_intro_hint",
329
+ label: "Accessibility skip intro hint",
330
+ initial_value: "Press to skip intro",
331
+ label_tooltip: "Hint for skip intro button accessibility",
332
+ type: "text_input",
333
+ },
320
334
  ],
321
335
  };
322
336
 
@@ -42,7 +42,7 @@ export function getNavigationType(
42
42
  R.unless(R.isNil, R.prop("navigation_type")),
43
43
  R.defaultTo(undefined),
44
44
  R.find(R.propEq("category", category))
45
- )(navigations);
45
+ )(navigations || []);
46
46
  }
47
47
 
48
48
  /**
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@applicaster/zapp-react-native-utils",
3
- "version": "14.0.0-alpha.6391068513",
3
+ "version": "14.0.0-alpha.6893149866",
4
4
  "description": "Applicaster Zapp React Native utilities package",
5
5
  "main": "index.js",
6
6
  "types": "index.d.ts",
@@ -27,7 +27,7 @@
27
27
  },
28
28
  "homepage": "https://github.com/applicaster/quickbrick#readme",
29
29
  "dependencies": {
30
- "@applicaster/applicaster-types": "14.0.0-alpha.6391068513",
30
+ "@applicaster/applicaster-types": "14.0.0-alpha.6893149866",
31
31
  "buffer": "^5.2.1",
32
32
  "camelize": "^1.0.0",
33
33
  "dayjs": "^1.11.10",
@@ -38,7 +38,6 @@
38
38
  "peerDependencies": {
39
39
  "@applicaster/zapp-pipes-v2-client": "*",
40
40
  "@react-native-community/netinfo": "*",
41
- "immer": "*",
42
41
  "react": "*",
43
42
  "react-native": "*",
44
43
  "uglify-js": "*",