@applicaster/zapp-react-native-utils 14.0.0-alpha.5114565431 → 14.0.0-alpha.5203410334
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 +1 -1
- package/analyticsUtils/__tests__/analyticsUtils.test.js +0 -11
- package/appUtils/contextKeysManager/contextResolver.ts +42 -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/focusManagerAux/utils/index.ts +106 -3
- package/configurationUtils/__tests__/manifestKeyParser.test.ts +0 -1
- package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
- package/navigationUtils/index.ts +6 -4
- package/package.json +2 -3
- package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +3 -1
- 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 +8 -2
- package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +71 -31
- package/reactHooks/feed/index.ts +2 -0
- package/reactHooks/feed/useBatchLoading.ts +15 -8
- package/reactHooks/feed/useFeedLoader.tsx +36 -34
- package/reactHooks/feed/useLoadPipesDataDispatch.ts +57 -0
- package/reactHooks/feed/usePipesCacheReset.ts +2 -2
- 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/useRoute.ts +7 -2
- package/reactHooks/navigation/useScreenStateStore.ts +8 -0
- 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 +204 -0
- package/storage/ScreenStateMultiSelectProvider.ts +293 -0
- package/storage/StorageMultiSelectProvider.ts +192 -0
- package/storage/StorageSingleSelectProvider.ts +108 -0
- 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
|
@@ -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,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
|
})();
|
|
@@ -1,14 +1,31 @@
|
|
|
1
|
-
import {
|
|
2
|
-
|
|
1
|
+
import {
|
|
2
|
+
isNil,
|
|
3
|
+
last,
|
|
4
|
+
startsWith,
|
|
5
|
+
find,
|
|
6
|
+
pathOr,
|
|
7
|
+
} from "@applicaster/zapp-react-native-utils/utils";
|
|
8
|
+
|
|
3
9
|
import {
|
|
4
10
|
QUICK_BRICK_CONTENT,
|
|
5
11
|
QUICK_BRICK_NAVBAR,
|
|
6
12
|
} from "@applicaster/quick-brick-core/const";
|
|
7
13
|
|
|
14
|
+
import {
|
|
15
|
+
getFocusableId,
|
|
16
|
+
SCREEN_PICKER_CONTAINER,
|
|
17
|
+
} from "@applicaster/zapp-react-native-utils/screenPickerUtils";
|
|
18
|
+
|
|
8
19
|
// run check each 300 ms
|
|
9
20
|
// run this check too often could lead to performance penalty on low-end devices
|
|
10
21
|
const HOW_OFTEN_TO_CHECK_CONDITION = 300; // ms
|
|
11
22
|
|
|
23
|
+
const isTopMenu = (node) => startsWith(QUICK_BRICK_NAVBAR, node?.id);
|
|
24
|
+
const isContent = (node) => startsWith(QUICK_BRICK_CONTENT, node?.id);
|
|
25
|
+
const isRoot = (node) => node?.id === "root";
|
|
26
|
+
|
|
27
|
+
const isScrenPicker = (node) => startsWith(SCREEN_PICKER_CONTAINER, node?.id);
|
|
28
|
+
|
|
12
29
|
type Props = {
|
|
13
30
|
maxTimeout: number;
|
|
14
31
|
conditionFn: () => boolean;
|
|
@@ -49,7 +66,7 @@ export const waitForActiveScreen = (currentRoute: string, focusableTree) => {
|
|
|
49
66
|
|
|
50
67
|
const route = find((route) => route.id === currentRoute, routes);
|
|
51
68
|
|
|
52
|
-
return
|
|
69
|
+
return !isNil(route);
|
|
53
70
|
};
|
|
54
71
|
|
|
55
72
|
return waitUntil({
|
|
@@ -99,3 +116,89 @@ export const waitForContent = (focusableTree) => {
|
|
|
99
116
|
conditionFn: contentHasAnyChildren,
|
|
100
117
|
});
|
|
101
118
|
};
|
|
119
|
+
|
|
120
|
+
export const findSelectedTabId = (item: ZappEntry): string => {
|
|
121
|
+
const selectedTabId = getFocusableId(item.id);
|
|
122
|
+
|
|
123
|
+
return selectedTabId;
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
export const findSelectedMenuId = (focusableTree) => {
|
|
127
|
+
// Set focus with back button context
|
|
128
|
+
const navbar = getNavbarNode(focusableTree);
|
|
129
|
+
|
|
130
|
+
const selectedMenuItemId = find(
|
|
131
|
+
(child) => child.component.props.selected,
|
|
132
|
+
navbar.children
|
|
133
|
+
)?.id;
|
|
134
|
+
|
|
135
|
+
return selectedMenuItemId;
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
export const isTabsScreenContentFocused = (node) => {
|
|
139
|
+
if (isRoot(node)) {
|
|
140
|
+
return false;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
if (isTopMenu(node)) {
|
|
144
|
+
return false;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
if (isContent(node)) {
|
|
148
|
+
return false;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
if (isScrenPicker(node)) {
|
|
152
|
+
return true;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
return isTabsScreenContentFocused(node.parent);
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
export const isCurrentFocusOnMenu = (node) => {
|
|
159
|
+
if (isRoot(node)) {
|
|
160
|
+
return false;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
if (isTopMenu(node)) {
|
|
164
|
+
return true;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (isContent(node)) {
|
|
168
|
+
return false;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
return isCurrentFocusOnMenu(node.parent);
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
export const isCurrentFocusOnContent = (node) => {
|
|
175
|
+
if (isRoot(node)) {
|
|
176
|
+
return false;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
if (isTopMenu(node)) {
|
|
180
|
+
return false;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
if (isContent(node)) {
|
|
184
|
+
return true;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
return isCurrentFocusOnContent(node.parent);
|
|
188
|
+
};
|
|
189
|
+
|
|
190
|
+
export const isCurrentFocusOn = (id, node) => {
|
|
191
|
+
if (!node) {
|
|
192
|
+
return false;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
if (isRoot(node)) {
|
|
196
|
+
return false;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
if (node?.id === id) {
|
|
200
|
+
return true;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
return isCurrentFocusOn(id, node.parent);
|
|
204
|
+
};
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
import { mapContentTypesToRivers } from "../index";
|
|
2
|
+
|
|
3
|
+
describe("mapContentTypesToRivers", () => {
|
|
4
|
+
it("should return the correct content types mapped to rivers", () => {
|
|
5
|
+
const state = {
|
|
6
|
+
rivers: {
|
|
7
|
+
"river-1": {
|
|
8
|
+
plugin_type: "river",
|
|
9
|
+
},
|
|
10
|
+
},
|
|
11
|
+
contentTypes: {
|
|
12
|
+
"content-type-1": {
|
|
13
|
+
screen_id: "river-1",
|
|
14
|
+
},
|
|
15
|
+
},
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
const result = mapContentTypesToRivers(state);
|
|
19
|
+
|
|
20
|
+
expect(result).toEqual({
|
|
21
|
+
"content-type-1": {
|
|
22
|
+
screenType: "river",
|
|
23
|
+
screen_id: "river-1",
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
it("should return null if contentTypes is undefined", () => {
|
|
29
|
+
const state = {
|
|
30
|
+
rivers: {
|
|
31
|
+
"river-1": {
|
|
32
|
+
plugin_type: "river",
|
|
33
|
+
},
|
|
34
|
+
},
|
|
35
|
+
// contentTypes is missing
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
const result = mapContentTypesToRivers(state);
|
|
39
|
+
|
|
40
|
+
expect(result).toBeNull();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it("should skip content types whose screen does not exist in rivers", () => {
|
|
44
|
+
const state = {
|
|
45
|
+
rivers: {
|
|
46
|
+
"river-1": {
|
|
47
|
+
plugin_type: "river",
|
|
48
|
+
},
|
|
49
|
+
},
|
|
50
|
+
contentTypes: {
|
|
51
|
+
"content-type-1": {
|
|
52
|
+
screen_id: "river-1",
|
|
53
|
+
},
|
|
54
|
+
"content-type-2": {
|
|
55
|
+
screen_id: "river-2", // river-2 does not exist
|
|
56
|
+
},
|
|
57
|
+
},
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
const result = mapContentTypesToRivers(state);
|
|
61
|
+
|
|
62
|
+
expect(result).toEqual({
|
|
63
|
+
"content-type-1": {
|
|
64
|
+
screenType: "river",
|
|
65
|
+
screen_id: "river-1",
|
|
66
|
+
},
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
// result is not null, but may be undefined for missing keys
|
|
70
|
+
expect(result && result["content-type-2"]).toBeUndefined();
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it("should use 'type' if 'plugin_type' is not present in river", () => {
|
|
74
|
+
const state = {
|
|
75
|
+
rivers: {
|
|
76
|
+
"river-1": {
|
|
77
|
+
type: "custom-type",
|
|
78
|
+
},
|
|
79
|
+
},
|
|
80
|
+
contentTypes: {
|
|
81
|
+
"content-type-1": {
|
|
82
|
+
screen_id: "river-1",
|
|
83
|
+
},
|
|
84
|
+
},
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
const result = mapContentTypesToRivers(state);
|
|
88
|
+
|
|
89
|
+
expect(result).toEqual({
|
|
90
|
+
"content-type-1": {
|
|
91
|
+
screenType: "custom-type",
|
|
92
|
+
screen_id: "river-1",
|
|
93
|
+
},
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
it("should skip content types if neither plugin_type nor type is present in river", () => {
|
|
98
|
+
const state = {
|
|
99
|
+
rivers: {
|
|
100
|
+
"river-1": {
|
|
101
|
+
// no plugin_type or type
|
|
102
|
+
},
|
|
103
|
+
},
|
|
104
|
+
contentTypes: {
|
|
105
|
+
"content-type-1": {
|
|
106
|
+
screen_id: "river-1",
|
|
107
|
+
},
|
|
108
|
+
},
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
const result = mapContentTypesToRivers(state);
|
|
112
|
+
|
|
113
|
+
expect(result).toEqual({});
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
it("should handle empty contentTypes object", () => {
|
|
117
|
+
const state = {
|
|
118
|
+
rivers: {
|
|
119
|
+
"river-1": {
|
|
120
|
+
plugin_type: "river",
|
|
121
|
+
},
|
|
122
|
+
},
|
|
123
|
+
contentTypes: {},
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
const result = mapContentTypesToRivers(state);
|
|
127
|
+
|
|
128
|
+
expect(result).toEqual({});
|
|
129
|
+
});
|
|
130
|
+
});
|
package/navigationUtils/index.ts
CHANGED
|
@@ -13,6 +13,7 @@ import {
|
|
|
13
13
|
isPlayable,
|
|
14
14
|
isV2River,
|
|
15
15
|
} from "./itemTypeMatchers";
|
|
16
|
+
import { RootState } from "@applicaster/zapp-react-native-redux/store";
|
|
16
17
|
|
|
17
18
|
type PathAttribute = {
|
|
18
19
|
screenType: string;
|
|
@@ -377,10 +378,11 @@ export const usesVideoModal = (
|
|
|
377
378
|
return targetScreenConfiguration?.styles?.use_video_modal;
|
|
378
379
|
};
|
|
379
380
|
|
|
380
|
-
export const mapContentTypesToRivers = (
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
381
|
+
export const mapContentTypesToRivers = (
|
|
382
|
+
state: Partial<RootState>
|
|
383
|
+
): ZappContentTypesMapped | null => {
|
|
384
|
+
const { rivers, contentTypes } = state;
|
|
385
|
+
|
|
384
386
|
if (!contentTypes) {
|
|
385
387
|
return null;
|
|
386
388
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@applicaster/zapp-react-native-utils",
|
|
3
|
-
"version": "14.0.0-alpha.
|
|
3
|
+
"version": "14.0.0-alpha.5203410334",
|
|
4
4
|
"description": "Applicaster Zapp React Native utilities package",
|
|
5
5
|
"main": "index.js",
|
|
6
6
|
"types": "index.d.ts",
|
|
@@ -27,7 +27,7 @@
|
|
|
27
27
|
},
|
|
28
28
|
"homepage": "https://github.com/applicaster/quickbrick#readme",
|
|
29
29
|
"dependencies": {
|
|
30
|
-
"@applicaster/applicaster-types": "14.0.0-alpha.
|
|
30
|
+
"@applicaster/applicaster-types": "14.0.0-alpha.5203410334",
|
|
31
31
|
"buffer": "^5.2.1",
|
|
32
32
|
"camelize": "^1.0.0",
|
|
33
33
|
"dayjs": "^1.11.10",
|
|
@@ -38,7 +38,6 @@
|
|
|
38
38
|
"peerDependencies": {
|
|
39
39
|
"@applicaster/zapp-pipes-v2-client": "*",
|
|
40
40
|
"@react-native-community/netinfo": "*",
|
|
41
|
-
"immer": "*",
|
|
42
41
|
"react": "*",
|
|
43
42
|
"react-native": "*",
|
|
44
43
|
"uglify-js": "*",
|
|
@@ -23,7 +23,9 @@ jest.mock(
|
|
|
23
23
|
|
|
24
24
|
jest.useFakeTimers();
|
|
25
25
|
|
|
26
|
-
jest.mock(
|
|
26
|
+
jest.mock(
|
|
27
|
+
"@applicaster/zapp-react-native-utils/reactHooks/navigation/useNavigation"
|
|
28
|
+
);
|
|
27
29
|
|
|
28
30
|
const mockStore = configureStore();
|
|
29
31
|
|
|
@@ -26,6 +26,9 @@ jest.mock("@applicaster/zapp-react-native-utils/analyticsUtils/", () => ({
|
|
|
26
26
|
}));
|
|
27
27
|
|
|
28
28
|
jest.mock("@applicaster/zapp-react-native-utils/reactHooks/screen", () => ({
|
|
29
|
+
...jest.requireActual(
|
|
30
|
+
"@applicaster/zapp-react-native-utils/reactHooks/screen"
|
|
31
|
+
),
|
|
29
32
|
useTargetScreenData: jest.fn(() => ({})),
|
|
30
33
|
useCurrentScreenData: jest.fn(() => ({})),
|
|
31
34
|
}));
|
|
@@ -16,7 +16,8 @@ import { ActionExecutorContext } from "@applicaster/zapp-react-native-utils/acti
|
|
|
16
16
|
import { isFunction, noop } from "../../functionUtils";
|
|
17
17
|
import { useSendAnalyticsOnPress } from "../analytics";
|
|
18
18
|
import { logOnPress, warnEmptyContentType } from "./helpers";
|
|
19
|
-
import { useCurrentScreenData } from "../screen";
|
|
19
|
+
import { useCurrentScreenData, useScreenContext } from "../screen";
|
|
20
|
+
import { useScreenStateStore } from "../navigation/useScreenStateStore";
|
|
20
21
|
|
|
21
22
|
/**
|
|
22
23
|
* If onCellTap is defined execute the function and
|
|
@@ -42,10 +43,12 @@ export const useCellClick = ({
|
|
|
42
43
|
}: Props): onPressReturnFn => {
|
|
43
44
|
const { push, currentRoute } = useNavigation();
|
|
44
45
|
const { pathname } = useRoute();
|
|
46
|
+
const screenStateStore = useScreenStateStore();
|
|
45
47
|
|
|
46
48
|
const onCellTap: Option<Function> = React.useContext(CellTapContext);
|
|
47
49
|
const actionExecutor = React.useContext(ActionExecutorContext);
|
|
48
50
|
const screenData = useCurrentScreenData();
|
|
51
|
+
const screenState = useScreenContext()?.options;
|
|
49
52
|
|
|
50
53
|
const cellSelectable = toBooleanWithDefaultTrue(
|
|
51
54
|
component?.rules?.component_cells_selectable
|
|
@@ -83,6 +86,9 @@ export const useCellClick = ({
|
|
|
83
86
|
await actionExecutor?.handleEntryActions(selectedItem, {
|
|
84
87
|
component,
|
|
85
88
|
screenData,
|
|
89
|
+
screenState,
|
|
90
|
+
screenRoute: pathname,
|
|
91
|
+
screenStateStore,
|
|
86
92
|
});
|
|
87
93
|
}
|
|
88
94
|
|
|
@@ -117,6 +123,7 @@ export const useCellClick = ({
|
|
|
117
123
|
push,
|
|
118
124
|
sendAnalyticsOnPress,
|
|
119
125
|
screenData,
|
|
126
|
+
screenState,
|
|
120
127
|
]
|
|
121
128
|
);
|
|
122
129
|
|
|
@@ -12,7 +12,6 @@ describe("Debug utils", () => {
|
|
|
12
12
|
// Clear the timers object
|
|
13
13
|
Object.keys(timers).forEach((key) => delete timers[key]);
|
|
14
14
|
|
|
15
|
-
// Mock performance.now()
|
|
16
15
|
// eslint-disable-next-line no-undef
|
|
17
16
|
performanceNowMock = jest.spyOn(performance, "now");
|
|
18
17
|
performanceNowMock.mockReturnValue(0); // Initial value
|