@applicaster/zapp-react-native-utils 14.0.0-alpha.5071825192 → 14.0.0-alpha.5243406255

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 (82) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +1 -1
  2. package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
  3. package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
  4. package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
  5. package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
  6. package/analyticsUtils/__tests__/analyticsUtils.test.js +14 -0
  7. package/analyticsUtils/events.ts +8 -0
  8. package/analyticsUtils/index.tsx +3 -4
  9. package/analyticsUtils/manager.ts +1 -1
  10. package/appUtils/HooksManager/Hook.ts +4 -4
  11. package/appUtils/HooksManager/index.ts +11 -1
  12. package/appUtils/accessibilityManager/index.ts +5 -5
  13. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +4 -0
  14. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  15. package/appUtils/focusManager/index.ts +81 -9
  16. package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
  17. package/appUtils/focusManagerAux/utils/index.ts +48 -2
  18. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +0 -15
  19. package/appUtils/playerManager/useChapterMarker.tsx +0 -1
  20. package/appUtils/playerManager/usePlayerControllerSetup.tsx +16 -0
  21. package/arrayUtils/__tests__/isEmptyArray.test.ts +63 -0
  22. package/arrayUtils/__tests__/isFilledArray.test.ts +1 -1
  23. package/arrayUtils/index.ts +8 -3
  24. package/audioPlayerUtils/__tests__/getArtworkImage.test.ts +144 -0
  25. package/audioPlayerUtils/__tests__/getBackgroundImage.test.ts +72 -0
  26. package/audioPlayerUtils/__tests__/getImageFromEntry.test.ts +110 -0
  27. package/audioPlayerUtils/assets/index.ts +2 -0
  28. package/audioPlayerUtils/index.ts +242 -0
  29. package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
  30. package/componentsUtils/index.ts +4 -1
  31. package/conf/player/__tests__/selectors.test.ts +34 -0
  32. package/conf/player/selectors.ts +10 -0
  33. package/configurationUtils/__tests__/configurationUtils.test.js +0 -31
  34. package/configurationUtils/__tests__/getMediaItems.test.ts +65 -0
  35. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +34 -0
  36. package/configurationUtils/__tests__/manifestKeyParser.test.ts +547 -0
  37. package/configurationUtils/index.ts +63 -34
  38. package/configurationUtils/manifestKeyParser.ts +57 -32
  39. package/focusManager/FocusManager.ts +26 -16
  40. package/focusManager/Tree.ts +25 -21
  41. package/focusManager/__tests__/FocusManager.test.ts +50 -8
  42. package/index.d.ts +0 -9
  43. package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
  44. package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
  45. package/manifestUtils/createConfig.js +4 -1
  46. package/manifestUtils/defaultManifestConfigurations/player.js +1239 -200
  47. package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
  48. package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
  49. package/navigationUtils/index.ts +6 -4
  50. package/package.json +2 -2
  51. package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
  52. package/playerUtils/__tests__/getPlayerActionButtons.test.ts +54 -0
  53. package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
  54. package/playerUtils/_internals/index.ts +1 -0
  55. package/playerUtils/_internals/utils.ts +31 -0
  56. package/playerUtils/configurationUtils.ts +0 -44
  57. package/playerUtils/getPlayerActionButtons.ts +17 -0
  58. package/playerUtils/index.ts +53 -0
  59. package/playerUtils/useValidatePlayerConfig.tsx +22 -19
  60. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
  61. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +39 -88
  62. package/reactHooks/feed/useBatchLoading.ts +11 -9
  63. package/reactHooks/feed/useFeedLoader.tsx +12 -8
  64. package/reactHooks/feed/usePipesCacheReset.ts +3 -3
  65. package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
  66. package/reactHooks/layout/__tests__/index.test.tsx +3 -1
  67. package/reactHooks/layout/isTablet/index.ts +12 -5
  68. package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
  69. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
  70. package/reactHooks/layout/useLayoutVersion.ts +5 -5
  71. package/reactHooks/navigation/index.ts +7 -5
  72. package/reactHooks/navigation/useIsScreenActive.ts +9 -5
  73. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
  74. package/reactHooks/screen/useScreenContext.ts +1 -1
  75. package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
  76. package/reactHooks/state/useRivers.ts +7 -8
  77. package/riverComponetsMeasurementProvider/index.tsx +1 -1
  78. package/services/js2native.ts +1 -0
  79. package/testUtils/index.tsx +7 -8
  80. package/time/BackgroundTimer.ts +6 -4
  81. package/utils/index.ts +17 -1
  82. package/playerUtils/configurationGenerator.ts +0 -2572
