@applicaster/zapp-react-native-ui-components 15.0.0-rc.7 → 15.0.0-rc.71

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 (88) 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 +8 -3
  4. package/Components/Cell/FocusableWrapper.tsx +47 -0
  5. package/Components/Cell/TvOSCellComponent.tsx +106 -19
  6. package/Components/Focusable/Focusable.tsx +4 -2
  7. package/Components/Focusable/FocusableTvOS.tsx +18 -1
  8. package/Components/Focusable/__tests__/__snapshots__/FocusableTvOS.test.tsx.snap +1 -0
  9. package/Components/FocusableGroup/FocusableTvOS.tsx +32 -1
  10. package/Components/GeneralContentScreen/utils/useCurationAPI.ts +19 -3
  11. package/Components/HandlePlayable/HandlePlayable.tsx +17 -65
  12. package/Components/HandlePlayable/const.ts +3 -0
  13. package/Components/HandlePlayable/utils.ts +74 -0
  14. package/Components/Layout/TV/LayoutBackground.tsx +5 -2
  15. package/Components/Layout/TV/ScreenContainer.tsx +2 -6
  16. package/Components/Layout/TV/index.tsx +3 -4
  17. package/Components/Layout/TV/index.web.tsx +3 -4
  18. package/Components/LinkHandler/LinkHandler.tsx +2 -2
  19. package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx +16 -1
  20. package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +30 -2
  21. package/Components/MasterCell/DefaultComponents/Image/Image.android.tsx +5 -1
  22. package/Components/MasterCell/DefaultComponents/Image/Image.ios.tsx +11 -3
  23. package/Components/MasterCell/DefaultComponents/Image/Image.web.tsx +9 -1
  24. package/Components/MasterCell/DefaultComponents/Image/hooks/useImage.ts +15 -14
  25. package/Components/MasterCell/DefaultComponents/LiveImage/index.tsx +10 -6
  26. package/Components/MasterCell/DefaultComponents/Text/index.tsx +8 -8
  27. package/Components/MasterCell/index.tsx +2 -0
  28. package/Components/MasterCell/utils/__tests__/resolveColor.test.js +82 -3
  29. package/Components/MasterCell/utils/index.ts +61 -31
  30. package/Components/MeasurmentsPortal/MeasurementsPortal.tsx +102 -87
  31. package/Components/MeasurmentsPortal/__tests__/MeasurementsPortal.test.tsx +355 -0
  32. package/Components/OfflineHandler/NotificationView/NotificationView.tsx +2 -2
  33. package/Components/OfflineHandler/NotificationView/__tests__/index.test.tsx +17 -18
  34. package/Components/OfflineHandler/__tests__/index.test.tsx +27 -18
  35. package/Components/PlayerContainer/PlayerContainer.tsx +51 -55
  36. package/Components/PlayerContainer/useRestrictMobilePlayback.tsx +101 -0
  37. package/Components/PlayerImageBackground/index.tsx +3 -22
  38. package/Components/River/TV/River.tsx +31 -14
  39. package/Components/River/TV/index.tsx +8 -4
  40. package/Components/River/TV/utils/__tests__/toStringOrEmpty.test.ts +30 -0
  41. package/Components/River/TV/utils/index.ts +4 -0
  42. package/Components/River/TV/withFocusableGroupForContent.tsx +71 -0
  43. package/Components/Screen/TV/hooks/useInitialFocus.ts +14 -4
  44. package/Components/Screen/TV/index.web.tsx +4 -2
  45. package/Components/Screen/__tests__/Screen.test.tsx +65 -42
  46. package/Components/Screen/__tests__/__snapshots__/Screen.test.tsx.snap +68 -42
  47. package/Components/Screen/hooks.ts +2 -3
  48. package/Components/Screen/index.tsx +24 -8
  49. package/Components/Screen/navigationHandler.ts +49 -24
  50. package/Components/Screen/orientationHandler.ts +3 -3
  51. package/Components/ScreenResolver/index.tsx +13 -7
  52. package/Components/ScreenRevealManager/ScreenRevealManager.ts +40 -8
  53. package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +86 -69
  54. package/Components/ScreenRevealManager/utils/index.ts +23 -0
  55. package/Components/ScreenRevealManager/withScreenRevealManager.tsx +54 -24
  56. package/Components/Tabs/TV/Tabs.tsx +20 -3
  57. package/Components/Transitioner/Scene.tsx +15 -2
  58. package/Components/Transitioner/index.js +3 -3
  59. package/Components/VideoLive/__tests__/__snapshots__/PlayerLiveImageComponent.test.tsx.snap +1 -0
  60. package/Components/VideoModal/ModalAnimation/ModalAnimationContext.tsx +118 -171
  61. package/Components/VideoModal/ModalAnimation/index.ts +2 -13
  62. package/Components/VideoModal/ModalAnimation/utils.ts +1 -327
  63. package/Components/VideoModal/PlayerWrapper.tsx +14 -88
  64. package/Components/VideoModal/VideoModal.tsx +1 -5
  65. package/Components/VideoModal/__tests__/PlayerWrapper.test.tsx +1 -0
  66. package/Components/VideoModal/hooks/useModalSize.ts +10 -5
  67. package/Components/VideoModal/playerWrapperStyle.ts +70 -0
  68. package/Components/VideoModal/playerWrapperUtils.ts +91 -0
  69. package/Components/VideoModal/utils.ts +19 -9
  70. package/Decorators/Analytics/index.tsx +6 -5
  71. package/Decorators/ConfigurationWrapper/__tests__/__snapshots__/withConfigurationProvider.test.tsx.snap +1 -0
  72. package/Decorators/ConfigurationWrapper/const.ts +1 -0
  73. package/Decorators/ZappPipesDataConnector/__tests__/zappPipesDataConnector.test.js +1 -1
  74. package/Decorators/ZappPipesDataConnector/index.tsx +2 -2
  75. package/Decorators/ZappPipesDataConnector/resolvers/StaticFeedResolver.tsx +1 -1
  76. package/Helpers/DataSourceHelper/__tests__/itemLimitForData.test.ts +80 -0
  77. package/Helpers/DataSourceHelper/index.ts +19 -0
  78. package/index.d.ts +7 -0
  79. package/package.json +6 -5
  80. package/Components/River/TV/withTVEventHandler.tsx +0 -27
  81. package/Components/VideoModal/ModalAnimation/AnimatedPlayerModalWrapper.tsx +0 -60
  82. package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.tsx +0 -417
  83. package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.web.tsx +0 -294
  84. package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.tsx +0 -176
  85. package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.web.tsx +0 -93
  86. package/Components/VideoModal/ModalAnimation/AnimationComponent.tsx +0 -500
  87. package/Components/VideoModal/ModalAnimation/__tests__/getMoveUpValue.test.ts +0 -108
  88. package/Helpers/DataSourceHelper/index.js +0 -19
