@applicaster/zapp-react-native-utils 14.0.0-alpha.1661204539 → 14.0.0-alpha.1740013076

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 (115) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +11 -6
  2. package/actionsExecutor/ScreenActions.ts +91 -17
  3. package/actionsExecutor/feedDecorator.ts +171 -0
  4. package/actionsExecutor/screenResolver.ts +6 -3
  5. package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
  6. package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
  7. package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
  8. package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
  9. package/analyticsUtils/__tests__/analyticsUtils.test.js +3 -0
  10. package/analyticsUtils/events.ts +8 -0
  11. package/analyticsUtils/index.tsx +3 -4
  12. package/analyticsUtils/manager.ts +1 -1
  13. package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
  14. package/appUtils/HooksManager/Hook.ts +4 -4
  15. package/appUtils/HooksManager/index.ts +11 -1
  16. package/appUtils/accessibilityManager/const.ts +13 -0
  17. package/appUtils/accessibilityManager/hooks.ts +35 -1
  18. package/appUtils/accessibilityManager/index.ts +154 -30
  19. package/appUtils/accessibilityManager/utils.ts +24 -0
  20. package/appUtils/contextKeysManager/contextResolver.ts +30 -3
  21. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +5 -0
  22. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  23. package/appUtils/focusManager/index.ios.ts +10 -0
  24. package/appUtils/focusManager/index.ts +82 -11
  25. package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
  26. package/appUtils/focusManagerAux/utils/index.ts +106 -3
  27. package/appUtils/platform/platformUtils.ts +31 -1
  28. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +0 -15
  29. package/appUtils/playerManager/useChapterMarker.tsx +0 -1
  30. package/appUtils/playerManager/usePlayerControllerSetup.tsx +16 -0
  31. package/arrayUtils/__tests__/isEmptyArray.test.ts +63 -0
  32. package/arrayUtils/__tests__/isFilledArray.test.ts +1 -1
  33. package/arrayUtils/index.ts +8 -3
  34. package/audioPlayerUtils/__tests__/getArtworkImage.test.ts +144 -0
  35. package/audioPlayerUtils/__tests__/getBackgroundImage.test.ts +72 -0
  36. package/audioPlayerUtils/__tests__/getImageFromEntry.test.ts +110 -0
  37. package/audioPlayerUtils/assets/index.ts +2 -0
  38. package/audioPlayerUtils/index.ts +242 -0
  39. package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
  40. package/componentsUtils/index.ts +4 -1
  41. package/conf/player/__tests__/selectors.test.ts +34 -0
  42. package/conf/player/selectors.ts +10 -0
  43. package/configurationUtils/__tests__/configurationUtils.test.js +0 -31
  44. package/configurationUtils/__tests__/getMediaItems.test.ts +65 -0
  45. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +34 -0
  46. package/configurationUtils/__tests__/manifestKeyParser.test.ts +546 -0
  47. package/configurationUtils/index.ts +64 -35
  48. package/configurationUtils/manifestKeyParser.ts +57 -32
  49. package/focusManager/FocusManager.ts +26 -16
  50. package/focusManager/Tree.ts +25 -21
  51. package/focusManager/__tests__/FocusManager.test.ts +50 -8
  52. package/index.d.ts +1 -10
  53. package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
  54. package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
  55. package/manifestUtils/createConfig.js +4 -1
  56. package/manifestUtils/defaultManifestConfigurations/player.js +1253 -200
  57. package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
  58. package/manifestUtils/sharedConfiguration/screenPicker/stylesFields.js +1 -2
  59. package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
  60. package/navigationUtils/index.ts +7 -5
  61. package/package.json +2 -3
  62. package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
  63. package/playerUtils/PlayerTTS/index.ts +1 -0
  64. package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
  65. package/playerUtils/__tests__/getPlayerActionButtons.test.ts +54 -0
  66. package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
  67. package/playerUtils/_internals/index.ts +1 -0
  68. package/playerUtils/_internals/utils.ts +31 -0
  69. package/playerUtils/configurationUtils.ts +0 -44
  70. package/playerUtils/getPlayerActionButtons.ts +17 -0
  71. package/playerUtils/index.ts +59 -0
  72. package/playerUtils/usePlayerTTS.ts +21 -0
  73. package/playerUtils/useValidatePlayerConfig.tsx +22 -19
  74. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
  75. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  76. package/reactHooks/cell-click/index.ts +3 -0
  77. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  78. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +47 -90
  79. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +71 -31
  80. package/reactHooks/feed/index.ts +2 -0
  81. package/reactHooks/feed/useBatchLoading.ts +17 -10
  82. package/reactHooks/feed/useFeedLoader.tsx +39 -44
  83. package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
  84. package/reactHooks/feed/usePipesCacheReset.ts +3 -3
  85. package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
  86. package/reactHooks/layout/__tests__/index.test.tsx +3 -1
  87. package/reactHooks/layout/isTablet/index.ts +12 -5
  88. package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
  89. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
  90. package/reactHooks/layout/useLayoutVersion.ts +5 -5
  91. package/reactHooks/navigation/index.ts +7 -5
  92. package/reactHooks/navigation/useIsScreenActive.ts +9 -5
  93. package/reactHooks/navigation/useRoute.ts +7 -2
  94. package/reactHooks/navigation/useScreenStateStore.ts +8 -0
  95. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
  96. package/reactHooks/screen/useScreenContext.ts +1 -1
  97. package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
  98. package/reactHooks/state/index.ts +1 -1
  99. package/reactHooks/state/useHomeRiver.ts +4 -2
  100. package/reactHooks/state/useRivers.ts +7 -8
  101. package/riverComponetsMeasurementProvider/index.tsx +1 -1
  102. package/screenPickerUtils/index.ts +7 -0
  103. package/services/js2native.ts +1 -0
  104. package/storage/ScreenSingleValueProvider.ts +138 -26
  105. package/storage/ScreenStateMultiSelectProvider.ts +210 -36
  106. package/testUtils/index.tsx +7 -8
  107. package/time/BackgroundTimer.ts +6 -4
  108. package/utils/__tests__/find.test.ts +36 -0
  109. package/utils/__tests__/pathOr.test.ts +37 -0
  110. package/utils/__tests__/startsWith.test.ts +30 -0
  111. package/utils/find.ts +3 -0
  112. package/utils/index.ts +24 -1
  113. package/utils/pathOr.ts +5 -0
  114. package/utils/startsWith.ts +9 -0
  115. package/playerUtils/configurationGenerator.ts +0 -2572