@@ -25,7 +25,7 @@ import {
25
25
  resolveObjectValues,
26
26
  } from "../appUtils/contextKeysManager/contextResolver";
27
27
  import { useNavigation } from "../reactHooks";
28
- import { get } from "lodash";
28
+ import { get } from "../utils";
29
29
 
30
30
  import {
31
31
  useContentTypes,
@@ -4,9 +4,12 @@ import {
4
4
  ANALYTICS_COMPONENT_EVENTS,
5
5
  ANALYTICS_CORE_EVENTS,
6
6
  ANALYTICS_ENTRY_EVENTS,
7
+ ANALYTICS_PREFERENCES_EVENTS,
7
8
  DOWNLOADS_EVENTS,
8
9
  } from "../events";
9
10
  import { isEmptyOrNil } from "../../cellUtils";
11
+ import { get } from "lodash";
12
+ import { StorageMultiSelectProvider } from "@applicaster/zapp-react-native-bridge/ZappStorage/StorageMultiSelectProvider";
10
13
 
11
14
  export enum OfflineItemState {
12
15
  notExist = "NOT_EXISTS",
@@ -102,6 +105,84 @@ export function eventForComponent(
102
105
  return analyticsProps;
103
106
  }
104
107
 
108
+ /**
109
+ * Checks if an item is currently selected in localStorage based on its actions
110
+ * @param item - The item to check
111
+ * @returns boolean indicating if the item is currently selected
112
+ */
113
+ function isItemPreviouslySelected(item: any): boolean {
114
+ const actions = item?.extensions?.tap_actions?.actions;
115
+
116
+ if (!actions) {
117
+ return false;
118
+ }
119
+
120
+ const localStorageAction = actions.find(
121
+ (action) => action?.type === "localStorageToggleFlag"
122
+ );
123
+
124
+ if (!localStorageAction?.options?.key) {
125
+ return false;
126
+ }
127
+
128
+ const keyNamespace = localStorageAction.options.key;
129
+
130
+ const tag = localStorageAction.options?.selector
131
+ ? get(item, localStorageAction.options.selector)
132
+ : (item.extensions?.tag ?? item.id);
133
+
134
+ if (!tag) {
135
+ return false;
136
+ }
137
+
138
+ try {
139
+ const multiSelectProvider =
140
+ StorageMultiSelectProvider.getProvider(keyNamespace);
141
+
142
+ const selectedItems = multiSelectProvider.getSelectedItems();
143
+
144
+ return selectedItems.includes(tag);
145
+ } catch (error) {
146
+ return false;
147
+ }
148
+ }
149
+
150
+ export function getLocalStorageSetPayload(extraProps) {
151
+ const { item } = extraProps;
152
+
153
+ const hasLocalStorageSetAction = item?.extensions?.tap_actions?.actions?.some(
154
+ (action) => action?.type === "localStorageSet"
155
+ );
156
+
157
+ if (!hasLocalStorageSetAction) {
158
+ return null;
159
+ }
160
+
161
+ return {
162
+ [ANALYTICS_PREFERENCES_EVENTS.ITEM_SELECTED_STATUS]: true,
163
+ };
164
+ }
165
+
166
+ export function getLocalStorageToggleFlagPayload(extraProps) {
167
+ const { item } = extraProps;
168
+
169
+ const hasLocalStorageToggleAction =
170
+ item?.extensions?.tap_actions?.actions?.some(
171
+ (action) => action?.type === "localStorageToggleFlag"
172
+ );
173
+
174
+ if (!hasLocalStorageToggleAction) {
175
+ return null;
176
+ }
177
+
178
+ const previouslySelected = isItemPreviouslySelected(item);
179
+
180
+ return {
181
+ [ANALYTICS_PREFERENCES_EVENTS.ITEM_SELECTED_STATUS]: !previouslySelected,
182
+ [ANALYTICS_PREFERENCES_EVENTS.PREVIOUS_SELECTED_STATE]: previouslySelected,
183
+ };
184
+ }
185
+
105
186
  export function playEventForType(item) {
106
187
  const itemType = item?.type && item.type?.value;
107
188
 
@@ -4,7 +4,7 @@ import { postAnalyticEvent } from "../manager";
4
4
  import { ANALYTICS_CORE_EVENTS } from "../events";
5
5
 
6
6
  type SendHeaderClickEventProps = {
7
- extraProps: ExtraProps;
7
+ extraProps: Record<string, any>;
8
8
  component?: ZappUIComponent;
9
9
  zappPipesData?: ZappPipesData;
10
10
  item?: ZappEntry;
@@ -1,10 +1,11 @@
1
+ /// <reference types="../../" />
1
2
  import { log_error, log_debug } from "../logger";
2
3
  import { replaceAnalyticsPropsNils } from "./helper";
3
4
  import { postAnalyticEvent } from "../manager";
4
5
 
5
6
  import { ANALYTICS_CORE_EVENTS } from "../events";
6
7
 
7
- declare type AnalyticsDefaultHelperProperties = {
8
+ type AnalyticsDefaultHelperProperties = {
8
9
  analyticsScreenData: AnalyticsScreenProperties;
9
10
  extraProps: any;
10
11
  props;
@@ -1,13 +1,13 @@
1
1
  import { log_error, log_debug } from "../logger";
2
-
3
- import { ANALYTICS_CORE_EVENTS } from "../events";
4
-
2
+ import { ANALYTICS_CORE_EVENTS, ACTION_TYPE } from "../events";
5
3
  import { postAnalyticEvent } from "../manager";
6
4
  import {
7
5
  replaceAnalyticsPropsNils,
8
6
  eventForEntry,
9
7
  eventForComponent,
10
8
  extensionsEvents,
9
+ getLocalStorageSetPayload,
10
+ getLocalStorageToggleFlagPayload,
11
11
  } from "./helper";
12
12
 
13
13
  declare type AnalyticsDefaultHelperProperties = {
@@ -26,7 +26,16 @@ export const sendOnClickEvent = ({
26
26
  const castedExtraProps: ExtraProps = extraProps;
27
27
  const componentData = component || extraProps.component;
28
28
  const data = zappPipesData || extraProps.zappPipesData;
29
- const eventName = ANALYTICS_CORE_EVENTS.TAP_CELL;
29
+
30
+ const actionCellPayload =
31
+ extraProps?.item?.type?.value === ACTION_TYPE
32
+ ? getLocalStorageSetPayload(extraProps) ||
33
+ getLocalStorageToggleFlagPayload(extraProps)
34
+ : null;
35
+
36
+ const eventName = actionCellPayload
37
+ ? ANALYTICS_CORE_EVENTS.TAP_SELECTABLE_CELL
38
+ : ANALYTICS_CORE_EVENTS.TAP_CELL;
30
39
 
31
40
  if (!analyticsScreenData) {
32
41
  log_error(
@@ -44,6 +53,7 @@ export const sendOnClickEvent = ({
44
53
  ...replaceAnalyticsPropsNils({
45
54
  ...analyticsScreenData,
46
55
  }),
56
+ ...actionCellPayload,
47
57
  };
48
58
 
49
59
  if (analyticsCustomProperties) {
@@ -3,8 +3,22 @@ import { ANALYTICS_CORE_EVENTS } from "../events";
3
3
 
4
4
  jest.mock("@applicaster/zapp-react-native-utils/reactUtils", () => ({
5
5
  isWeb: jest.fn(),
6
+ platformSelect: jest.fn(
7
+ (options) => options.android || options.ios || options.web
8
+ ),
6
9
  }));
7
10
 
11
+ jest.mock(
12
+ "@applicaster/zapp-react-native-bridge/ZappStorage/StorageMultiSelectProvider",
13
+ () => ({
14
+ StorageMultiSelectProvider: {
15
+ getProvider: jest.fn(() => ({
16
+ getSelectedItems: jest.fn(() => []),
17
+ })),
18
+ },
19
+ })
20
+ );
21
+
8
22
  const mock_postAnalyticEvent = jest.fn();
9
23
  const mock_startAnalyticsTimedEvent = jest.fn();
10
24
  const mock_endAnalyticsTimedEvent = jest.fn();
@@ -17,8 +17,11 @@ export const SCREEN_VIEW_EVENTS = {
17
17
  TIME_ON_SCREEN: "time_on_screen",
18
18
  };
19
19
 
20
+ export const ACTION_TYPE = "action";
21
+
20
22
  export const TAPPING_EVENTS = {
21
23
  TAP_CELL: "tap_cell",
24
+ TAP_SELECTABLE_CELL: "tap_selectable_cell",
22
25
  TAP_MENU: "tap_menu",
23
26
  TAP_NAVBAR_BACK_BUTTON: "tap_navbar_back_button",
24
27
  };
@@ -99,6 +102,11 @@ export const ANALYTICS_COMPONENT_EVENTS = {
99
102
  COMPONENT_SOURCE: "component_source",
100
103
  };
101
104
 
105
+ export const ANALYTICS_PREFERENCES_EVENTS = {
106
+ ITEM_SELECTED_STATUS: "item_selected_status",
107
+ PREVIOUS_SELECTED_STATE: "previous_selected_state",
108
+ };
109
+
102
110
  // ---------------- EVENTS ---------------------
103
111
  export const AD_EVENT = {
104
112
  ad_break_start: "player_ad_break_start",
@@ -1,4 +1,3 @@
1
- /// <reference types="@applicaster/zapp-react-native-utils" />
2
1
  import * as R from "ramda";
3
2
  import * as React from "react";
4
3
  import { isWeb } from "@applicaster/zapp-react-native-utils/reactUtils";
@@ -31,7 +30,7 @@ import { ANALYTICS_CORE_EVENTS } from "./events";
31
30
  import { noop } from "../functionUtils";
32
31
 
33
32
  type ComponentWithChildrenProps = {
34
- children: React.ReactChildren;
33
+ children: React.ReactElement;
35
34
  };
36
35
 
37
36
  export function sendSelectCellEvent(item, component, headerTitle, itemIndex) {
@@ -120,11 +119,11 @@ export function getAnalyticsFunctions({
120
119
  export const AnalyticsContext =
121
120
  React.createContext<GetAnalyticsFunctions>(noop);
122
121
 
123
- export function AnalyticsProvider(props: ComponentWithChildrenProps) {
122
+ export function AnalyticsProvider({ children }: ComponentWithChildrenProps) {
124
123
  return (
125
124
  // @ts-ignore - this is a valid context provider
126
125
  <AnalyticsContext.Provider value={getAnalyticsFunctions}>
127
- {props?.children}
126
+ {children}
128
127
  </AnalyticsContext.Provider>
129
128
  );
130
129
  }
@@ -1,7 +1,7 @@
1
1
  /* eslint-disable @typescript-eslint/no-use-before-define */
2
2
  import * as R from "ramda";
3
3
  import { NativeModules } from "react-native";
4
- import { ANALYTICS_CORE_EVENTS } from "@applicaster/zapp-react-native-utils/analyticsUtils/events";
4
+ import { ANALYTICS_CORE_EVENTS } from "./events";
5
5
 
6
6
  import { analyticsUtilsLogger } from "./logger";
7
7
 
@@ -65,10 +65,6 @@ export class Hook implements HookInterface {
65
65
  event: (typeof HOOKS_EVENTS)[keyof typeof HOOKS_EVENTS],
66
66
  ...args
67
67
  ) {
68
- if (this.state === hookState(HOOKS_EVENTS.CANCEL)) {
69
- return;
70
- }
71
-
72
68
  this.state = hookState(event);
73
69
  this.manager.subscriber.invokeHandler(event, ...args);
74
70
  }
@@ -198,4 +194,8 @@ export class Hook implements HookInterface {
198
194
  R.eqProps("weight", nextHook, this)
199
195
  );
200
196
  }
197
+
198
+ isCancelled(): boolean {
199
+ return this.state === hookState(HOOKS_EVENTS.CANCEL);
200
+ }
201
201
  }
@@ -255,7 +255,7 @@ export function HooksManager({
255
255
  * @param {Array<Hook>} restOfHooks to run
256
256
  * @returns {function} callback function
257
257
  */
258
- function hookCallback(hookPlugin, restOfHooks, initialPayload) {
258
+ function hookCallback(hookPlugin: Hook, restOfHooks: Hook[], initialPayload) {
259
259
  /**
260
260
  * callback invoked after a hook is executed
261
261
  * @param {object} options
@@ -273,6 +273,16 @@ export function HooksManager({
273
273
  }) {
274
274
  let callback = callbackArg;
275
275
 
276
+ if (hookPlugin.isCancelled()) {
277
+ logHookEvent(
278
+ hooksManagerLogger.info,
279
+ `hookCallback: hook was cancelled: ${hookPlugin["identifier"]}`,
280
+ {}
281
+ );
282
+
283
+ return;
284
+ }
285
+
276
286
  if (error) {
277
287
  logHookEvent(
278
288
  hooksManagerLogger.error,
@@ -1,9 +1,9 @@
1
1
  import { BehaviorSubject } from "rxjs";
2
2
  import { accessibilityManagerLogger as logger } from "./logger";
3
- import { TTSManager } from "../platform/platformUtils";
3
+ import { TTSManager } from "../platform";
4
4
  import { BUTTON_ACCESSIBILITY_KEYS } from "./const";
5
+ import { toString } from "../../utils";
5
6
  import { AccessibilityRole } from "react-native";
6
- import _ from "lodash";
7
7
 
8
8
  export class AccessibilityManager {
9
9
  private static _instance: AccessibilityManager | null = null;
@@ -137,7 +137,7 @@ export class AccessibilityManager {
137
137
  }
138
138
 
139
139
  public getButtonAccessibilityProps(name: string): AccessibilityProps {
140
- const buttonName = _.toString(name);
140
+ const buttonName = toString(name);
141
141
 
142
142
  const buttonConfig = BUTTON_ACCESSIBILITY_KEYS[buttonName];
143
143
 
@@ -147,7 +147,7 @@ export class AccessibilityManager {
147
147
  accessibilityHint: `Press button to perform action on ${buttonName}`,
148
148
  "aria-label": buttonName,
149
149
  "aria-description": `Press button to perform action on ${buttonName}`,
150
- accessibilityRole: "button" as AccessibilityRole,
150
+ accessibilityRole: "button",
151
151
  "aria-role": "button",
152
152
  };
153
153
  }
@@ -166,7 +166,7 @@ export class AccessibilityManager {
166
166
  accessibilityHint: hint,
167
167
  "aria-label": label,
168
168
  "aria-description": hint,
169
- accessibilityRole: "button" as AccessibilityRole,
169
+ accessibilityRole: "button",
170
170
  "aria-role": "button",
171
171
  };
172
172
  }
@@ -6,6 +6,7 @@ exports[`focusManager should be defined 1`] = `
6
6
  "disableFocus": [Function],
7
7
  "enableFocus": [Function],
8
8
  "findPreferredFocusChild": [Function],
9
+ "focusTopNavigation": [Function],
9
10
  "focusableTree": Tree {
10
11
  "loadingCounter": 0,
11
12
  "root": {
@@ -24,7 +25,10 @@ exports[`focusManager should be defined 1`] = `
24
25
  "invokeHandler": [Function],
25
26
  "isCurrentFocusOnTheTopScreen": [Function],
26
27
  "isFocusDisabled": [Function],
28
+ "isFocusOnContent": [Function],
29
+ "isFocusOnMenu": [Function],
27
30
  "isGroupItemFocused": [Function],
31
+ "isOnRootScreen": [Function],
28
32
  "longPress": [Function],
29
33
  "moveFocus": [Function],
30
34
  "on": [Function],
@@ -33,7 +33,7 @@ describe("focusManager", () => {
33
33
 
34
34
  expect(success).toBe(true);
35
35
  expect(mockSetFocus).toBeCalledTimes(1);
36
- expect(mockSetFocus).toBeCalledWith(null);
36
+ expect(mockSetFocus).toBeCalledWith(null, undefined);
37
37
  });
38
38
 
39
39
  describe("register", () => {});
@@ -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
+ isTabsScreen,
19
+ findSelectedTabId,
20
+ findSelectedMenuId,
21
+ isTabsMenuFocused,
22
+ isCurrentFocusOnContent,
23
+ isCurrentFocusOnMenu,
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,15 +281,61 @@ 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
+ function landFocusTo(id) {
304
+ if (id) {
305
+ // set focus on selected menu item
306
+ const direction = undefined;
307
+
308
+ const context: FocusManager.FocusContext = {
309
+ source: "back",
310
+ preserveScroll: true,
311
+ };
312
+
313
+ blur(direction);
314
+ setFocus(id, direction, context);
315
+ }
316
+ }
317
+
318
+ // Move focus to appropriate top navigation tab with context
319
+ function focusTopNavigation() {
320
+ // Store current focus for restoration
321
+ // this.storeFocusState();
322
+
323
+ if (isTabsScreen(focusableTree) && !isTabsMenuFocused(currentFocusNode)) {
324
+ const selectedTabId = findSelectedTabId(focusableTree);
325
+
326
+ console.log("debug_2", "FM - moveFocusToSelectedTab", { selectedTabId });
327
+
328
+ landFocusTo(selectedTabId);
329
+
330
+ return;
331
+ }
332
+
333
+ // Set focus with back button context
334
+ const selectedMenuItemId = findSelectedMenuId(focusableTree);
335
+
336
+ console.log("debug_2", "IM - moveFocusToTopMenu", { selectedMenuItemId });
337
+
338
+ landFocusTo(selectedMenuItemId);
278
339
  }
279
340
 
280
341
  /**
@@ -483,6 +544,12 @@ export const focusManager = (function () {
483
544
  return haveSameParentBeforeRoot(currentFocusNode, R.last(routes));
484
545
  }
485
546
 
547
+ function isOnRootScreen() {
548
+ const routes = R.pathOr([], ["root", "children"], focusableTree);
549
+
550
+ return routes.length <= 1;
551
+ }
552
+
486
553
  function recoverFocus() {
487
554
  if (!isCurrentFocusOnTheTopScreen()) {
488
555
  // We've failed to set focused node on the new screen => run focus recovery
@@ -576,5 +643,10 @@ export const focusManager = (function () {
576
643
  recoverFocus,
577
644
  isCurrentFocusOnTheTopScreen,
578
645
  findPreferredFocusChild,
646
+
647
+ focusTopNavigation,
648
+ isFocusOnContent,
649
+ isFocusOnMenu,
650
+ isOnRootScreen,
579
651
  };
580
652
  })();
@@ -142,7 +142,7 @@ export class Tree {
142
142
  this.hasGroupID(node)
143
143
  ? "Make sure that there are no id duplicates inside the " +
144
144
  existingNode.parent.id +
145
- " group."
145
+ " group. This can as well happen when the component is re-mounted"
146
146
  : ""
147
147
  }`,
148
148
  });
@@ -1,4 +1,4 @@
1
- import { isNotNil } from "@applicaster/zapp-react-native-utils/reactUtils/helpers";
1
+ import { isNil } from "@applicaster/zapp-react-native-utils/utils";
2
2
  import { find, last, pathOr, startsWith } from "ramda";
3
3
  import {
4
4
  QUICK_BRICK_CONTENT,
@@ -9,6 +9,8 @@ import {
9
9
  // run this check too often could lead to performance penalty on low-end devices
10
10
  const HOW_OFTEN_TO_CHECK_CONDITION = 300; // ms
11
11
 
12
+ const TABS_GROUP_ID = "PickerSelector.sp-river";
13
+
12
14
  type Props = {
13
15
  maxTimeout: number;
14
16
  conditionFn: () => boolean;
@@ -49,7 +51,7 @@ export const waitForActiveScreen = (currentRoute: string, focusableTree) => {
49
51
 
50
52
  const route = find((route) => route.id === currentRoute, routes);
51
53
 
52
- return isNotNil(route);
54
+ return !isNil(route);
53
55
  };
54
56
 
55
57
  return waitUntil({
@@ -99,3 +101,47 @@ export const waitForContent = (focusableTree) => {
99
101
  conditionFn: contentHasAnyChildren,
100
102
  });
101
103
  };
104
+
105
+ export function isTabsScreen(focusableTree) {
106
+ const tabsGroup = focusableTree.findInTree(TABS_GROUP_ID);
107
+
108
+ return !!tabsGroup;
109
+ }
110
+
111
+ export const findSelectedTabId = (focusableTree) => {
112
+ // FIXME - find elegant way how to get ID of selected tab
113
+ const tabsGroup = focusableTree.findInTree(TABS_GROUP_ID);
114
+
115
+ const selectedTabId = tabsGroup.children.find(
116
+ (child) => child.component.props.preferredFocus // ?? selected
117
+ )?.id;
118
+
119
+ return selectedTabId;
120
+ };
121
+
122
+ export const findSelectedMenuId = (focusableTree) => {
123
+ // Set focus with back button context
124
+ const navbar = getNavbarNode(focusableTree);
125
+
126
+ const selectedMenuItemId = find(
127
+ (child) => child.component.props.selected,
128
+ navbar.children
129
+ )?.id;
130
+
131
+ return selectedMenuItemId;
132
+ };
133
+
134
+ export const isTabsMenuFocused = (node) => {
135
+ // FIXME - find elegant way how to get ID of selected tab
136
+ return node.parent.id === TABS_GROUP_ID;
137
+ };
138
+
139
+ export const isCurrentFocusOnMenu = (node) => {
140
+ // FIXME
141
+ return node.parent.id.startsWith(QUICK_BRICK_NAVBAR);
142
+ };
143
+
144
+ export const isCurrentFocusOnContent = (node) => {
145
+ // FIXME
146
+ return !isCurrentFocusOnMenu(node);
147
+ };
@@ -18,13 +18,6 @@ export const { log_verbose, log_debug, log_info, log_error } = createLogger({
18
18
  parent: utilsLogger,
19
19
  });
20
20
 
21
- type ActionChapter = {
22
- type: string;
23
- options?: {
24
- title: string;
25
- };
26
- };
27
-
28
21
  type ChapterMarkerOriginal = {
29
22
  id: string;
30
23
  title: string;
@@ -33,14 +26,6 @@ type ChapterMarkerOriginal = {
33
26
  actions: ActionChapter[];
34
27
  };
35
28
 
36
- export type ChapterMarkerEvent = {
37
- id: string;
38
- title: string;
39
- start_time: number;
40
- end_time: number;
41
- actions: ActionChapter[];
42
- };
43
-
44
29
  export type TitleSummaryEvent = {
45
30
  title: string | number;
46
31
  summary: string | number;
@@ -1,4 +1,3 @@
1
- import { ChapterMarkerEvent } from "./OverlayObserver/OverlaysObserver";
2
1
  import { usePlayer } from "./usePlayer";
3
2
  import * as React from "react";
4
3
 
@@ -5,6 +5,9 @@ import { playerManager } from "./index";
5
5
  import { useValidatePlayerConfig } from "../../playerUtils/useValidatePlayerConfig";
6
6
  import { PlayerRole } from "./conts";
7
7
  import { isAppleTV } from "@applicaster/zapp-react-native-ui-components/Helpers/Platform";
8
+ import { TVSeekController } from "../../reactHooks/player/TVSeekControlller/TVSeekController";
9
+ import { KeyInputHandler } from "../keyInputHandler/KeyInputHandler";
10
+ import { isTV } from "../../reactUtils";
8
11
 
9
12
  // TODO: Rename to ControllerType
10
13
  export const usePlayerControllerSetup = ({
@@ -76,5 +79,18 @@ export const usePlayerControllerSetup = ({
76
79
  };
77
80
  }, [playerId, playerController]);
78
81
 
82
+ useEffect(() => {
83
+ if (!isTV()) {
84
+ return;
85
+ }
86
+
87
+ if (playerController) {
88
+ const seekController = new TVSeekController(playerController);
89
+ playerController.seekController = seekController;
90
+
91
+ return KeyInputHandler.getInstance().addListener(seekController);
92
+ }
93
+ }, [playerController]);
94
+
79
95
  return playerController;
80
96
  };