@@ -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
+ });
@@ -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", () => ({
@@ -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", () => ({
@@ -17,7 +17,7 @@ import {
17
17
 
18
18
  import { TVEventHandlerComponent } from "@applicaster/zapp-react-native-tvos-ui-components/Components/TVEventHandlerComponent";
19
19
  import { usePrevious } from "@applicaster/zapp-react-native-utils/reactHooks/utils";
20
- import { usePickFromState } from "@applicaster/zapp-react-native-redux/hooks";
20
+
21
21
  import {
22
22
  useBackHandler,
23
23
  useNavigation,
@@ -56,15 +56,12 @@ import { toNumber } from "@applicaster/zapp-react-native-utils/numberUtils";
56
56
  import { usePlayNextOverlay } from "@applicaster/zapp-react-native-utils/appUtils/playerManager/usePlayNextOverlay";
57
57
  import { PlayNextState } from "@applicaster/zapp-react-native-utils/appUtils/playerManager/OverlayObserver/OverlaysObserver";
58
58
 
59
- import {
60
- PlayerAnimationStateEnum,
61
- useModalAnimationContext,
62
- } from "@applicaster/zapp-react-native-ui-components/Components/VideoModal/ModalAnimation";
63
-
64
59
  import {
65
60
  PlayerNativeCommandTypes,
66
61
  PlayerNativeSendCommand,
67
62
  } from "@applicaster/zapp-react-native-utils/appUtils/playerManager/playerNativeCommand";
63
+ import { useAppData } from "@applicaster/zapp-react-native-redux";
64
+ import { useRestrictMobilePlayback } from "./useRestrictMobilePlayback";
68
65
 
69
66
  type Props = {
70
67
  Player: React.ComponentType<any>;
@@ -243,14 +240,11 @@ const PlayerContainerComponent = (props: Props) => {
243
240
  const [isLoadingNextVideo, setIsLoadingNextVideo] = React.useState(false);
244
241
 
245
242
  const navigator = useNavigation();
246
- const { appData } = usePickFromState(["appData"]);
243
+ const { isTabletPortrait } = useAppData();
247
244
  const prevItemId = usePrevious(item?.id);
248
245
  const screenData = useTargetScreenData(item);
249
246
  const { setVisible: showNavBar } = useSetNavbarState();
250
247
 
251
- const { isActiveGesture, startComponentsAnimation, setPlayerAnimationState } =
252
- useModalAnimationContext();
253
-
254
248
  const playerEvent = (event, ...args) => {
255
249
  playerManager.invokeHandler(event, ...args);
256
250
  };
@@ -271,7 +265,21 @@ const PlayerContainerComponent = (props: Props) => {
271
265
 
272
266
  showNavBar(true);
273
267
  navigator.goBack();
274
- }, [isModal, navigator.goBack, state.playerId, showNavBar]);
268
+ }, [isModal, state.playerId, showNavBar, navigator]);
269
+
270
+ const pluginConfiguration = React.useMemo(() => {
271
+ return (
272
+ playerManager.getPluginConfiguration() ||
273
+ R.prop("__plugin_configuration", Player)
274
+ );
275
+ }, [playerManager.isRegistered()]);
276
+
277
+ const { isRestricted } = useRestrictMobilePlayback({
278
+ player,
279
+ entry: item,
280
+ pluginConfiguration,
281
+ close,
282
+ });
275
283
 
276
284
  const playEntry = (entry) => navigator.replaceTop(entry, { mode });
277
285
 
@@ -463,13 +471,6 @@ const PlayerContainerComponent = (props: Props) => {
463
471
  }
464
472
  }, []);
