@applicaster/zapp-react-native-utils 14.0.0-alpha.8387612031 → 14.0.0-alpha.8419134002

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 (31) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +1 -1
  2. package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
  3. package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
  4. package/analyticsUtils/__tests__/analyticsUtils.test.js +14 -0
  5. package/analyticsUtils/events.ts +8 -0
  6. package/appUtils/accessibilityManager/index.ts +3 -3
  7. package/audioPlayerUtils/index.ts +104 -0
  8. package/focusManager/FocusManager.ts +4 -6
  9. package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
  10. package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
  11. package/manifestUtils/createConfig.js +4 -1
  12. package/manifestUtils/defaultManifestConfigurations/player.js +1239 -200
  13. package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
  14. package/package.json +2 -2
  15. package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
  16. package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
  17. package/playerUtils/_internals/index.ts +1 -0
  18. package/playerUtils/_internals/utils.ts +31 -0
  19. package/playerUtils/configurationUtils.ts +0 -44
  20. package/playerUtils/getPlayerActionButtons.ts +1 -1
  21. package/playerUtils/useValidatePlayerConfig.tsx +22 -19
  22. package/reactHooks/feed/useBatchLoading.ts +1 -1
  23. package/reactHooks/feed/usePipesCacheReset.ts +1 -1
  24. package/reactHooks/navigation/useIsScreenActive.ts +9 -5
  25. package/reactHooks/screen/useScreenContext.ts +1 -1
  26. package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
  27. package/riverComponetsMeasurementProvider/index.tsx +1 -1
  28. package/services/js2native.ts +1 -0
  29. package/time/BackgroundTimer.ts +5 -3
  30. package/utils/index.ts +7 -0
  31. package/playerUtils/configurationGenerator.ts +0 -2572
@@ -25,7 +25,7 @@ import {
25
25
  resolveObjectValues,
26
26
  } from "../appUtils/contextKeysManager/contextResolver";
27
27
  import { useNavigation } from "../reactHooks";
28
- import { get } from "lodash";
28
+ import { get } from "../utils";
29
29
 
