@applicaster/zapp-react-native-utils 14.0.0-rc.9 → 15.0.0-rc.1

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 (135) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +60 -84
  2. package/actionsExecutor/ScreenActions.ts +164 -0
  3. package/actionsExecutor/StorageActions.ts +110 -0
  4. package/actionsExecutor/feedDecorator.ts +171 -0
  5. package/actionsExecutor/screenResolver.ts +11 -0
  6. package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
  7. package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
  8. package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
  9. package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
  10. package/analyticsUtils/__tests__/analyticsUtils.test.js +3 -0
  11. package/analyticsUtils/events.ts +8 -0
  12. package/analyticsUtils/index.tsx +3 -4
  13. package/analyticsUtils/manager.ts +1 -1
  14. package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
  15. package/appUtils/HooksManager/Hook.ts +4 -4
  16. package/appUtils/HooksManager/index.ts +11 -1
  17. package/appUtils/accessibilityManager/const.ts +13 -0
  18. package/appUtils/accessibilityManager/hooks.ts +35 -1
  19. package/appUtils/accessibilityManager/index.ts +154 -30
  20. package/appUtils/accessibilityManager/utils.ts +24 -0
  21. package/appUtils/contextKeysManager/contextResolver.ts +42 -1
  22. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +7 -0
  23. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  24. package/appUtils/focusManager/events.ts +2 -0
  25. package/appUtils/focusManager/index.ios.ts +10 -0
  26. package/appUtils/focusManager/index.ts +86 -11
  27. package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
  28. package/appUtils/focusManagerAux/utils/index.ts +94 -3
  29. package/appUtils/platform/platformUtils.ts +31 -1
  30. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +0 -15
  31. package/appUtils/playerManager/useChapterMarker.tsx +0 -1
  32. package/appUtils/playerManager/usePlayerControllerSetup.tsx +16 -0
  33. package/arrayUtils/__tests__/isEmptyArray.test.ts +63 -0
  34. package/arrayUtils/__tests__/isFilledArray.test.ts +1 -1
  35. package/arrayUtils/index.ts +8 -3
  36. package/audioPlayerUtils/__tests__/getArtworkImage.test.ts +144 -0
  37. package/audioPlayerUtils/__tests__/getBackgroundImage.test.ts +72 -0
  38. package/audioPlayerUtils/__tests__/getImageFromEntry.test.ts +110 -0
  39. package/audioPlayerUtils/assets/index.ts +2 -0
  40. package/audioPlayerUtils/index.ts +242 -0
  41. package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
  42. package/componentsUtils/index.ts +4 -1
  43. package/conf/player/__tests__/selectors.test.ts +34 -0
  44. package/conf/player/selectors.ts +10 -0
  45. package/configurationUtils/__tests__/configurationUtils.test.js +0 -31
  46. package/configurationUtils/__tests__/getMediaItems.test.ts +65 -0
  47. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +34 -0
  48. package/configurationUtils/__tests__/manifestKeyParser.test.ts +546 -0
  49. package/configurationUtils/index.ts +64 -35
  50. package/configurationUtils/manifestKeyParser.ts +57 -32
  51. package/focusManager/FocusManager.ts +104 -20
  52. package/focusManager/Tree.ts +25 -21
  53. package/focusManager/__tests__/FocusManager.test.ts +50 -8
  54. package/focusManager/aux/index.ts +98 -0
  55. package/focusManager/utils.ts +12 -6
  56. package/index.d.ts +1 -10
  57. package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
  58. package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
  59. package/manifestUtils/createConfig.js +4 -1
  60. package/manifestUtils/defaultManifestConfigurations/player.js +2764 -1539
  61. package/manifestUtils/index.js +4 -0
  62. package/manifestUtils/keys.js +12 -0
  63. package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
  64. package/manifestUtils/sharedConfiguration/screenPicker/stylesFields.js +6 -0
  65. package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
  66. package/navigationUtils/index.ts +7 -5
  67. package/package.json +2 -3
  68. package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
  69. package/playerUtils/PlayerTTS/index.ts +1 -0
  70. package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
  71. package/playerUtils/__tests__/getPlayerActionButtons.test.ts +54 -0
  72. package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
  73. package/playerUtils/_internals/index.ts +1 -0
  74. package/playerUtils/_internals/utils.ts +31 -0
  75. package/playerUtils/configurationUtils.ts +0 -44
  76. package/playerUtils/getPlayerActionButtons.ts +17 -0
  77. package/playerUtils/index.ts +53 -0
  78. package/playerUtils/usePlayerTTS.ts +21 -0
  79. package/playerUtils/useValidatePlayerConfig.tsx +22 -19
  80. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
  81. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  82. package/reactHooks/cell-click/index.ts +8 -1
  83. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  84. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +47 -90
  85. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +71 -31
  86. package/reactHooks/feed/index.ts +2 -0
  87. package/reactHooks/feed/useBatchLoading.ts +17 -10
  88. package/reactHooks/feed/useFeedLoader.tsx +36 -34
  89. package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
  90. package/reactHooks/feed/usePipesCacheReset.ts +3 -3
  91. package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
  92. package/reactHooks/layout/__tests__/index.test.tsx +3 -1
  93. package/reactHooks/layout/isTablet/index.ts +12 -5
  94. package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
  95. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
  96. package/reactHooks/layout/useLayoutVersion.ts +5 -5
  97. package/reactHooks/navigation/index.ts +7 -5
  98. package/reactHooks/navigation/useIsScreenActive.ts +9 -5
  99. package/reactHooks/navigation/useRoute.ts +7 -2
  100. package/reactHooks/navigation/useScreenStateStore.ts +8 -0
  101. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
  102. package/reactHooks/screen/useScreenContext.ts +1 -1
  103. package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
  104. package/reactHooks/state/index.ts +1 -1
  105. package/reactHooks/state/useHomeRiver.ts +4 -2
  106. package/reactHooks/state/useRivers.ts +7 -8
  107. package/riverComponetsMeasurementProvider/index.tsx +1 -1
  108. package/screenPickerUtils/index.ts +13 -0
  109. package/services/js2native.ts +1 -0
  110. package/storage/ScreenSingleValueProvider.ts +204 -0
  111. package/storage/ScreenStateMultiSelectProvider.ts +293 -0
  112. package/storage/StorageMultiSelectProvider.ts +192 -0
  113. package/storage/StorageSingleSelectProvider.ts +108 -0
  114. package/testUtils/index.tsx +7 -8
  115. package/time/BackgroundTimer.ts +6 -4
  116. package/utils/__tests__/endsWith.test.ts +30 -0
  117. package/utils/__tests__/find.test.ts +36 -0
  118. package/utils/__tests__/omit.test.ts +19 -0
  119. package/utils/__tests__/path.test.ts +33 -0
  120. package/utils/__tests__/pathOr.test.ts +37 -0
  121. package/utils/__tests__/startsWith.test.ts +30 -0
  122. package/utils/__tests__/take.test.ts +40 -0
  123. package/utils/endsWith.ts +9 -0
  124. package/utils/find.ts +3 -0
  125. package/utils/index.ts +33 -1
  126. package/utils/omit.ts +5 -0
  127. package/utils/path.ts +5 -0
  128. package/utils/pathOr.ts +5 -0
  129. package/utils/startsWith.ts +9 -0
  130. package/utils/take.ts +5 -0
  131. package/zappFrameworkUtils/HookCallback/callbackNavigationAction.ts +164 -0
  132. package/zappFrameworkUtils/HookCallback/hookCallbackManifestExtensions.config.js +60 -0
  133. package/zappFrameworkUtils/HookCallback/useCallbackActions.ts +22 -0
  134. package/zappFrameworkUtils/loginPluginUtils.ts +1 -1
  135. package/playerUtils/configurationGenerator.ts +0 -2572
