@applicaster/zapp-react-native-utils 14.0.0-alpha.5333112458 → 14.0.0-alpha.5351122050

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 (58) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +0 -1
  2. package/actionsExecutor/ScreenActions.ts +20 -19
  3. package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
  4. package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
  5. package/analyticsUtils/__tests__/analyticsUtils.test.js +0 -11
  6. package/analyticsUtils/index.tsx +3 -4
  7. package/analyticsUtils/manager.ts +1 -1
  8. package/appUtils/HooksManager/Hook.ts +4 -4
  9. package/appUtils/HooksManager/index.ts +11 -1
  10. package/appUtils/accessibilityManager/index.ts +3 -12
  11. package/appUtils/contextKeysManager/contextResolver.ts +29 -1
  12. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +5 -0
  13. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  14. package/appUtils/focusManager/index.ios.ts +10 -0
  15. package/appUtils/focusManager/index.ts +82 -11
  16. package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
  17. package/appUtils/focusManagerAux/utils/index.ts +106 -3
  18. package/arrayUtils/index.ts +1 -1
  19. package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
  20. package/componentsUtils/index.ts +4 -1
  21. package/configurationUtils/__tests__/manifestKeyParser.test.ts +0 -1
  22. package/index.d.ts +0 -12
  23. package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
  24. package/navigationUtils/index.ts +7 -5
  25. package/package.json +2 -3
  26. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
  27. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  28. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  29. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +47 -90
  30. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +57 -37
  31. package/reactHooks/feed/index.ts +2 -0
  32. package/reactHooks/feed/useBatchLoading.ts +15 -8
  33. package/reactHooks/feed/useFeedLoader.tsx +39 -44
  34. package/reactHooks/feed/useLoadPipesDataDispatch.ts +57 -0
  35. package/reactHooks/feed/usePipesCacheReset.ts +2 -2
  36. package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
  37. package/reactHooks/layout/__tests__/index.test.tsx +3 -1
  38. package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
  39. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
  40. package/reactHooks/layout/useLayoutVersion.ts +5 -5
  41. package/reactHooks/navigation/index.ts +5 -7
  42. package/reactHooks/navigation/useScreenStateStore.ts +3 -3
  43. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
  44. package/reactHooks/state/index.ts +1 -1
  45. package/reactHooks/state/useHomeRiver.ts +4 -2
  46. package/reactHooks/state/useRivers.ts +7 -8
  47. package/screenPickerUtils/index.ts +7 -0
  48. package/storage/ScreenSingleValueProvider.ts +25 -22
  49. package/storage/ScreenStateMultiSelectProvider.ts +26 -23
  50. package/testUtils/index.tsx +7 -8
  51. package/time/BackgroundTimer.ts +1 -1
  52. package/utils/__tests__/find.test.ts +36 -0
  53. package/utils/__tests__/pathOr.test.ts +37 -0
  54. package/utils/__tests__/startsWith.test.ts +30 -0
  55. package/utils/find.ts +3 -0
  56. package/utils/index.ts +8 -0
  57. package/utils/pathOr.ts +5 -0
  58. package/utils/startsWith.ts +9 -0
@@ -120,7 +120,6 @@ const prepareDefaultActions = (actionExecutor) => {
120
120
  loadPipesData(dataSource, {
121
121
  silentRefresh: false,
122
122
  clearCache: true,
123
- callback: (_data) => {},
124
123
  riverId: context?.screenData?.id,
125
124
  })
126
125
  );
@@ -7,10 +7,11 @@ import { get } from "lodash";
7
7
  import { onMaxTagsReached } from "./StorageActions";
8
8
  import { ScreenMultiSelectProvider } from "../storage/ScreenStateMultiSelectProvider";
9
9
  import { ScreenSingleValueProvider } from "../storage/ScreenSingleValueProvider";
10
+ import { useScreenStateStore } from "../reactHooks/navigation/useScreenStateStore";
10
11
 
