@applicaster/zapp-react-native-ui-components 15.0.0-rc.98 → 15.1.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 (115) hide show
  1. package/Components/BaseFocusable/index.ios.ts +2 -12
  2. package/Components/Cell/FocusableWrapper.tsx +0 -3
  3. package/Components/Cell/TvOSCellComponent.tsx +0 -5
  4. package/Components/Focusable/Focusable.tsx +2 -4
  5. package/Components/Focusable/FocusableTvOS.tsx +1 -18
  6. package/Components/Focusable/__tests__/__snapshots__/FocusableTvOS.test.tsx.snap +0 -1
  7. package/Components/FocusableGroup/FocusableTvOS.tsx +1 -30
  8. package/Components/GeneralContentScreen/GeneralContentScreen.tsx +39 -28
  9. package/Components/GeneralContentScreen/__tests__/GeneralContentScreen.test.tsx +104 -0
  10. package/Components/GeneralContentScreen/utils/__tests__/getScreenDataSource.test.ts +19 -0
  11. package/Components/GeneralContentScreen/utils/__tests__/useCurationAPI.test.js +1 -1
  12. package/Components/GeneralContentScreen/utils/getScreenDataSource.ts +9 -0
  13. package/Components/HandlePlayable/HandlePlayable.tsx +24 -42
  14. package/Components/HandlePlayable/utils.ts +31 -0
  15. package/Components/HookRenderer/HookRenderer.tsx +40 -10
  16. package/Components/HookRenderer/__tests__/HookRenderer.test.tsx +60 -0
  17. package/Components/Layout/TV/LayoutBackground.tsx +2 -5
  18. package/Components/Layout/TV/ScreenContainer.tsx +6 -2
  19. package/Components/Layout/TV/__tests__/__snapshots__/index.test.tsx.snap +5 -0
  20. package/Components/Layout/TV/index.tsx +4 -3
  21. package/Components/Layout/TV/index.web.tsx +4 -3
  22. package/Components/LinkHandler/LinkHandler.tsx +2 -2
  23. package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +10 -4
  24. package/Components/MasterCell/DefaultComponents/Image/Image.android.tsx +1 -5
  25. package/Components/MasterCell/DefaultComponents/Image/Image.ios.tsx +3 -11
  26. package/Components/MasterCell/DefaultComponents/Image/Image.web.tsx +1 -9
  27. package/Components/MasterCell/DefaultComponents/Image/hooks/useImage.ts +14 -15
  28. package/Components/MasterCell/DefaultComponents/LiveImage/__tests__/prepareEntry.test.ts +352 -0
  29. package/Components/MasterCell/DefaultComponents/LiveImage/executePreloadHooks.ts +136 -0
  30. package/Components/MasterCell/DefaultComponents/LiveImage/index.tsx +34 -16
  31. package/Components/MasterCell/DefaultComponents/SecondaryImage/hooks/__tests__/useGetImageDimensions.test.ts +6 -7
  32. package/Components/MasterCell/DefaultComponents/Text/index.tsx +2 -6
  33. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/index.ts +2 -6
  34. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/utils/__tests__/getPluginIdentifier.test.ts +11 -233
  35. package/Components/MasterCell/DefaultComponents/tv/TvActionButtons/utils/index.ts +15 -19
  36. package/Components/Navigator/StackNavigator.tsx +6 -0
  37. package/Components/OfflineHandler/NotificationView/NotificationView.tsx +2 -2
  38. package/Components/OfflineHandler/NotificationView/__tests__/index.test.tsx +18 -17
  39. package/Components/OfflineHandler/__tests__/index.test.tsx +18 -27
  40. package/Components/PlayerContainer/PlayerContainer.tsx +14 -32
  41. package/Components/PreloaderWrapper/__tests__/index.test.tsx +26 -0
  42. package/Components/PreloaderWrapper/index.tsx +15 -0
  43. package/Components/River/ComponentsMap/ComponentsMap.tsx +15 -0
  44. package/Components/River/ComponentsMap/hooks/__tests__/useLoadingState.test.ts +1 -1
  45. package/Components/River/RefreshControl.tsx +9 -3
  46. package/Components/River/RiverItem.tsx +26 -20
  47. package/Components/River/TV/River.tsx +14 -31
  48. package/Components/River/TV/index.tsx +4 -8
  49. package/Components/River/TV/withTVEventHandler.tsx +36 -0
  50. package/Components/River/__tests__/__snapshots__/componentsMap.test.js.snap +1 -0
  51. package/Components/Screen/TV/index.web.tsx +2 -4
  52. package/Components/Screen/__tests__/Screen.test.tsx +43 -65
  53. package/Components/Screen/__tests__/__snapshots__/Screen.test.tsx.snap +44 -68
  54. package/Components/Screen/hooks.ts +76 -5
  55. package/Components/Screen/index.tsx +10 -3
  56. package/Components/Screen/orientationHandler.ts +3 -3
  57. package/Components/ScreenFeedLoader/ScreenFeedLoader.tsx +46 -0
  58. package/Components/ScreenFeedLoader/__tests__/ScreenFeedLoader.test.tsx +94 -0
  59. package/Components/ScreenFeedLoader/index.ts +1 -0
  60. package/Components/ScreenResolver/__tests__/screenResolver.test.js +24 -0
  61. package/Components/ScreenResolver/hooks/index.ts +3 -0
  62. package/Components/ScreenResolver/hooks/useGetComponent.ts +15 -0
  63. package/Components/ScreenResolver/hooks/useScreenComponentResolver.tsx +90 -0
  64. package/Components/ScreenResolver/index.tsx +9 -115
  65. package/Components/ScreenResolver/utils/__tests__/getScreenTypeProps.test.ts +45 -0
  66. package/Components/ScreenResolver/utils/getScreenTypeProps.ts +43 -0
  67. package/Components/ScreenResolver/utils/index.ts +1 -0
  68. package/Components/ScreenResolver/withDefaultScreenContext.tsx +16 -0
  69. package/Components/ScreenResolverFeedProvider/ScreenResolverFeedProvider.tsx +25 -0
  70. package/Components/ScreenResolverFeedProvider/__tests__/ScreenResolverFeedProvider.test.tsx +44 -0
  71. package/Components/ScreenResolverFeedProvider/index.ts +1 -0
  72. package/Components/ScreenRevealManager/ScreenRevealManager.ts +8 -40
  73. package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +69 -86
  74. package/Components/ScreenRevealManager/withScreenRevealManager.tsx +4 -1
  75. package/Components/Tabs/TabContent.tsx +4 -7
  76. package/Components/Transitioner/Scene.tsx +9 -15
  77. package/Components/Transitioner/index.js +3 -3
  78. package/Components/VideoLive/LiveImageManager.ts +199 -54
  79. package/Components/VideoLive/PlayerLiveImageComponent.tsx +31 -33
  80. package/Components/VideoLive/__tests__/PlayerLiveImageComponent.test.tsx +2 -17
  81. package/Components/VideoModal/ModalAnimation/ModalAnimationContext.tsx +5 -5
  82. package/Components/VideoModal/hooks/__tests__/useDelayedPlayerDetails.test.ts +7 -15
  83. package/Components/VideoModal/utils.ts +9 -12
  84. package/Components/Viewport/ViewportEvents/__tests__/viewportEvents.test.js +1 -1
  85. package/Components/ZappFrameworkComponents/BarView/BarView.tsx +6 -4
  86. package/Components/ZappFrameworkComponents/BarView/__tests__/BarView.test.tsx +2 -2
  87. package/Components/ZappUIComponent/index.tsx +12 -6
  88. package/Components/index.js +1 -1
  89. package/Contexts/ScreenContext/__tests__/index.test.tsx +57 -0
  90. package/Contexts/ScreenContext/index.tsx +64 -26
  91. package/Contexts/ScreenTrackedViewPositionsContext/__tests__/index.test.tsx +1 -1
  92. package/Contexts/ZappPipesContext/ZappPipesContextFactory.tsx +18 -7
  93. package/Decorators/Analytics/index.tsx +5 -6
  94. package/Decorators/ConfigurationWrapper/__tests__/__snapshots__/withConfigurationProvider.test.tsx.snap +0 -1
  95. package/Decorators/ConfigurationWrapper/const.ts +0 -1
  96. package/Decorators/ZappPipesDataConnector/ResolverSelector.tsx +25 -7
  97. package/Decorators/ZappPipesDataConnector/__tests__/ResolverSelector.test.tsx +212 -5
  98. package/Decorators/ZappPipesDataConnector/__tests__/zappPipesDataConnector.test.js +1 -1
  99. package/Decorators/ZappPipesDataConnector/index.tsx +2 -2
  100. package/Decorators/ZappPipesDataConnector/resolvers/StaticFeedResolver.tsx +1 -1
  101. package/Helpers/DataSourceHelper/index.js +19 -0
  102. package/events/index.ts +2 -0
  103. package/events/scrollEndReached.ts +15 -0
  104. package/package.json +5 -5
  105. package/Components/MasterCell/DefaultComponents/Text/utils/__tests__/withAdjustedLineHeight.test.ts +0 -46
  106. package/Components/MasterCell/DefaultComponents/Text/utils/index.ts +0 -21
  107. package/Components/PlayerContainer/ErrorDisplay/ErrorDisplay.tsx +0 -57
  108. package/Components/PlayerContainer/ErrorDisplay/index.ts +0 -9
  109. package/Components/PlayerContainer/useRestrictMobilePlayback.tsx +0 -101
  110. package/Components/River/TV/utils/__tests__/toStringOrEmpty.test.ts +0 -30
  111. package/Components/River/TV/utils/index.ts +0 -4
  112. package/Components/River/TV/withFocusableGroupForContent.tsx +0 -71
  113. package/Helpers/DataSourceHelper/__tests__/itemLimitForData.test.ts +0 -80
  114. package/Helpers/DataSourceHelper/index.ts +0 -19
  115. /package/Components/HookRenderer/{index.tsx → index.ts} +0 -0