@@ -0,0 +1,108 @@
1
+ import { bridgeLogger } from "../../zapp-react-native-bridge/logger";
2
+ import { BehaviorSubject } from "rxjs";
3
+ import { localStorage } from "@applicaster/zapp-react-native-bridge/ZappStorage/LocalStorage";
4
+ import { getNamespaceAndKey } from "../appUtils/contextKeysManager/utils";
5
+ import { createLogger } from "../logger";
6
+
7
+ export const { log_verbose, log_debug, log_warning, log_info, log_error } =
8
+ createLogger({
9
+ category: "StorageSingleValueProvider",
10
+ subsystem: "zapp-react-native-bridge",
11
+ parent: bridgeLogger,
12
+ });
13
+
14
+ export interface SingleValueProvider {
15
+ getObservable(): BehaviorSubject<string | null>;
16
+
17
+ setValue(value: string): Promise<void>;
18
+ getValue(): string | null;
19
+ getValueAsync(): Promise<string | null>;
20
+ clearValue(): Promise<void>;
21
+ }
22
+
23
+ interface StorageListenerArgs {
24
+ value: string | null;
25
+ }
26
+
27
+ export class StorageSingleValueProvider implements SingleValueProvider {
28
+ // Static cache of providers by namespace
29
+ private static singleValueProviders: Record<
30
+ string,
31
+ StorageSingleValueProvider
32
+ > = {};
33
+
34
+ public static getProvider(keyNamespace: string): SingleValueProvider {
35
+ if (!this.singleValueProviders[keyNamespace]) {
36
+ this.singleValueProviders[keyNamespace] = new StorageSingleValueProvider(
37
+ keyNamespace
38
+ );
39
+ }
40
+
41
+ return this.singleValueProviders[keyNamespace];
42
+ }
43
+
44
+ private valueSubject: BehaviorSubject<string | null>;
45
+
46
+ private readonly key: string;
47
+ private readonly namespace: string;
48
+
49
+ private constructor(keyNamespace: string) {
50
+ const { namespace, key } = getNamespaceAndKey(keyNamespace);
51
+
52
+ if (!key) {
53
+ throw new Error("StorageSingleValueProvider: Key is required");
54
+ }
55
+
56
+ this.key = key;
57
+ this.namespace = namespace;
58
+ localStorage.addListener?.({ key, namespace }, this.reloadValue);
59
+
60
+ this.valueSubject = new BehaviorSubject<string | null>(null);
61
+
62
+ void this.getValueAsync();
63
+ log_debug("StorageSingleValueProvider: Initializing");
64
+ }
65
+
66
+ private reloadValue = async ({ value }: StorageListenerArgs) => {
67
+ log_debug(`reloadValue: request to reload value: ${value}`);
68
+
69
+ this.valueSubject.next(value);
70
+ };
71
+
72
+ public getObservable = (): BehaviorSubject<string | null> =>
73
+ this.valueSubject;
74
+
75
+ private async updateValueAndNotifyObservers(value: string | null) {
76
+ if (value === null) {
77
+ await localStorage.removeItem(this.key, this.namespace);
78
+ } else {
79
+ await localStorage.setItem(this.key, value, this.namespace);
80
+ }
81
+
82
+ this.valueSubject.next(value);
83
+ }
84
+
85
+ public setValue = async (value: string): Promise<void> => {
86
+ log_debug(`setValue: Setting new value: ${value}`);
87
+
88
+ await this.updateValueAndNotifyObservers(value);
89
+ };
90
+
91
+ public clearValue = async (): Promise<void> => {
92
+ await localStorage.removeItem(this.key, this.namespace);
93
+ log_debug("clearValue: Removing value");
94
+
95
+ this.valueSubject.next(null);
96
+ };
97
+
98
+ public getValueAsync = async (): Promise<string | null> => {
99
+ const value = await localStorage.getItem(this.key, this.namespace);
100
+ this.valueSubject.next(value);
101
+
102
+ return value;
103
+ };
104
+
105
+ public getValue = (): string | null => {
106
+ return this.valueSubject.getValue();
107
+ };
108
+ }
@@ -1,17 +1,16 @@
1
- import * as R from "ramda";
2
-
1
+ import { SafeAreaProvider } from "react-native-safe-area-context";
2
+ import { render } from "@testing-library/react-native";
3
3
  import React, { PropsWithChildren } from "react";