11
12
  export const screenSetVariable = async (
12
13
  screenRoute: string,
13
- screenStateStore: ScreenStateStore,
14
+ screenStateStore: ReturnType<typeof useScreenStateStore>,
14
15
  context: Record<string, any>,
15
16
  action: ActionType
16
17
  ): Promise<ActionResult> => {
@@ -34,11 +35,11 @@ export const screenSetVariable = async (
34
35
  ? get(entry, action.options.selector)
35
36
  : (entry.extensions?.tag ?? entry.id);
36
37
 
37
- const keyNamespace = action.options?.key;
38
+ const key = action.options?.key;
38
39
 
39
- if (!keyNamespace) {
40
- log_error("handleAction: screenSetVariable action missing key namespace", {
41
- keyNamespace,
40
+ if (!key) {
41
+ log_error("handleAction: screenSetVariable action missing argument 'key'", {
42
+ key,
42
43
  });
43
44
 
44
45
  return ActionResult.Error;
@@ -55,7 +56,7 @@ export const screenSetVariable = async (
55
56
 
56
57
  try {
57
58
  const singleValueProvider = ScreenSingleValueProvider.getProvider(
58
- keyNamespace,
59
+ key,
59
60
  screenRoute,
60
61
  screenStateStore
61
62
  );
@@ -63,19 +64,19 @@ export const screenSetVariable = async (
63
64
  const currentValue = await singleValueProvider.getValueAsync();
64
65
 
65
66
  log_info(
66
- `handleAction: screenSetVariable setting value: ${tag} for keyNamespace: ${keyNamespace}, previous value: ${currentValue}`
67
+ `handleAction: screenSetVariable setting value: ${tag} for key: ${key}, previous value: ${currentValue}`
67
68
  );
68
69
 
69
70
  await singleValueProvider.setValue(String(tag));
70
71
 
71
72
  log_info(
72
- `handleAction: screenSetVariable successfully set value: ${tag} for keyNamespace: ${keyNamespace}`
73
+ `handleAction: screenSetVariable successfully set value: ${tag} for key: ${key}`
73
74
  );
74
75
 
75
76
  return ActionResult.Success;
76
77
  } catch (error) {
77
78
  log_error("handleAction: screenSetVariable failed to set value", {
78
- keyNamespace,
79
+ key,
79
80
  tag,
80
81
  error,
81
82
  });
@@ -86,7 +87,7 @@ export const screenSetVariable = async (
86
87
 
87
88
  export const screenToggleFlag = async (
88
89
  screenRoute: string,
89
- screenStateStore: ScreenStateStore,
90
+ screenStateStore: ReturnType<typeof useScreenStateStore>,
90
91
  context: Record<string, any>,
91
92
  action: ActionType
92
93
  ) => {
@@ -110,11 +111,11 @@ export const screenToggleFlag = async (
110
111
  ? get(entry, action.options.selector)
111
112
  : (entry.extensions?.tag ?? entry.id);
112
113
 
113
- const keyNamespace = action.options?.key;
114
+ const key = action.options?.key;
114
115
 
115
- if (keyNamespace && tag) {
116
+ if (key && tag) {
116
117
  const multiSelectProvider = ScreenMultiSelectProvider.getProvider(
117
- keyNamespace,
118
+ key,
118
119
  screenRoute,
119
120
  screenStateStore
120
121
  );
@@ -125,7 +126,7 @@ export const screenToggleFlag = async (
125
126
  log_info(
126
127
  `handleAction: screenToggleFlag event will ${
127
128
  isTagInSelectedItems ? "remove" : "add"
128
- } tag: ${tag} for keyNamespace: ${keyNamespace}, current selectedItems: ${selectedItems}`
129
+ } tag: ${tag} for key: ${key}, current selectedItems: ${selectedItems}`
129
130
  );
130
131
 
131
132
  if (selectedItems.includes(tag)) {
@@ -142,7 +143,7 @@ export const screenToggleFlag = async (
142
143
  selectedItems,
143
144
  maxItems,
144
145
  tag,
145
- keyNamespace,
146
+ keyNamespace: key,
146
147
  });
147
148
 
148
149
  return ActionResult.Cancel;
@@ -151,10 +152,10 @@ export const screenToggleFlag = async (
151
152
  await multiSelectProvider.addItem(tag);
152
153
  }
153
154
  } else {
154
- log_error(
155
- "handleAction: screenToggleFlag event missing keyNamespace or tag",
156
- { keyNamespace, tag }
157
- );
155
+ log_error("handleAction: screenToggleFlag event missing key or tag", {
156
+ key,
157
+ tag,
158
+ });
158
159
 
159
160
  return ActionResult.Error;
160
161
  }
@@ -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;
@@ -8,17 +8,6 @@ jest.mock("@applicaster/zapp-react-native-utils/reactUtils", () => ({
8
8
  ),
9
9
  }));
10
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
-
22
11
  const mock_postAnalyticEvent = jest.fn();
23
12
  const mock_startAnalyticsTimedEvent = jest.fn();
24
13
  const mock_endAnalyticsTimedEvent = jest.fn();
@@ -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,
@@ -2,8 +2,8 @@ import { BehaviorSubject } from "rxjs";
2
2
  import { accessibilityManagerLogger as logger } from "./logger";
3
3
  import { TTSManager } from "../platform";
4
4
  import { BUTTON_ACCESSIBILITY_KEYS } from "./const";
5
- import { AccessibilityRole } from "react-native";
6
5
  import { toString } from "../../utils";
6
+ import { AccessibilityRole } from "react-native";
7
7
 
8
8
  export class AccessibilityManager {
9
9
  private static _instance: AccessibilityManager | null = null;
@@ -143,15 +143,12 @@ export class AccessibilityManager {
143
143
 
144
144
  if (!buttonConfig) {
145
145
  return {
146
- accessible: true,
147
146
  accessibilityLabel: buttonName,
148
147
  accessibilityHint: `Press button to perform action on ${buttonName}`,
149
148
  "aria-label": buttonName,
150
149
  "aria-description": `Press button to perform action on ${buttonName}`,
151
- accessibilityRole: "button" as AccessibilityRole,
150
+ accessibilityRole: "button",
152
151
  "aria-role": "button",
153
- role: "button",
154
- tabindex: 0,
155
152
  };
156
153
  }
157
154
 
@@ -165,29 +162,23 @@ export class AccessibilityManager {
165
162
  `Press button to perform action on ${buttonName}`;
166
163
 
167
164
  return {
168
- accessible: true,
169
165
  accessibilityLabel: label,
170
166
  accessibilityHint: hint,
171
167
  "aria-label": label,
172
168
  "aria-description": hint,
173
- accessibilityRole: "button" as AccessibilityRole,
169
+ accessibilityRole: "button",
174
170
  "aria-role": "button",
175
- role: "button",
176
- tabindex: 0,
177
171
  };
178
172
  }
179
173
 
180
174
  public getInputAccessibilityProps(inputName: string): AccessibilityProps {
181
175
  return {
182
- accessible: true,
183
176
  accessibilityLabel: inputName,
184
177
  accessibilityHint: `Enter text into ${inputName}`,
185
178
  "aria-label": inputName,
186
179
  "aria-description": `Enter text into ${inputName}`,
187
180
  accessibilityRole: "textbox" as AccessibilityRole,
188
181
  "aria-role": "textbox",
189
- role: "textbox",
190
- tabindex: 0,
191
182
  };
192
183
  }
193
184
 
@@ -1,5 +1,7 @@
1
1
  import { ContextKeysManager } from "./index";
2
2
  import * as R from "ramda";
3
+ import * as _ from "lodash";
4
+ import { useScreenStateStore } from "../../reactHooks/navigation/useScreenStateStore";
3
5
 
4
6
  export interface IResolver {
5
7
  resolve: (string) => Promise<string | number | object>;
@@ -25,11 +27,21 @@ export class EntryResolver implements IResolver {
25
27
  // TODO: Move to proper place
26
28
 
27
29
  export class ScreenStateResolver implements IResolver {
28
- constructor(private screenStateStore: ScreenStateStore) {}
30
+ constructor(
31
+ private screenStateStore: ReturnType<typeof useScreenStateStore>
32
+ ) {}
29
33
 
30
34
  async resolve(key: string) {
31
35
  const screenState = this.screenStateStore.getState().data;
32
36
 
37
+ if (!key || key.length === 0) {
38
+ return screenState;
39
+ }
40
+
41
+ if (key.includes(".")) {
42
+ return R.view(R.lensPath(key.split(".")), screenState);
43
+ }
44
+
33
45
  return screenState?.[key];
34
46
  }
35
47
  }
@@ -77,3 +89,19 @@ export const resolveObjectValues = async (
77
89
 
78
90
  return Object.fromEntries(resolvedEntries);
79
91
  };
92
+
93
+ export const extractAtValues = _.memoize((input: any): string[] => {
94
+ return _.flatMapDeep(input, (value: any) => {
95
+ if (_.isString(value)) {
96
+ const matches = value.match(/@\{([^}]*)\}/g);
97
+
98
+ return matches ? matches.map((match) => match.slice(2, -1)) : [];
99
+ }
100
+
101
+ if (_.isObject(value)) {
102
+ return extractAtValues(_.values(value));
103
+ }
104
+
105
+ return [];
106
+ });
107
+ });
@@ -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,6 +25,9 @@ exports[`focusManager should be defined 1`] = `
24
25
  "invokeHandler": [Function],
25
26
  "isCurrentFocusOnTheTopScreen": [Function],
26
27
  "isFocusDisabled": [Function],
28
+ "isFocusOn": [Function],
29
+ "isFocusOnContent": [Function],
30
+ "isFocusOnMenu": [Function],
27
31
  "isGroupItemFocused": [Function],
28
32
  "longPress": [Function],
29
33
  "moveFocus": [Function],
@@ -63,6 +67,7 @@ exports[`focusManagerIOS should be defined 1`] = `
63
67
  "getGroupRootById": [Function],
64
68
  "getPreferredFocusChild": [Function],
65
69
  "invokeHandler": [Function],
70
+ "isFocusOn": [Function],
66
71
  "isGroupItemFocused": [Function],
67
72
  "moveFocus": [Function],
68
73
  "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", () => {});
@@ -1,6 +1,7 @@
1
1
  import { NativeModules } from "react-native";
2
2
  import * as R from "ramda";
3
3
 
4
+ import { isCurrentFocusOn } from "../focusManagerAux/utils";
4
5
  import { Tree } from "./treeDataStructure/Tree";
5
6
  import { findFocusableNode } from "./treeDataStructure/Utils";
6
7
  import { subscriber } from "../../functionUtils";
@@ -391,6 +392,14 @@ export const focusManager = (function () {
391
392
  return node;
392
393
  }
393
394
 
395
+ function isFocusOn(id): boolean {
396
+ const currentFocusNode = focusableTree.findInTree(
397
+ getCurrentFocus()?.props?.id
398
+ );
399
+
400
+ return id && isCurrentFocusOn(id, currentFocusNode);
401
+ }
402
+
394
403
  return {
395
404
  on,
396
405
  invokeHandler,
@@ -412,5 +421,6 @@ export const focusManager = (function () {
412
421
  getGroupRootById,
413
422
  isGroupItemFocused,
414
423
  getPreferredFocusChild,
424
+ isFocusOn,
415
425
  };
416
426
  })();
@@ -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
  })();
@@ -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
  });