465
473
 
466
- const pluginConfiguration = React.useMemo(() => {
467
- return (
468
- playerManager.getPluginConfiguration() ||
469
- R.prop("__plugin_configuration", Player)
470
- );
471
- }, [playerManager.isRegistered()]);
472
-
473
474
  const disableMiniPlayer = React.useMemo(() => {
474
475
  return pluginConfiguration?.disable_mini_player_when_inline;
475
476
  }, [pluginConfiguration]);
@@ -482,8 +483,6 @@ const PlayerContainerComponent = (props: Props) => {
482
483
  if (isModal && mode === VideoModalMode.MAXIMIZED) {
483
484
  if (disableMiniPlayer) {
484
485
  navigator.closeVideoModal();
485
- } else {
486
- setPlayerAnimationState(PlayerAnimationStateEnum.minimize);
487
486
  }
488
487
  }
489
488
 
@@ -671,41 +670,38 @@ const PlayerContainerComponent = (props: Props) => {
671
670
  <PlayerFocusableWrapperView
672
671
  nextFocusDown={context.bottomFocusableId}
673
672
  >
674
- <Player
675
- source={{
676
- uri,
677
- entry: item,
678
- }}
679
- focused={isInlineTV ? true : undefined}
680
- autoplay={true}
681
- controls={false}
682
- disableCastAction={disableCastAction}
683
- docked={
684
- navigator.isVideoModalDocked() &&
685
- !startComponentsAnimation &&
686
- !isActiveGesture
687
- }
688
- entry={item}
689
- fullscreen={mode === VideoModalMode.FULLSCREEN}
690
- inline={inline}
691
- isModal={isModal}
692
- isTabletPortrait={appData.isTabletPortrait}
693
- muted={false}
694
- playableItem={item}
695
- playerEvent={playerEvent}
696
- playerId={state.playerId}
697
- pluginConfiguration={pluginConfiguration}
698
- ref={playerRef}
699
- toggleFullscreen={toggleFullscreen}
700
- style={videoStyle}
701
- playNextData={playNextData}
702
- setNextVideoPreloadThresholdPercentage={
703
- setNextVideoPreloadThresholdPercentage
704
- }
705
- startComponentsAnimation={startComponentsAnimation}
706
- >
707
- {renderApplePlayer(applePlayerProps)}
708
- </Player>
673
+ {isRestricted ? null : (
674
+ <Player
675
+ source={{
676
+ uri,
677
+ entry: item,
678
+ }}
679
+ focused={isInlineTV ? true : undefined}
680
+ autoplay={true}
681
+ controls={false}
682
+ disableCastAction={disableCastAction}
683
+ docked={navigator.isVideoModalDocked()}
684
+ entry={item}
685
+ fullscreen={mode === VideoModalMode.FULLSCREEN}
686
+ inline={inline}
687
+ isModal={isModal}
688
+ isTabletPortrait={isTabletPortrait}
689
+ muted={false}
690
+ playableItem={item}
691
+ playerEvent={playerEvent}
692
+ playerId={state.playerId}
693
+ pluginConfiguration={pluginConfiguration}
694
+ ref={playerRef}
695
+ toggleFullscreen={toggleFullscreen}
696
+ style={videoStyle}
697
+ playNextData={playNextData}
698
+ setNextVideoPreloadThresholdPercentage={
699
+ setNextVideoPreloadThresholdPercentage
700
+ }
701
+ >
702
+ {renderApplePlayer(applePlayerProps)}
703
+ </Player>
704
+ )}
709
705
  </PlayerFocusableWrapperView>
710
706
 
711
707
  {state.error ? <ErrorDisplay error={state.error} /> : null}