30
30
  import {
31
31
  useContentTypes,
@@ -4,9 +4,12 @@ import {
4
4
  ANALYTICS_COMPONENT_EVENTS,
5
5
  ANALYTICS_CORE_EVENTS,
6
6
  ANALYTICS_ENTRY_EVENTS,
7
+ ANALYTICS_PREFERENCES_EVENTS,
7
8
  DOWNLOADS_EVENTS,
8
9
  } from "../events";
9
10
  import { isEmptyOrNil } from "../../cellUtils";
11
+ import { get } from "lodash";
12
+ import { StorageMultiSelectProvider } from "@applicaster/zapp-react-native-bridge/ZappStorage/StorageMultiSelectProvider";
10
13
 
11
14
  export enum OfflineItemState {
12
15
  notExist = "NOT_EXISTS",
@@ -102,6 +105,84 @@ export function eventForComponent(
102
105
  return analyticsProps;
103
106
  }
104
107
 
108
+ /**
109
+ * Checks if an item is currently selected in localStorage based on its actions
110
+ * @param item - The item to check
111
+ * @returns boolean indicating if the item is currently selected
112
+ */
113
+ function isItemPreviouslySelected(item: any): boolean {
114
+ const actions = item?.extensions?.tap_actions?.actions;
115
+
116
+ if (!actions) {
117
+ return false;
118
+ }
119
+
120
+ const localStorageAction = actions.find(
121
+ (action) => action?.type === "localStorageToggleFlag"
122
+ );
123
+
124
+ if (!localStorageAction?.options?.key) {
125
+ return false;
126
+ }
127
+
128
+ const keyNamespace = localStorageAction.options.key;
129
+
130
+ const tag = localStorageAction.options?.selector
131
+ ? get(item, localStorageAction.options.selector)
132
+ : (item.extensions?.tag ?? item.id);
133
+
134
+ if (!tag) {
135
+ return false;
136
+ }
137
+
138
+ try {
139
+ const multiSelectProvider =
140
+ StorageMultiSelectProvider.getProvider(keyNamespace);
141
+
142
+ const selectedItems = multiSelectProvider.getSelectedItems();
143
+
144
+ return selectedItems.includes(tag);
145
+ } catch (error) {
146
+ return false;
147
+ }
148
+ }
149
+
150
+ export function getLocalStorageSetPayload(extraProps) {
151
+ const { item } = extraProps;
152
+
153
+ const hasLocalStorageSetAction = item?.extensions?.tap_actions?.actions?.some(
154
+ (action) => action?.type === "localStorageSet"
155
+ );
156
+
157
+ if (!hasLocalStorageSetAction) {
158
+ return null;
159
+ }
160
+
161
+ return {
162
+ [ANALYTICS_PREFERENCES_EVENTS.ITEM_SELECTED_STATUS]: true,
163
+ };
164
+ }
165
+
166
+ export function getLocalStorageToggleFlagPayload(extraProps) {
167
+ const { item } = extraProps;
168
+
169
+ const hasLocalStorageToggleAction =
170
+ item?.extensions?.tap_actions?.actions?.some(
171
+ (action) => action?.type === "localStorageToggleFlag"
172
+ );
173
+
174
+ if (!hasLocalStorageToggleAction) {
175
+ return null;
176
+ }
177
+
178
+ const previouslySelected = isItemPreviouslySelected(item);
179
+
180
+ return {
181
+ [ANALYTICS_PREFERENCES_EVENTS.ITEM_SELECTED_STATUS]: !previouslySelected,
182
+ [ANALYTICS_PREFERENCES_EVENTS.PREVIOUS_SELECTED_STATE]: previouslySelected,
183
+ };
184
+ }
185
+
105
186
  export function playEventForType(item) {
106
187
  const itemType = item?.type && item.type?.value;
107
188
 
@@ -1,13 +1,13 @@
1
1
  import { log_error, log_debug } from "../logger";
2
-
3
- import { ANALYTICS_CORE_EVENTS } from "../events";
4
-
2
+ import { ANALYTICS_CORE_EVENTS, ACTION_TYPE } from "../events";
5
3
  import { postAnalyticEvent } from "../manager";
6
4
  import {
7
5
  replaceAnalyticsPropsNils,
8
6
  eventForEntry,
9
7
  eventForComponent,
10
8
  extensionsEvents,
9
+ getLocalStorageSetPayload,
10
+ getLocalStorageToggleFlagPayload,
11
11
  } from "./helper";
12
12
 
13
13
  declare type AnalyticsDefaultHelperProperties = {
@@ -26,7 +26,16 @@ export const sendOnClickEvent = ({
26
26
  const castedExtraProps: ExtraProps = extraProps;
27
27
  const componentData = component || extraProps.component;
28
28
  const data = zappPipesData || extraProps.zappPipesData;
29
- const eventName = ANALYTICS_CORE_EVENTS.TAP_CELL;
29
+
30
+ const actionCellPayload =
31
+ extraProps?.item?.type?.value === ACTION_TYPE
32
+ ? getLocalStorageSetPayload(extraProps) ||
33
+ getLocalStorageToggleFlagPayload(extraProps)
34
+ : null;
35
+
36
+ const eventName = actionCellPayload
37
+ ? ANALYTICS_CORE_EVENTS.TAP_SELECTABLE_CELL
38
+ : ANALYTICS_CORE_EVENTS.TAP_CELL;
30
39
 
31
40
  if (!analyticsScreenData) {
32
41
  log_error(
@@ -44,6 +53,7 @@ export const sendOnClickEvent = ({
44
53
  ...replaceAnalyticsPropsNils({
45
54
  ...analyticsScreenData,
46
55
  }),
56
+ ...actionCellPayload,
47
57
  };
48
58
 
49
59
  if (analyticsCustomProperties) {
@@ -3,8 +3,22 @@ import { ANALYTICS_CORE_EVENTS } from "../events";
3
3
 
4
4
  jest.mock("@applicaster/zapp-react-native-utils/reactUtils", () => ({
5
5
  isWeb: jest.fn(),
6
+ platformSelect: jest.fn(
7
+ (options) => options.android || options.ios || options.web
8
+ ),
6
9
  }));
7
10
 
11
+ jest.mock(
12
+ "@applicaster/zapp-react-native-bridge/ZappStorage/StorageMultiSelectProvider",
13
+ () => ({
14
+ StorageMultiSelectProvider: {
15
+ getProvider: jest.fn(() => ({
16
+ getSelectedItems: jest.fn(() => []),
17
+ })),
18
+ },
19
+ })
20
+ );
21
+
8
22
  const mock_postAnalyticEvent = jest.fn();
9
23
  const mock_startAnalyticsTimedEvent = jest.fn();
10
24
  const mock_endAnalyticsTimedEvent = jest.fn();
@@ -17,8 +17,11 @@ export const SCREEN_VIEW_EVENTS = {
17
17
  TIME_ON_SCREEN: "time_on_screen",
18
18
  };
19
19
 
20
+ export const ACTION_TYPE = "action";
21
+
20
22
  export const TAPPING_EVENTS = {
21
23
  TAP_CELL: "tap_cell",
24
+ TAP_SELECTABLE_CELL: "tap_selectable_cell",
22
25
  TAP_MENU: "tap_menu",
23
26
  TAP_NAVBAR_BACK_BUTTON: "tap_navbar_back_button",
24
27
  };
@@ -99,6 +102,11 @@ export const ANALYTICS_COMPONENT_EVENTS = {
99
102
  COMPONENT_SOURCE: "component_source",
100
103
  };
101
104
 
105
+ export const ANALYTICS_PREFERENCES_EVENTS = {
106
+ ITEM_SELECTED_STATUS: "item_selected_status",
107
+ PREVIOUS_SELECTED_STATE: "previous_selected_state",
108
+ };
109
+
102
110
  // ---------------- EVENTS ---------------------
103
111
  export const AD_EVENT = {
104
112
  ad_break_start: "player_ad_break_start",
@@ -1,9 +1,9 @@
1
1
  import { BehaviorSubject } from "rxjs";
2
2
  import { accessibilityManagerLogger as logger } from "./logger";
3
- import { TTSManager } from "../platform/platformUtils";
3
+ import { TTSManager } from "../platform";
4
4
  import { BUTTON_ACCESSIBILITY_KEYS } from "./const";
5
5
  import { AccessibilityRole } from "react-native";
6
- import _ from "lodash";
6
+ import { toString } from "../../utils";
7
7
 
8
8
  export class AccessibilityManager {
9
9
  private static _instance: AccessibilityManager | null = null;
@@ -137,7 +137,7 @@ export class AccessibilityManager {
137
137
  }
138
138
 
139
139
  public getButtonAccessibilityProps(name: string): AccessibilityProps {
140
- const buttonName = _.toString(name);
140
+ const buttonName = toString(name);
141
141
 
142
142
  const buttonConfig = BUTTON_ACCESSIBILITY_KEYS[buttonName];
143
143
 
@@ -1,3 +1,8 @@
1
+ import * as React from "react";
2
+ import { get, has } from "@applicaster/zapp-react-native-utils/utils";
3
+ import { useZStore } from "@applicaster/zapp-react-native-utils/reactHooks";
4
+ import { useRoute } from "@applicaster/zapp-react-native-utils/reactHooks/navigation";
5
+
1
6
  import { isNotEmptyString } from "@applicaster/zapp-react-native-utils/stringUtils";
2
7
  import { getMediaItems } from "@applicaster/zapp-react-native-utils/configurationUtils";
3
8
 
@@ -136,3 +141,102 @@ export const getArtworkImage = ({
136
141
  // default image
137
142
  return DEFAULT_IMAGE;
138
143
  };
144
+
145
+ const useAdjustedKeyFromScreenData = ({ config, keys }) => {
146
+ const { screenData } = useRoute();
147
+
148
+ const adjustedConfig = { ...config };
149
+
150
+ keys.forEach((key) => {
151
+ const path = ["targetScreen", "styles", key];
152
+
153
+ if (has(screenData, path)) {
154
+ const value = get(screenData, path);
155
+
156
+ adjustedConfig[key] = value;
157
+ }
158
+ });
159
+
160
+ return adjustedConfig;
161
+ };
162
+
163
+ const AUDIO_PLAYER_ARTWORK_IMAGE_KEY = "audio_player_artwork_image_key";
164
+
165
+ const audioPlayerArtworkImageKeySelector = (config) =>
166
+ config?.[AUDIO_PLAYER_ARTWORK_IMAGE_KEY];
167
+
168
+ export const useArtworkImage = (entry: ZappEntry): string => {
169
+ const configuration = useZStore("playerConfiguration");
170
+
171
+ const audio_player_artwork_image_key_value = configuration(
172
+ audioPlayerArtworkImageKeySelector
173
+ );
174
+
175
+ const pluginConfiguration = React.useMemo(
176
+ () => ({
177
+ audio_player_artwork_image_key: audio_player_artwork_image_key_value,
178
+ }),
179
+ [audio_player_artwork_image_key_value]
180
+ );
181
+
182
+ // HACK: for override [key] from screenData, because we treat empty string value as missing key and replace it to initial_value from manifest(which is wrong)
183
+ const adjustedPluginConfiguration = useAdjustedKeyFromScreenData({
184
+ keys: [AUDIO_PLAYER_ARTWORK_IMAGE_KEY],
185
+ config: pluginConfiguration,
186
+ });
187
+
188
+ return React.useMemo(
189
+ () =>
190
+ getArtworkImage({
191
+ key: AUDIO_PLAYER_ARTWORK_IMAGE_KEY,
192
+ entry,
193
+ plugin_configuration: adjustedPluginConfiguration,
194
+ }),
195
+ []
196
+ );
197
+ };
198
+
199
+ const AUDIO_PLAYER_IMAGE_KEY = "audio_player_image_key";
200
+ const AUDIO_PLAYER_BACKGROUND_IMAGE = "audio_player_background_image";
201
+
202
+ const audioPlayerImageKeySelector = (config) =>
203
+ config?.[AUDIO_PLAYER_IMAGE_KEY];
204
+
205
+ const audioPlayerBackgroundImageSelector = (config) =>
206
+ config?.[AUDIO_PLAYER_BACKGROUND_IMAGE];
207
+
208
+ export const useBackgroundImage = (entry: ZappEntry): { uri: string } => {
209
+ const configuration = useZStore("playerConfiguration");
210
+
211
+ const audio_player_image_key_value = configuration(
212
+ audioPlayerImageKeySelector
213
+ );
214
+
215
+ const audio_player_background_image_value = configuration(
216
+ audioPlayerBackgroundImageSelector
217
+ );
218
+
219
+ const pluginConfiguration = React.useMemo(
220
+ () => ({
221
+ audio_player_image_key: audio_player_image_key_value,
222
+ audio_player_background_image: audio_player_background_image_value,
223
+ }),
224
+ [audio_player_image_key_value, audio_player_background_image_value]
225
+ );
226
+
227
+ // HACK: for override [key] from screenData, because we treat empty string value as missing key and replace it to initial_value from manifest(which is wrong)
228
+ const adjustedPluginConfiguration = useAdjustedKeyFromScreenData({
229
+ keys: [AUDIO_PLAYER_IMAGE_KEY, AUDIO_PLAYER_BACKGROUND_IMAGE],
230
+ config: pluginConfiguration,
231
+ });
232
+
233
+ return React.useMemo(
234
+ () => ({
235
+ uri: getBackgroundImage({
236
+ entry,
237
+ plugin_configuration: adjustedPluginConfiguration,
238
+ }),
239
+ }),
240
+ [entry, adjustedPluginConfiguration]
241
+ );
242
+ };
@@ -243,12 +243,10 @@ class FocusManager {
243
243
  }
244
244
 
245
245
  blurPrevious(options?: FocusManager.Android.CallbackOptions) {
246
- if (options) {
247
- FocusManager.instance.prevFocused?.onBlur?.(
248
- FocusManager.instance.prevFocused,
249
- options
250
- );
251
- }
246
+ FocusManager.instance.prevFocused?.onBlur?.(
247
+ FocusManager.instance.prevFocused,
248
+ options ?? {} // Adding fallback to avoid potential regression caused by #7509
249
+ );
252
250
  }
253
251
 
254
252
  onDisableFocusChange = (id) => {
@@ -0,0 +1,28 @@
1
+ const R = require("ramda");
2
+ const { defaultConfigurations } = require("../defaultManifestConfigurations");
3
+
4
+ /**
5
+ * returns default configuration keys for provided plugin type
6
+ * @param {('general-content'|'player')} pluginType
7
+ * @param options manifest generator information
8
+ * @param {string} options.version manifest version
9
+ * @param {string} options.platform qb platform value
10
+ */
11
+ function getDefaultConfiguration(pluginType, options) {
12
+ const defConfig = R.compose(
13
+ R.unless(R.isNil, (fn) => fn(options)),
14
+ R.propOr(null, pluginType)
15
+ )(defaultConfigurations);
16
+
17
+ if (!defConfig) {
18
+ const availableKeys = R.keys(defaultConfigurations);
19
+
20
+ const message = `Requested key "${pluginType}" doesn't exist in the default configuration\nAvailable keys: ${availableKeys}`;
21
+ // eslint-disable-next-line no-console
22
+ console.warn(message);
23
+ }
24
+
25
+ return defConfig;
26
+ }
27
+
28
+ module.exports = { getDefaultConfiguration };
@@ -1,6 +1,7 @@
1
1
  const R = require("ramda");
2
2
  const camelize = require("camelize");
3
- const { defaultConfigurations } = require("./defaultManifestConfigurations");
3
+
4
+ const { getDefaultConfiguration } = require("./getDefaultConfiguration");
4
5
 
5
6
  const toSnakeCase = R.compose(R.replace(/\s/g, "_"), R.toLower, R.trim);
6
7
 
@@ -173,30 +174,6 @@ function generateFieldsFromDefaultsWithoutPrefixedLabel(
173
174
  )(fields);
174
175
  }
175
176
 
176
- /**
177
- * returns default configuration keys for provided plugin type
178
- * @param {('general-content'|'player')} pluginType
179
- * @param options manifest generator information
180
- * @param {string} options.version manifest version
181
- * @param {string} options.platform qb platform value
182
- */
183
- function getDefaultConfiguration(pluginType, options) {
184
- const defConfig = R.compose(
185
- R.unless(R.isNil, (fn) => fn(options)),
186
- R.propOr(null, pluginType)
187
- )(defaultConfigurations);
188
-
189
- if (!defConfig) {
190
- const availableKeys = R.keys(defaultConfigurations);
191
-
192
- const message = `Requested key "${pluginType}" doesn't exist in the default configuration\nAvailable keys: ${availableKeys}`;
193
- // eslint-disable-next-line no-console
194
- console.warn(message);
195
- }
196
-
197
- return defConfig;
198
- }
199
-
200
177
  module.exports = {
201
178
  toSnakeCase,
202
179
  toCamelCase,
@@ -1,5 +1,8 @@
1
1
  const R = require("ramda");
2
- const { getDefaultConfiguration } = require("./_internals");
2
+
3
+ const {
4
+ getDefaultConfiguration,
5
+ } = require("./_internals/getDefaultConfiguration");
3
6
 
4
7
  const indexByKey = R.indexBy((obj) => R.prop(obj.group ? "label" : "key", obj));
5
8
  const propIfExist = R.curry((prop) => R.when(R.prop(prop), R.prop(prop)));