4
- import { View } from "react-native";
5
-
4
+ import configureStore from "redux-mock-store";
6
5
  import { Provider } from "react-redux";
6
+ import { View } from "react-native";
7
7
  import thunk from "redux-thunk";
8
- import configureStore from "redux-mock-store";
9
- import { SafeAreaProvider } from "react-native-safe-area-context";
8
+ import * as R from "ramda";
9
+
10
10
  import { appStore } from "@applicaster/zapp-react-native-redux/AppStore";
11
11
 
12
- import { render } from "@testing-library/react-native";
13
- import { AnalyticsProvider } from "../analyticsUtils";
14
12
  import { ThemeContext } from "../theme";
13
+ import { AnalyticsProvider } from "../analyticsUtils";
15
14
 
16
15
  export { getByTestId } from "./getByTestId";
17
16
 
@@ -11,15 +11,17 @@ class BackgroundTimer {
11
11
  this.uniqueId = 0;
12
12
  this.callbacks = {};
13
13
 
14
- const EventEmitter = platformSelect({
14
+ const EventEmitter: typeof DeviceEventEmitter | undefined = platformSelect({
15
15
  android: DeviceEventEmitter,
16
- ios: undefined,
16
+ android_tv: DeviceEventEmitter,
17
+ amazon: DeviceEventEmitter, // probably does not exist and uses android_tv
17
18
  default: undefined,
18
19
  });
19
20
 
20
21
  EventEmitter?.addListener("BackgroundTimer.timer.fired", (id: number) => {
21
- if (this.callbacks[id]) {
22
- const callback = this.callbacks[id];
22
+ const callback = this.callbacks[id];
23
+
24
+ if (callback) {
23
25
  delete this.callbacks[id];
24
26
  callback();
25
27
  }
@@ -0,0 +1,30 @@
1
+ import { endsWith } from "../endsWith";
2
+
3
+ describe("endsWith", () => {
4
+ it("returns false when str is null", () => {
5
+ expect(endsWith("a", null)).toBe(false);
6
+ });
7
+
8
+ it("returns false when str is undefined", () => {
9
+ expect(endsWith("a", undefined)).toBe(false);
10
+ });
11
+
12
+ it("returns true when string ends with target", () => {
13
+ expect(endsWith("lo", "hello")).toBe(true);
14
+ expect(endsWith("", "hello")).toBe(true); // empty target always matches
15
+ });
16
+
17
+ it("returns false when string does not end with target", () => {
18
+ expect(endsWith("yo", "hello")).toBe(false);
19
+ });
20
+
21
+ it("works with single character target", () => {
22
+ expect(endsWith("o", "hello")).toBe(true);
23
+ expect(endsWith("x", "hello")).toBe(false);
24
+ });
25
+
26
+ it("is case-sensitive", () => {
27
+ expect(endsWith("Lo", "hello")).toBe(false);
28
+ expect(endsWith("lo", "hello")).toBe(true);
29
+ });
30
+ });
@@ -0,0 +1,36 @@
1
+ import { find } from "../find";
2
+
3
+ test("example 1", () => {
4
+ const predicate = <T>(_: T, index: number): boolean => index === 0;
5
+ const xs = ["1", "2", "2", "3", "4"];
6
+
7
+ expect(find(predicate, xs)).toBe("1");
8
+ });
9
+
10
+ test("example 2", () => {
11
+ const predicate = <T>(_: T, index: number): boolean => index === 0;
12
+ const xs: string[] = [];
13
+
14
+ expect(find(predicate, xs)).toBe(undefined);
15
+ });
16
+
17
+ test("example 3", () => {
18
+ const predicate = () => false;
19
+ const xs = ["1", "2", "2", "3"];
20
+
21
+ expect(find(predicate, xs)).toBe(undefined);
22
+ });
23
+
24
+ test("example 4", () => {
25
+ const predicate = <T>(_: T, index: number): boolean => index === 1;
26
+ const xs = ["1", "2", "2", "3"];
27
+
28
+ expect(find(predicate, xs)).toBe("2");
29
+ });
30
+
31
+ test("example 5", () => {
32
+ const predicate = <T>(_: T, index: number): boolean => index === 2;
33
+ const xs = ["1", "2.1", "2", "3", "2", "4"];
34
+
35
+ expect(find(predicate, xs)).toBe("2");
36
+ });
@@ -0,0 +1,19 @@
1
+ import { omit } from "../omit";
2
+
3
+ test("example 1", () => {
4
+ const path = ["a", "b", "c"];
5
+ const record = { a: 1, b: 2, c: 3 };
6
+
7
+ const output = {};
8
+
9
+ expect(omit(path, record)).toEqual(output);
10
+ });
11
+
12
+ test("example 2", () => {
13
+ const path = ["a", "b"];
14
+ const record = { a: 1, b: 2, c: 3 };
15
+
16
+ const output = { c: 3 };
17
+
18
+ expect(omit(path, record)).toEqual(output);
19
+ });
@@ -0,0 +1,33 @@
1
+ import { path } from "../path";
2
+
3
+ test("example 1", () => {
4
+ const route = ["a", "b", "c"];
5
+ const xs = { a: { b: { c: 1 } } };
6
+
7
+ const output = 1;
8
+
9
+ expect(path(route, xs)).toEqual(output);
10
+ });
11
+
12
+ test("example 2", () => {
13
+ const route = ["a", "b"];
14
+ const xs = { a: { b: { c: 1 } } };
15
+
16
+ const output = { c: 1 };
17
+
18
+ expect(path(route, xs)).toEqual(output);
19
+ });
20
+
21
+ test("example 3", () => {
22
+ const route = ["a", "b", "x"];
23
+ const xs = { a: { b: { c: 1 } } };
24
+
25
+ expect(path(route, xs)).toBeUndefined();
26
+ });
27
+
28
+ test("example 4", () => {
29
+ const route = ["a", "b", "c"];
30
+ const xs = undefined;
31
+
32
+ expect(path(route, xs)).toBeUndefined();
33
+ });
@@ -0,0 +1,37 @@
1
+ import { pathOr } from "../pathOr";
2
+
3
+ test("example 1", () => {
4
+ const defaultValue = "defaultValue";
5
+ const path = ["a", "b", "c"];
6
+ const xs = { a: { b: { c: 1 } } };
7
+
8
+ const output = 1;
9
+
10
+ expect(pathOr(defaultValue, path, xs)).toEqual(output);
11
+ });
12
+
13
+ test("example 2", () => {
14
+ const defaultValue = "defaultValue";
15
+ const path = ["a", "b"];
16
+ const xs = { a: { b: { c: 1 } } };
17
+
18
+ const output = { c: 1 };
19
+
20
+ expect(pathOr(defaultValue, path, xs)).toEqual(output);
21
+ });
22
+
23
+ test("example 3", () => {
24
+ const defaultValue = "defaultValue";
25
+ const path = ["a", "b", "x"];
26
+ const xs = { a: { b: { c: 1 } } };
27
+
28
+ expect(pathOr(defaultValue, path, xs)).toBe(defaultValue);
29
+ });
30
+
31
+ test("example 4", () => {
32
+ const defaultValue = "defaultValue";
33
+ const path = ["a", "b", "c"];
34
+ const xs = undefined;
35
+
36
+ expect(pathOr(defaultValue, path, xs)).toBe(defaultValue);
37
+ });
@@ -0,0 +1,30 @@
1
+ import { startsWith } from "../startsWith";
2
+
3
+ describe("startsWith", () => {
4
+ it("returns false when str is null", () => {
5
+ expect(startsWith("a", null)).toBe(false);
6
+ });
7
+
8
+ it("returns false when str is undefined", () => {
9
+ expect(startsWith("a", undefined)).toBe(false);
10
+ });
11
+
12
+ it("returns true when string starts with target", () => {
13
+ expect(startsWith("he", "hello")).toBe(true);
14
+ expect(startsWith("", "hello")).toBe(true); // empty target always matches
15
+ });
16
+
17
+ it("returns false when string does not start with target", () => {
18
+ expect(startsWith("yo", "hello")).toBe(false);
19
+ });
20
+
21
+ it("works with single character target", () => {
22
+ expect(startsWith("h", "hello")).toBe(true);
23
+ expect(startsWith("x", "hello")).toBe(false);
24
+ });
25
+
26
+ it("is case-sensitive", () => {
27
+ expect(startsWith("He", "hello")).toBe(false);
28
+ expect(startsWith("he", "hello")).toBe(true);
29
+ });
30
+ });
@@ -0,0 +1,40 @@
1
+ import { take } from "../take";
2
+
3
+ describe("take", () => {
4
+ it("takes n elements from the beginning", () => {
5
+ expect(take(2, [1, 2, 3])).toEqual([1, 2]);
6
+ });
7
+
8
+ it("returns the whole array if n is larger than length", () => {
9
+ expect(take(5, [1, 2, 3])).toEqual([1, 2, 3]);
10
+ });
11
+
12
+ it("returns empty array if n is 0", () => {
13
+ expect(take(0, [1, 2, 3])).toEqual([]);
14
+ });
15
+
16
+ it("returns empty array for empty input array", () => {
17
+ expect(take(2, [])).toEqual([]);
18
+ });
19
+
20
+ it("returns empty array if n is negative", () => {
21
+ expect(take(-1, [1, 2, 3])).toEqual([]);
22
+ });
23
+
24
+ it("works with strings in array", () => {
25
+ expect(take(2, ["a", "b", "c"])).toEqual(["a", "b"]);
26
+ });
27
+
28
+ it("works with objects in array", () => {
29
+ const arr = [{ id: 1 }, { id: 2 }];
30
+ expect(take(1, arr)).toEqual([{ id: 1 }]);
31
+ });
32
+
33
+ it("returns empty array if input is not an array", () => {
34
+ // @ts-expect-error testing non-array input
35
+ expect(take(2, null)).toEqual([]);
36
+
37
+ // @ts-expect-error testing non-array input
38
+ expect(take(2, undefined)).toEqual([]);
39
+ });
40
+ });
@@ -0,0 +1,9 @@
1
+ import { isNil } from "lodash";
2
+
3
+ export const endsWith = (target, str) => {
4
+ if (isNil(str)) {
5
+ return false;
6
+ }
7
+
8
+ return str.endsWith(target);
9
+ };
package/utils/find.ts ADDED
@@ -0,0 +1,3 @@
1
+ export const find = (predicate, xs) => {
2
+ return (xs || []).find((x, index) => predicate(x, index));
3
+ };
package/utils/index.ts CHANGED
@@ -2,4 +2,36 @@ export { chunk } from "./chunk";
2
2
 
3
3
  export { times } from "./times";
4
4
 
5
- export { cloneDeep as clone, flatten, drop, size, isNil } from "lodash";
5
+ export { startsWith } from "./startsWith";
6
+
7
+ export { find } from "./find";
8
+
9
+ export { pathOr } from "./pathOr";
10
+
11
+ export { path } from "./path";
12
+
13
+ export { omit } from "./omit";
14
+
15
+ export { endsWith } from "./endsWith";
16
+
17
+ export { take } from "./take";
18
+
19
+ export {
20
+ cloneDeep as clone,
21
+ flatten,
22
+ drop,
23
+ size,
24
+ isNil,
25
+ isEmpty,
26
+ get,
27
+ has,
28
+ flatMap,
29
+ difference,
30
+ pick,
31
+ map,
32
+ trim,
33
+ toString,
34
+ last,
35
+ toLower,
36
+ isEqual as equals,
37
+ } from "lodash";
package/utils/omit.ts ADDED
@@ -0,0 +1,5 @@
1
+ import { omit as Lodash_omit } from "lodash";
2
+
3
+ export const omit = (path, record) => {
4
+ return Lodash_omit(record, path);
5
+ };
package/utils/path.ts ADDED
@@ -0,0 +1,5 @@
1
+ import { get } from "lodash";
2
+
3
+ export const path = (route, record) => {
4
+ return get(record, route, undefined);
5
+ };
@@ -0,0 +1,5 @@
1
+ import { get } from "lodash";
2
+
3
+ export const pathOr = (defaultValue, path, record) => {
4
+ return get(record, path, defaultValue);
5
+ };
@@ -0,0 +1,9 @@
1
+ import { isNil } from "lodash";
2
+
3
+ export const startsWith = (target, str) => {
4
+ if (isNil(str)) {
5
+ return false;
6
+ }
7
+
8
+ return str.startsWith(target);
9
+ };
package/utils/take.ts ADDED
@@ -0,0 +1,5 @@
1
+ import { take as Ltake } from "lodash";
2
+
3
+ export function take<T>(n: number, xs: T[]): T[] {
4
+ return Ltake(xs, n);
5
+ }
@@ -0,0 +1,164 @@
1
+ import { useNavigation, useRivers, useScreenContext } from "../../reactHooks";
2
+ import { createLogger } from "../../logger";
3
+ import { useCallback, useMemo } from "react";
4
+
5
+ export enum NavigationCallbackOptions {
6
+ DEFAULT = "default",
7
+ GO_HOME = "go_home",
8
+ GO_BACK = "go_back",
9
+ GO_TO_SCREEN = "go_to_screen",
10
+ }
11
+
12
+ export const CALLBACK_NAVIGATION_KEY = "hook_callback_navigation";
13
+
14
+ export const CALLBACK_NAVIGATION_GO_TO_SCREEN_KEY =
15
+ "hook_callback_navigation_go_to_screen";
16
+
17
+ type NavKeys = {
18
+ action: NavigationCallbackOptions;
19
+ targetScreenId: string | null;
20
+ } | null;
21
+
22
+ type General = Record<string, unknown>;
23
+
24
+ const LogPrefix = "useCallbackNavigationAction:";
25
+
26
+ const { log_info, log_verbose } = createLogger({
27
+ subsystem: "hook-navigation-callback",
28
+ });
29
+
30
+ const legacyMappingKeys = {
31
+ "quick-brick-login-flow": {
32
+ actionType: "login_completion_action",
33
+ targetScreen: "navigate_to_login_screen",
34
+ },
35
+ "quick-brick-user-account-ui-component": {
36
+ actionType: "callbackAction",
37
+ },
38
+ };
39
+
40
+ const isNonEmptyString = (v: unknown): v is string =>
41
+ typeof v === "string" && v.trim().length > 0;
42
+
43
+ const NAV_ACTIONS = Object.values(NavigationCallbackOptions) as string[];
44
+
45
+ const isNavAction = (v: unknown): v is NavigationCallbackOptions =>
46
+ typeof v === "string" && NAV_ACTIONS.includes(v.trim());
47
+
48
+ export const getNavigationKeys = (
49
+ item?: ZappUIComponent | ZappRiver
50
+ ): NavKeys => {
51
+ const general = (item?.general ?? {}) as General;
52
+ const pluginIdentifier = item?.type ?? "";
53
+ const legacy = legacyMappingKeys[pluginIdentifier] ?? {};
54
+
55
+ const rawAction =
56
+ (general as General).hook_callback_navigation ??
57
+ (legacy.actionType ? (general as General)[legacy.actionType] : undefined);
58
+
59
+ let action: NavigationCallbackOptions | null = null;
60
+
61
+ if (isNonEmptyString(rawAction)) {
62
+ const trimmed = rawAction.trim();
63
+ action = isNavAction(trimmed) ? trimmed : null;
64
+ }
65
+
66
+ if (!action) return null;
67
+
68
+ let targetScreenId: string | null = null;
69
+
70
+ if (action === NavigationCallbackOptions.GO_TO_SCREEN) {
71
+ const screenId: string | null =
72
+ ((general as General)[CALLBACK_NAVIGATION_GO_TO_SCREEN_KEY] as string) ??
73
+ (legacy.targetScreen
74
+ ? ((general as General)[legacy.targetScreen] as string)
75
+ : undefined);
76
+
77
+ if (screenId) {
78
+ const trimmedTargetScreenId = screenId.trim();
79
+
80
+ targetScreenId =
81
+ trimmedTargetScreenId.length > 0 ? trimmedTargetScreenId : null;
82
+ }
83
+ }
84
+
85
+ return { action, targetScreenId };
86
+ };
87
+
88
+ export const useCallbackNavigationAction = (
89
+ item?: ZappUIComponent | ZappRiver
90
+ ): hookCallback | undefined => {
91
+ const navigation = useNavigation();
92
+ const rivers = useRivers();
93
+ const enabled = Boolean(item?.general?.[CALLBACK_NAVIGATION_KEY]);
94
+ const screenContext = useScreenContext();
95
+
96
+ const overrideCallbackFromComponent = useMemo(() => {
97
+ log_verbose(`${LogPrefix}: overridden callbackAction by component`);
98
+
99
+ // TODO: Check if we have better option where to store overridden callback action
100
+ return screenContext?.options?.callback;
101
+ }, [screenContext?.options?.callback]);
102
+
103
+ if (typeof __DEV__ !== "undefined" && __DEV__) {
104
+ log_verbose(`${LogPrefix} screenContext`, { screenContext, item });
105
+ }
106
+
107
+ const callbackAction = useCallback<hookCallback>(
108
+ (_args: hookCallbackArgs) => {
109
+ const data = getNavigationKeys(item) ?? {
110
+ action: NavigationCallbackOptions.DEFAULT,
111
+ targetScreenId: null,
112
+ };
113
+
114
+ switch (data.action) {
115
+ case NavigationCallbackOptions.GO_BACK: {
116
+ if (navigation.canGoBack()) {
117
+ navigation.goBack();
118
+ log_info(`${LogPrefix} performing 'GO BACK' action`);
119
+ } else {
120
+ log_info(`${LogPrefix} cannot perform 'GO BACK' action — ignoring`);
121
+ }
122
+
123
+ break;
124
+ }
125
+
126
+ case NavigationCallbackOptions.GO_HOME: {
127
+ navigation.goHome();
128
+ log_info(`${LogPrefix} performing 'GO HOME' action`);
129
+ break;
130
+ }
131
+
132
+ case NavigationCallbackOptions.GO_TO_SCREEN: {
133
+ const screenId = data.targetScreenId;
134
+
135
+ if (!screenId) {
136
+ log_info(`${LogPrefix} no screenId provided — ignoring`);
137
+ break;
138
+ }
139
+
140
+ const screen = rivers[screenId];
141
+
142
+ if (screen) {
143
+ navigation.replace(screen);
144
+
145
+ log_info(
146
+ `${LogPrefix} performing 'GO TO SCREEN' action to screen: ${screenId}`
147
+ );
148
+ } else {
149
+ log_info(`${LogPrefix} no screen provided — ignoring`);
150
+ }
151
+
152
+ break;
153
+ }
154
+
155
+ default: {
156
+ break;
157
+ }
158
+ }
159
+ },
160
+ [item, navigation, rivers]
161
+ );
162
+
163
+ return enabled ? overrideCallbackFromComponent || callbackAction : undefined;
164
+ };