@applicaster/zapp-react-native-ui-components 15.0.0-rc.5 → 15.0.0-rc.50
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/Cell/Cell.tsx +8 -3
- package/Components/Cell/FocusableWrapper.tsx +44 -0
- package/Components/Cell/TvOSCellComponent.tsx +101 -19
- 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 +5 -19
- package/Components/PlayerImageBackground/index.tsx +3 -22
- 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/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/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
|
@@ -2,7 +2,7 @@ import * as React from "react";
|
|
|
2
2
|
import * as R from "ramda";
|
|
3
3
|
|
|
4
4
|
import { findPluginByIdentifier } from "@applicaster/zapp-react-native-utils/pluginUtils";
|
|
5
|
-
import {
|
|
5
|
+
import { usePlugins } from "@applicaster/zapp-react-native-redux/hooks";
|
|
6
6
|
import {
|
|
7
7
|
inflateString,
|
|
8
8
|
objectToReadableString,
|
|
@@ -40,7 +40,7 @@ export async function inflateUrl(url) {
|
|
|
40
40
|
|
|
41
41
|
export function LinkHandler(props: Props) {
|
|
42
42
|
const screenData = props?.screenData;
|
|
43
|
-
const
|
|
43
|
+
const plugins = usePlugins();
|
|
44
44
|
|
|
45
45
|
const ScreenPlugin = findPluginByIdentifier(
|
|
46
46
|
WEBVIEW_SCREEN_IDENTIFIER,
|
package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
+
import * as React from "react";
|
|
1
2
|
import {
|
|
2
3
|
BorderContainerView,
|
|
3
4
|
getBorderPadding, // Export for testing (using a double underscore prefix is a common convention)
|
|
4
5
|
} from "../index";
|
|
5
|
-
import * as React from "react";
|
|
6
6
|
import { render } from "@testing-library/react-native";
|
|
7
7
|
import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/numberUtils";
|
|
8
8
|
import { View } from "react-native";
|
|
@@ -11,6 +11,15 @@ jest.mock("@applicaster/zapp-react-native-utils/numberUtils", () => ({
|
|
|
11
11
|
toNumberWithDefaultZero: jest.fn((value) => Number(value) || 0),
|
|
12
12
|
}));
|
|
13
13
|
|
|
14
|
+
jest.mock(
|
|
15
|
+
"@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks",
|
|
16
|
+
() => ({
|
|
17
|
+
useAccessibilityManager: jest.fn(() => ({
|
|
18
|
+
addHeading: jest.fn(),
|
|
19
|
+
})),
|
|
20
|
+
})
|
|
21
|
+
);
|
|
22
|
+
|
|
14
23
|
describe("BorderContainerView", () => {
|
|
15
24
|
describe("getBorderPadding", () => {
|
|
16
25
|
it("returns 0 for inside", () => {
|
|
@@ -42,6 +51,8 @@ describe("BorderContainerView", () => {
|
|
|
42
51
|
};
|
|
43
52
|
|
|
44
53
|
const borderPosition = null;
|
|
54
|
+
const mockEntry = { id: "test-entry" } as ZappEntry;
|
|
55
|
+
const mockHasFocusableInside = jest.fn(() => false);
|
|
45
56
|
|
|
46
57
|
const { queryByTestId } = render(
|
|
47
58
|
<BorderContainerView
|
|
@@ -52,6 +63,10 @@ describe("BorderContainerView", () => {
|
|
|
52
63
|
borderPaddingRight={toNumberWithDefaultZero(padding.paddingRight)}
|
|
53
64
|
borderPaddingBottom={toNumberWithDefaultZero(padding.paddingBottom)}
|
|
54
65
|
borderPaddingLeft={toNumberWithDefaultZero(padding.paddingLeft)}
|
|
66
|
+
hasFocusableInside={mockHasFocusableInside}
|
|
67
|
+
entry={mockEntry}
|
|
68
|
+
state="focused"
|
|
69
|
+
hasTextLabels={false}
|
|
55
70
|
>
|
|
56
71
|
<View testID="child" />
|
|
57
72
|
</BorderContainerView>
|
|
@@ -1,10 +1,16 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import * as React from "react";
|
|
1
|
+
import React, { useMemo, useContext, useEffect } from "react";
|
|
3
2
|
import { ImageStyle, StyleSheet, View, ViewStyle } from "react-native";
|
|
3
|
+
import { useAccessibilityManager } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
|
|
4
|
+
import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/numberUtils";
|
|
5
|
+
import { MeasurementPortalContext } from "../../../MeasurmentsPortal/MeasurementsPortal";
|
|
4
6
|
|
|
5
7
|
type BorderPosition = "inside" | "outside" | "center";
|
|
6
8
|
|
|
7
9
|
interface Props {
|
|
10
|
+
hasFocusableInside: (entry: ZappEntry) => boolean;
|
|
11
|
+
entry: ZappEntry;
|
|
12
|
+
state: CellState;
|
|
13
|
+
hasTextLabels: boolean;
|
|
8
14
|
style: ImageStyle | ViewStyle;
|
|
9
15
|
borderPosition: BorderPosition;
|
|
10
16
|
borderPaddingTop: number;
|
|
@@ -118,8 +124,30 @@ export const BorderContainerView = (props: Props) => {
|
|
|
118
124
|
borderPaddingLeft,
|
|
119
125
|
style,
|
|
120
126
|
children,
|
|
127
|
+
hasFocusableInside,
|
|
128
|
+
entry,
|
|
129
|
+
state,
|
|
130
|
+
hasTextLabels,
|
|
121
131
|
} = props;
|
|
122
132
|
|
|
133
|
+
const accessibilityManager = useAccessibilityManager();
|
|
134
|
+
const isMeasurement = useContext(MeasurementPortalContext);
|
|
135
|
+
|
|
136
|
+
const isImageOnlyCell = useMemo(
|
|
137
|
+
() =>
|
|
138
|
+
state === "focused" &&
|
|
139
|
+
!hasTextLabels &&
|
|
140
|
+
!isMeasurement?.measuringInProgress &&
|
|
141
|
+
!hasFocusableInside(entry),
|
|
142
|
+
[entry, hasTextLabels, state, isMeasurement?.measuringInProgress]
|
|
143
|
+
);
|
|
144
|
+
|
|
145
|
+
useEffect(() => {
|
|
146
|
+
if (isImageOnlyCell && entry?.title) {
|
|
147
|
+
accessibilityManager.addHeading(String(entry.title));
|
|
148
|
+
}
|
|
149
|
+
}, [isImageOnlyCell, entry?.title]);
|
|
150
|
+
|
|
123
151
|
const padding =
|
|
124
152
|
borderPosition === "outside"
|
|
125
153
|
? {
|
|
@@ -32,6 +32,7 @@ export default function Image({
|
|
|
32
32
|
placeholderImage,
|
|
33
33
|
entry,
|
|
34
34
|
withDimensions,
|
|
35
|
+
source: sourceProp,
|
|
35
36
|
...otherProps
|
|
36
37
|
}: Props) {
|
|
37
38
|
const [showDefault, setShowDefault] = React.useState(false);
|
|
@@ -48,7 +49,10 @@ export default function Image({
|
|
|
48
49
|
entry,
|
|
49
50
|
showDefault,
|
|
50
51
|
placeholderImage: placeholderImage || "",
|
|
51
|
-
otherProps
|
|
52
|
+
otherProps: {
|
|
53
|
+
source: sourceProp,
|
|
54
|
+
state: otherProps.state,
|
|
55
|
+
},
|
|
52
56
|
});
|
|
53
57
|
|
|
54
58
|
const onError = React.useCallback(() => {
|
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import * as React from "react";
|
|
2
2
|
import { Image as RnImage, ImageStyle } from "react-native";
|
|
3
|
-
import { equals, omit } from "ramda";
|
|
4
3
|
|
|
5
4
|
import { useImageSource } from "./hooks";
|
|
5
|
+
import { equals } from "@applicaster/zapp-react-native-utils/utils";
|
|
6
6
|
|
|
7
7
|
type Source = {
|
|
8
8
|
uri: string;
|
|
@@ -25,11 +25,19 @@ function Image({
|
|
|
25
25
|
placeholderImage,
|
|
26
26
|
entry,
|
|
27
27
|
withDimensions,
|
|
28
|
+
source: sourceProp,
|
|
28
29
|
...otherProps
|
|
29
30
|
}: Props) {
|
|
30
31
|
const [error, setErrorState] = React.useState(null);
|
|
31
32
|
|
|
32
|
-
const source = useImageSource({
|
|
33
|
+
const source = useImageSource({
|
|
34
|
+
uri,
|
|
35
|
+
entry,
|
|
36
|
+
otherProps: {
|
|
37
|
+
source: sourceProp,
|
|
38
|
+
state: otherProps.state,
|
|
39
|
+
},
|
|
40
|
+
});
|
|
33
41
|
|
|
34
42
|
React.useEffect(() => {
|
|
35
43
|
// reset error state on URI change as the error is referencing previous uri
|
|
@@ -49,7 +57,7 @@ function Image({
|
|
|
49
57
|
onError={React.useCallback(() => setErrorState(true), [])}
|
|
50
58
|
// as we have defaults as "" for placeholder image, we need to pass undefined to source to not throw warnings
|
|
51
59
|
source={_source?.uri ? _source : undefined}
|
|
52
|
-
{...
|
|
60
|
+
{...otherProps}
|
|
53
61
|
/>
|
|
54
62
|
);
|
|
55
63
|
}
|
|
@@ -23,9 +23,17 @@ function Image({
|
|
|
23
23
|
placeholderImage,
|
|
24
24
|
entry,
|
|
25
25
|
withDimensions,
|
|
26
|
+
source: sourceProp,
|
|
26
27
|
...otherProps
|
|
27
28
|
}: Props) {
|
|
28
|
-
const source = useImageSource({
|
|
29
|
+
const source = useImageSource({
|
|
30
|
+
uri,
|
|
31
|
+
entry,
|
|
32
|
+
otherProps: {
|
|
33
|
+
source: sourceProp,
|
|
34
|
+
state: otherProps.state,
|
|
35
|
+
},
|
|
36
|
+
});
|
|
29
37
|
|
|
30
38
|
const updatedSource = source ? withDimensions(source) : { uri: "" };
|
|
31
39
|
|
|
@@ -1,19 +1,21 @@
|
|
|
1
1
|
import * as React from "react";
|
|
2
|
-
import { path } from "ramda";
|
|
3
2
|
|
|
4
3
|
import { isTV } from "@applicaster/zapp-react-native-utils/reactUtils";
|
|
5
4
|
import { useActions } from "@applicaster/zapp-react-native-utils/reactHooks/actions";
|
|
6
5
|
import { extractAsset } from "./utils";
|
|
7
6
|
|
|
8
7
|
type Return = { uri: string } | null;
|
|
8
|
+
type Source = { context?: string; uri?: string } | null | undefined;
|
|
9
9
|
|
|
10
|
-
const getSourceContext =
|
|
11
|
-
const getSourceUri =
|
|
12
|
-
const getState = path(["state"]);
|
|
10
|
+
const getSourceContext = (source: Source) => source?.context;
|
|
11
|
+
const getSourceUri = (source: Source) => source?.uri;
|
|
13
12
|
|
|
14
|
-
export const useImageSource = ({
|
|
15
|
-
|
|
16
|
-
|
|
13
|
+
export const useImageSource = ({
|
|
14
|
+
uri,
|
|
15
|
+
entry,
|
|
16
|
+
otherProps: { source, state: uriState },
|
|
17
|
+
}): Return => {
|
|
18
|
+
const uriContext = getSourceContext(source);
|
|
17
19
|
|
|
18
20
|
const action = useActions(uriContext);
|
|
19
21
|
|
|
@@ -38,7 +40,7 @@ export const useImageSource = ({ uri, entry, otherProps }): Return => {
|
|
|
38
40
|
return { uri };
|
|
39
41
|
}
|
|
40
42
|
|
|
41
|
-
const uriFromSource = getSourceUri(
|
|
43
|
+
const uriFromSource = getSourceUri(source);
|
|
42
44
|
|
|
43
45
|
if (uriFromSource) {
|
|
44
46
|
return { uri: uriFromSource };
|
|
@@ -47,7 +49,7 @@ export const useImageSource = ({ uri, entry, otherProps }): Return => {
|
|
|
47
49
|
return null;
|
|
48
50
|
};
|
|
49
51
|
|
|
50
|
-
const getSource = (uri, showDefault, placeholderImage,
|
|
52
|
+
const getSource = (uri, showDefault, placeholderImage, source) => {
|
|
51
53
|
const placeholderName = placeholderImage || "";
|
|
52
54
|
|
|
53
55
|
const defaultPath = {
|
|
@@ -60,7 +62,7 @@ const getSource = (uri, showDefault, placeholderImage, otherProps) => {
|
|
|
60
62
|
return { uri };
|
|
61
63
|
}
|
|
62
64
|
|
|
63
|
-
const uriFromSource = getSourceUri(
|
|
65
|
+
const uriFromSource = getSourceUri(source);
|
|
64
66
|
|
|
65
67
|
if (uriFromSource) {
|
|
66
68
|
return { uri: uriFromSource };
|
|
@@ -74,10 +76,9 @@ export const useImageSourceWithDefault = ({
|
|
|
74
76
|
entry,
|
|
75
77
|
showDefault,
|
|
76
78
|
placeholderImage,
|
|
77
|
-
otherProps,
|
|
79
|
+
otherProps: { state: uriState, source },
|
|
78
80
|
}): Return => {
|
|
79
|
-
const uriContext = getSourceContext(
|
|
80
|
-
const uriState = getState(otherProps);
|
|
81
|
+
const uriContext = getSourceContext(source);
|
|
81
82
|
|
|
82
83
|
const action = useActions(uriContext);
|
|
83
84
|
|
|
@@ -98,5 +99,5 @@ export const useImageSourceWithDefault = ({
|
|
|
98
99
|
return extractAsset(!isTV(), entryStateLocal.asset, uriState);
|
|
99
100
|
}
|
|
100
101
|
|
|
101
|
-
return getSource(uri, showDefault, placeholderImage,
|
|
102
|
+
return getSource(uri, showDefault, placeholderImage, source);
|
|
102
103
|
};
|
|
@@ -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?.
|
|
120
|
+
...entry?.extensions?.brightcove,
|
|
121
121
|
video_id: previewPlayback,
|
|
122
122
|
},
|
|
123
123
|
},
|
|
124
124
|
};
|
|
125
125
|
}
|
|
126
126
|
|
|
127
|
-
if (entry.extensions?.
|
|
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?.
|
|
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
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
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]);
|
|
@@ -32,14 +32,16 @@ describe("resolveColor", () => {
|
|
|
32
32
|
color: "invalid_path",
|
|
33
33
|
};
|
|
34
34
|
|
|
35
|
-
expect(resolveColor(entry, style)).toEqual(
|
|
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(
|
|
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 =
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
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
|
|
52
|
-
|
|
53
|
-
|
|
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,
|
|
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(
|
|
78
|
+
logWarning(colorFromProp, colorFromProp, nestedEntryValue);
|
|
68
79
|
|
|
69
|
-
return
|
|
80
|
+
return undefined;
|
|
70
81
|
}
|
|
71
82
|
|
|
72
|
-
return
|
|
83
|
+
return colorValue;
|
|
73
84
|
}
|
|
74
85
|
|
|
75
|
-
|
|
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
|
|
85
|
-
|
|
86
|
-
|
|
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,
|