@@ -23,6 +23,7 @@ import { isLast } from "@applicaster/zapp-react-native-utils/arrayUtils";
23
23
  import { withComponentsMapProvider } from "@applicaster/zapp-react-native-ui-components/Decorators/ComponentsMapWrapper";
24
24
  import { useScreenContextV2 } from "@applicaster/zapp-react-native-utils/reactHooks/screen/useScreenContext";
25
25
  import { useShallow } from "zustand/react/shallow";
26
+ import { emitScrollEndReached } from "@applicaster/zapp-react-native-ui-components/events";
26
27
 
27
28
  import { isAndroidPlatform } from "@applicaster/zapp-react-native-utils/reactUtils";
28
29
  import { ComponentsMapHeightContext } from "./ContextProviders/ComponentsMapHeightContext";
@@ -73,6 +74,7 @@ function ComponentsMapComponent(props: Props) {
73
74
 
74
75
  const flatListRef = React.useRef<FlatList | null>(null);
75
76
  const flatListWrapperRef = React.useRef<View | null>(null);
77
+ const hasUserScrolledRef = React.useRef(false);
76
78
  const screenConfig = useScreenConfiguration(riverId);
77
79
  const screenData = useScreenData(riverId);
78
80
  const pullToRefreshEnabled = screenData?.rules?.pull_to_refresh_enabled;
@@ -236,6 +238,8 @@ function ComponentsMapComponent(props: Props) {
236
238
  }, []);
