@applicaster/zapp-react-native-utils 15.0.0-rc.99 → 16.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 (63) hide show
  1. package/README.md +0 -6
  2. package/actionUtils/index.ts +7 -0
  3. package/actionsExecutor/ActionExecutorContext.tsx +83 -6
  4. package/appUtils/HooksManager/index.ts +35 -0
  5. package/appUtils/focusManager/treeDataStructure/Tree/__tests__/Tree.test.js +46 -0
  6. package/appUtils/focusManager/treeDataStructure/Tree/index.js +18 -18
  7. package/appUtils/focusManagerAux/utils/index.ts +12 -6
  8. package/appUtils/focusManagerAux/utils/utils.ios.ts +6 -3
  9. package/appUtils/localizationsHelper.ts +4 -0
  10. package/appUtils/playerManager/index.ts +9 -0
  11. package/appUtils/playerManager/player.ts +1 -1
  12. package/appUtils/playerManager/playerNative.ts +2 -1
  13. package/appUtils/playerManager/usePlayer.tsx +5 -3
  14. package/cellUtils/__tests__/cellUtils.test.ts +39 -0
  15. package/cellUtils/index.ts +11 -1
  16. package/componentsUtils/index.ts +8 -0
  17. package/dateUtils/__tests__/dayjs.test.ts +0 -3
  18. package/dateUtils/index.ts +2 -0
  19. package/manifestUtils/_internals/__tests__/index.test.js +41 -0
  20. package/manifestUtils/_internals/index.js +33 -0
  21. package/manifestUtils/defaultManifestConfigurations/player.js +6 -16
  22. package/manifestUtils/fieldUtils/__tests__/fieldUtils.test.js +49 -0
  23. package/manifestUtils/fieldUtils/index.js +54 -0
  24. package/manifestUtils/index.js +2 -0
  25. package/manifestUtils/keys.js +228 -0
  26. package/manifestUtils/mobileAction/button/__tests__/mobileActionButton.test.js +168 -0
  27. package/manifestUtils/mobileAction/button/index.js +140 -0
  28. package/manifestUtils/mobileAction/container/__tests__/mobileActionButtonsContainer.test.js +102 -0
  29. package/manifestUtils/mobileAction/container/index.js +73 -0
  30. package/manifestUtils/mobileAction/groups/__tests__/buildMobileActionButtonGroups.test.js +127 -0
  31. package/manifestUtils/mobileAction/groups/defaults.js +76 -0
  32. package/manifestUtils/mobileAction/groups/index.js +80 -0
  33. package/numberUtils/__tests__/toNumber.test.ts +27 -12
  34. package/numberUtils/__tests__/toPositiveNumber.test.ts +32 -4
  35. package/numberUtils/index.ts +5 -1
  36. package/package.json +3 -3
  37. package/pluginUtils/index.ts +4 -5
  38. package/reactHooks/casting/index.ts +1 -0
  39. package/reactHooks/casting/useIsCasting.tsx +57 -0
  40. package/reactHooks/cell-click/index.ts +2 -1
  41. package/reactHooks/feed/index.ts +0 -2
  42. package/reactHooks/feed/useInflatedUrl.ts +1 -1
  43. package/reactHooks/resolvers/useComponentResolver.ts +13 -3
  44. package/reactHooks/screen/__tests__/useCurrentScreenIsHook.test.ts +103 -0
  45. package/reactHooks/screen/__tests__/useCurrentScreenIsStartupHook.test.ts +94 -0
  46. package/reactHooks/screen/index.ts +4 -0
  47. package/reactHooks/screen/useCurrentScreenIsHook.ts +9 -0
  48. package/reactHooks/screen/useCurrentScreenIsStartupHook.ts +8 -0
  49. package/reactHooks/state/__tests__/useComponentScreenState.test.ts +246 -0
  50. package/reactHooks/state/index.ts +2 -0
  51. package/reactHooks/state/useComponentScreenState.ts +45 -0
  52. package/refreshUtils/RefreshCoordinator/__tests__/refreshCoordinator.test.ts +206 -0
  53. package/refreshUtils/RefreshCoordinator/index.ts +245 -0
  54. package/refreshUtils/RefreshCoordinator/utils/__tests__/getDataRefreshConfig.test.ts +104 -0
  55. package/refreshUtils/RefreshCoordinator/utils/index.ts +29 -0
  56. package/screenPickerUtils/index.ts +5 -0
  57. package/screenUtils/index.ts +3 -0
  58. package/utils/__tests__/clone.test.ts +158 -0
  59. package/utils/__tests__/path.test.ts +7 -0
  60. package/utils/clone.ts +7 -0
  61. package/utils/index.ts +2 -1
  62. package/reactHooks/feed/__tests__/useFeedRefresh.test.tsx +0 -75
  63. package/reactHooks/feed/useFeedRefresh.tsx +0 -65
