@applicaster/zapp-react-native-utils 14.0.0-rc.9 → 15.0.0-alpha.2239032089
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.
- package/actionsExecutor/ActionExecutorContext.tsx +60 -84
- package/actionsExecutor/ScreenActions.ts +164 -0
- package/actionsExecutor/StorageActions.ts +110 -0
- package/actionsExecutor/feedDecorator.ts +171 -0
- package/actionsExecutor/screenResolver.ts +11 -0
- package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
- package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
- package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
- package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
- package/analyticsUtils/__tests__/analyticsUtils.test.js +3 -0
- package/analyticsUtils/events.ts +8 -0
- package/analyticsUtils/index.tsx +3 -4
- package/analyticsUtils/manager.ts +1 -1
- package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
- package/appUtils/HooksManager/Hook.ts +4 -4
- package/appUtils/HooksManager/index.ts +11 -1
- package/appUtils/RiverFocusManager/{index.js → index.ts} +25 -18
- package/appUtils/accessibilityManager/const.ts +13 -0
- package/appUtils/accessibilityManager/hooks.ts +35 -1
- package/appUtils/accessibilityManager/index.ts +154 -30
- package/appUtils/accessibilityManager/utils.ts +24 -0
- package/appUtils/contextKeysManager/contextResolver.ts +42 -1
- package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +10 -0
- package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
- package/appUtils/focusManager/events.ts +2 -0
- package/appUtils/focusManager/index.ios.ts +46 -1
- package/appUtils/focusManager/index.ts +86 -11
- package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
- package/appUtils/focusManagerAux/utils/index.ios.ts +104 -0
- package/appUtils/focusManagerAux/utils/index.ts +94 -3
- package/appUtils/focusManagerAux/utils/utils.ios.ts +63 -0
- package/appUtils/platform/platformUtils.ts +31 -1
- package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +0 -15
- package/appUtils/playerManager/useChapterMarker.tsx +0 -1
- package/appUtils/playerManager/usePlayerControllerSetup.tsx +16 -0
- package/arrayUtils/__tests__/isEmptyArray.test.ts +63 -0
- package/arrayUtils/__tests__/isFilledArray.test.ts +1 -1
- package/arrayUtils/index.ts +8 -3
- package/audioPlayerUtils/__tests__/getArtworkImage.test.ts +144 -0
- package/audioPlayerUtils/__tests__/getBackgroundImage.test.ts +72 -0
- package/audioPlayerUtils/__tests__/getImageFromEntry.test.ts +110 -0
- package/audioPlayerUtils/assets/index.ts +2 -0
- package/audioPlayerUtils/index.ts +242 -0
- package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
- package/componentsUtils/index.ts +4 -1
- package/conf/player/__tests__/selectors.test.ts +34 -0
- package/conf/player/selectors.ts +10 -0
- package/configurationUtils/__tests__/configurationUtils.test.js +0 -31
- package/configurationUtils/__tests__/getMediaItems.test.ts +65 -0
- package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +34 -0
- package/configurationUtils/__tests__/manifestKeyParser.test.ts +546 -0
- package/configurationUtils/index.ts +64 -35
- package/configurationUtils/manifestKeyParser.ts +57 -32
- package/focusManager/FocusManager.ts +104 -20
- package/focusManager/Tree.ts +25 -21
- package/focusManager/__tests__/FocusManager.test.ts +50 -8
- package/focusManager/aux/index.ts +98 -0
- package/focusManager/utils.ts +12 -6
- package/index.d.ts +1 -10
- package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
- package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
- package/manifestUtils/createConfig.js +4 -1
- package/manifestUtils/defaultManifestConfigurations/player.js +2764 -1539
- package/manifestUtils/index.js +4 -0
- package/manifestUtils/keys.js +12 -0
- package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
- package/manifestUtils/sharedConfiguration/screenPicker/stylesFields.js +6 -0
- package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
- package/navigationUtils/index.ts +7 -5
- package/package.json +2 -3
- package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
- package/playerUtils/PlayerTTS/index.ts +1 -0
- package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
- package/playerUtils/__tests__/getPlayerActionButtons.test.ts +54 -0
- package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
- package/playerUtils/_internals/index.ts +1 -0
- package/playerUtils/_internals/utils.ts +31 -0
- package/playerUtils/configurationUtils.ts +0 -44
- package/playerUtils/getPlayerActionButtons.ts +17 -0
- package/playerUtils/index.ts +53 -0
- package/playerUtils/usePlayerTTS.ts +21 -0
- package/playerUtils/useValidatePlayerConfig.tsx +22 -19
- package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
- package/reactHooks/cell-click/__tests__/index.test.js +3 -0
- package/reactHooks/cell-click/index.ts +8 -1
- package/reactHooks/debugging/__tests__/index.test.js +0 -1
- package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +47 -90
- package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +71 -31
- package/reactHooks/feed/index.ts +2 -0
- package/reactHooks/feed/useBatchLoading.ts +17 -10
- package/reactHooks/feed/useFeedLoader.tsx +36 -34
- package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
- package/reactHooks/feed/usePipesCacheReset.ts +3 -3
- package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
- package/reactHooks/layout/__tests__/index.test.tsx +3 -1
- package/reactHooks/layout/isTablet/index.ts +12 -5
- package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
- package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
- package/reactHooks/layout/useLayoutVersion.ts +5 -5
- package/reactHooks/navigation/index.ts +7 -5
- package/reactHooks/navigation/useIsScreenActive.ts +9 -5
- package/reactHooks/navigation/useRoute.ts +7 -2
- package/reactHooks/navigation/useScreenStateStore.ts +8 -0
- package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
- package/reactHooks/screen/useScreenContext.ts +1 -1
- package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
- package/reactHooks/state/index.ts +1 -1
- package/reactHooks/state/useHomeRiver.ts +4 -2
- package/reactHooks/state/useRivers.ts +7 -8
- package/riverComponetsMeasurementProvider/index.tsx +1 -1
- package/screenPickerUtils/index.ts +13 -0
- package/services/js2native.ts +1 -0
- package/storage/ScreenSingleValueProvider.ts +204 -0
- package/storage/ScreenStateMultiSelectProvider.ts +293 -0
- package/storage/StorageMultiSelectProvider.ts +192 -0
- package/storage/StorageSingleSelectProvider.ts +108 -0
- package/testUtils/index.tsx +7 -8
- package/time/BackgroundTimer.ts +6 -4
- package/utils/__tests__/endsWith.test.ts +30 -0
- package/utils/__tests__/find.test.ts +36 -0
- package/utils/__tests__/omit.test.ts +19 -0
- package/utils/__tests__/path.test.ts +33 -0
- package/utils/__tests__/pathOr.test.ts +37 -0
- package/utils/__tests__/startsWith.test.ts +30 -0
- package/utils/__tests__/take.test.ts +40 -0
- package/utils/endsWith.ts +9 -0
- package/utils/find.ts +3 -0
- package/utils/index.ts +34 -1
- package/utils/omit.ts +5 -0
- package/utils/path.ts +5 -0
- package/utils/pathOr.ts +5 -0
- package/utils/startsWith.ts +9 -0
- package/utils/take.ts +5 -0
- package/zappFrameworkUtils/HookCallback/callbackNavigationAction.ts +164 -0
- package/zappFrameworkUtils/HookCallback/hookCallbackManifestExtensions.config.js +60 -0
- package/zappFrameworkUtils/HookCallback/useCallbackActions.ts +22 -0
- package/zappFrameworkUtils/loginPluginUtils.ts +1 -1
- package/playerUtils/configurationGenerator.ts +0 -2572
|
@@ -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
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
115
|
-
|
|
116
|
-
|
|
141
|
+
const focusId = this.pendingFocusId || Date.now().toString();
|
|
142
|
+
this.currentFocusId = focusId;
|
|
143
|
+
this.pendingFocusId = null;
|
|
117
144
|
|
|
118
|
-
|
|
119
|
-
clearTimeout(this.headingTimeout);
|
|
120
|
-
}
|
|
145
|
+
this.clearAnnouncement();
|
|
121
146
|
|
|
122
|
-
|
|
147
|
+
this.announcementDelayTimeout = setTimeout(() => {
|
|
148
|
+
this.executeAnnouncement(textToRead, keyOfLocalizedText, focusId);
|
|
149
|
+
}, this.ANNOUNCEMENT_DELAY);
|
|
150
|
+
}
|
|
123
151
|
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
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
|
-
|
|
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: "
|
|
178
|
-
"aria-role": "
|
|
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,10 +1,13 @@
|
|
|
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
|
-
interface IResolver {
|
|
6
|
+
export interface IResolver {
|
|
5
7
|
resolve: (string) => Promise<string | number | object>;
|
|
6
8
|
}
|
|
7
9
|
|
|
10
|
+
// TODO: Rename to ObjectKeyResolver or similar
|
|
8
11
|
export class EntryResolver implements IResolver {
|
|
9
12
|
entry: ZappEntry;
|
|
10
13
|
|
|
@@ -21,6 +24,28 @@ export class EntryResolver implements IResolver {
|
|
|
21
24
|
}
|
|
22
25
|
}
|
|
23
26
|
|
|
27
|
+
// TODO: Move to proper place
|
|
28
|
+
|
|
29
|
+
export class ScreenStateResolver implements IResolver {
|
|
30
|
+
constructor(
|
|
31
|
+
private screenStateStore: ReturnType<typeof useScreenStateStore>
|
|
32
|
+
) {}
|
|
33
|
+
|
|
34
|
+
async resolve(key: string) {
|
|
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
|
+
}
|
|
44
|
+
|
|
45
|
+
return screenState?.[key];
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
24
49
|
export class ContextResolver implements IResolver {
|
|
25
50
|
resolve = async (compositeKey: string) =>
|
|
26
51
|
ContextKeysManager.instance.getKey(compositeKey);
|
|
@@ -64,3 +89,19 @@ export const resolveObjectValues = async (
|
|
|
64
89
|
|
|
65
90
|
return Object.fromEntries(resolvedEntries);
|
|
66
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,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,10 @@ exports[`focusManagerIOS should be defined 1`] = `
|
|
|
63
69
|
"getGroupRootById": [Function],
|
|
64
70
|
"getPreferredFocusChild": [Function],
|
|
65
71
|
"invokeHandler": [Function],
|
|
72
|
+
"isFocusOn": [Function],
|
|
73
|
+
"isFocusOnContent": [Function],
|
|
74
|
+
"isFocusOnMenu": [Function],
|
|
75
|
+
"isFocusOnTabsScreenContent": [Function],
|
|
66
76
|
"isGroupItemFocused": [Function],
|
|
67
77
|
"moveFocus": [Function],
|
|
68
78
|
"on": [Function],
|
|
@@ -6,6 +6,13 @@ import { findFocusableNode } from "./treeDataStructure/Utils";
|
|
|
6
6
|
import { subscriber } from "../../functionUtils";
|
|
7
7
|
import { findChild } from "./utils";
|
|
8
8
|
|
|
9
|
+
import {
|
|
10
|
+
isCurrentFocusOn,
|
|
11
|
+
isPartOfMenu,
|
|
12
|
+
isPartOfContent,
|
|
13
|
+
isPartOfTabsScreenContent,
|
|
14
|
+
} from "../focusManagerAux/utils/index.ios";
|
|
15
|
+
|
|
9
16
|
const { FocusableManagerModule } = NativeModules;
|
|
10
17
|
|
|
11
18
|
/**
|
|
@@ -260,7 +267,9 @@ export const focusManager = (function () {
|
|
|
260
267
|
function setFocus(
|
|
261
268
|
id: string,
|
|
262
269
|
direction?: FocusManager.IOS.Direction,
|
|
263
|
-
options?: Partial<{
|
|
270
|
+
options?: Partial<{
|
|
271
|
+
groupFocusedChanged: boolean;
|
|
272
|
+
}>,
|
|
264
273
|
callback?: any
|
|
265
274
|
) {
|
|
266
275
|
blur(direction);
|
|
@@ -391,6 +400,38 @@ export const focusManager = (function () {
|
|
|
391
400
|
return node;
|
|
392
401
|
}
|
|
393
402
|
|
|
403
|
+
function isFocusOn(id): boolean {
|
|
404
|
+
const currentFocusNode = focusableTree.findInTree(
|
|
405
|
+
getCurrentFocus()?.props?.id
|
|
406
|
+
);
|
|
407
|
+
|
|
408
|
+
return id && isCurrentFocusOn(id, currentFocusNode);
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
function isFocusOnMenu(): boolean {
|
|
412
|
+
const currentFocusable = getCurrentFocus();
|
|
413
|
+
|
|
414
|
+
return isPartOfMenu(focusableTree, currentFocusable?.props?.id);
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
function isFocusOnContent(): boolean {
|
|
418
|
+
const currentFocusable = getCurrentFocus();
|
|
419
|
+
|
|
420
|
+
return isPartOfContent(focusableTree, currentFocusable?.props?.id);
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
function isFocusOnTabsScreenContent(
|
|
424
|
+
screenPickerContentContainerId: string
|
|
425
|
+
): boolean {
|
|
426
|
+
const currentFocusable = getCurrentFocus();
|
|
427
|
+
|
|
428
|
+
return isPartOfTabsScreenContent(
|
|
429
|
+
focusableTree,
|
|
430
|
+
screenPickerContentContainerId,
|
|
431
|
+
currentFocusable?.props?.id
|
|
432
|
+
);
|
|
433
|
+
}
|
|
434
|
+
|
|
394
435
|
return {
|
|
395
436
|
on,
|
|
396
437
|
invokeHandler,
|
|
@@ -412,5 +453,9 @@ export const focusManager = (function () {
|
|
|
412
453
|
getGroupRootById,
|
|
413
454
|
isGroupItemFocused,
|
|
414
455
|
getPreferredFocusChild,
|
|
456
|
+
isFocusOn,
|
|
457
|
+
isFocusOnMenu,
|
|
458
|
+
isFocusOnContent,
|
|
459
|
+
isFocusOnTabsScreenContent,
|
|
415
460
|
};
|
|
416
461
|
})();
|
|
@@ -3,6 +3,7 @@ import {
|
|
|
3
3
|
isNilOrEmpty,
|
|
4
4
|
isNotNil,
|
|
5
5
|
} from "@applicaster/zapp-react-native-utils/reactUtils/helpers";
|
|
6
|
+
import { getFocusableId } from "@applicaster/zapp-react-native-utils/screenPickerUtils";
|
|
6
7
|
|
|
7
8
|
import { Tree } from "./treeDataStructure/Tree";
|
|
8
9
|
import {
|
|
@@ -14,6 +15,13 @@ import { subscriber } from "../../functionUtils";
|
|
|
14
15
|
import { coreLogger } from "../../logger";
|
|
15
16
|
import { ACTION } from "./utils/enums";
|
|
16
17
|
|
|
18
|
+
import {
|
|
19
|
+
isTabsScreenOnContentFocused,
|
|
20
|
+
isCurrentFocusOnContent,
|
|
21
|
+
isCurrentFocusOnMenu,
|
|
22
|
+
isCurrentFocusOn,
|
|
23
|
+
} from "../focusManagerAux/utils";
|
|
24
|
+
|
|
17
25
|
const logger = coreLogger.addSubsystem("focusManager");
|
|
18
26
|
|
|
19
27
|
const isFocusEnabled = (focusableItem): boolean => {
|
|
@@ -100,7 +108,7 @@ export const focusManager = (function () {
|
|
|
100
108
|
* @private
|
|
101
109
|
* @param {Object} direction of the navigation which led to this action
|
|
102
110
|
*/
|
|
103
|
-
function focus(direction) {
|
|
111
|
+
function focus(direction, context?: FocusManager.FocusContext) {
|
|
104
112
|
const currentFocusable = getCurrentFocus();
|
|
105
113
|
|
|
106
114
|
if (
|
|
@@ -108,7 +116,7 @@ export const focusManager = (function () {
|
|
|
108
116
|
!currentFocusable.isGroup &&
|
|
109
117
|
currentFocusable.isMounted()
|
|
110
118
|
) {
|
|
111
|
-
currentFocusable.setFocus(direction);
|
|
119
|
+
currentFocusable.setFocus(direction, context);
|
|
112
120
|
}
|
|
113
121
|
}
|
|
114
122
|
|
|
@@ -205,7 +213,7 @@ export const focusManager = (function () {
|
|
|
205
213
|
* @param {Array<string>} ids - An array of node IDs to update.
|
|
206
214
|
* @param {boolean} setFocus - A flag indicating whether to set focus (true) or blur (false) on the nodes.
|
|
207
215
|
*/
|
|
208
|
-
const updateNodeFocus = (ids, action) => {
|
|
216
|
+
const updateNodeFocus = (ids, action, context: FocusManager.FocusContext) => {
|
|
209
217
|
if (!ids || ids.length === 0) {
|
|
210
218
|
return; // Nothing to do
|
|
211
219
|
}
|
|
@@ -222,11 +230,13 @@ export const focusManager = (function () {
|
|
|
222
230
|
|
|
223
231
|
// Function to apply the action (focus or blur)
|
|
224
232
|
const applyAction = (node) => {
|
|
233
|
+
const direction = undefined;
|
|
234
|
+
|
|
225
235
|
if (node && node.component) {
|
|
226
236
|
if (action === "focus") {
|
|
227
|
-
node.component.setFocus();
|
|
237
|
+
node.component.setFocus(direction, context);
|
|
228
238
|
} else if (action === "blur") {
|
|
229
|
-
node.component.setBlur();
|
|
239
|
+
node.component.setBlur(direction, context);
|
|
230
240
|
}
|
|
231
241
|
}
|
|
232
242
|
};
|
|
@@ -253,7 +263,11 @@ export const focusManager = (function () {
|
|
|
253
263
|
* @param {Object} direction of the navigation, which led to this focus change
|
|
254
264
|
* to another group or not. defaults to false
|
|
255
265
|
*/
|
|
256
|
-
function setFocus(
|
|
266
|
+
function setFocus(
|
|
267
|
+
id: string,
|
|
268
|
+
direction?: FocusManager.Web.Direction,
|
|
269
|
+
context?: FocusManager.FocusContext
|
|
270
|
+
) {
|
|
257
271
|
if (focusDisabled) return false;
|
|
258
272
|
|
|
259
273
|
// due to optimisiation it's recommanded to set currentFocusNode before setFocus
|
|
@@ -266,21 +280,68 @@ export const focusManager = (function () {
|
|
|
266
280
|
);
|
|
267
281
|
|
|
268
282
|
// Set focus on current node parents and blur on previous node parents
|
|
269
|
-
updateNodeFocus(currentNodeParentsIDs, ACTION.FOCUS);
|
|
270
|
-
updateNodeFocus(previousNodeParentsIDs, ACTION.BLUR);
|
|
283
|
+
updateNodeFocus(currentNodeParentsIDs, ACTION.FOCUS, context);
|
|
284
|
+
updateNodeFocus(previousNodeParentsIDs, ACTION.BLUR, context);
|
|
271
285
|
|
|
272
286
|
currentFocusNode = focusableTree.findInTree(id);
|
|
273
287
|
}
|
|
274
288
|
|
|
275
289
|
setLastFocusOnParentNode(currentFocusNode);
|
|
276
290
|
|
|
277
|
-
focus(direction);
|
|
291
|
+
focus(direction, context);
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
function isFocusOnContent() {
|
|
295
|
+
return isCurrentFocusOnContent(currentFocusNode);
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
function isFocusOnMenu() {
|
|
299
|
+
return isCurrentFocusOnMenu(currentFocusNode);
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
function landFocusToWithoutScrolling(id) {
|
|
303
|
+
if (id) {
|
|
304
|
+
// set focus on selected menu item
|
|
305
|
+
const direction = undefined;
|
|
306
|
+
|
|
307
|
+
const context: FocusManager.FocusContext = {
|
|
308
|
+
source: "back",
|
|
309
|
+
preserveScroll: true,
|
|
310
|
+
};
|
|
311
|
+
|
|
312
|
+
logger.log({ message: "landFocusTo", data: { id } });
|
|
313
|
+
|
|
314
|
+
blur(direction);
|
|
315
|
+
setFocus(id, direction, context);
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
function isTabsScreenContentFocused() {
|
|
320
|
+
return isTabsScreenOnContentFocused(currentFocusNode);
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
function focusOnSelectedTab(item: ZappEntry): void {
|
|
324
|
+
// Move focus to appropriate top navigation tab with context
|
|
325
|
+
const selectedTabId = getFocusableId(item.id);
|
|
326
|
+
|
|
327
|
+
// Set focus with back button context to tabs-menu
|
|
328
|
+
landFocusToWithoutScrolling(selectedTabId);
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
function focusOnSelectedTopMenuItem(
|
|
332
|
+
selectedMenuItemId: Option<string>
|
|
333
|
+
): void {
|
|
334
|
+
// Set focus with back button context to top-menu
|
|
335
|
+
landFocusToWithoutScrolling(selectedMenuItemId);
|
|
278
336
|
}
|
|
279
337
|
|
|
280
338
|
/**
|
|
281
339
|
* sets the initial focus when the screen loads, or when focus is lost
|
|
282
340
|
*/
|
|
283
|
-
function setInitialFocus(
|
|
341
|
+
function setInitialFocus(
|
|
342
|
+
lastAddedParentNode?: any,
|
|
343
|
+
context?: FocusManager.FocusContext
|
|
344
|
+
) {
|
|
284
345
|
const preferredFocus = findPriorityItem(
|
|
285
346
|
lastAddedParentNode?.children || focusableTree.root.children
|
|
286
347
|
);
|
|
@@ -326,7 +387,7 @@ export const focusManager = (function () {
|
|
|
326
387
|
},
|
|
327
388
|
});
|
|
328
389
|
|
|
329
|
-
focusableItem && setFocus(focusCandidate.id, null);
|
|
390
|
+
focusableItem && setFocus(focusCandidate.id, null, context);
|
|
330
391
|
|
|
331
392
|
return { success: true };
|
|
332
393
|
}
|
|
@@ -546,6 +607,14 @@ export const focusManager = (function () {
|
|
|
546
607
|
return preferredFocus[0];
|
|
547
608
|
}
|
|
548
609
|
|
|
610
|
+
function isFocusOn(id): boolean {
|
|
611
|
+
return (
|
|
612
|
+
id &&
|
|
613
|
+
isCurrentFocusOnTheTopScreen() &&
|
|
614
|
+
isCurrentFocusOn(id, currentFocusNode)
|
|
615
|
+
);
|
|
616
|
+
}
|
|
617
|
+
|
|
549
618
|
/**
|
|
550
619
|
* this is the list of the functions available externally
|
|
551
620
|
* when importing the focus manager
|
|
@@ -576,5 +645,11 @@ export const focusManager = (function () {
|
|
|
576
645
|
recoverFocus,
|
|
577
646
|
isCurrentFocusOnTheTopScreen,
|
|
578
647
|
findPreferredFocusChild,
|
|
648
|
+
isFocusOnContent,
|
|
649
|
+
isFocusOnMenu,
|
|
650
|
+
isFocusOn,
|
|
651
|
+
focusOnSelectedTopMenuItem,
|
|
652
|
+
focusOnSelectedTab,
|
|
653
|
+
isTabsScreenContentFocused,
|
|
579
654
|
};
|
|
580
655
|
})();
|