@@ -65,4 +65,17 @@ export const BUTTON_ACCESSIBILITY_KEYS = {
65
65
  label: "accessibility_menu_item_label",
66
66
  hint: "accessibility_menu_item_hint",
67
67
  },
68
+ skip_intro: {
69
+ label: "accessibility_skip_intro_label",
70
+ hint: "accessibility_skip_intro_hint",
71
+ },
72
+ // Top Menu Bar-specific buttons
73
+ top_menu_bar_item_selected: {
74
+ label: "accessibility_top_menu_bar_item_selected_label",
75
+ hint: "accessibility_top_menu_bar_item_selected_hint",
76
+ },
77
+ top_menu_title: {
78
+ label: "accessibility_top_menu_title_label",
79
+ hint: "accessibility_top_menu_hint",
80
+ },
68
81
  } as const;
@@ -1,4 +1,4 @@
1
- import { useEffect, useMemo } from "react";
1
+ import { useEffect, useMemo, useState } from "react";
2
2
  import { AccessibilityManager } from "./index";
3
3
 
4
4
  /**
@@ -38,3 +38,37 @@ export const useAccessibilityManager = (
38
38
 
39
39
  return accessibilityManager;
40
40
  };
41
+
42
+ export const useInitialAnnouncementReady = (
43
+ accessibilityManager: AccessibilityManager
44
+ ) => {
45
+ const [isReady, setIsReady] = useState(
46
+ accessibilityManager.isInitialPlayerAnnouncementReady
47
+ );
48
+
49
+ useEffect(() => {
50
+ const subscription = accessibilityManager
51
+ .getInitialAnnouncementReadyObservable()
52
+ .subscribe(setIsReady);
53
+
54
+ return () => subscription.unsubscribe();
55
+ }, [accessibilityManager]);
56
+
57
+ return isReady;
58
+ };
59
+
60
+ export const useAnnouncementActive = (
61
+ accessibilityManager: AccessibilityManager
62
+ ) => {
63
+ const [isActive, setIsActive] = useState(false);
64
+
65
+ useEffect(() => {
66
+ const subscription = accessibilityManager
67
+ .getTTSStateObservable()
68
+ .subscribe(setIsActive);
69
+
70
+ return () => subscription.unsubscribe();
71
+ }, [accessibilityManager]);
72
+
73
+ return isActive;
74
+ };
@@ -1,14 +1,19 @@
1
+ import * as R from "ramda";
1
2
  import { BehaviorSubject } from "rxjs";
2
3
  import { accessibilityManagerLogger as logger } from "./logger";
3
- import { TTSManager } from "../platform/platformUtils";
4
+ import { TTSManager } from "../platform";
4
5
  import { BUTTON_ACCESSIBILITY_KEYS } from "./const";
6
+ import { toString } from "../../utils";
7
+ import { calculateReadingTime } from "./utils";
5
8
  import { AccessibilityRole } from "react-native";
6
9
 
7
10
  export class AccessibilityManager {
8
11
  private static _instance: AccessibilityManager | null = null;
9
12
  private headingTimeout: NodeJS.Timeout | null = null;
10
- private WORDS_PER_MINUTE = 160;
13
+ private announcementDelayTimeout: NodeJS.Timeout | null = null;
14
+ private WORDS_PER_MINUTE = 140;
11
15
  private MINIMUM_PAUSE = 500;
16
+ private ANNOUNCEMENT_DELAY = 700;
12
17
  private state$ = new BehaviorSubject<AccessibilityState>({
13
18
  screenReaderEnabled: false,
14
19
  reduceMotionEnabled: false,
@@ -24,6 +29,12 @@ export class AccessibilityManager {
24
29
  private ttsManager = TTSManager.getInstance();
25
30
  private localizations: { [key: string]: string } = {};
26
31
  private headingQueue: string[] = [];
32
+ private currentFocusId: string | null = null;
33
+ private headingFocusMap: Map<string, string> = new Map();
34
+ private pendingFocusId: string | null = null;
35
+ private isInitialPlayerAnnouncementReady$ = new BehaviorSubject<boolean>(
36
+ false
37
+ );
27
38
 
28
39
  private constructor() {}
29
40
 
@@ -35,6 +46,26 @@ export class AccessibilityManager {
35
46
  return AccessibilityManager._instance;
36
47
  }
37
48
 
49
+ public get isInitialPlayerAnnouncementReady(): boolean {
50
+ return this.isInitialPlayerAnnouncementReady$.getValue();
51
+ }
52
+
53
+ public setInitialPlayerAnnouncementReady(): void {
54
+ this.isInitialPlayerAnnouncementReady$.next(true);
55
+ }
56
+
57
+ public resetInitialPlayerAnnouncementReady(): void {
58
+ this.isInitialPlayerAnnouncementReady$.next(false);
59
+ }
60
+
61
+ public getInitialAnnouncementReadyObservable() {
62
+ return this.isInitialPlayerAnnouncementReady$.asObservable();
63
+ }
64
+
65
+ public getTTSStateObservable() {
66
+ return this.ttsManager.getStateAsObservable();
67
+ }
68
+
38
69
  /**
39
70
  * The method now accepts any object with localizations using a flattened structure
40
71
  *
@@ -45,7 +76,9 @@ export class AccessibilityManager {
45
76
  * i.e. localizations: [{ en: { accessibility_close_label: "Close", accessibility_close_hint: "Press here to close" } }]
46
77
  */