@@ -0,0 +1,246 @@
1
+ import { renderHook, act } from "@testing-library/react-native";
2
+ import { useComponentScreenState } from "../useComponentScreenState";
3
+ import { useScreenContextV2 } from "@applicaster/zapp-react-native-utils/reactHooks/screen/useScreenContext";
4
+
5
+ import { create } from "zustand";
6
+
7
+ const createMockStore = () => create(() => ({}));
8
+
9
+ // Mock the external dependencies
10
+ jest.mock(
11
+ "@applicaster/zapp-react-native-utils/reactHooks/screen/useScreenContext",
12
+ () => ({
13
+ useScreenContextV2: jest.fn(),
14
+ })
15
+ );
16
+
17
+ const mockUseScreenContextV2 = useScreenContextV2 as jest.MockedFunction<
18
+ typeof useScreenContextV2
19
+ >;
20
+
21
+ describe("useComponentScreenState", () => {
22
+ beforeEach(() => {
23
+ jest.clearAllMocks();
24
+ });
25
+
26
+ it("should return initial value when no state exists for componentId", () => {
27
+ const mockStore = createMockStore();
28
+
29
+ mockUseScreenContextV2.mockReturnValue({
30
+ _componentStateStore: mockStore,
31
+ } as any);
32
+
33
+ const initialValue = "default-value";
34
+ const componentId = "test-component";
35
+
36
+ const { result } = renderHook(() =>
37
+ useComponentScreenState(componentId, initialValue)
38
+ );
39
+
40
+ expect(result.current[0]).toBe(initialValue);
41
+ });
42
+
43
+ it("should return stored value when state exists for componentId", () => {
44
+ const mockStore = createMockStore();
45
+ mockStore.setState({ "existing-component": "stored-value" });
46
+
47
+ mockUseScreenContextV2.mockReturnValue({
48
+ _componentStateStore: mockStore,
49
+ } as any);
50
+
51
+ const { result } = renderHook(() =>
52
+ useComponentScreenState("existing-component", "default")
53
+ );
54
+
55
+ expect(result.current[0]).toBe("stored-value");
56
+ });
57
+
58
+ it("should update state when setter is called", () => {
59
+ const mockStore = createMockStore();
60
+
61
+ mockUseScreenContextV2.mockReturnValue({
62
+ _componentStateStore: mockStore,
63
+ } as any);
64
+
65
+ const { result } = renderHook(() =>
66
+ useComponentScreenState("update-test", "initial")
67
+ );
68
+
69
+ // Verify initial value
70
+ expect(result.current[0]).toBe("initial");
71
+
72
+ // Update the value
73
+ act(() => {
74
+ result.current[1]("updated-value");
75
+ });
76
+
77
+ // Verify updated value
78
+ expect(result.current[0]).toBe("updated-value");
79
+ });
80
+
81
+ it("should handle different data types correctly", () => {
82
+ const mockStore = createMockStore();
83
+
84
+ mockUseScreenContextV2.mockReturnValue({
85
+ _componentStateStore: mockStore,
86
+ } as any);
87
+
88
+ // Test with number
89
+ const { result: numberResult } = renderHook(() =>
90
+ useComponentScreenState<number>("number-test", 42)
91
+ );
92
+
93
+ expect(numberResult.current[0]).toBe(42);
94
+
95
+ // Test with boolean
96
+ const { result: boolResult } = renderHook(() =>
97
+ useComponentScreenState<boolean>("bool-test", true)
98
+ );
99
+
100
+ expect(boolResult.current[0]).toBe(true);
101
+
102
+ // Test with object
103
+ const testObj = { key: "value" };
104
+
105
+ const { result: objResult } = renderHook(() =>
106
+ useComponentScreenState<{ key: string }>("obj-test", testObj)
107
+ );
108
+
109
+ expect(objResult.current[0]).toEqual(testObj);
110
+ });
111
+
112
+ it("should maintain separate states for different componentIds", () => {
113
+ const mockStore = createMockStore();
114
+
115
+ mockUseScreenContextV2.mockReturnValue({
116
+ _componentStateStore: mockStore,
117
+ } as any);
118
+
119
+ const { result: firstResult } = renderHook(() =>
120
+ useComponentScreenState("first", "first-initial")
121
+ );
122
+
123
+ const { result: secondResult } = renderHook(() =>
124
+ useComponentScreenState("second", "second-initial")
125
+ );
126
+
127
+ // Verify initial values are separate
128
+ expect(firstResult.current[0]).toBe("first-initial");
129
+ expect(secondResult.current[0]).toBe("second-initial");
130
+
131
+ // Update first component's state
132
+ act(() => {
133
+ firstResult.current[1]("first-updated");
134
+ });
135
+
136
+ // Verify only first component's state changed
137
+ expect(firstResult.current[0]).toBe("first-updated");
138
+ expect(secondResult.current[0]).toBe("second-initial");
139
+ });
140
+
141
+ it("should return a memoized setter function", () => {
142
+ const mockStore = createMockStore();
143
+
144
+ mockUseScreenContextV2.mockReturnValue({
145
+ _componentStateStore: mockStore,
146
+ } as any);
147
+
148
+ const { result, rerender } = renderHook(
149
+ ({ id, initial }) => useComponentScreenState(id, initial),
150
+ {
151
+ initialProps: { id: "memo-test", initial: "initial" },
152
+ }
153
+ );
154
+
155
+ const firstSetter = result.current[1];
156
+
157
+ // Rerender with same props
158
+ rerender({ id: "memo-test", initial: "initial" });
159
+
160
+ const secondSetter = result.current[1];
161
+
162
+ // Setter should be the same instance due to useCallback
163
+ expect(firstSetter).toBe(secondSetter);
164
+ });
165
+
166
+ it("should update setter when componentId changes", () => {
167
+ const mockStore = createMockStore();
168
+
169
+ mockUseScreenContextV2.mockReturnValue({
170
+ _componentStateStore: mockStore,
171
+ } as any);
172
+
173
+ const { result, rerender } = renderHook(
174
+ ({ id, initial }) => useComponentScreenState(id, initial),
175
+ {
176
+ initialProps: { id: "old-id", initial: "initial" },
177
+ }
178
+ );
179
+
180
+ const firstSetter = result.current[1];
181
+
182
+ // Rerender with different componentId
183
+ rerender({ id: "new-id", initial: "initial" });
184
+
185
+ const secondSetter = result.current[1];
186
+
187
+ // Setter should be different because componentId changed
188
+ expect(firstSetter).not.toBe(secondSetter);
189
+ });
190
+
191
+ it("should call store.setState with correct parameters", () => {
192
+ const mockStore = createMockStore();
193
+ const setStateSpy = jest.spyOn(mockStore, "setState");
194
+
195
+ mockUseScreenContextV2.mockReturnValue({
196
+ _componentStateStore: mockStore,
197
+ } as any);
198
+
199
+ const { result } = renderHook(() =>
200
+ useComponentScreenState("state-test", "initial")
201
+ );
202
+
203
+ const newValue = "new-value";
204
+
205
+ act(() => {
206
+ result.current[1](newValue);
207
+ });
208
+
209
+ expect(setStateSpy).toHaveBeenCalledWith({
210
+ "state-test": newValue,
211
+ });
212
+ });
213
+
214
+ it("should handle falsy initial values correctly", () => {
215
+ const mockStore = createMockStore();
216
+
217
+ mockUseScreenContextV2.mockReturnValue({
218
+ _componentStateStore: mockStore,
219
+ } as any);
220
+
221
+ // Test with falsy values
222
+ const { result: nullResult } = renderHook(() =>
223
+ useComponentScreenState("null-test", null)
224
+ );
225
+
226
+ expect(nullResult.current[0]).toBeNull();
227
+
228
+ const { result: falseResult } = renderHook(() =>
229
+ useComponentScreenState("false-test", false)
230
+ );
231
+
232
+ expect(falseResult.current[0]).toBe(false);
233
+
234
+ const { result: zeroResult } = renderHook(() =>
235
+ useComponentScreenState("zero-test", 0)
236
+ );
237
+
238
+ expect(zeroResult.current[0]).toBe(0);
239
+
240
+ const { result: emptyStrResult } = renderHook(() =>
241
+ useComponentScreenState("empty-test", "")
242
+ );
243
+
244
+ expect(emptyStrResult.current[0]).toBe("");
245
+ });
246
+ });
@@ -3,3 +3,5 @@ export { useRivers } from "./useRivers";
3
3
  export { useHomeRiver, getHomeRiver } from "./useHomeRiver";
