@applicaster/zapp-react-native-ui-components 15.0.0-rc.12 → 15.0.0-rc.120

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 (159) hide show
  1. package/Components/AnimatedInOut/index.tsx +69 -26
  2. package/Components/BaseFocusable/index.ios.ts +12 -2
  3. package/Components/Cell/Cell.tsx +14 -3
  4. package/Components/Cell/CellWithFocusable.tsx +9 -0
  5. package/Components/Cell/FocusableWrapper.tsx +47 -0
  6. package/Components/Cell/TvOSCellComponent.tsx +106 -19
  7. package/Components/Focusable/Focusable.tsx +4 -2
  8. package/Components/Focusable/FocusableTvOS.tsx +18 -1
  9. package/Components/Focusable/__tests__/__snapshots__/FocusableTvOS.test.tsx.snap +1 -0
  10. package/Components/FocusableGroup/FocusableTvOS.tsx +32 -1
  11. package/Components/GeneralContentScreen/GeneralContentScreen.tsx +39 -28
  12. package/Components/GeneralContentScreen/__tests__/GeneralContentScreen.test.tsx +104 -0
  13. package/Components/GeneralContentScreen/utils/__tests__/getScreenDataSource.test.ts +19 -0
  14. package/Components/GeneralContentScreen/utils/__tests__/useCurationAPI.test.js +1 -1
  15. package/Components/GeneralContentScreen/utils/getScreenDataSource.ts +9 -0
  16. package/Components/GeneralContentScreen/utils/useCurationAPI.ts +22 -6
  17. package/Components/HandlePlayable/HandlePlayable.tsx +33 -94
  18. package/Components/HandlePlayable/const.ts +3 -0
  19. package/Components/HandlePlayable/utils.ts +105 -0
  20. package/Components/HookRenderer/HookRenderer.tsx +40 -10
  21. package/Components/HookRenderer/__tests__/HookRenderer.test.tsx +60 -0
  22. package/Components/Layout/TV/LayoutBackground.tsx +5 -2
  23. package/Components/Layout/TV/NavBarContainer.tsx +1 -10
  24. package/Components/Layout/TV/ScreenContainer.tsx +2 -6
  25. package/Components/Layout/TV/__tests__/__snapshots__/NavBarContainer.test.tsx.snap +7 -12
  26. package/Components/Layout/TV/__tests__/__snapshots__/ScreenContainer.test.tsx.snap +7 -12
  27. package/Components/Layout/TV/index.tsx +3 -4
  28. package/Components/Layout/TV/index.web.tsx +3 -4
  29. package/Components/LinkHandler/LinkHandler.tsx +2 -2
  30. package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx +16 -1
  31. package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +30 -2
  32. package/Components/MasterCell/DefaultComponents/Image/Image.android.tsx +5 -1
  33. package/Components/MasterCell/DefaultComponents/Image/Image.ios.tsx +11 -3
  34. package/Components/MasterCell/DefaultComponents/Image/Image.web.tsx +9 -1
  35. package/Components/MasterCell/DefaultComponents/Image/hooks/useImage.ts +15 -14
  36. package/Components/MasterCell/DefaultComponents/LiveImage/__tests__/prepareEntry.test.ts +352 -0
  37. package/Components/MasterCell/DefaultComponents/LiveImage/executePreloadHooks.ts +136 -0
  38. package/Components/MasterCell/DefaultComponents/LiveImage/index.tsx +43 -22
  39. package/Components/MasterCell/DefaultComponents/SecondaryImage/Image.tsx +40 -39
  40. package/Components/MasterCell/DefaultComponents/SecondaryImage/__tests__/Image.test.tsx +95 -0
  41. package/Components/MasterCell/DefaultComponents/SecondaryImage/__tests__/__snapshots__/Image.test.tsx.snap +86 -0
  42. package/Components/MasterCell/DefaultComponents/SecondaryImage/__tests__/index.test.ts +141 -0
  43. package/Components/MasterCell/DefaultComponents/SecondaryImage/hooks/__tests__/useGetImageDimensions.test.ts +7 -6
  44. package/Components/MasterCell/DefaultComponents/SecondaryImage/index.ts +1 -1
  45. package/Components/MasterCell/DefaultComponents/Text/index.tsx +8 -8
  46. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/index.ts +6 -2
  47. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/utils/__tests__/getPluginIdentifier.test.ts +233 -11
  48. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/utils/index.ts +19 -15
  49. package/Components/MasterCell/hoc/__tests__/withAsyncRender.test.tsx +219 -0
  50. package/Components/MasterCell/hoc/withAsyncRender.tsx +9 -7
  51. package/Components/MasterCell/index.tsx +2 -0
  52. package/Components/MasterCell/utils/__tests__/resolveColor.test.js +82 -3
  53. package/Components/MasterCell/utils/index.ts +61 -31
  54. package/Components/MeasurmentsPortal/MeasurementsPortal.tsx +102 -87
  55. package/Components/MeasurmentsPortal/__tests__/MeasurementsPortal.test.tsx +355 -0
  56. package/Components/OfflineHandler/NotificationView/NotificationView.lg.tsx +17 -9
  57. package/Components/OfflineHandler/NotificationView/NotificationView.samsung.tsx +16 -8
  58. package/Components/OfflineHandler/NotificationView/NotificationView.tsx +2 -2
  59. package/Components/OfflineHandler/NotificationView/__tests__/index.test.tsx +17 -18
  60. package/Components/OfflineHandler/NotificationView/utils.ts +34 -0
  61. package/Components/OfflineHandler/__tests__/index.test.tsx +27 -18
  62. package/Components/PlayerContainer/PlayerContainer.tsx +43 -64
  63. package/Components/PlayerImageBackground/index.tsx +3 -22
  64. package/Components/PreloaderWrapper/__tests__/index.test.tsx +26 -0
  65. package/Components/PreloaderWrapper/index.tsx +15 -0
  66. package/Components/River/ComponentsMap/ComponentsMap.tsx +16 -0
  67. package/Components/River/ComponentsMap/hooks/__tests__/useLoadingState.test.ts +1 -1
  68. package/Components/River/RefreshControl.tsx +9 -3
  69. package/Components/River/RiverItem.tsx +26 -20
  70. package/Components/River/TV/River.tsx +31 -14
  71. package/Components/River/TV/index.tsx +8 -4
  72. package/Components/River/TV/utils/__tests__/toStringOrEmpty.test.ts +30 -0
  73. package/Components/River/TV/utils/index.ts +4 -0
  74. package/Components/River/TV/withFocusableGroupForContent.tsx +71 -0
  75. package/Components/River/__tests__/__snapshots__/componentsMap.test.js.snap +2 -0
  76. package/Components/River/__tests__/componentsMap.test.js +38 -0
  77. package/Components/Screen/TV/index.web.tsx +4 -2
  78. package/Components/Screen/__tests__/Screen.test.tsx +66 -42
  79. package/Components/Screen/__tests__/__snapshots__/Screen.test.tsx.snap +68 -44
  80. package/Components/Screen/hooks.ts +75 -6
  81. package/Components/Screen/index.tsx +9 -4
  82. package/Components/Screen/navigationHandler.ts +49 -24
  83. package/Components/Screen/orientationHandler.ts +10 -13
  84. package/Components/ScreenFeedLoader/ScreenFeedLoader.tsx +46 -0
  85. package/Components/ScreenFeedLoader/__tests__/ScreenFeedLoader.test.tsx +94 -0
  86. package/Components/ScreenFeedLoader/index.ts +1 -0
  87. package/Components/ScreenResolver/__tests__/screenResolver.test.js +24 -0
  88. package/Components/ScreenResolver/hooks/index.ts +3 -0
  89. package/Components/ScreenResolver/hooks/useGetComponent.ts +15 -0
  90. package/Components/ScreenResolver/hooks/useScreenComponentResolver.tsx +90 -0
  91. package/Components/ScreenResolver/index.tsx +15 -111
  92. package/Components/ScreenResolver/utils/__tests__/getScreenTypeProps.test.ts +45 -0
  93. package/Components/ScreenResolver/utils/getScreenTypeProps.ts +43 -0
  94. package/Components/ScreenResolver/utils/index.ts +1 -0
  95. package/Components/ScreenResolver/withDefaultScreenContext.tsx +16 -0
  96. package/Components/ScreenResolverFeedProvider/ScreenResolverFeedProvider.tsx +25 -0
  97. package/Components/ScreenResolverFeedProvider/__tests__/ScreenResolverFeedProvider.test.tsx +44 -0
  98. package/Components/ScreenResolverFeedProvider/index.ts +1 -0
  99. package/Components/ScreenRevealManager/ScreenRevealManager.ts +40 -8
  100. package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +86 -69
  101. package/Components/ScreenRevealManager/withScreenRevealManager.tsx +44 -26
  102. package/Components/Tabs/TV/Tabs.tsx +20 -3
  103. package/Components/Tabs/TabContent.tsx +7 -4
  104. package/Components/Transitioner/Scene.tsx +10 -3
  105. package/Components/Transitioner/index.js +3 -3
  106. package/Components/VideoLive/LiveImageManager.ts +199 -54
  107. package/Components/VideoLive/PlayerLiveImageComponent.tsx +31 -33
  108. package/Components/VideoLive/__tests__/PlayerLiveImageComponent.test.tsx +2 -17
  109. package/Components/VideoLive/__tests__/__snapshots__/PlayerLiveImageComponent.test.tsx.snap +1 -0
  110. package/Components/VideoModal/ModalAnimation/ModalAnimationContext.tsx +118 -171
  111. package/Components/VideoModal/ModalAnimation/index.ts +2 -13
  112. package/Components/VideoModal/ModalAnimation/utils.ts +1 -327
  113. package/Components/VideoModal/PlayerWrapper.tsx +14 -88
  114. package/Components/VideoModal/VideoModal.tsx +1 -5
  115. package/Components/VideoModal/__tests__/PlayerWrapper.test.tsx +1 -0
  116. package/Components/VideoModal/hooks/__tests__/useDelayedPlayerDetails.test.ts +15 -7
  117. package/Components/VideoModal/hooks/useModalSize.ts +10 -5
  118. package/Components/VideoModal/playerWrapperStyle.ts +70 -0
  119. package/Components/VideoModal/playerWrapperUtils.ts +91 -0
  120. package/Components/VideoModal/utils.ts +19 -9
  121. package/Components/Viewport/ViewportAware/__tests__/viewportAware.test.js +0 -2
  122. package/Components/Viewport/ViewportAware/index.tsx +16 -7
  123. package/Components/Viewport/ViewportEvents/__tests__/viewportEvents.test.js +1 -1
  124. package/Components/ZappUIComponent/index.tsx +12 -6
  125. package/Components/index.js +1 -1
  126. package/Contexts/ScreenContext/__tests__/index.test.tsx +57 -0
  127. package/Contexts/ScreenContext/index.tsx +71 -19
  128. package/Contexts/ScreenTrackedViewPositionsContext/__tests__/index.test.tsx +1 -1
  129. package/Contexts/ZappHookModalContext/index.tsx +37 -61
  130. package/Contexts/ZappPipesContext/ZappPipesContextFactory.tsx +18 -7
  131. package/Contexts/index.ts +0 -2
  132. package/Decorators/Analytics/index.tsx +6 -5
  133. package/Decorators/ConfigurationWrapper/__tests__/__snapshots__/withConfigurationProvider.test.tsx.snap +1 -0
  134. package/Decorators/ConfigurationWrapper/const.ts +1 -0
  135. package/Decorators/ZappPipesDataConnector/ResolverSelector.tsx +25 -7
  136. package/Decorators/ZappPipesDataConnector/__tests__/ResolverSelector.test.tsx +212 -5
  137. package/Decorators/ZappPipesDataConnector/__tests__/UrlFeedResolver.test.tsx +39 -21
  138. package/Decorators/ZappPipesDataConnector/__tests__/zappPipesDataConnector.test.js +1 -1
  139. package/Decorators/ZappPipesDataConnector/index.tsx +2 -2
  140. package/Decorators/ZappPipesDataConnector/resolvers/StaticFeedResolver.tsx +1 -1
  141. package/Decorators/ZappPipesDataConnector/resolvers/UrlFeedResolver.tsx +18 -7
  142. package/Helpers/DataSourceHelper/__tests__/itemLimitForData.test.ts +80 -0
  143. package/Helpers/DataSourceHelper/index.ts +19 -0
  144. package/events/index.ts +3 -0
  145. package/events/scrollEndReached.ts +15 -0
  146. package/index.d.ts +7 -0
  147. package/package.json +6 -5
  148. package/Components/PlayerContainer/ErrorDisplay/ErrorDisplay.tsx +0 -57
  149. package/Components/PlayerContainer/ErrorDisplay/index.ts +0 -9
  150. package/Components/River/TV/withTVEventHandler.tsx +0 -27
  151. package/Components/VideoModal/ModalAnimation/AnimatedPlayerModalWrapper.tsx +0 -60
  152. package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.tsx +0 -417
  153. package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.web.tsx +0 -294
  154. package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.tsx +0 -176
  155. package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.web.tsx +0 -93
  156. package/Components/VideoModal/ModalAnimation/AnimationComponent.tsx +0 -500
  157. package/Components/VideoModal/ModalAnimation/__tests__/getMoveUpValue.test.ts +0 -108
  158. package/Helpers/DataSourceHelper/index.js +0 -19
  159. /package/Components/HookRenderer/{index.tsx → index.ts} +0 -0
