@applicaster/zapp-react-native-utils 14.0.0-alpha.5974411329 → 14.0.0-alpha.6000342231

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 (69) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +0 -1
  2. package/actionsExecutor/ScreenActions.ts +20 -19
  3. package/analyticsUtils/AnalyticPlayerListener.ts +5 -2
  4. package/analyticsUtils/__tests__/analyticsUtils.test.js +0 -11
  5. package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
  6. package/appUtils/accessibilityManager/const.ts +13 -0
  7. package/appUtils/accessibilityManager/hooks.ts +35 -1
  8. package/appUtils/accessibilityManager/index.ts +151 -30
  9. package/appUtils/accessibilityManager/utils.ts +24 -0
  10. package/appUtils/contextKeysManager/contextResolver.ts +12 -1
  11. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +8 -0
  12. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  13. package/appUtils/focusManager/events.ts +2 -0
  14. package/appUtils/focusManager/index.ios.ts +27 -0
  15. package/appUtils/focusManager/index.ts +86 -11
  16. package/appUtils/focusManagerAux/utils/index.ts +112 -3
  17. package/appUtils/focusManagerAux/utils/utils.ios.ts +35 -0
  18. package/appUtils/platform/platformUtils.ts +33 -3
  19. package/appUtils/playerManager/conts.ts +21 -0
  20. package/arrayUtils/__tests__/allTruthy.test.ts +24 -0
  21. package/arrayUtils/__tests__/anyThruthy.test.ts +24 -0
  22. package/arrayUtils/index.ts +5 -0
  23. package/configurationUtils/__tests__/manifestKeyParser.test.ts +0 -1
  24. package/configurationUtils/index.ts +1 -1
  25. package/focusManager/FocusManager.ts +78 -4
  26. package/focusManager/aux/index.ts +98 -0
  27. package/focusManager/utils.ts +12 -6
  28. package/index.d.ts +1 -1
  29. package/manifestUtils/defaultManifestConfigurations/player.js +188 -2
  30. package/manifestUtils/index.js +4 -0
  31. package/manifestUtils/keys.js +12 -0
  32. package/manifestUtils/sharedConfiguration/screenPicker/stylesFields.js +6 -0
  33. package/navigationUtils/index.ts +20 -17
  34. package/package.json +2 -2
  35. package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
  36. package/playerUtils/PlayerTTS/index.ts +1 -0
  37. package/playerUtils/getPlayerActionButtons.ts +1 -1
  38. package/playerUtils/usePlayerTTS.ts +21 -0
  39. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  40. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  41. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +8 -2
  42. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +57 -37
  43. package/reactHooks/feed/index.ts +2 -0
  44. package/reactHooks/feed/useBatchLoading.ts +14 -9
  45. package/reactHooks/feed/useFeedLoader.tsx +39 -59
  46. package/reactHooks/feed/useInflatedUrl.ts +23 -29
  47. package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
  48. package/reactHooks/layout/index.ts +1 -1
  49. package/reactHooks/navigation/useScreenStateStore.ts +3 -3
  50. package/reactHooks/state/index.ts +1 -1
  51. package/reactHooks/state/useHomeRiver.ts +4 -2
  52. package/screenPickerUtils/index.ts +13 -0
  53. package/storage/ScreenSingleValueProvider.ts +25 -22
  54. package/storage/ScreenStateMultiSelectProvider.ts +26 -23
  55. package/utils/__tests__/endsWith.test.ts +30 -0
  56. package/utils/__tests__/find.test.ts +36 -0
  57. package/utils/__tests__/omit.test.ts +19 -0
  58. package/utils/__tests__/path.test.ts +33 -0
  59. package/utils/__tests__/pathOr.test.ts +37 -0
  60. package/utils/__tests__/startsWith.test.ts +30 -0
  61. package/utils/__tests__/take.test.ts +40 -0
  62. package/utils/endsWith.ts +9 -0
  63. package/utils/find.ts +3 -0
  64. package/utils/index.ts +19 -1
  65. package/utils/omit.ts +5 -0
  66. package/utils/path.ts +5 -0
  67. package/utils/pathOr.ts +5 -0
  68. package/utils/startsWith.ts +9 -0
  69. package/utils/take.ts +5 -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
  }
@@ -35,8 +35,11 @@ export class AnalyticPlayerListener
35
35
  this.handleAnalyticEvent(PLAYBACK_EVENT.complete);
