@applicaster/zapp-react-native-ui-components 15.0.0-alpha.7591121530 → 15.0.0-alpha.7607942912

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 (54) hide show
  1. package/Components/BaseFocusable/index.ios.ts +12 -2
  2. package/Components/Cell/Cell.tsx +8 -3
  3. package/Components/Cell/FocusableWrapper.tsx +3 -0
  4. package/Components/Cell/TvOSCellComponent.tsx +26 -5
  5. package/Components/Focusable/FocusableTvOS.tsx +12 -2
  6. package/Components/Focusable/__tests__/__snapshots__/FocusableTvOS.test.tsx.snap +1 -0
  7. package/Components/FocusableGroup/FocusableTvOS.tsx +11 -0
  8. package/Components/HandlePlayable/HandlePlayable.tsx +10 -7
  9. package/Components/Layout/TV/LayoutBackground.tsx +5 -2
  10. package/Components/Layout/TV/ScreenContainer.tsx +2 -6
  11. package/Components/Layout/TV/index.tsx +3 -4
  12. package/Components/Layout/TV/index.web.tsx +3 -4
  13. package/Components/LinkHandler/LinkHandler.tsx +2 -2
  14. package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx +16 -1
  15. package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +30 -2
  16. package/Components/MasterCell/DefaultComponents/Image/Image.android.tsx +5 -1
  17. package/Components/MasterCell/DefaultComponents/Image/Image.ios.tsx +11 -3
  18. package/Components/MasterCell/DefaultComponents/Image/Image.web.tsx +9 -1
  19. package/Components/MasterCell/DefaultComponents/Image/hooks/useImage.ts +15 -14
  20. package/Components/MasterCell/DefaultComponents/LiveImage/index.tsx +10 -6
  21. package/Components/MasterCell/DefaultComponents/Text/index.tsx +8 -8
  22. package/Components/MasterCell/index.tsx +2 -0
  23. package/Components/MasterCell/utils/__tests__/resolveColor.test.js +82 -3
  24. package/Components/MasterCell/utils/index.ts +61 -31
  25. package/Components/MeasurmentsPortal/MeasurementsPortal.tsx +102 -87
  26. package/Components/MeasurmentsPortal/__tests__/MeasurementsPortal.test.tsx +355 -0
  27. package/Components/OfflineHandler/NotificationView/NotificationView.tsx +2 -2
  28. package/Components/OfflineHandler/NotificationView/__tests__/index.test.tsx +17 -18
  29. package/Components/OfflineHandler/__tests__/index.test.tsx +27 -18
  30. package/Components/PlayerContainer/PlayerContainer.tsx +4 -3
  31. package/Components/Screen/TV/index.web.tsx +4 -2
  32. package/Components/Screen/__tests__/Screen.test.tsx +65 -42
  33. package/Components/Screen/__tests__/__snapshots__/Screen.test.tsx.snap +68 -44
  34. package/Components/Screen/hooks.ts +2 -3
  35. package/Components/Screen/index.tsx +2 -3
  36. package/Components/Screen/navigationHandler.ts +49 -24
  37. package/Components/Screen/orientationHandler.ts +3 -3
  38. package/Components/ScreenResolver/index.tsx +13 -7
  39. package/Components/ScreenRevealManager/ScreenRevealManager.ts +40 -8
  40. package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +86 -69
  41. package/Components/Tabs/TV/Tabs.tsx +20 -3
  42. package/Components/Transitioner/Scene.tsx +15 -2
  43. package/Components/Transitioner/index.js +3 -3
  44. package/Components/VideoModal/ModalAnimation/ModalAnimationContext.tsx +124 -27
  45. package/Components/VideoModal/VideoModal.tsx +1 -5
  46. package/Components/VideoModal/utils.ts +19 -9
  47. package/Decorators/Analytics/index.tsx +6 -5
  48. package/Decorators/ZappPipesDataConnector/index.tsx +2 -2
  49. package/Decorators/ZappPipesDataConnector/resolvers/StaticFeedResolver.tsx +1 -1
  50. package/Helpers/DataSourceHelper/__tests__/itemLimitForData.test.ts +80 -0
  51. package/Helpers/DataSourceHelper/index.ts +19 -0
  52. package/index.d.ts +7 -0
  53. package/package.json +6 -5
  54. package/Helpers/DataSourceHelper/index.js +0 -19