@@ -0,0 +1,355 @@
1
+ import React, { useContext } from "react";
2
+ import { View, Text } from "react-native";
3
+ import { render, fireEvent, waitFor, act } from "@testing-library/react-native";
4
+
5
+ import {
6
+ MeasurementsPortal,
7
+ MeasurementsPortalContextProvider,
8
+ MeasurementPortalContext,
9
+ } from "../MeasurementsPortal";
10
+
11
+ // Mock uuid
12
+ jest.mock("uuid", () => ({
13
+ v4: jest.fn(() => "mock-uuid-key"),
14
+ }));
15
+
16
+ describe("MeasurementsPortal", () => {
17
+ const mockOnLayout = jest.fn();
18
+ const MockComponent = jest.fn(() => <View testID="mock-component" />);
19
+
20
+ beforeEach(() => {
21
+ jest.clearAllMocks();
22
+ });
23
+
24
+ it("renders with correct testID", () => {
25
+ const { getByTestId } = render(
26
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
27
+ );
28
+
29
+ expect(getByTestId("MeasurementsPortal")).toBeTruthy();
30
+ });
31
+
32
+ it("renders the passed Component", () => {
33
+ const { getByTestId } = render(
34
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
35
+ );
36
+
37
+ expect(getByTestId("mock-component")).toBeTruthy();
38
+ expect(MockComponent).toHaveBeenCalled();
39
+ });
40
+
41
+ it("passes componentProps to the Component", () => {
42
+ const componentProps = {
43
+ testProp: "test-value",
44
+ anotherProp: 123,
45
+ };
46
+
47
+ render(
48
+ <MeasurementsPortal
49
+ Component={MockComponent}
50
+ onLayout={mockOnLayout}
51
+ componentProps={componentProps}
52
+ />
53
+ );
54
+
55
+ expect(MockComponent).toHaveBeenCalledWith(componentProps, {});
56
+ });
57
+
58
+ it("calls onLayout when layout changes", () => {
59
+ const { getByTestId } = render(
60
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
61
+ );
62
+
63
+ const container = getByTestId("MeasurementsPortal");
64
+ const wrapper = container.children[0] as any;
65
+
66
+ const layoutEvent = {
67
+ nativeEvent: {
68
+ layout: { width: 100, height: 200 },
69
+ },
70
+ };
71
+
72
+ fireEvent(wrapper, "layout", layoutEvent);
73
+
74
+ expect(mockOnLayout).toHaveBeenCalledWith(layoutEvent);
75
+ });
76
+
77
+ it("applies correct container styles", () => {
78
+ const { getByTestId } = render(
79
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
80
+ );
81
+
82
+ const container = getByTestId("MeasurementsPortal");
83
+
84
+ expect(container.props.style).toEqual({
85
+ position: "absolute",
86
+ opacity: 0,
87
+ top: 0,
88
+ left: 0,
89
+ right: 0,
90
+ bottom: 0,
91
+ });
92
+ });
93
+
94
+ it("renders wrapper with onLayout", () => {
95
+ const { getByTestId } = render(
96
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
97
+ );
98
+
99
+ const container = getByTestId("MeasurementsPortal");
100
+ const wrapper = container.children[0] as any;
101
+
102
+ expect(wrapper.props.onLayout).toBe(mockOnLayout);
103
+ });
104
+
105
+ it("works with different Component types", () => {
106
+ const CustomComponent = ({ title }: { title: string }) => (
107
+ <Text testID="custom-text">{title}</Text>
108
+ );
109
+
110
+ const props = { title: "Test Title" };
111
+
112
+ const { getByTestId, getByText } = render(
113
+ <MeasurementsPortal
114
+ Component={CustomComponent}
115
+ onLayout={mockOnLayout}
116
+ componentProps={props}
117
+ />
118
+ );
119
+
120
+ expect(getByTestId("custom-text")).toBeTruthy();
121
+ expect(getByText("Test Title")).toBeTruthy();
122
+ });
123
+
124
+ it("handles component without componentProps", () => {
125
+ render(
126
+ <MeasurementsPortal Component={MockComponent} onLayout={mockOnLayout} />
127
+ );
128
+
129
+ expect(MockComponent).toHaveBeenCalledWith({}, {});
130
+ });
131
+ });
132
+
133
+ describe("MeasurementsPortalContextProvider", () => {
134
+ const TestConsumer = () => {
135
+ const context = useContext(MeasurementPortalContext);
136
+
137
+ return (
138
+ <View testID="test-consumer">
139
+ <Text testID="measuring-status">
140
+ {context?.measuringInProgress ? "measuring" : "idle"}
141
+ </Text>
142
+ </View>
143
+ );
144
+ };
145
+
146
+ it("provides context to children", () => {
147
+ const { getByTestId } = render(
148
+ <MeasurementsPortalContextProvider>
149
+ <TestConsumer />
150
+ </MeasurementsPortalContextProvider>
151
+ );
152
+
153
+ expect(getByTestId("test-consumer")).toBeTruthy();
154
+ expect(getByTestId("measuring-status")).toBeTruthy();
155
+ });
156
+
157
+ it("initially sets measuringInProgress to false", () => {
158
+ const { getByText } = render(
159
+ <MeasurementsPortalContextProvider>
160
+ <TestConsumer />
161
+ </MeasurementsPortalContextProvider>
162
+ );
163
+
164
+ expect(getByText("idle")).toBeTruthy();
165
+ });
166
+
167
+ it("renders MeasurementsPortal with default View component", () => {
168
+ const { getByTestId } = render(
169
+ <MeasurementsPortalContextProvider>
170
+ <View />
171
+ </MeasurementsPortalContextProvider>
172
+ );
173
+
174
+ expect(getByTestId("MeasurementsPortal")).toBeTruthy();
175
+ });
176
+
177
+ describe("measureComponent function", () => {
178
+ const TestMeasureComponent = () => {
179
+ const context = useContext(MeasurementPortalContext);
180
+ const [result, setResult] = React.useState<any>(null);
181
+
182
+ const handleMeasure = async () => {
183
+ if (context?.measureComponent) {
184
+ const measurement = await context.measureComponent(View, {
185
+ index: 0,
186
+ onLoadFinished: () => {},
187
+ });
188
+
189
+ setResult(measurement);
190
+ }
191
+ };
192
+
193
+ return (
194
+ <View testID="measure-component">
195
+ <Text testID="measuring-status">
196
+ {context?.measuringInProgress ? "measuring" : "idle"}
197
+ </Text>
198
+ <Text testID="measure-button" onPress={handleMeasure}>
199
+ Measure
200
+ </Text>
201
+ {result ? (
202
+ <Text testID="result">
203
+ {result.width}x{result.height}
204
+ </Text>
205
+ ) : null}
206
+ </View>
207
+ );
208
+ };
209
+
210
+ it("sets measuringInProgress to true when measureComponent is called", async () => {
211
+ const { getByTestId, getByText } = render(
212
+ <MeasurementsPortalContextProvider>
213
+ <TestMeasureComponent />
214
+ </MeasurementsPortalContextProvider>
215
+ );
216
+
217
+ expect(getByText("idle")).toBeTruthy();
218
+
219
+ await act(async () => {
220
+ fireEvent.press(getByTestId("measure-button"));
221
+ });
222
+
223
+ expect(getByText("measuring")).toBeTruthy();
224
+ });
225
+
226
+ it("can measure a component through the context", async () => {
227
+ const TestMeasureComponent = () => {
228
+ const context = useContext(MeasurementPortalContext);
229
+ const [measured, setMeasured] = React.useState(false);
230
+
231
+ React.useEffect(() => {
232
+ if (context?.measureComponent) {
233
+ context
234
+ .measureComponent(View, {
235
+ index: 5,
236
+ onLoadFinished: () => {},
237
+ })
238
+ .then(() => {
239
+ setMeasured(true);
240
+ });
241
+ }
242
+ }, [context]);
243
+
244
+ return (
245
+ <View testID="measure-component">
246
+ <Text testID="measure-status">
247
+ {measured ? "measured" : "not-measured"}
248
+ </Text>
249
+ </View>
250
+ );
251
+ };
252
+
253
+ const { getByText } = render(
254
+ <MeasurementsPortalContextProvider>
255
+ <TestMeasureComponent />
256
+ </MeasurementsPortalContextProvider>
257
+ );
258
+
259
+ // Initially should be not measured
260
+ expect(getByText("not-measured")).toBeTruthy();
261
+ });
262
+
263
+ it("handles error in onLoadFinished", async () => {
264
+ const ErrorComponent = ({ onLoadFinished }: any) => {
265
+ React.useEffect(() => {
266
+ // Simulate error
267
+ setTimeout(
268
+ () => onLoadFinished({ error: new Error("Test error") }),
269
+ 10
270
+ );
271
+ }, [onLoadFinished]);
272
+
273
+ return <View testID="error-component" />;
274
+ };
275
+
276
+ const TestErrorFlow = () => {
277
+ const context = useContext(MeasurementPortalContext);
278
+ const [result, setResult] = React.useState<any>(null);
279
+
280
+ const handleMeasure = React.useCallback(async () => {
281
+ if (context?.measureComponent) {
282
+ const measurement = await context.measureComponent(ErrorComponent, {
283
+ index: 1,
284
+ onLoadFinished: () => {},
285
+ });
286
+
287
+ setResult(measurement);
288
+ }
289
+ }, [context]);
290
+
291
+ React.useEffect(() => {
292
+ handleMeasure();
293
+ }, [handleMeasure]);
294
+
295
+ return (
296
+ <View testID="error-flow">
297
+ {result ? (
298
+ <Text testID="error-result">
299
+ {result.width}x{result.height}
300
+ </Text>
301
+ ) : null}
302
+ </View>
303
+ );
304
+ };
305
+
306
+ const { queryByText } = render(
307
+ <MeasurementsPortalContextProvider>
308
+ <TestErrorFlow />
309
+ </MeasurementsPortalContextProvider>
310
+ );
311
+
312
+ // Wait for error handling
313
+ await waitFor(() => {
314
+ expect(queryByText("0x0")).toBeTruthy();
315
+ });
316
+ });
317
+ });
318
+ });
319
+
320
+ describe("MeasurementPortalContext", () => {
321
+ it("returns null when used outside of provider", () => {
322
+ const TestComponent = () => {
323
+ const context = useContext(MeasurementPortalContext);
324
+
325
+ return (
326
+ <Text testID="context-value">
327
+ {context ? "has-context" : "no-context"}
328
+ </Text>
329
+ );
330
+ };
331
+
332
+ const { getByText } = render(<TestComponent />);
333
+ expect(getByText("no-context")).toBeTruthy();
334
+ });
335
+
336
+ it("returns context value when used inside provider", () => {
337
+ const TestComponent = () => {
338
+ const context = useContext(MeasurementPortalContext);
339
+
340
+ return (
341
+ <Text testID="context-value">
342
+ {context ? "has-context" : "no-context"}
343
+ </Text>
344
+ );
345
+ };
346
+
347
+ const { getByText } = render(
348
+ <MeasurementsPortalContextProvider>
349
+ <TestComponent />
350
+ </MeasurementsPortalContextProvider>
351
+ );
352
+
353
+ expect(getByText("has-context")).toBeTruthy();
354
+ });
355
+ });
@@ -1,4 +1,6 @@
1
1
  import * as React from "react";
