@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.
- package/Components/AnimatedInOut/index.tsx +69 -26
- package/Components/BaseFocusable/index.ios.ts +12 -2
- package/Components/Cell/Cell.tsx +8 -3
- package/Components/Cell/FocusableWrapper.tsx +47 -0
- package/Components/Cell/TvOSCellComponent.tsx +106 -19
- package/Components/Focusable/Focusable.tsx +4 -2
- package/Components/Focusable/FocusableTvOS.tsx +18 -1
- package/Components/Focusable/__tests__/__snapshots__/FocusableTvOS.test.tsx.snap +1 -0
- package/Components/FocusableGroup/FocusableTvOS.tsx +32 -1
- package/Components/GeneralContentScreen/utils/useCurationAPI.ts +19 -3
- package/Components/HandlePlayable/HandlePlayable.tsx +17 -65
- package/Components/HandlePlayable/const.ts +3 -0
- package/Components/HandlePlayable/utils.ts +74 -0
- package/Components/Layout/TV/LayoutBackground.tsx +5 -2
- package/Components/Layout/TV/ScreenContainer.tsx +2 -6
- package/Components/Layout/TV/index.tsx +3 -4
- package/Components/Layout/TV/index.web.tsx +3 -4
- package/Components/LinkHandler/LinkHandler.tsx +2 -2
- package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx +16 -1
- package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +30 -2
- package/Components/MasterCell/DefaultComponents/Image/Image.android.tsx +5 -1
- package/Components/MasterCell/DefaultComponents/Image/Image.ios.tsx +11 -3
- package/Components/MasterCell/DefaultComponents/Image/Image.web.tsx +9 -1
- package/Components/MasterCell/DefaultComponents/Image/hooks/useImage.ts +15 -14
- package/Components/MasterCell/DefaultComponents/LiveImage/index.tsx +10 -6
- package/Components/MasterCell/DefaultComponents/Text/index.tsx +8 -8
- package/Components/MasterCell/index.tsx +2 -0
- package/Components/MasterCell/utils/__tests__/resolveColor.test.js +82 -3
- package/Components/MasterCell/utils/index.ts +61 -31
- package/Components/MeasurmentsPortal/MeasurementsPortal.tsx +102 -87
- package/Components/MeasurmentsPortal/__tests__/MeasurementsPortal.test.tsx +355 -0
- package/Components/OfflineHandler/NotificationView/NotificationView.tsx +2 -2
- package/Components/OfflineHandler/NotificationView/__tests__/index.test.tsx +17 -18
- package/Components/OfflineHandler/__tests__/index.test.tsx +27 -18
- package/Components/PlayerContainer/PlayerContainer.tsx +51 -55
- package/Components/PlayerContainer/useRestrictMobilePlayback.tsx +101 -0
- package/Components/PlayerImageBackground/index.tsx +3 -22
- package/Components/River/TV/River.tsx +31 -14
- package/Components/River/TV/index.tsx +8 -4
- package/Components/River/TV/utils/__tests__/toStringOrEmpty.test.ts +30 -0
- package/Components/River/TV/utils/index.ts +4 -0
- package/Components/River/TV/withFocusableGroupForContent.tsx +71 -0
- package/Components/Screen/TV/hooks/useInitialFocus.ts +14 -4
- package/Components/Screen/TV/index.web.tsx +4 -2
- package/Components/Screen/__tests__/Screen.test.tsx +65 -42
- package/Components/Screen/__tests__/__snapshots__/Screen.test.tsx.snap +68 -42
- package/Components/Screen/hooks.ts +2 -3
- package/Components/Screen/index.tsx +24 -8
- package/Components/Screen/navigationHandler.ts +49 -24
- package/Components/Screen/orientationHandler.ts +3 -3
- package/Components/ScreenResolver/index.tsx +13 -7
- package/Components/ScreenRevealManager/ScreenRevealManager.ts +40 -8
- package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +86 -69
- package/Components/ScreenRevealManager/utils/index.ts +23 -0
- package/Components/ScreenRevealManager/withScreenRevealManager.tsx +54 -24
- package/Components/Tabs/TV/Tabs.tsx +20 -3
- package/Components/Transitioner/Scene.tsx +15 -2
- package/Components/Transitioner/index.js +3 -3
- package/Components/VideoLive/__tests__/__snapshots__/PlayerLiveImageComponent.test.tsx.snap +1 -0
- package/Components/VideoModal/ModalAnimation/ModalAnimationContext.tsx +118 -171
- package/Components/VideoModal/ModalAnimation/index.ts +2 -13
- package/Components/VideoModal/ModalAnimation/utils.ts +1 -327
- package/Components/VideoModal/PlayerWrapper.tsx +14 -88
- package/Components/VideoModal/VideoModal.tsx +1 -5
- package/Components/VideoModal/__tests__/PlayerWrapper.test.tsx +1 -0
- package/Components/VideoModal/hooks/useModalSize.ts +10 -5
- package/Components/VideoModal/playerWrapperStyle.ts +70 -0
- package/Components/VideoModal/playerWrapperUtils.ts +91 -0
- package/Components/VideoModal/utils.ts +19 -9
- package/Decorators/Analytics/index.tsx +6 -5
- package/Decorators/ConfigurationWrapper/__tests__/__snapshots__/withConfigurationProvider.test.tsx.snap +1 -0
- package/Decorators/ConfigurationWrapper/const.ts +1 -0
- package/Decorators/ZappPipesDataConnector/__tests__/zappPipesDataConnector.test.js +1 -1
- package/Decorators/ZappPipesDataConnector/index.tsx +2 -2
- package/Decorators/ZappPipesDataConnector/resolvers/StaticFeedResolver.tsx +1 -1
- package/Helpers/DataSourceHelper/__tests__/itemLimitForData.test.ts +80 -0
- package/Helpers/DataSourceHelper/index.ts +19 -0
- package/index.d.ts +7 -0
- package/package.json +6 -5
- package/Components/River/TV/withTVEventHandler.tsx +0 -27
- package/Components/VideoModal/ModalAnimation/AnimatedPlayerModalWrapper.tsx +0 -60
- package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.tsx +0 -417
- package/Components/VideoModal/ModalAnimation/AnimatedScrollModal.web.tsx +0 -294
- package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.tsx +0 -176
- package/Components/VideoModal/ModalAnimation/AnimatedVideoPlayerComponent.web.tsx +0 -93
- package/Components/VideoModal/ModalAnimation/AnimationComponent.tsx +0 -500
- package/Components/VideoModal/ModalAnimation/__tests__/getMoveUpValue.test.ts +0 -108
- 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 {
|
|
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
|
|
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
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
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
|
-
|
|
24
|
-
|
|
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
|
-
|
|
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 {
|
|
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,
|
|
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
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
684
|
-
navigator.isVideoModalDocked()
|
|
685
|
-
|
|
686
|
-
|
|
687
|
-
|
|
688
|
-
|
|
689
|
-
|
|
690
|
-
|
|
691
|
-
|
|
692
|
-
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
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}
|