@@ -9,6 +9,7 @@ import { useTrackCurrentAutoScrollingElement } from "@applicaster/zapp-react-nat
9
9
  import { useUIComponentContext } from "@applicaster/zapp-react-native-ui-components/Contexts/UIComponentContext";
10
10
  import { getPropComponentType } from "@applicaster/zapp-react-native-utils/cellUtils";
11
11
  import { findPluginByIdentifier } from "@applicaster/zapp-react-native-utils/pluginUtils";
12
+ import { useAccessibilityState } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
12
13
 
13
14
  type LiveImageProps = {
14
15
  item: ZappEntry;
@@ -108,8 +109,7 @@ const prepareEntry = (entry) => {
108
109
  };
109
110
  }
110
111
 
111
- const previewPlayback =
112
- entry.extensions?.["brightcove"]?.["preview_playback"];
112
+ const previewPlayback = entry.extensions?.brightcove?.preview_playback;
113
113
 
114
114
  if (previewPlayback) {
115
115
  return {
@@ -117,14 +117,14 @@ const prepareEntry = (entry) => {
117
117
  extensions: {
118
118
  ...entry.extensions,
119
119
  brightcove: {
120
- ...entry?.extensions?.["brightcove"],
120
+ ...entry?.extensions?.brightcove,
121
121
  video_id: previewPlayback,
122
122
  },
123
123
  },
124
124
  };
125
125
  }
126
126
 
127
- if (entry.extensions?.["brightcove"]?.["video_id"]) {
127
+ if (entry.extensions?.brightcove?.video_id) {
128
128
  return entry;
129
129
  }
130
130
 
@@ -174,7 +174,7 @@ const getPlayerConfig = (player_screen_id, actionIdentifier) => {
174
174
  // TODO: Add more dict if needed from the screen component, styles, data etc
175
175
  return {
176
176
  playerPluginId: playerScreen?.type ?? DEFAULT_PLAYER_IDENTIFIER,
177
- screenConfig: playerScreen?.["general"],
177
+ screenConfig: playerScreen?.general,
178
178
  };
179
179
  }
180
180
 
@@ -206,6 +206,9 @@ const LiveImageComponent = (props: LiveImageProps) => {
206
206
  state,
207
207
  } = props;
208
208
 
209
+ const accessibilityState = useAccessibilityState({});
210
+ const isScreenReaderEnabled = accessibilityState.screenReaderEnabled;
211
+
209
212
  const component = useUIComponentContext();
210
213
 
211
214
  // Fix for blinking on state change
@@ -239,7 +242,8 @@ const LiveImageComponent = (props: LiveImageProps) => {
239
242
  getFocusedState(state, componentType, isCurrentlyFocused) &&
240
243
  playableEntry &&
241
244
  cellUUID &&
242
- isSupportedTVForLiveImage();
245
+ isSupportedTVForLiveImage() &&
246
+ !isScreenReaderEnabled;
243
247
 
244
248
  return (
245
249
  <>
@@ -52,14 +52,14 @@ const _Text = ({
52
52
  : textTransform(transformText, _label);
53
53
 
54
54
  React.useLayoutEffect(() => {
55
- // For FocusableCells with action buttons
56
- if (otherProps.state) {
57
- if (otherProps.state === "focused" && cellFocused === true) {
58
- accessibilityManager.addHeading(textLabel);
59
- }
60
- } else {
61
- if (cellFocused === true) {
62
- accessibilityManager.addHeading(textLabel);
55
+ if (cellFocused) {
56
+ switch (otherProps.state) {
57
+ case "focused":
58
+ accessibilityManager.addHeading(textLabel);
59
+ break;
60
+ case "focused_selected":
61
+ accessibilityManager.addHeading(`${textLabel}, Selected`);
62
+ break;
63
63
  }
64
64
  }
65
65
  }, [cellFocused, otherProps.state, textLabel]);
@@ -103,6 +103,8 @@ export function masterCellBuilder({
103
103
  wrapperRef,
104
104
  cellUUID,
105
105
  skipButtons,
106
+ hasFocusableInside,
107
+ entry: item,
106
108
  })
107
109
  );
108
110
 
@@ -32,14 +32,16 @@ describe("resolveColor", () => {
32
32
  color: "invalid_path",
33
33
  };
34
34
 
35
- expect(resolveColor(entry, style)).toEqual(style);
35
+ expect(resolveColor(entry, style)).toEqual({
36
+ color: undefined,
37
+ });
36
38
 
37
39
  expect(loggerSpy).toHaveBeenCalledWith(
38
40
  expect.objectContaining({
39
41
  message: "Cannot resolve property invalid_path from the entry.",
40
42
  data: {
43
+ colorFromProp: "invalid_path",
41
44
  configurationValue: "invalid_path",
42
- entry,
43
45
  },
44
46
  })
45
47
  );
@@ -102,7 +104,9 @@ describe("resolveColor", () => {
102
104
  color: "not.exist.path",
103
105
  };
104
106
 
105
- expect(resolveColor(entry, style)).toEqual(style);
107
+ expect(resolveColor(entry, style)).toEqual({
108
+ color: undefined,
109
+ });
106
110
  });
107
111
 
108
112
  it("not modify style with empty path", () => {
@@ -112,4 +116,79 @@ describe("resolveColor", () => {
112
116
 
113
117
  expect(resolveColor(entry, style)).toEqual(style);
114
118
  });
119
+
120
+ describe("memoization", () => {
121
+ beforeEach(() => {
122
+ // Clear memoization cache before each test
123
+ resolveColor.clear && resolveColor.clear();
124
+ });
125
+
126
+ it("hits cache with same entry and style references", () => {
127
+ const style = { color: "extensions.color" };
128
+
129
+ const result1 = resolveColor(entry, style);
130
+ const result2 = resolveColor(entry, style);
131
+
132
+ expect(result1).toBe(result2); // Same object reference
133
+ });
134
+
135
+ it("hits cache with new references but equal entry/style values", () => {
136
+ const entryClone = {
137
+ extensions: {
138
+ color: "red",
139
+ green_color: "green",
140
+ },
141
+ };
142
+
143
+ const style = { color: "extensions.color" };
144
+ const styleClone = { color: "extensions.color" };
145
+
146
+ const result1 = resolveColor(entry, style);
147
+ const result2 = resolveColor(entryClone, styleClone);
148
+
149
+ expect(result1).toBe(result2);
150
+ });
151
+
152
+ it("misses cache when entry is new object", () => {
153
+ const entry2 = { extensions: { color: "blue" } }; // Same values, different object
154
+ const style = { color: "extensions.color" };
155
+
156
+ const result1 = resolveColor(entry, style);
157
+ const result2 = resolveColor(entry2, style);
158
+
159
+ expect(result1).not.toBe(result2); // Different object references
160
+ });
161
+
162
+ it("misses cache when entry property changes", () => {
163
+ const myEntry = {
164
+ extensions: {
165
+ color: "red",
166
+ green_color: "green",
167
+ },
168
+ };
169
+
170
+ const style = { color: "extensions.color" };
171
+
172
+ const result1 = resolveColor(myEntry, style);
173
+
174
+ myEntry.extensions.color = "blue"; // Change property
175
+ const result2 = resolveColor(myEntry, style);
176
+
177
+ expect(result1).toEqual({ color: "red" });
178
+ expect(result2).toEqual({ color: "blue" });
179
+ expect(result1).not.toBe(result2);
180
+ });
181
+
182
+ it("misses cache when style changes", () => {
183
+ const style1 = { color: "extensions.color" };
184
+ const style2 = { backgroundColor: "extensions.color" };
185
+
186
+ const result1 = resolveColor(entry, style1);
187
+ const result2 = resolveColor(entry, style2);
188
+
189
+ expect(result1).toEqual({ color: "red" });
190
+ expect(result2).toEqual({ backgroundColor: "red" });
191
+ expect(result1).not.toBe(result2);
192
+ });
193
+ });
115
194
  });
@@ -7,9 +7,12 @@ import { useActions } from "@applicaster/zapp-react-native-utils/reactHooks/acti
7
7
  import { masterCellLogger } from "../logger";
8
8
  import { getCellState } from "../../Cell/utils";
9
9
  import { getColorFromData } from "@applicaster/zapp-react-native-utils/cellUtils";
10
+ import { get } from "@applicaster/zapp-react-native-utils/utils";
10
11
  import { isCellSelected, useBehaviorUpdate } from "./behaviorProvider";
11
12
  import { useRoute } from "@applicaster/zapp-react-native-utils/reactHooks";
12
13
  import { useScreenStateStore } from "@applicaster/zapp-react-native-utils/reactHooks/navigation/useScreenStateStore";
14
+ import memoizee from "memoizee";
15
+ import stringify from "fast-json-stable-stringify";
13
16
 
14
17
  const hasElementSpecificViewType = (viewType) => (element) => {
15
18
  if (R.isNil(element)) {
@@ -24,21 +27,21 @@ const hasElementSpecificViewType = (viewType) => (element) => {
24
27
  return hasElementsSpecificViewType(viewType)(element.elements);
25
28
  };
26
29
 
27
- const logWarning = R.curry(
28
- (colorValueFromCellStyle, style, entry, colorValueFromEntry) => {
29
- if (R.isNil(colorValueFromEntry)) {
30
- masterCellLogger.warn({
31
- message: `Cannot resolve property ${colorValueFromCellStyle} from the entry.`,
32
- data: {
33
- configurationValue: colorValueFromCellStyle,
34
- entry,
35
- },
36
- });
37
- }
38
-
39
- return style;
30
+ const logWarning = (
31
+ colorValueFromCellStyle,
32
+ colorFromProp,
33
+ colorValueFromEntry
34
+ ) => {
35
+ if (R.isNil(colorValueFromEntry)) {
36
+ masterCellLogger.warn({
37
+ message: `Cannot resolve property ${colorValueFromCellStyle} from the entry.`,
38
+ data: {
39
+ configurationValue: colorValueFromCellStyle,
40
+ colorFromProp,
41
+ },
42
+ });
40
43
  }
41
- );
44
+ };
42
45
 
43
46
  export const hasElementsSpecificViewType = (viewType) => (elements) => {
44
47
  if (R.isNil(elements) || R.isEmpty(elements)) {
@@ -48,14 +51,22 @@ export const hasElementsSpecificViewType = (viewType) => (elements) => {
48
51
  return R.any(hasElementSpecificViewType(viewType))(elements);
49
52
  };
50
53
 
51
- function resolveColorForProp(entry, style, colorProp) {
52
- const colorFromProp = style[colorProp];
53
- const nestedEntryValue = R.path(colorFromProp.split("."), entry);
54
+ function resolveColorForProp(entry: any, colorFromProp: string | undefined) {
55
+ if (!colorFromProp) {
56
+ return undefined;
57
+ }
58
+
59
+ const nestedEntryValue: string | undefined = get(
60
+ entry,
61
+ colorFromProp.split(".")
62
+ );
54
63
 
55
64
  const color = colorFromProp.replace(".00", "").replace(".0", ""); // https://github.com/dreamyguy/validate-color/issues/44
56
65
 
57
66
  if (nestedEntryValue === undefined && !validateColor(color)) {
58
- logWarning(colorFromProp, style, entry, nestedEntryValue);
67
+ logWarning(colorFromProp, colorFromProp, nestedEntryValue);
68
+
69
+ return undefined;
59
70
  }
60
71
 
61
72
  const colorValue = getColorFromData({
@@ -64,27 +75,46 @@ function resolveColorForProp(entry, style, colorProp) {
64
75
  });
65
76
 
66
77
  if (!colorValue) {
67
- logWarning(colorProp, style, entry, colorValue);
78
+ logWarning(colorFromProp, colorFromProp, nestedEntryValue);
68
79
 
69
- return style;
80
+ return undefined;
70
81
  }
71
82
 
72
- return { ...style, [colorProp]: colorValue };
83
+ return colorValue;
73
84
  }
74
85
 
75
- export function resolveColor(entry, style) {
76
- if (style === null || style === undefined) {
77
- return style;
78
- }
79
-
80
- // TODO can be optimized to remove 3 O(n) loops
86
+ const getColorKeys = memoizee((style) => {
81
87
  const styleKeys = Object.keys(style);
82
88
  const colorKeys = styleKeys.filter((key) => /color/i.test(key));
83
89
 
84
- return colorKeys.reduce((acc, value) => {
85
- return { ...style, ...resolveColorForProp(entry, acc, value) };
86
- }, style);
87
- }
90
+ return colorKeys;
91
+ });
92
+
93
+ export const resolveColor = memoizee(
94
+ (entry, style) => {
95
+ if (style === null || style === undefined) {
96
+ return style;
97
+ }
98
+
99
+ return getColorKeys(style).reduce(
100
+ (acc, value) => {
101
+ if (acc[value] && typeof acc[value] === "string") {
102
+ const colorStyle = resolveColorForProp(entry, acc[value]);
103
+
104
+ acc[value] = colorStyle;
105
+ }
106
+
107
+ return acc;
108
+ },
109
+ { ...style }
110
+ );
111
+ },
112
+ {
113
+ normalizer: (args) => {
114
+ return [stringify(args[0]), stringify(args[1])].join("|");
115
+ },
116
+ }
117
+ );
88
118
 
89
119
  export function isVideoPreviewEnabled({
90
120
  enable_video_preview = false,
@@ -1,8 +1,12 @@
1
- /* eslint react/prop-types: off */
2
-
3
- import React from "react";
1
+ import React, {
2
+ memo,
3
+ PropsWithChildren,
4
+ useCallback,
5
+ useMemo,
6
+ useRef,
7
+ useState,
8
+ } from "react";
4
9
  import { View, StyleSheet } from "react-native";
5
- import * as R from "ramda";
6
10
  import { v4 } from "uuid";
7
11
 
8
12
  type Props = {
@@ -46,93 +50,104 @@ type MeasurementPortalContextType = {
46
50
  const MeasurementPortalContext =
47
51
  React.createContext<null | MeasurementPortalContextType>(null);
48
52
 
49
- const MeasurementsPortalContextProvider = ({ children }) => {
50
- const Component = React.useRef(View);
51
- const [measuringInProgress, setMeasuringInProgress] = React.useState(false);
52
-
53
- const measureComponentCallback = React.useRef(null);
54
- const componentProps = React.useRef(null);
55
- const onLoadFinishedIsCalled = React.useRef(null);
56
-
57
- const setComponent = (comp) => {
58
- Component.current = comp;
59
- };
60
-
61
- const setMeasureComponentCallback = (cb) => {
62
- measureComponentCallback.current = cb;
63
- };
64
-
65
- const finalize = ({ width, height }) => {
66
- measureComponentCallback?.current?.({
67
- width,
68
- height,
69
- index: componentProps.current.index,
70
- });
71
-
72
- setMeasureComponentCallback(null);
73
- setMeasuringInProgress(false);
74
- onLoadFinishedIsCalled.current = false;
75
- };
76
-
77
- const setComponentProps = (props) => {
78
- const handleOnLoadFinish = (...args) => {
79
- const error = R.path([0, "error"], args);
80
-
81
- if (isError(error)) {
82
- finalize({ width: 0, height: 0 });
83
- } else {
84
- onLoadFinishedIsCalled.current = true;
85
- props.onLoadFinished(...args);
86
- }
87
- };
88
-
89
- componentProps.current = {
90
- ...props,
91
- onLoadFinished: handleOnLoadFinish,
92
- };
93
- };
94
-
95
- const measureComponent = React.useCallback(async (comp, props) => {
96
- return new Promise((resolve) => {
97
- setMeasureComponentCallback(resolve);
98
- setMeasuringInProgress(true);
99
-
100
- setComponentProps(props);
101
- setComponent(comp);
102
- });
103
- }, []);
104
-
105
- const onLayout = ({
106
- nativeEvent: {
107
- layout: { width, height },
108
- },
109
- }) => {
110
- if (measureComponentCallback.current && onLoadFinishedIsCalled.current) {
111
- finalize({
53
+ const MeasurementsPortalContextProvider = memo(
54
+ ({ children }: PropsWithChildren) => {
55
+ const Component = useRef(View);
56
+ const [measuringInProgress, setMeasuringInProgress] = useState(false);
57
+
58
+ const measureComponentCallback = useRef(null);
59
+ const componentProps = useRef(null);
60
+ const onLoadFinishedIsCalled = useRef(null);
61
+
62
+ const setComponent = useCallback((comp) => {
63
+ Component.current = comp;
64
+ }, []);
65
+
66
+ const setMeasureComponentCallback = useCallback((cb) => {
67
+ measureComponentCallback.current = cb;
68
+ }, []);
69
+
70
+ const finalize = useCallback(({ width, height }) => {
71
+ measureComponentCallback?.current?.({
112
72
  width,
113
73
  height,
74
+ index: componentProps.current.index,
114
75
  });
115
- }
116
- };
117
76
 
118
- return (
119
- <>
120
- <MeasurementsPortal
121
- Component={Component.current}
122
- componentProps={componentProps.current}
123
- onLayout={onLayout}
124
- />
125
- <MeasurementPortalContext.Provider
126
- value={{
127
- measureComponent,
128
- measuringInProgress,
129
- }}
130
- >
131
- {children}
132
- </MeasurementPortalContext.Provider>
133
- </>
134
- );
135
- };
77
+ setMeasureComponentCallback(null);
78
+ setMeasuringInProgress(false);
79
+ onLoadFinishedIsCalled.current = false;
80
+ }, []);
81
+
82
+ const setComponentProps = useCallback((props) => {
83
+ const handleOnLoadFinish = (...args) => {
84
+ const error = args[0]?.error;
85
+
86
+ if (isError(error)) {
87
+ finalize({ width: 0, height: 0 });
88
+ } else {
89
+ onLoadFinishedIsCalled.current = true;
90
+ props.onLoadFinished(...args);
91
+ }
92
+ };
93
+
94
+ componentProps.current = {
95
+ ...props,
96
+ onLoadFinished: handleOnLoadFinish,
97
+ };
98
+ }, []);
99
+
100
+ const measureComponent = useCallback(async (comp, props) => {
101
+ return new Promise((resolve) => {
102
+ setMeasureComponentCallback(resolve);
103
+ setMeasuringInProgress(true);
104
+
105
+ setComponentProps(props);
106
+ setComponent(comp);
107
+ });
108
+ }, []);
109
+
110
+ const onLayout = useCallback(
111
+ ({
112
+ nativeEvent: {
113
+ layout: { width, height },
114
+ },
115
+ }) => {
116
+ if (
117
+ measureComponentCallback.current &&
118
+ onLoadFinishedIsCalled.current
119
+ ) {
120
+ finalize({
121
+ width,
122
+ height,
123
+ });
124
+ }
125
+ },
126
+ []
127
+ );
128
+
129
+ const contextValue = useMemo(
130
+ () => ({
131
+ measureComponent,
132
+ measuringInProgress,
133
+ }),
134
+ [measuringInProgress]
135
+ );
136
+
137
+ return (
138
+ <>
139
+ <MeasurementsPortal
140
+ Component={Component.current}
141
+ componentProps={componentProps.current}
142
+ onLayout={onLayout}
143
+ />
144
+ <MeasurementPortalContext.Provider value={contextValue}>
145
+ {children}
146
+ </MeasurementPortalContext.Provider>
147
+ </>
148
+ );
149
+ }
150
+ );
136
151
 
137
152
  export {
138
153
  MeasurementsPortal,