36
36
  };
37
37
 
38
- onError = (err: Error) => {
39
- this.handleAnalyticEvent(PLAYBACK_EVENT.error, err); // TODO: Check error format
38
+ onError = (err: QuickBrickPlayer.PlayerErrorI) => {
39
+ this.handleAnalyticEvent(
40
+ PLAYBACK_EVENT.error,
41
+ err.toObject?.() || { message: err.message }
42
+ );
40
43
  };
41
44
 
42
45
  onPlayerPause = (event) => {
@@ -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();
@@ -105,7 +105,8 @@ export class PlayerAnalyticsTracker implements PlayerAnalyticsTrackerI {
105
105
  return this.getDateTimestamp();
106
106
  }
107
107
 
108
- this.mediaTime = this.playerState?.currentTime || eventData?.currentTime;
108
+ this.mediaTime =
109
+ this.playerState?.contentPosition || eventData?.currentTime;
109
110
 
110
111
  return this.mediaTime;
111
112
  }
@@ -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,15 +1,19 @@
1
+ import * as R from "ramda";
1
2
  import { BehaviorSubject } from "rxjs";
2
3
  import { accessibilityManagerLogger as logger } from "./logger";
3
4
  import { TTSManager } from "../platform";
4
5
  import { BUTTON_ACCESSIBILITY_KEYS } from "./const";
5
6
  import { toString } from "../../utils";
7
+ import { calculateReadingTime } from "./utils";
6
8
  import { AccessibilityRole } from "react-native";
7
9
 
8
10
  export class AccessibilityManager {
9
11
  private static _instance: AccessibilityManager | null = null;
10
12
  private headingTimeout: NodeJS.Timeout | null = null;
11
- private WORDS_PER_MINUTE = 160;
13
+ private announcementDelayTimeout: NodeJS.Timeout | null = null;
14
+ private WORDS_PER_MINUTE = 140;
12
15
  private MINIMUM_PAUSE = 500;
16
+ private ANNOUNCEMENT_DELAY = 700;
13
17
  private state$ = new BehaviorSubject<AccessibilityState>({
14
18
  screenReaderEnabled: false,
15
19
  reduceMotionEnabled: false,
@@ -25,6 +29,12 @@ export class AccessibilityManager {
25
29
  private ttsManager = TTSManager.getInstance();
26
30
  private localizations: { [key: string]: string } = {};
27
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
+ );
28
38
 
29
39
  private constructor() {}
30
40
 
@@ -36,6 +46,26 @@ export class AccessibilityManager {
36
46
  return AccessibilityManager._instance;
37
47
  }
38
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
+
39
69
  /**
40
70
  * The method now accepts any object with localizations using a flattened structure
41
71
  *
@@ -46,7 +76,9 @@ export class AccessibilityManager {
46
76
  * i.e. localizations: [{ en: { accessibility_close_label: "Close", accessibility_close_hint: "Press here to close" } }]
47
77
  */
48
78
  public updateLocalizations(localizations: { [key: string]: string }) {
49
- this.localizations = localizations;
79
+ if (!R.isEmpty(localizations)) {
80
+ this.localizations = localizations;
81
+ }
50
82
  }
51
83
 
52
84
  public getState(): AccessibilityState {
@@ -57,31 +89,25 @@ export class AccessibilityManager {
57
89
  return this.state$.asObservable();
58
90
  }
59
91
 
60
- /** Calculates the reading time for a given text
61
- * This method is a bit of a hack because we don't have a callback, or promise from VIZIO API
62
- * @param text - The text to calculate the reading time for
63
- * @returns The reading time in milliseconds
64
- */
65
- private calculateReadingTime(text: string): number {
66
- const words = text.trim().split(/\s+/).length;
67
-
68
- return Math.max(
69
- this.MINIMUM_PAUSE,
70
- (words / this.WORDS_PER_MINUTE) * 60 * 1000
71
- );
72
- }
73
-
74
92
  /**
75
93
  * Adds a heading to the queue, headings will be read before the next text
76
94
  * Each heading will be read once and removed from the queue
77
95
  */
78
96
  public addHeading(heading: string) {
97
+ if (!this.pendingFocusId) {
98
+ this.pendingFocusId = Date.now().toString();
99
+ }
100
+
101
+ this.headingFocusMap.set(heading, this.pendingFocusId);
79
102
  this.headingQueue.push(heading);
80
103
  }
81
104
 
82
105
  /**
83
106
  * text you want to be read, if you want to use localized text pass keyOfLocalizedText instead
84
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.
85
111
  */
86
112
  public readText({
87
113
  text,
@@ -112,19 +138,27 @@ export class AccessibilityManager {
112
138
  textToRead = localizedMessage;
113
139
  }
114
140
 
115
- if (this.headingQueue.length > 0) {
116
- const heading = this.headingQueue.shift()!;
117
- this.ttsManager?.readText(heading);
141
+ const focusId = this.pendingFocusId || Date.now().toString();
142
+ this.currentFocusId = focusId;
143
+ this.pendingFocusId = null;
118
144
 
119
- if (this.headingTimeout) {
120
- clearTimeout(this.headingTimeout);
121
- }
145
+ this.clearAnnouncement();
122
146
 
123
- const pauseTime = this.calculateReadingTime(heading);
147
+ this.announcementDelayTimeout = setTimeout(() => {
148
+ this.executeAnnouncement(textToRead, keyOfLocalizedText, focusId);
149
+ }, this.ANNOUNCEMENT_DELAY);
150
+ }
124
151
 
125
- this.headingTimeout = setTimeout(() => {
126
- this.ttsManager?.readText(textToRead);
127
- }, 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);
128
162
  } else {
129
163
  this.ttsManager?.readText(textToRead);
130
164
  }
@@ -136,6 +170,54 @@ export class AccessibilityManager {
136
170
  });
137
171
  }
138
172
 
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
+
139
221
  public getButtonAccessibilityProps(name: string): AccessibilityProps {
140
222
  const buttonName = toString(name);
141
223
 
@@ -143,12 +225,15 @@ export class AccessibilityManager {
143
225
 
144
226
  if (!buttonConfig) {
145
227
  return {
228
+ accessible: true,
146
229
  accessibilityLabel: buttonName,
147
230
  accessibilityHint: `Press button to perform action on ${buttonName}`,
148
231
  "aria-label": buttonName,
149
232
  "aria-description": `Press button to perform action on ${buttonName}`,
150
- accessibilityRole: "button",
233
+ accessibilityRole: "button" as AccessibilityRole,
151
234
  "aria-role": "button",
235
+ role: "button",
236
+ tabindex: 0,
152
237
  };
153
238
  }
154
239
 
@@ -162,23 +247,52 @@ export class AccessibilityManager {
162
247
  `Press button to perform action on ${buttonName}`;
163
248
 
164
249
  return {
250
+ accessible: true,
165
251
  accessibilityLabel: label,
166
252
  accessibilityHint: hint,
167
253
  "aria-label": label,
168
254
  "aria-description": hint,
169
- accessibilityRole: "button",
255
+ accessibilityRole: "button" as AccessibilityRole,
170
256
  "aria-role": "button",
257
+ role: "button",
258
+ tabindex: 0,
171
259
  };
172
260
  }
173
261
 
174
262
  public getInputAccessibilityProps(inputName: string): AccessibilityProps {
175
263
  return {
264
+ accessible: true,
176
265
  accessibilityLabel: inputName,
177
266
  accessibilityHint: `Enter text into ${inputName}`,
178
267
  "aria-label": inputName,
179
268
  "aria-description": `Enter text into ${inputName}`,
180
- accessibilityRole: "textbox" as AccessibilityRole,
181
- "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,
182
296
  };
183
297
  }
184
298
 
@@ -196,4 +310,11 @@ export class AccessibilityManager {
196
310
 
197
311
  return this.localizations[key];
198
312
  }
313
+
314
+ private clearAnnouncement() {
315
+ if (this.announcementDelayTimeout) {
316
+ clearTimeout(this.announcementDelayTimeout);
317
+ this.announcementDelayTimeout = null;
318
+ }
319
+ }
199
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
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,11 +27,21 @@ export class EntryResolver implements IResolver {
26
27
  // TODO: Move to proper place
27
28
 
28
29
  export class ScreenStateResolver implements IResolver {
29
- constructor(private screenStateStore: ScreenStateStore) {}
30
+ constructor(
31
+ private screenStateStore: ReturnType<typeof useScreenStateStore>
32
+ ) {}
30
33
 
31
34
  async resolve(key: string) {
32
35
  const screenState = this.screenStateStore.getState().data;
33
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
+
34
45
  return screenState?.[key];
35
46
  }
36
47
  }
@@ -6,6 +6,8 @@ exports[`focusManager should be defined 1`] = `
6
6
  "disableFocus": [Function],
7
7
  "enableFocus": [Function],
8
8
  "findPreferredFocusChild": [Function],
9
+ "focusOnSelectedTab": [Function],
10
+ "focusOnSelectedTopMenuItem": [Function],
9
11
  "focusableTree": Tree {
10
12
  "loadingCounter": 0,
11
13
  "root": {
@@ -24,7 +26,11 @@ exports[`focusManager should be defined 1`] = `
24
26
  "invokeHandler": [Function],
25
27
  "isCurrentFocusOnTheTopScreen": [Function],
26
28
  "isFocusDisabled": [Function],
29
+ "isFocusOn": [Function],
30
+ "isFocusOnContent": [Function],
31
+ "isFocusOnMenu": [Function],
27
32
  "isGroupItemFocused": [Function],
33
+ "isTabsScreenContentFocused": [Function],
28
34
  "longPress": [Function],
29
35
  "moveFocus": [Function],
30
36
  "on": [Function],
@@ -63,6 +69,8 @@ exports[`focusManagerIOS should be defined 1`] = `
63
69
  "getGroupRootById": [Function],
64
70
  "getPreferredFocusChild": [Function],
65
71
  "invokeHandler": [Function],
72
+ "isChildOf": [Function],
73
+ "isFocusOn": [Function],
66
74
  "isGroupItemFocused": [Function],
67
75
  "moveFocus": [Function],
68
76
  "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", () => {});
@@ -9,3 +9,5 @@ export const WILL_LOSE_FOCUS = "willLoseFocus";
9
9
  export const BLUR = "blur";
10
10
 
11
11
  export const HAS_LOST_FOCUS = "hasLostFocus";
12
+
13
+ export const RESET = "reset";
@@ -1,11 +1,20 @@
1
1
  import { NativeModules } from "react-native";
2
2
  import * as R from "ramda";
3
3
 
4
+ import {
5
+ isCurrentFocusOn,
6
+ isChildOf as isChildOfUtils,
7
+ } from "../focusManagerAux/utils";
4
8
  import { Tree } from "./treeDataStructure/Tree";
5
9
  import { findFocusableNode } from "./treeDataStructure/Utils";
6
10
  import { subscriber } from "../../functionUtils";
7
11
  import { findChild } from "./utils";
8
12
 
13
+ import {
14
+ emitRegistered,
15
+ emitUnregistered,
16
+ } from "../focusManagerAux/utils/utils.ios";
17
+
9
18
  const { FocusableManagerModule } = NativeModules;
10
19
 
11
20
  /**
@@ -179,10 +188,14 @@ export const focusManager = (function () {
179
188
  function register({ id, component }) {
180
189
  const { isGroup = false } = component;
181
190
 
191
+ emitRegistered(id);
192
+
182
193
  return isGroup ? registerGroup(id, component) : registerItem(id, component);
183
194
  }
184
195
 
185
196
  function unregister(id, { group = false } = {}) {
197
+ emitUnregistered(id);
198
+
186
199
  group ? unregisterGroup(id) : unregisterItem(id);
187
200
  }
188
201
 
@@ -391,6 +404,18 @@ export const focusManager = (function () {
391
404
  return node;
392
405
  }
393
406
 
407
+ function isFocusOn(id): boolean {
408
+ const currentFocusNode = focusableTree.findInTree(
409
+ getCurrentFocus()?.props?.id
410
+ );
411
+
412
+ return id && isCurrentFocusOn(id, currentFocusNode);
413
+ }
414
+
415
+ function isChildOf(childId, parentId): boolean {
416
+ return isChildOfUtils(focusableTree, childId, parentId);
417
+ }
418
+
394
419
  return {
395
420
  on,
396
421
  invokeHandler,
@@ -412,5 +437,7 @@ export const focusManager = (function () {
412
437
  getGroupRootById,
413
438
  isGroupItemFocused,
414
439
  getPreferredFocusChild,
440
+ isFocusOn,
441
+ isChildOf,
415
442
  };
416
443
  })();