4
4
 
5
5
  export { ZStoreProvider, useZStore } from "./ZStoreProvider";
6
+
7
+ export { useComponentScreenState } from "./useComponentScreenState";
@@ -0,0 +1,45 @@
1
+ import * as React from "react";
2
+ import { useScreenContextV2 } from "@applicaster/zapp-react-native-utils/reactHooks/screen/useScreenContext";
3
+ import { useStore } from "zustand";
4
+
5
+ /**
6
+ * A custom hook that provides persistent state storage across component re-mounts
7
+ * using the screen's component state store.
8
+ *
9
+ * @see {@link DOCS/adr/0010-useComponentScreenState.md} - ADR explaining why useComponentScreenState is used for List/Hero/Grid/Gallery components
10
+ *
11
+ * The state is shared across all components using the same componentId within the same screen context.
12
+ * If no value exists for the componentId, the initial value is returned as a fallback but is not persisted in the store.
13
+ * The initial value must be explicitly set using the returned setter function to persist it.
14
+ *
15
+ * @template T - The type of value being stored
16
+ * @param componentId - Unique identifier for this state in the component state store
17
+ * @param initialValue - The default value if no value exists for the componentId
18
+ * @returns A tuple containing the current value and a memoized setter function
19
+ *
20
+ * @example
21
+ * ```tsx
22
+ * const [count, setCount] = useComponentScreenState<number>('counter', 0);
23
+ * const [name, setName] = useComponentScreenState<string>('username', 'Anonymous');
24
+ * ```
25
+ */
26
+ export const useComponentScreenState = <T = unknown>(
27
+ componentId: string,
28
+ initialValue: T
29
+ ): [T, (value: T) => void] => {
30
+ const store = useScreenContextV2()._componentStateStore;
31
+
32
+ const value = useStore(
33
+ store,
34
+ (state) => (state[componentId] as T | undefined) ?? initialValue
35
+ );
36
+
37
+ const setValue = React.useCallback(
38
+ (nextValue: T) => {
39
+ store.setState({ [componentId]: nextValue });
40
+ },
41
+ [componentId, store]
42
+ );
43
+
44
+ return [value, setValue] as const;
45
+ };
@@ -0,0 +1,206 @@
1
+ import { refreshCoordinator } from "..";
2
+ import { getDataRefreshConfig } from "../utils";
3
+ import { Subscription } from "rxjs";
4
+
5
+ jest.mock("../utils", () => ({
6
+ getDataRefreshConfig: jest.fn(),
7
+ }));
8
+
9
+ describe("RefreshCoordinator", () => {
10
+ const component = { id: "c1" } as ZappUIComponent;
11
+ const screenId = "screen-1";
12
+
13
+ beforeEach(() => {
14
+ jest.clearAllMocks();
15
+ refreshCoordinator.clear();
16
+ jest.useFakeTimers();
17
+ });
18
+
19
+ afterEach(() => {
20
+ jest.runOnlyPendingTimers();
21
+ jest.useRealTimers();
22
+ });
23
+
24
+ // ----------------------------------------
25
+ // Singleton
26
+ // ----------------------------------------
27
+ it("returns the same instance via getInstance", () => {
28
+ const instance1 = refreshCoordinator;
29
+ const instance2 = (refreshCoordinator as any).constructor.getInstance();
30
+ expect(instance1).toBe(instance2);
31
+ });
32
+
33
+ // ----------------------------------------
34
+ // register & unregister
35
+ // ----------------------------------------
36
+ it("register starts a timer and returns an unregister function", () => {
37
+ (getDataRefreshConfig as jest.Mock).mockReturnValue({
38
+ refreshIntervalMs: 1000,
39
+ isRefreshingEnabled: true,
40
+ });
41
+
42
+ const unregister = refreshCoordinator.register(component, screenId);
43
+
44
+ expect(typeof unregister).toBe("function");
45
+ expect((refreshCoordinator as any).timers.size).toBe(1);
46
+
47
+ // Calling unregister should remove timer
48
+ unregister();
49
+ expect((refreshCoordinator as any).timers.size).toBe(0);
50
+ });
51
+
52
+ it("does not register if refreshing is disabled", () => {
53
+ (getDataRefreshConfig as jest.Mock).mockReturnValue({
54
+ refreshIntervalMs: 1000,
55
+ isRefreshingEnabled: false,
56
+ });
57
+
58
+ const unregister = refreshCoordinator.register(component, screenId);
59
+
60
+ expect(unregister).toBeInstanceOf(Function);
61
+ expect((refreshCoordinator as any).timers.size).toBe(0);
62
+ });
63
+
64
+ it("replaces existing timer if component is re-registered", () => {
65
+ (getDataRefreshConfig as jest.Mock).mockReturnValue({
66
+ refreshIntervalMs: 1000,
67
+ isRefreshingEnabled: true,
68
+ });
69
+
70
+ const unregister1 = refreshCoordinator.register(component, screenId);
71
+ const unregister2 = refreshCoordinator.register(component, screenId);
72
+
73
+ expect((refreshCoordinator as any).timers.size).toBe(1);
74
+ expect(unregister1).toBeInstanceOf(Function);
75
+ expect(unregister2).toBeInstanceOf(Function);
76
+ });
77
+
78
+ // ----------------------------------------
79
+ // subscribeTo
80
+ // ----------------------------------------
81
+ it("subscribeTo only triggers listener for matching component & screen", () => {
82
+ (getDataRefreshConfig as jest.Mock).mockReturnValue({
83
+ refreshIntervalMs: 1000,
84
+ isRefreshingEnabled: true,
85
+ });
86
+
87
+ const listener = jest.fn();
88
+ refreshCoordinator.register(component, screenId);
89
+ const sub = refreshCoordinator.subscribeTo(component, screenId, listener);
90
+
91
+ // Emit a refresh manually via private Subject
92
+ (refreshCoordinator as any).refresh$.next({ component, screenId });
93
+ expect(listener).toHaveBeenCalledTimes(1);
94
+
95
+ // Non-matching component should not call listener
96
+ (refreshCoordinator as any).refresh$.next({
97
+ component: { id: "other" } as ZappUIComponent,
98
+ screenId,
99
+ });
100
+
101
+ expect(listener).toHaveBeenCalledTimes(1);
102
+
103
+ sub.unsubscribe();
104
+ });
105
+
106
+ // ----------------------------------------
107
+ // subscribeToAny
108
+ // ----------------------------------------
109
+ it("subscribeToAny triggers listener with throttling and replaces previous listener", () => {
110
+ const listener1 = jest.fn();
111
+ const listener2 = jest.fn();
112
+
113
+ // First subscription
114
+ const sub1: Subscription = refreshCoordinator.subscribeToAny(listener1);
115
+ expect(sub1).toBeInstanceOf(Subscription);
116
+
117
+ // Second subscription replaces first
118
+ const sub2: Subscription = refreshCoordinator.subscribeToAny(listener2);
119
+ expect(sub2).toBeInstanceOf(Subscription);
120
+
121
+ // Emit multiple events quickly
122
+ (refreshCoordinator as any).refresh$.next({ component, screenId });
123
+ (refreshCoordinator as any).refresh$.next({ component, screenId });
124
+
125
+ // Because of throttleTime(500ms), fast events are ignored
126
+ jest.advanceTimersByTime(500);
127
+ expect(listener1).not.toHaveBeenCalled();
128
+ expect(listener2).toHaveBeenCalledTimes(1);
129
+
130
+ sub2.unsubscribe();
131
+ });
132
+
133
+ // ----------------------------------------
134
+ // triggerRefresh
135
+ // ----------------------------------------
136
+ it("triggerRefresh emits refresh events for each component", () => {
137
+ const component1 = { id: "c1" } as ZappUIComponent;
138
+ const component2 = { id: "c2" } as ZappUIComponent;
139
+ const listener = jest.fn();
140
+
141
+ // Subscribe to all events
142
+ (refreshCoordinator as any).refresh$.subscribe(listener);
143
+
144
+ refreshCoordinator.triggerRefresh([component1, component2], screenId);
145
+
146
+ expect(listener).toHaveBeenCalledTimes(2);
147
+
148
+ expect(listener).toHaveBeenCalledWith({
149
+ component: component1,
150
+ screenId,
151
+ });
152
+
153
+ expect(listener).toHaveBeenCalledWith({
154
+ component: component2,
155
+ screenId,
156
+ });
157
+ });
158
+
159
+ it("triggerRefresh with empty components does nothing", () => {
160
+ const listener = jest.fn();
161
+
162
+ (refreshCoordinator as any).refresh$.subscribe(listener);
163
+
164
+ refreshCoordinator.triggerRefresh([], screenId);
165
+
166
+ expect(listener).not.toHaveBeenCalled();
167
+ });
168
+
169
+ it("triggerRefresh events are picked up by subscribeTo", () => {
170
+ const listener = jest.fn();
171
+
172
+ refreshCoordinator.subscribeTo(component, screenId, listener);
173
+ refreshCoordinator.triggerRefresh([component], screenId);
174
+
175
+ expect(listener).toHaveBeenCalledTimes(1);
176
+ });
177
+
178
+ // ----------------------------------------
179
+ // clear
180
+ // ----------------------------------------
181
+ it("clear stops all timers, anyListener, and recreates subject", () => {
182
+ (getDataRefreshConfig as jest.Mock).mockReturnValue({
183
+ refreshIntervalMs: 1000,
184
+ isRefreshingEnabled: true,
185
+ });
186
+
187
+ const listener = jest.fn();
188
+ refreshCoordinator.register(component, screenId);
189
+ refreshCoordinator.subscribeToAny(listener);
190
+
191
+ expect((refreshCoordinator as any).timers.size).toBe(1);
192
+ expect((refreshCoordinator as any).anyListenerSubscription).toBeDefined();
193
+
194
+ refreshCoordinator.clear();
195
+
196
+ expect((refreshCoordinator as any).timers.size).toBe(0);
197
+ expect((refreshCoordinator as any).anyListenerSubscription).toBeUndefined();
198
+
199
+ // New events on new Subject still work
200
+ const newListener = jest.fn();
201
+ refreshCoordinator.subscribeToAny(newListener);
202
+ (refreshCoordinator as any).refresh$.next({ component, screenId });
203
+ jest.advanceTimersByTime(500);
204
+ expect(newListener).toHaveBeenCalled();
205
+ });
206
+ });