2
+ import { useTheme } from "@applicaster/zapp-react-native-utils/theme";
3
+ import { useNetworkStatusLocalizations } from "./utils";
2
4
 
3
5
  type Props = {
4
6
  children?: React.ReactNode;
@@ -10,13 +12,6 @@ type Props = {
10
12
 
11
13
  const NOTIF_DURATION = 4500;
12
14
 
13
- export const ONLINE_MSG = "You are back online";
14
-
15
- export const OFFLINE_MSG = "No internet connection";
16
-
17
- const EXTRA_OFFLINE_MSG = "Please check your connection";
18
- const EXTRA_ONLINE_MSG = "Feel free to continue where you left off";
19
-
20
15
  const styles: Record<any, React.CSSProperties> = {
21
16
  body: {
22
17
  position: "absolute",
@@ -47,6 +42,9 @@ const styles: Record<any, React.CSSProperties> = {
47
42
  export const NotificationView = (props: Props) => {
48
43
  const { children, hidden, dismiss, previousOnline, online } = props;
49
44
 
45
+ const theme = useTheme<BaseThemePropertiesTV>();
46
+ const storedLocalizations = useNetworkStatusLocalizations();
47
+
50
48
  const [open, setOpen] = React.useState<boolean | null>(null);
51
49
 
52
50
  const timeout = React.useRef<NodeJS.Timeout>();
@@ -84,8 +82,18 @@ export const NotificationView = (props: Props) => {
84
82
  }, [hidden, online]);
85
83
 
86
84
  const showOnlineMsg = wentOnline || online;
87
- const MSG = showOnlineMsg ? ONLINE_MSG : OFFLINE_MSG;
88
- const EXTRA_MSG = showOnlineMsg ? EXTRA_ONLINE_MSG : EXTRA_OFFLINE_MSG;
85
+
86
+ const MSG = showOnlineMsg
87
+ ? (theme?.online_notification_title ??
88
+ storedLocalizations?.online_notification_title)
89
+ : (theme?.offline_notification_title ??
90
+ storedLocalizations?.offline_notification_title);
91
+
92
+ const EXTRA_MSG = showOnlineMsg
93
+ ? (theme?.online_notification_subtitle ??
94
+ storedLocalizations?.online_notification_subtitle)
95
+ : (theme?.offline_notification_subtitle ??
96
+ storedLocalizations?.offline_notification_subtitle);
89
97
 
90
98
  return (
91
99
  <div onClick={onClose}>
@@ -1,4 +1,6 @@
1
1
  import * as React from "react";
2
+ import { useTheme } from "@applicaster/zapp-react-native-utils/theme";
3
+ import { useNetworkStatusLocalizations } from "./utils";
2
4
 
3
5
  type Props = {
4
6
  children?: React.ReactNode;
@@ -9,12 +11,6 @@ type Props = {
9
11
 
10
12
  const DURATION_TO_HIDE_AFTER_BACK_TO_ONLINE = 4500; // ms
11
13
 
12
- const ONLINE_TITLE = "You are back online";
13
- const ONLINE_SUBTITLE = "Feel free to continue where you left off";
14
-
15
- const OFFLINE_TITLE = "No internet connection";
16
- const OFFLINE_SUBTITLE = "Please check your connection";
17
-
18
14
  const styles: Record<any, React.CSSProperties> = {
19
15
  body: {
20
16
  position: "absolute",
@@ -47,6 +43,9 @@ let timer: NodeJS.Timeout;
47
43
  export const NotificationView = (props: Props) => {
48
44
  const { children, dismiss, previousOnline = true, online } = props;
49
45
 
46
+ const theme = useTheme<BaseThemePropertiesTV>();
47
+ const storedLocalizations = useNetworkStatusLocalizations();
48
+
50
49
  const [shown, setShown] = React.useState<boolean>(false);
51
50
 
52
51
  const onClose = () => {
@@ -79,8 +78,17 @@ export const NotificationView = (props: Props) => {
79
78
  }
80
79
  }, [online, previousOnline]);
81
80
 
82
- const title = online ? ONLINE_TITLE : OFFLINE_TITLE;
83
- const subtitle = online ? ONLINE_SUBTITLE : OFFLINE_SUBTITLE;
81
+ const title = online
82
+ ? (theme?.online_notification_title ??
83
+ storedLocalizations?.online_notification_title)
84
+ : (theme?.offline_notification_title ??
85
+ storedLocalizations?.offline_notification_title);
86
+
87
+ const subtitle = online
88
+ ? (theme?.online_notification_subtitle ??
89
+ storedLocalizations?.online_notification_subtitle)
90
+ : (theme?.offline_notification_subtitle ??
91
+ storedLocalizations?.offline_notification_subtitle);
84
92
 
85
93
  return (
86
94
  <div onClick={onClose}>
@@ -8,7 +8,7 @@ import {
8
8
  TouchableWithoutFeedback,
9
9
  } from "react-native";
10
10
 
11
- import { usePickFromState } from "@applicaster/zapp-react-native-redux/hooks";
11
+ import { usePlugins } from "@applicaster/zapp-react-native-redux";
12
12
  import { platformSelect } from "@applicaster/zapp-react-native-utils/reactUtils";
13
13
  import { textTransform } from "@applicaster/zapp-react-native-utils/cellUtils";
14
14
  import { useNotificationHeight } from "../utils";
@@ -46,7 +46,7 @@ export const NotificationView = ({
46
46
  online = true,
47
47
  dismiss,
48
48
  }: Props) => {
49
- const { plugins } = usePickFromState(["plugins"]);
49
+ const plugins = usePlugins();
50
50
  const { statusHeight, notificationHeight } = useNotificationHeight();
51
51
 
52
52
  const offlinePlugin = R.find(
@@ -8,25 +8,24 @@ import {
8
8
  offlinePhrase,
9
9
  } from "../NotificationView";
10
10
 
11
- jest.mock("@applicaster/zapp-react-native-redux/hooks", () => ({
12
- usePickFromState: () => ({
13
- plugins: [
14
- {
15
- name: "offline experience",
16
- identifier: "offline-experience",
17
- type: "general",
18
- module: {
19
- useOfflineExperienceConfiguration: () => ({
20
- configurationFields: {},
21
- localizations: {
22
- offline_toast_message: "No internet connection",
23
- online_toast_message: "You are back online",
24
- },
25
- }),
26
- },
11
+ jest.mock("@applicaster/zapp-react-native-redux", () => ({
12
+ ...jest.requireActual("@applicaster/zapp-react-native-redux"),
13
+ usePlugins: () => [
14
+ {
15
+ name: "offline experience",
16
+ identifier: "offline-experience",
17
+ type: "general",
18
+ module: {
19
+ useOfflineExperienceConfiguration: () => ({
20
+ configurationFields: {},
21
+ localizations: {
22
+ offline_toast_message: "No internet connection",
23
+ online_toast_message: "You are back online",
24
+ },
25
+ }),
27
26
  },
28
- ],
29
- }),
27
+ },
28
+ ],
30
29
  }));
31
30
 
32
31
  jest.mock("react-native-safe-area-context", () => ({
@@ -0,0 +1,34 @@
1
+ import * as React from "react";
2
+ import { sessionStorage } from "@applicaster/zapp-react-native-bridge/ZappStorage/SessionStorage";
3
+
4
+ export const NETWORK_STATUS_LOCALIZATIONS_KEY = "network_status_localizations";
5
+
6
+ export const THEME_STORAGE_NAMESPACE = "quick-brick-theme";
7
+
8
+ export function useNetworkStatusLocalizations() {
9
+ const [storedLocalizations, setStoredLocalizations] = React.useState<Record<
10
+ string,
11
+ string
12
+ > | null>(null);
13
+
14
+ React.useEffect(() => {
15
+ async function loadStoredLocalizations() {
16
+ try {
17
+ const stored = await sessionStorage.getItem(
18
+ NETWORK_STATUS_LOCALIZATIONS_KEY,
19
+ THEME_STORAGE_NAMESPACE
20
+ );
21
+
22
+ if (stored) {
23
+ setStoredLocalizations(stored);
24
+ }
25
+ } catch (error) {
26
+ console.error("Error loading network status localizations", error);
27
+ }
28
+ }
29
+
30
+ loadStoredLocalizations();
31
+ }, []);
32
+
33
+ return storedLocalizations;
34
+ }
@@ -8,36 +8,45 @@ const mockPreviousValue = null;
8
8
 
9
9
  jest.mock("@applicaster/zapp-react-native-utils/reactHooks/connection", () => {
10
10
  return {
11
+ ...jest.requireActual(
12
+ "@applicaster/zapp-react-native-utils/reactHooks/connection"
13
+ ),
11
14
  useConnectionInfo: jest.fn(() => mockConnectionStatus),
12
15
  };
13
16
  });
14
17
 
15
18
  jest.mock("@applicaster/zapp-react-native-utils/reactHooks/utils", () => {
16
19
  return {
20
+ ...jest.requireActual(
21
+ "@applicaster/zapp-react-native-utils/reactHooks/utils"
22
+ ),
17
23
  usePrevious: jest.fn(() => mockPreviousValue),
18
24
  };
19
25
  });
20
26
 
27
+ const mock_storeObj = {
28
+ plugins: [
29
+ {
30
+ name: "offline experience",
31
+ identifier: "offline-experience",
32
+ type: "general",
33
+ module: {
34
+ useOfflineExperienceConfiguration: () => ({
35
+ configurationFields: {},
36
+ localizations: {
37
+ offline_toast_message: "No internet connection",
38
+ online_toast_message: "You are back online",
39
+ },
40
+ }),
41
+ },
42
+ },
43
+ ],
44
+ };
45
+
21
46
  jest.mock("@applicaster/zapp-react-native-redux/hooks", () => ({
22
47
  ...jest.requireActual("@applicaster/zapp-react-native-redux/hooks"),
23
- usePickFromState: () => ({
24
- plugins: [
25
- {
26
- name: "offline experience",
27
- identifier: "offline-experience",
28
- type: "general",
29
- module: {
30
- useOfflineExperienceConfiguration: () => ({
31
- configurationFields: {},
32
- localizations: {
33
- offline_toast_message: "No internet connection",
34
- online_toast_message: "You are back online",
35
- },
36
- }),
37
- },
38
- },
39
- ],
40
- }),
48
+ usePlugins: () => mock_storeObj.plugins,
49
+ usePickFromState: () => ({}),
41
50
  }));
42
51
 
43
52
  jest.mock("react-native-safe-area-context", () => ({