@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.
- package/actionsExecutor/ActionExecutorContext.tsx +0 -1
- package/actionsExecutor/ScreenActions.ts +20 -19
- package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
- package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
- package/analyticsUtils/__tests__/analyticsUtils.test.js +0 -11
- package/analyticsUtils/index.tsx +3 -4
- package/analyticsUtils/manager.ts +1 -1
- package/appUtils/HooksManager/Hook.ts +4 -4
- package/appUtils/HooksManager/index.ts +11 -1
- package/appUtils/accessibilityManager/index.ts +3 -12
- package/appUtils/contextKeysManager/contextResolver.ts +29 -1
- package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +5 -0
- package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
- package/appUtils/focusManager/index.ios.ts +10 -0
- package/appUtils/focusManager/index.ts +82 -11
- package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
- package/appUtils/focusManagerAux/utils/index.ts +106 -3
- package/arrayUtils/index.ts +1 -1
- package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
- package/componentsUtils/index.ts +4 -1
- package/configurationUtils/__tests__/manifestKeyParser.test.ts +0 -1
- package/index.d.ts +0 -12
- package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
- package/navigationUtils/index.ts +7 -5
- package/package.json +2 -3
- package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
- package/reactHooks/cell-click/__tests__/index.test.js +3 -0
- 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 +57 -37
- package/reactHooks/feed/index.ts +2 -0
- package/reactHooks/feed/useBatchLoading.ts +15 -8
- package/reactHooks/feed/useFeedLoader.tsx +39 -44
- package/reactHooks/feed/useLoadPipesDataDispatch.ts +57 -0
- package/reactHooks/feed/usePipesCacheReset.ts +2 -2
- package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
- package/reactHooks/layout/__tests__/index.test.tsx +3 -1
- 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 +5 -7
- package/reactHooks/navigation/useScreenStateStore.ts +3 -3
- package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
- package/reactHooks/state/index.ts +1 -1
- package/reactHooks/state/useHomeRiver.ts +4 -2
- package/reactHooks/state/useRivers.ts +7 -8
- package/screenPickerUtils/index.ts +7 -0
- package/storage/ScreenSingleValueProvider.ts +25 -22
- package/storage/ScreenStateMultiSelectProvider.ts +26 -23
- package/testUtils/index.tsx +7 -8
- package/time/BackgroundTimer.ts +1 -1
- package/utils/__tests__/find.test.ts +36 -0
- package/utils/__tests__/pathOr.test.ts +37 -0
- package/utils/__tests__/startsWith.test.ts +30 -0
- package/utils/find.ts +3 -0
- package/utils/index.ts +8 -0
- package/utils/pathOr.ts +5 -0
- package/utils/startsWith.ts +9 -0
|
@@ -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:
|
|
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
|
|
38
|
+
const key = action.options?.key;
|
|
38
39
|
|
|
39
|
-
if (!
|
|
40
|
-
log_error("handleAction: screenSetVariable action missing key
|
|
41
|
-
|
|
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
|
-
|
|
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
|
|
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
|
|
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
|
-
|
|
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:
|
|
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
|
|
114
|
+
const key = action.options?.key;
|
|
114
115
|
|
|
115
|
-
if (
|
|
116
|
+
if (key && tag) {
|
|
116
117
|
const multiSelectProvider = ScreenMultiSelectProvider.getProvider(
|
|
117
|
-
|
|
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
|
|
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
|
-
|
|
156
|
-
|
|
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:
|
|
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
|
-
|
|
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();
|
package/analyticsUtils/index.tsx
CHANGED
|
@@ -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.
|
|
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(
|
|
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
|
-
{
|
|
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 "
|
|
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"
|
|
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"
|
|
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(
|
|
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],
|
|
@@ -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(
|
|
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(
|
|
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
|
})();
|