237
239
 
238
240
  const onScroll = React.useCallback((event) => {
241
+ hasUserScrolledRef.current = true;
242
+
239
243
  const {
240
244
  nativeEvent: {
241
245
  contentOffset: { y },
@@ -308,6 +312,17 @@ function ComponentsMapComponent(props: Props) {
308
312
  onScrollEndDrag={_onScrollEndDrag}
309
313
  scrollEventThrottle={16}
310
314
  {...scrollViewExtraProps}
315
+ /* When wrapped in a parent ScrollView (e.g. tabs),
316
+ this FlatList doesn't scroll so onEndReached can fire repeatedly;
317
+ skip it here and let the parent ScrollView emit scroll-end instead. */
318
+ onEndReached={
319
+ isScreenWrappedInContainer
320
+ ? undefined
321
+ : () => {
322
+ if (!hasUserScrolledRef.current) return;
323
+ emitScrollEndReached();
324
+ }
325
+ }
311
326
  />
312
327
  </ViewportTracker>
313
328
  </ScreenLoadingMeasurements>
@@ -1,4 +1,4 @@
1
- import { renderHook, act } from "@testing-library/react-native";
1
+ import { renderHook, act } from "@testing-library/react-hooks";
2
2
  import { BehaviorSubject } from "rxjs";
3
3
  import { useLoadingState } from "../useLoadingState";
4
4
 
@@ -5,6 +5,8 @@ import {
5
5
  StyleSheet,
6
6
  } from "react-native";
7
7
  import * as R from "ramda";
8
+ import { path } from "@applicaster/zapp-react-native-utils/utils";
9
+ import { isNilOrEmpty } from "@applicaster/zapp-react-native-utils/reactUtils/helpers";
8
10
  import { useTheme } from "@applicaster/zapp-react-native-utils/theme";
9
11
  import { useLocalizedStrings } from "@applicaster/zapp-react-native-utils/localizationUtils";
10
12
  import { useAnalytics } from "@applicaster/zapp-react-native-utils/analyticsUtils";
@@ -62,9 +64,13 @@ export const usePullToRefresh = (
62
64
 
63
65
  const [refreshing, setRefreshing] = React.useState(false);
64
66
 
65
- const feeds: string[] =
66
- riverComponents?.map(R.path(["data", "source"])).filter((feed) => !!feed) ??
67
- [];
67
+ const feeds: string[] = React.useMemo(
68
+ () =>
69
+ (riverComponents || [])
70
+ .map((riverComponent) => path(["data", "source"], riverComponent))
71
+ .filter((feed) => !isNilOrEmpty(feed)),
72
+ [riverComponents]
73
+ );
68
74
 
69
75
  const feedsLength = feeds.length;
70
76
 
@@ -14,6 +14,7 @@ import { tvPluginsWithCellRenderer } from "../../const";
14
14
  import { isTV } from "@applicaster/zapp-react-native-utils/reactUtils";
15
15
  import type { BehaviorSubject } from "rxjs";
16
16
  import { useCallbackActions } from "@applicaster/zapp-react-native-utils/zappFrameworkUtils/HookCallback/useCallbackActions";
17
+ import { isNilOrEmpty } from "@applicaster/zapp-react-native-utils/reactUtils/helpers";
17
18
 
18
19
  export type RiverItemType = {
19
20
  item: ZappUIComponent;
@@ -112,33 +113,38 @@ function RiverItemComponent(props: RiverItemType) {
112
113
  CellRenderer = undefined;
113
114
  }
114
115
 
115
- React.useEffect(() => {
116
- riverLogger.log({
117
- message: "mounting component",
118
- data: { item, feedUrl, Component, CellRenderer },
119
- jsOnly: true,
120
- });
116
+ const isComponentMissing = isNilOrEmpty(Component);
121
117
 
122
- if (!CellRenderer && !isGroup(item)) {
118
+ /**
119
+ * TODO: Move this plugin existence check further up the stack (before ComponentsMap).
120
+ * Filtering items at the list-rendering or data-processing level would prevent
121
+ * mounting RiverItem entirely for missing components.
122
+ */
123
+ React.useEffect(() => {
124
+ if (isComponentMissing) {
123
125
  riverLogger.warning({
124
- message: "Cell Renderer is null - will fallback to default cell",
125
- data: { item, CellRenderer },
126
+ message: `Component ${item.component_type} is null - skipping rendering`,
127
+ });
128
+
129
+ onLoadFinished(index);
130
+ } else {
131
+ riverLogger.log({
132
+ message: "mounting component",
133
+ data: { item, feedUrl, Component, CellRenderer },
126
134
  jsOnly: true,
127
135
  });
136
+
137
+ if (!CellRenderer && !isGroup(item)) {
138
+ riverLogger.warning({
139
+ message: "Cell Renderer is null - will fallback to default cell",
140
+ data: { item, CellRenderer },
141
+ jsOnly: true,
142
+ });
143
+ }
128
144
  }
129
145
  }, []);
130
146
 
131
- if (!readyToBeDisplayed) {
132
- return null;
133
- }
134
-
135
- if (Component === null || typeof Component === "undefined") {
136
- riverLogger.warning({
137
- message: `Component ${item.component_type} is null - skipping rendering`,
138
- });
139
-
140
- onLoadFinished(index);
141
-
147
+ if (!readyToBeDisplayed || isComponentMissing) {
142
148
  return null;
143
149
  }
144
150
 
@@ -2,8 +2,7 @@
2
2
 
3
3
  import * as React from "react";
4
4
  import { Text } from "react-native";
5
-
6
- import { mergeRight } from "@applicaster/zapp-react-native-utils/utils";
5
+ import * as R from "ramda";
7
6
 
8
7
  import { GeneralContentScreen } from "../../GeneralContentScreen";
9
8
  import { ScreenResolver } from "@applicaster/zapp-react-native-ui-components/Components/ScreenResolver";
@@ -14,8 +13,6 @@ import {
14
13
  } from "@applicaster/zapp-react-native-utils/reactHooks/screen/useScreenContext";
15
14
  import { useRivers } from "@applicaster/zapp-react-native-utils/reactHooks/state";
16
15
 
17
- import { toStringOrEmpty } from "./utils";
18
-
19
16
  type Props = {
20
17
  screenId: string;
21
18
  screenData: ZappRiver | ZappEntry;
@@ -27,7 +24,6 @@ type Props = {
27
24
  isInsideContainer?: boolean;
28
25
  extraAnchorPointYOffset: number;
29
26
  river?: ZappRiver | ZappEntry;
30
- groupId: string;
31
27
  };
32
28
 
33
29
  export const River = (props: Props) => {
@@ -39,7 +35,6 @@ export const River = (props: Props) => {
39
35
  componentsMapExtraProps,
40
36
  isInsideContainer,
41
37
  extraAnchorPointYOffset,
42
- groupId,
43
38
  } = props;
44
39
 
45
40
  const { title: screenTitle, summary: screenSummary } = useNavbarState();
@@ -56,41 +51,28 @@ export const River = (props: Props) => {
56
51
  [screenId]
57
52
  );
58
53
 
59
- const screenResolverData = React.useMemo(() => {
60
- const extraData = mergeRight(extraProps, screenResolverExtraProps);
61
-
62
- return {
63
- extraData,
64
- screenData: mergeRight(river, { groupId: extraData?.groupId }),
65
- componentsMapExtraProps: mergeRight(componentsMapExtraProps, { groupId }),
66
- };
67
- }, [
68
- extraProps,
69
- screenResolverExtraProps,
70
- river,
71
- componentsMapExtraProps,
72
- groupId,
73
- ]);
54
+ const stringOrEmpty = (value: string | number | undefined): string =>
55
+ R.isNil(value) ? "" : String(value);
74
56
 
75
57
  React.useEffect(() => {
76
58
  if (!isInsideContainer) {
77
- setScreenTitle(toStringOrEmpty(screenData?.title));
78
- setScreenSummary(toStringOrEmpty(screenData?.summary));
59
+ setScreenTitle(stringOrEmpty(screenData?.title));
60
+ setScreenSummary(stringOrEmpty(screenData?.summary));
79
61
  }
80
62
  }, [screenData.id]);
81
63
 
82
64
  React.useEffect(() => {
83
65
  if (feedData && !isInsideContainer) {
84
66
  if (feedData.title && feedData.title !== screenTitle) {
85
- setScreenTitle(toStringOrEmpty(feedData.title));
67
+ setScreenTitle(stringOrEmpty(feedData.title));
86
68
  }
87
69
 
88
70
  if (feedData.summary && feedData.summary !== screenSummary) {
89
- setScreenSummary(toStringOrEmpty(feedData.summary));
71
+ setScreenSummary(stringOrEmpty(feedData.summary));
90
72
  }
91
73
  } else {
92
- setScreenTitle(toStringOrEmpty(screenData?.title));
93
- setScreenSummary(toStringOrEmpty(screenData?.summary));
74
+ setScreenTitle(stringOrEmpty(screenData?.title));
75
+ setScreenSummary(stringOrEmpty(screenData?.summary));
94
76
  }
95
77
  }, [feedData, screenData, screenTitle, screenSummary]);
96
78
 
@@ -104,13 +86,15 @@ export const River = (props: Props) => {
104
86
  }
105
87
 
106
88
  if (river.type !== "general_content") {
89
+ const extraData = { ...R.mergeRight(extraProps, screenResolverExtraProps) };
90
+
107
91
  return (
108
92
  <ScreenResolver
109
93
  screenType={river.type}
110
94
  screenId={screenId}
111
- screenData={screenResolverData.screenData}
112
- componentsMapExtraProps={screenResolverData.componentsMapExtraProps}
113
- {...screenResolverData.extraData}
95
+ screenData={R.merge(river, { groupId: extraData?.groupId })}
96
+ componentsMapExtraProps={componentsMapExtraProps}
97
+ {...extraData}
114
98
  />
115
99
  );
116
100
  }
@@ -122,7 +106,6 @@ export const River = (props: Props) => {
122
106
  isScreenWrappedInContainer={isInsideContainer}
123
107
  extraAnchorPointYOffset={extraAnchorPointYOffset}
124
108
  componentsMapExtraProps={componentsMapExtraProps}
125
- groupId={groupId}
126
109
  />
127
110
  );
128
111
  };
@@ -1,15 +1,11 @@
1
- import { compose, identity } from "@applicaster/zapp-react-native-utils/utils";
2
- import { isTvOSPlatform } from "@applicaster/zapp-react-native-utils/reactUtils";
3
-
1
+ import { compose } from "ramda";
4
2
  import { River as RiverComponent } from "./River";
3
+ import { withTvEventHandler } from "./withTVEventHandler";
5
4
  import { withComponentsMapOffsetContext } from "../../../Contexts/ComponentsMapOffsetContext";
6
5
  import { withRiverDataLoader } from "./withRiverDataLoader";
7
- import { withFocusableGroupForContent } from "./withFocusableGroupForContent";
8
-
9
- const isTVOS = isTvOSPlatform();
10
6
 
11
7
  export const River = compose(
8
+ withTvEventHandler,
12
9
  withComponentsMapOffsetContext,
13
- withRiverDataLoader,
14
- isTVOS ? withFocusableGroupForContent : identity
10
+ withRiverDataLoader
15
11
  )(RiverComponent);
@@ -0,0 +1,36 @@
1
+ /* eslint max-len: off */
2
+
3
+ import React from "react";
4
+ import { TVEventHandlerComponent } from "@applicaster/zapp-react-native-tvos-ui-components/Components/TVEventHandlerComponent";
5
+ import { useNavigation } from "@applicaster/zapp-react-native-utils/reactHooks";
6
+ import { useIsStandaloneFullscreen } from "@applicaster/zapp-react-native-utils/reactHooks/screen";
7
+
8
+ export const withTvEventHandler = (Component) => {
9
+ return function WithTVEventHandler(props) {
10
+ const navigator = useNavigation();
11
+
12
+ const isStandaloneFullscreen = useIsStandaloneFullscreen();
13
+
14
+ const remoteHandler = (event) => {
15
+ const { eventType } = event;
16
+
17
+ const canGoBack = navigator.canGoBack();
18
+
19
+ if (eventType === "menu" && isStandaloneFullscreen) {
20
+ navigator.goHome();
21
+
22
+ return;
23
+ }
24
+
25
+ if (eventType === "menu" && canGoBack) {
26
+ navigator.goBack();
27
+ }
28
+ };
29
+
30
+ return (
31
+ <TVEventHandlerComponent tvEventHandler={remoteHandler}>
32
+ <Component {...props} />
33
+ </TVEventHandlerComponent>
34
+ );
35
+ };
36
+ };
@@ -137,6 +137,7 @@ exports[`componentsMap renders renders components map correctly 1`] = `
137
137
  keyExtractor={[Function]}
138
138
  maxToRenderPerBatch={10}
139
139
  onContentSizeChange={[Function]}
140
+ onEndReached={[Function]}
140
141
  onLayout={[Function]}
141
142
  onMomentumScrollBegin={[Function]}
142
143
  onMomentumScrollEnd={[Function]}
@@ -9,10 +9,9 @@ import {
9
9
  import {
10
10
  useIsScreenActive,
11
11
  useNavigation,
12
- useRivers,
13
12
  } from "@applicaster/zapp-react-native-utils/reactHooks";
14
13
  import { noop } from "@applicaster/zapp-react-native-utils/functionUtils";
15
- import { usePlugins } from "@applicaster/zapp-react-native-redux/hooks";
14
+ import { usePickFromState } from "@applicaster/zapp-react-native-redux/hooks";
16
15
  import {
17
16
  useNavbarState,
18
17
  useScreenBackgroundColor,
@@ -103,8 +102,7 @@ export const Screen = ({ route, Components }: Props) => {
103
102
  }
104
103
 
105
104
  const navigator = useNavigation();
106
- const plugins = usePlugins();
107
- const rivers = useRivers();
105
+ const { plugins = [], rivers = [] } = usePickFromState(["plugins", "rivers"]);
108
106
 
109
107
  const pathAttributes = getPathAttributes({ pathname: route });
110
108
  const routeState = navigator.getStackForPathname(route);
@@ -1,16 +1,20 @@
1
1
  import * as React from "react";
2
2
  import { View } from "react-native";
3
- import { renderWithProviders } from "@applicaster/zapp-react-native-utils/testUtils";
3
+ import { render } from "@testing-library/react-native";
4
4
 
5
5
  const Mocked_RouteManager = jest.fn((props) => (
6
6
  <View testID="routeManager" {...props} />
7
7
  ));
8
8
 
9
+ const mock_navBarVisibleFlag = true;
10
+
11
+ const mockIsNavBarVisible = jest.fn(() => mock_navBarVisibleFlag);
12
+
9
13
  const mockIsOrientationCompatible = jest.fn(() => true);
10
14
 
11
15
  jest.mock("react-native-safe-area-context", () => ({
12
- ...jest.requireActual("react-native-safe-area-context"),
13
16
  useSafeAreaInsets: () => ({ top: 44 }),
17
+ useSafeAreaFrame: () => ({ x: 0, y: 0, width: 375, height: 812 }),
14
18
  }));
15
19
 
16
20
  jest.mock("../../RouteManager", () => ({
@@ -32,14 +36,12 @@ jest.mock(
32
36
  );
33
37
 
34
38
  jest.mock("@applicaster/zapp-react-native-utils/analyticsUtils", () => ({
35
- ...jest.requireActual("@applicaster/zapp-react-native-utils/analyticsUtils"),
36
39
  useAnalytics: jest.fn(() => ({
37
40
  sendScreenEvent: jest.fn(),
38
41
  })),
39
42
  }));
40
43
 
41
44
  jest.mock("@applicaster/zapp-react-native-utils/theme", () => ({
42
- ...jest.requireActual("@applicaster/zapp-react-native-utils/theme"),
43
45
  useTheme: jest.fn(() => ({
44
46
  app_background_color: "blue",
45
47
  })),
@@ -76,44 +78,21 @@ jest.mock(
76
78
  })
77
79
  );
78
80
 
79
- jest.mock(
80
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useNavigation",
81
- () => ({
82
- ...jest.requireActual(
83
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useNavigation"
84
- ),
85
- useNavigation: jest.fn(() => ({
86
- canGoBack: () => false,
87
- currentRoute: "/river/testId",
88
- activeRiver: { id: "testId" },
89
- screenData: { id: "testId" },
90
- data: { screen: { id: "testId" } },
91
- })),
92
- })
93
- );
94
-
95
- jest.mock(
96
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useIsScreenActive",
97
- () => ({
98
- ...jest.requireActual(
99
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useIsScreenActive"
100
- ),
101
- useIsScreenActive: jest.fn().mockReturnValue(true),
102
- })
103
- );
104
-
105
- jest.mock(
106
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useRoute",
107
- () => ({
108
- ...jest.requireActual(
109
- "@applicaster/zapp-react-native-utils/reactHooks/navigation/useRoute"
110
- ),
111
- useRoute: jest.fn(() => ({
112
- pathname: "/river/testId",
113
- screenData: { id: "testId" },
114
- })),
115
- })
116
- );
81
+ jest.mock("@applicaster/zapp-react-native-utils/reactHooks/navigation", () => ({
82
+ isNavBarVisible: mockIsNavBarVisible,
83
+ useIsScreenActive: jest.fn().mockReturnValue(true),
84
+ useNavigation: jest.fn(() => ({
85
+ canGoBack: () => false,
86
+ currentRoute: "/river/testId",
87
+ activeRiver: { id: "testId" },
88
+ screenData: { id: "testId" },
89
+ data: { screen: { id: "testId" } },
90
+ })),
91
+ useRoute: jest.fn(() => ({
92
+ pathname: "/river/testId",
93
+ screenData: { id: "testId" },
94
+ })),
95
+ }));
117
96
 
118
97
  jest.mock("@applicaster/zapp-react-native-utils/reactHooks", () => ({
119
98
  ...jest.requireActual("@applicaster/zapp-react-native-utils/reactHooks"),
@@ -134,6 +113,26 @@ jest.mock("@applicaster/zapp-react-native-utils/reactHooks", () => ({
134
113
  useIsTablet: jest.fn(() => false),
135
114
  }));
136
115
 
116
+ jest.mock("@applicaster/zapp-react-native-redux/hooks", () => {
117
+ const View = jest.requireActual("react-native").View;
118
+
119
+ return {
120
+ ...jest.requireActual("@applicaster/zapp-react-native-redux/hooks"),
121
+ usePickFromState: () => ({
122
+ plugins: [
123
+ {
124
+ name: "Offline Plugin",
125
+ identifier: "offline-experience",
126
+ type: "general",
127
+ module: {
128
+ OfflineFallbackScreen: ({ children }) => <View>{children}</View>, // eslint-disable-line
129
+ },
130
+ },
131
+ ],
132
+ }),
133
+ };
134
+ });
135
+
137
136
  const {
138
137
  allowedOrientationsForScreen,
139
138
  getOrientation,
@@ -153,19 +152,6 @@ const screenProps = {
153
152
 
154
153
  const { Screen } = require("..");
155
154
 
156
- const store = {
157
- plugins: [
158
- {
159
- name: "Offline Plugin",
160
- identifier: "offline-experience",
161
- type: "general",
162
- module: {
163
- OfflineFallbackScreen: ({ children }) => <View>{children}</View>, // eslint-disable-line
164
- },
165
- },
166
- ],
167
- };
168
-
169
155
  describe("<Screen Component />", () => {
170
156
  beforeEach(() => {
171
157
  allowedOrientationsForScreen.mockClear();
@@ -175,22 +161,14 @@ describe("<Screen Component />", () => {
175
161
 
176
162
  describe("when the navbar should show", () => {
177
163
  it("renders correctly", () => {
178
- const { toJSON } = renderWithProviders(
179
- <Screen {...screenProps} />,
180
- store
181
- );
182
-
164
+ const { toJSON } = render(<Screen {...screenProps} />);
183
165
  expect(toJSON()).toMatchSnapshot();
184
166
  });
185
167
  });
186
168
 
187
169
  describe("when the navbar should be hidden", () => {
188
170
  it("renders correctly", () => {
189
- const { toJSON } = renderWithProviders(
190
- <Screen {...screenProps} />,
191
- store
192
- );
193
-
171
+ const { toJSON } = render(<Screen {...screenProps} />);
194
172
  expect(toJSON()).toMatchSnapshot();
195
173
  });
196
174
  });
@@ -1,91 +1,67 @@
1
1
  // Jest Snapshot v1, https://goo.gl/fbAQLP
2
2
 
3
3
  exports[`<Screen Component /> when the navbar should be hidden renders correctly 1`] = `
4
- <RNCSafeAreaProvider
5
- onInsetsChange={[Function]}
4
+ <View
5
+ importantForAccessibility="yes"
6
6
  style={
7
- [
8
- {
9
- "flex": 1,
10
- },
11
- undefined,
12
- ]
7
+ {
8
+ "backgroundColor": "blue",
9
+ "flex": 1,
10
+ "paddingTop": 0,
11
+ }
13
12
  }
14
13
  >
15
14
  <View
16
- importantForAccessibility="yes"
17
- style={
18
- {
19
- "backgroundColor": "blue",
20
- "flex": 1,
21
- "paddingTop": 0,
22
- }
23
- }
24
- >
15
+ hasMenu={false}
16
+ id="/river/testId"
17
+ pathname="/river/testId"
18
+ selected="testId"
19
+ testID="navBar"
20
+ title="Test Title"
21
+ />
22
+ <View>
25
23
  <View
26
- hasMenu={false}
27
- id="/river/testId"
28
24
  pathname="/river/testId"
29
- selected="testId"
30
- testID="navBar"
31
- title="Test Title"
32
- />
33
- <View>
34
- <View
35
- pathname="/river/testId"
36
- screenData={
37
- {
38
- "id": "testId",
39
- }
25
+ screenData={
26
+ {
27
+ "id": "testId",
40
28
  }
41
- testID="routeManager"
42
- />
43
- </View>
29
+ }
30
+ testID="routeManager"
31
+ />
44
32
  </View>
45
- </RNCSafeAreaProvider>
33
+ </View>
46
34
  `;
47
35
 
48
36
  exports[`<Screen Component /> when the navbar should show renders correctly 1`] = `
49
- <RNCSafeAreaProvider
50
- onInsetsChange={[Function]}
37
+ <View
38
+ importantForAccessibility="yes"
51
39
  style={
52
- [
53
- {
54
- "flex": 1,
55
- },
56
- undefined,
57
- ]
40
+ {
41
+ "backgroundColor": "blue",
42
+ "flex": 1,
43
+ "paddingTop": 0,
44
+ }
58
45
  }
59
46
  >
60
47
  <View
61
- importantForAccessibility="yes"
62
- style={
63
- {
64
- "backgroundColor": "blue",
65
- "flex": 1,
66
- "paddingTop": 0,
67
- }
68
- }
69
- >
48
+ hasMenu={false}
49
+ id="/river/testId"
50
+ pathname="/river/testId"
51
+ selected="testId"
52
+ testID="navBar"
53
+ title="Test Title"
54
+ />
55
+ <View>
70
56
  <View
71
- hasMenu={false}
72
- id="/river/testId"
73
57
  pathname="/river/testId"
74
- selected="testId"
75
- testID="navBar"
76
- title="Test Title"
77
- />
78
- <View>
79
- <View
80
- pathname="/river/testId"
81
- screenData={
82
- {
83
- "id": "testId",
84
- }
58
+ screenData={
59
+ {
60
+ "id": "testId",
85
61
  }
86
- testID="routeManager"
87
- />
88
- </View>
62
+ }
63
+ testID="routeManager"
64
+ />
89
65
  </View>
90
- </RNCSafeAreaProvider>
66
+ </View>
91
67
  `;