47
78
  public updateLocalizations(localizations: { [key: string]: string }) {
48
- this.localizations = localizations;
79
+ if (!R.isEmpty(localizations)) {
80
+ this.localizations = localizations;
81
+ }
49
82
  }
50
83
 
51
84
  public getState(): AccessibilityState {
@@ -56,31 +89,25 @@ export class AccessibilityManager {
56
89
  return this.state$.asObservable();
57
90
  }
58
91
 
59
- /** Calculates the reading time for a given text
60
- * This method is a bit of a hack because we don't have a callback, or promise from VIZIO API
61
- * @param text - The text to calculate the reading time for
62
- * @returns The reading time in milliseconds
63
- */
64
- private calculateReadingTime(text: string): number {
65
- const words = text.trim().split(/\s+/).length;
66
-
67
- return Math.max(
68
- this.MINIMUM_PAUSE,
69
- (words / this.WORDS_PER_MINUTE) * 60 * 1000
70
- );
71
- }
72
-
73
92
  /**
74
93
  * Adds a heading to the queue, headings will be read before the next text
75
94
  * Each heading will be read once and removed from the queue
76
95
  */
77
96
  public addHeading(heading: string) {
97
+ if (!this.pendingFocusId) {
98
+ this.pendingFocusId = Date.now().toString();
99
+ }
100
+
101
+ this.headingFocusMap.set(heading, this.pendingFocusId);
78
102
  this.headingQueue.push(heading);
79
103
  }
80
104
 
81
105
  /**
82
106
  * text you want to be read, if you want to use localized text pass keyOfLocalizedText instead
83
107
  * keyOfLocalizedText is the key to the localized text
108
+ *
109
+ * Implements a delay mechanism to reduce noise during rapid navigation.
110
+ * Only the most recent announcement will be read after the delay period.
84
111
  */
85
112
  public readText({
86
113
  text,
@@ -111,19 +138,27 @@ export class AccessibilityManager {
111
138
  textToRead = localizedMessage;
112
139
  }
113
140
 
114
- if (this.headingQueue.length > 0) {
115
- const heading = this.headingQueue.shift()!;
116
- this.ttsManager?.readText(heading);
141
+ const focusId = this.pendingFocusId || Date.now().toString();
142
+ this.currentFocusId = focusId;
143
+ this.pendingFocusId = null;
117
144
 
118
- if (this.headingTimeout) {
119
- clearTimeout(this.headingTimeout);
120
- }
145
+ this.clearAnnouncement();
121
146
 
122
- const pauseTime = this.calculateReadingTime(heading);
147
+ this.announcementDelayTimeout = setTimeout(() => {
148
+ this.executeAnnouncement(textToRead, keyOfLocalizedText, focusId);
149
+ }, this.ANNOUNCEMENT_DELAY);
150
+ }
123
151
 
124
- this.headingTimeout = setTimeout(() => {
125
- this.ttsManager?.readText(textToRead);
126
- }, pauseTime);
152
+ /**
153
+ * Internal method to execute the actual announcement after the delay
154
+ */
155
+ private executeAnnouncement(
156
+ textToRead: string,
157
+ keyOfLocalizedText?: string,
158
+ focusId?: string
159
+ ) {
160
+ if (this.headingQueue.length > 0) {
161
+ this.processHeadingQueue(textToRead, focusId);
127
162
  } else {
128
163
  this.ttsManager?.readText(textToRead);
129
164
  }
@@ -135,17 +170,70 @@ export class AccessibilityManager {
135
170
  });
136
171
  }
137
172
 
138
- public getButtonAccessibilityProps(buttonName: string): AccessibilityProps {
173
+ /**
174
+ * Recursively processes all headings in the queue, reading them one by one
175
+ */
176
+ private processHeadingQueue(textToRead: string, focusId?: string) {
177
+ // If focus has changed, abort this announcement
178
+ if (focusId && this.currentFocusId !== focusId) {
179
+ return;
180
+ }
181
+
182
+ if (this.headingQueue.length === 0) {
183
+ if (focusId && this.currentFocusId === focusId) {
184
+ this.ttsManager?.readText(textToRead);
185
+ }
186
+
187
+ return;
188
+ }
189
+
190
+ const heading = this.headingQueue.shift()!;
191
+
192
+ const headingFocusId = this.headingFocusMap.get(heading);
193
+
194
+ if (headingFocusId && headingFocusId !== focusId) {
195
+ // This heading belongs to a previous focus, skip it
196
+ this.headingFocusMap.delete(heading);
197
+ this.processHeadingQueue(textToRead, focusId);
198
+
199
+ return;
200
+ }
201
+
202
+ this.ttsManager?.readText(heading);
203
+ this.headingFocusMap.delete(heading); // Clean up after reading
204
+
205
+ if (this.headingTimeout) {
206
+ clearTimeout(this.headingTimeout);
207
+ }
208
+
209
+ const pauseTime = calculateReadingTime(
210
+ heading,
211
+ this.WORDS_PER_MINUTE,
212
+ this.MINIMUM_PAUSE,
213
+ this.ANNOUNCEMENT_DELAY
214
+ );
215
+
216
+ this.headingTimeout = setTimeout(() => {
217
+ this.processHeadingQueue(textToRead, focusId);
218
+ }, pauseTime);
219
+ }
220
+
221
+ public getButtonAccessibilityProps(name: string): AccessibilityProps {
222
+ const buttonName = toString(name);
223
+
139
224
  const buttonConfig = BUTTON_ACCESSIBILITY_KEYS[buttonName];
140
225
 
141
226
  if (!buttonConfig) {
142
227
  return {
228
+ accessible: true,
143
229
  accessibilityLabel: buttonName,
144
230
  accessibilityHint: `Press button to perform action on ${buttonName}`,
145
231
  "aria-label": buttonName,
146
232
  "aria-description": `Press button to perform action on ${buttonName}`,
147
233
  accessibilityRole: "button" as AccessibilityRole,
148
234
  "aria-role": "button",
235
+ role: "button",
236
+ tabindex: 0,
149
237
  };
150
238
  }
151
239
 
@@ -159,23 +247,52 @@ export class AccessibilityManager {
159
247
  `Press button to perform action on ${buttonName}`;
160
248
 
161
249
  return {
250
+ accessible: true,
162
251
  accessibilityLabel: label,
163
252
  accessibilityHint: hint,
164
253
  "aria-label": label,
165
254
  "aria-description": hint,
166
255
  accessibilityRole: "button" as AccessibilityRole,
167
256
  "aria-role": "button",
257
+ role: "button",
258
+ tabindex: 0,
168
259
  };
169
260
  }
170
261
 
171
262
  public getInputAccessibilityProps(inputName: string): AccessibilityProps {
172
263
  return {
264
+ accessible: true,
173
265
  accessibilityLabel: inputName,
174
266
  accessibilityHint: `Enter text into ${inputName}`,
175
267
  "aria-label": inputName,
176
268
  "aria-description": `Enter text into ${inputName}`,
177
- accessibilityRole: "textbox" as AccessibilityRole,
178
- "aria-role": "textbox",
269
+ accessibilityRole: "searchbox" as AccessibilityRole,
270
+ "aria-role": "searchbox",
271
+ role: "searchbox",
272
+ tabindex: 0,
273
+ };
274
+ }
275
+
276
+ /**
277
+ * Extracts accessibility props from component props and returns them as HTML attributes
278
+ * @param props - Component props containing accessibility properties
279
+ * @returns Object with accessibility HTML attributes
280
+ */
281
+ public getWebAccessibilityProps(props: any): AccessibilityProps {
282
+ const {
283
+ "aria-label": ariaLabel,
284
+ "aria-description": ariaDescription,
285
+ "aria-role": ariaRole,
286
+ role,
287
+ tabindex,
288
+ } = props;
289
+
290
+ return {
291
+ "aria-label": ariaLabel,
292
+ "aria-description": ariaDescription,
293
+ "aria-role": ariaRole,
294
+ role: role || ariaRole,
295
+ tabindex,
179
296
  };
180
297
  }
181
298
 
@@ -193,4 +310,11 @@ export class AccessibilityManager {
193
310
 
194
311
  return this.localizations[key];
195
312
  }
313
+
314
+ private clearAnnouncement() {
315
+ if (this.announcementDelayTimeout) {
316
+ clearTimeout(this.announcementDelayTimeout);
317
+ this.announcementDelayTimeout = null;
318
+ }
319
+ }
196
320
  }
@@ -0,0 +1,24 @@
1
+ /**
2
+ * Calculates the reading time for a given text based on word count
3
+ * @param text - The text to calculate the reading time for
4
+ * @param wordsPerMinute - Words per minute reading speed (default: 160)
5
+ * @param minimumPause - Minimum pause time in milliseconds (default: 500)
6
+ * @param announcementDelay - Additional delay for announcement in milliseconds (default: 700)
7
+ * @returns The reading time in milliseconds
8
+ */
9
+ export function calculateReadingTime(
10
+ text: string,
11
+ wordsPerMinute: number = 140,
12
+ minimumPause: number = 500,
13
+ announcementDelay: number = 700
14
+ ): number {
15
+ const words = text
16
+ .trim()
17
+ .split(/(?<=\d)(?=[a-zA-Z])|(?<=[a-zA-Z])(?=\d)|[^\w\s]+|\s+/)
18
+ .filter(Boolean).length;
19
+
20
+ return (
21
+ Math.max(minimumPause, (words / wordsPerMinute) * 60 * 1000) +
22
+ announcementDelay
23
+ );
24
+ }
@@ -1,6 +1,7 @@
1
1
  import { ContextKeysManager } from "./index";
2
2
  import * as R from "ramda";
3
- import { screenStates } from "@applicaster/zapp-react-native-utils/storage/ScreenSingleValueProvider";
3
+ import * as _ from "lodash";
4
+ import { useScreenStateStore } from "../../reactHooks/navigation/useScreenStateStore";
4
5
 
5
6
  export interface IResolver {
6
7
  resolve: (string) => Promise<string | number | object>;
@@ -26,10 +27,20 @@ export class EntryResolver implements IResolver {
26
27
  // TODO: Move to proper place
27
28
 
28
29
  export class ScreenStateResolver implements IResolver {
29
- constructor(private route: string) {}
30
+ constructor(
31
+ private screenStateStore: ReturnType<typeof useScreenStateStore>
32
+ ) {}
30
33
 
31
34
  async resolve(key: string) {
32
- const screenState = screenStates[this.route];
35
+ const screenState = this.screenStateStore.getState().data;
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
+ }
33
44
 
34
45
  return screenState?.[key];
35
46
  }
@@ -78,3 +89,19 @@ export const resolveObjectValues = async (
78
89
 
79
90
  return Object.fromEntries(resolvedEntries);
80
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
  });