@applicaster/zapp-react-native-utils 14.0.0-alpha.1661204539 → 14.0.0-alpha.1740013076

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/actionsExecutor/ActionExecutorContext.tsx +11 -6
  2. package/actionsExecutor/ScreenActions.ts +91 -17
  3. package/actionsExecutor/feedDecorator.ts +171 -0
  4. package/actionsExecutor/screenResolver.ts +6 -3
  5. package/analyticsUtils/AnalyticsEvents/helper.ts +81 -0
  6. package/analyticsUtils/AnalyticsEvents/sendHeaderClickEvent.ts +1 -1
  7. package/analyticsUtils/AnalyticsEvents/sendMenuClickEvent.ts +2 -1
  8. package/analyticsUtils/AnalyticsEvents/sendOnClickEvent.ts +14 -4
  9. package/analyticsUtils/__tests__/analyticsUtils.test.js +3 -0
  10. package/analyticsUtils/events.ts +8 -0
  11. package/analyticsUtils/index.tsx +3 -4
  12. package/analyticsUtils/manager.ts +1 -1
  13. package/analyticsUtils/playerAnalyticsTracker.ts +2 -1
  14. package/appUtils/HooksManager/Hook.ts +4 -4
  15. package/appUtils/HooksManager/index.ts +11 -1
  16. package/appUtils/accessibilityManager/const.ts +13 -0
  17. package/appUtils/accessibilityManager/hooks.ts +35 -1
  18. package/appUtils/accessibilityManager/index.ts +154 -30
  19. package/appUtils/accessibilityManager/utils.ts +24 -0
  20. package/appUtils/contextKeysManager/contextResolver.ts +30 -3
  21. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +5 -0
  22. package/appUtils/focusManager/__tests__/focusManager.test.js +1 -1
  23. package/appUtils/focusManager/index.ios.ts +10 -0
  24. package/appUtils/focusManager/index.ts +82 -11
  25. package/appUtils/focusManager/treeDataStructure/Tree/index.js +1 -1
  26. package/appUtils/focusManagerAux/utils/index.ts +106 -3
  27. package/appUtils/platform/platformUtils.ts +31 -1
  28. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +0 -15
  29. package/appUtils/playerManager/useChapterMarker.tsx +0 -1
  30. package/appUtils/playerManager/usePlayerControllerSetup.tsx +16 -0
  31. package/arrayUtils/__tests__/isEmptyArray.test.ts +63 -0
  32. package/arrayUtils/__tests__/isFilledArray.test.ts +1 -1
  33. package/arrayUtils/index.ts +8 -3
  34. package/audioPlayerUtils/__tests__/getArtworkImage.test.ts +144 -0
  35. package/audioPlayerUtils/__tests__/getBackgroundImage.test.ts +72 -0
  36. package/audioPlayerUtils/__tests__/getImageFromEntry.test.ts +110 -0
  37. package/audioPlayerUtils/assets/index.ts +2 -0
  38. package/audioPlayerUtils/index.ts +242 -0
  39. package/componentsUtils/__tests__/isTabsScreen.test.ts +38 -0
  40. package/componentsUtils/index.ts +4 -1
  41. package/conf/player/__tests__/selectors.test.ts +34 -0
  42. package/conf/player/selectors.ts +10 -0
  43. package/configurationUtils/__tests__/configurationUtils.test.js +0 -31
  44. package/configurationUtils/__tests__/getMediaItems.test.ts +65 -0
  45. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +34 -0
  46. package/configurationUtils/__tests__/manifestKeyParser.test.ts +546 -0
  47. package/configurationUtils/index.ts +64 -35
  48. package/configurationUtils/manifestKeyParser.ts +57 -32
  49. package/focusManager/FocusManager.ts +26 -16
  50. package/focusManager/Tree.ts +25 -21
  51. package/focusManager/__tests__/FocusManager.test.ts +50 -8
  52. package/index.d.ts +1 -10
  53. package/manifestUtils/_internals/getDefaultConfiguration.js +28 -0
  54. package/manifestUtils/{_internals.js → _internals/index.js} +2 -25
  55. package/manifestUtils/createConfig.js +4 -1
  56. package/manifestUtils/defaultManifestConfigurations/player.js +1253 -200
  57. package/manifestUtils/progressBar/__tests__/mobileProgressBar.test.js +0 -30
  58. package/manifestUtils/sharedConfiguration/screenPicker/stylesFields.js +1 -2
  59. package/navigationUtils/__tests__/mapContentTypesToRivers.test.ts +130 -0
  60. package/navigationUtils/index.ts +7 -5
  61. package/package.json +2 -3
  62. package/playerUtils/PlayerTTS/PlayerTTS.ts +359 -0
  63. package/playerUtils/PlayerTTS/index.ts +1 -0
  64. package/playerUtils/__tests__/configurationUtils.test.ts +1 -65
  65. package/playerUtils/__tests__/getPlayerActionButtons.test.ts +54 -0
  66. package/playerUtils/_internals/__tests__/utils.test.ts +71 -0
  67. package/playerUtils/_internals/index.ts +1 -0
  68. package/playerUtils/_internals/utils.ts +31 -0
  69. package/playerUtils/configurationUtils.ts +0 -44
  70. package/playerUtils/getPlayerActionButtons.ts +17 -0
  71. package/playerUtils/index.ts +59 -0
  72. package/playerUtils/usePlayerTTS.ts +21 -0
  73. package/playerUtils/useValidatePlayerConfig.tsx +22 -19
  74. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +15 -14
  75. package/reactHooks/cell-click/__tests__/index.test.js +3 -0
  76. package/reactHooks/cell-click/index.ts +3 -0
  77. package/reactHooks/debugging/__tests__/index.test.js +0 -1
  78. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +47 -90
  79. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +71 -31
  80. package/reactHooks/feed/index.ts +2 -0
  81. package/reactHooks/feed/useBatchLoading.ts +17 -10
  82. package/reactHooks/feed/useFeedLoader.tsx +39 -44
  83. package/reactHooks/feed/useLoadPipesDataDispatch.ts +63 -0
  84. package/reactHooks/feed/usePipesCacheReset.ts +3 -3
  85. package/reactHooks/flatList/useSequentialRenderItem.tsx +3 -3
  86. package/reactHooks/layout/__tests__/index.test.tsx +3 -1
  87. package/reactHooks/layout/isTablet/index.ts +12 -5
  88. package/reactHooks/layout/useDimensions/__tests__/useDimensions.test.ts +34 -36
  89. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -3
  90. package/reactHooks/layout/useLayoutVersion.ts +5 -5
  91. package/reactHooks/navigation/index.ts +7 -5
  92. package/reactHooks/navigation/useIsScreenActive.ts +9 -5
  93. package/reactHooks/navigation/useRoute.ts +7 -2
  94. package/reactHooks/navigation/useScreenStateStore.ts +8 -0
  95. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +4 -0
  96. package/reactHooks/screen/useScreenContext.ts +1 -1
  97. package/reactHooks/state/__tests__/ZStoreProvider.test.tsx +2 -1
  98. package/reactHooks/state/index.ts +1 -1
  99. package/reactHooks/state/useHomeRiver.ts +4 -2
  100. package/reactHooks/state/useRivers.ts +7 -8
  101. package/riverComponetsMeasurementProvider/index.tsx +1 -1
  102. package/screenPickerUtils/index.ts +7 -0
  103. package/services/js2native.ts +1 -0
  104. package/storage/ScreenSingleValueProvider.ts +138 -26
  105. package/storage/ScreenStateMultiSelectProvider.ts +210 -36
  106. package/testUtils/index.tsx +7 -8
  107. package/time/BackgroundTimer.ts +6 -4
  108. package/utils/__tests__/find.test.ts +36 -0
  109. package/utils/__tests__/pathOr.test.ts +37 -0
  110. package/utils/__tests__/startsWith.test.ts +30 -0
  111. package/utils/find.ts +3 -0
  112. package/utils/index.ts +24 -1
  113. package/utils/pathOr.ts +5 -0
  114. package/utils/startsWith.ts +9 -0
  115. package/playerUtils/configurationGenerator.ts +0 -2572
@@ -36,7 +36,6 @@ import {
36
36
  sessionStorageToggleFlag,
37
37
  } from "./StorageActions";
38
38
 
39
- import { ScreenSingleValueProvider } from "@applicaster/zapp-react-native-utils/storage/ScreenSingleValueProvider";
40
39
  import { screenSetVariable, screenToggleFlag } from "./ScreenActions";
41
40
 
42
41
  export const { log_error, log_info, log_debug } = createLogger({
@@ -121,7 +120,6 @@ const prepareDefaultActions = (actionExecutor) => {
121
120
  loadPipesData(dataSource, {
122
121
  silentRefresh: false,
123
122
  clearCache: true,
124
- callback: (_data) => {},
125
123
  riverId: context?.screenData?.id,
126
124
  })
127
125
  );
@@ -169,8 +167,7 @@ const prepareDefaultActions = (actionExecutor) => {
169
167
 
170
168
  const entry = context?.entry || {};
171
169
  const entryResolver = new EntryResolver(entry);
172
- const route = context.screenRoute;
173
- const screenData = ScreenSingleValueProvider.getState(route) as any;
170
+ const screenData = context?.screenStateStore.getState().data || {};
174
171
  const screenResolver = new EntryResolver(screenData || {});
175
172
 
176
173
  const data =
@@ -239,8 +236,14 @@ const prepareDefaultActions = (actionExecutor) => {
239
236
  context?: Record<string, any>
240
237
  ): Promise<ActionResult> => {
241
238
  const route = context?.screenRoute;
242
- const { key, value } = action.options;
243
- screenSetVariable(route, key, value);
239
+ const screenStateStore = context?.screenStateStore;
240
+
241
+ await screenSetVariable(
242
+ route,
243
+ screenStateStore,
244
+ { entry: context?.entry, options: action.options },
245
+ action
246
+ );
244
247
 
245
248
  return Promise.resolve(ActionResult.Success);
246
249
  }
@@ -253,9 +256,11 @@ const prepareDefaultActions = (actionExecutor) => {
253
256
  context?: Record<string, any>
254
257
  ): Promise<ActionResult> => {
255
258
  const screenRoute = context?.screenRoute;
259
+ const screenStateStore = context?.screenStateStore;
256
260
 
257
261
  await screenToggleFlag(
258
262
  screenRoute,
263
+ screenStateStore,
259
264
  { entry: context?.entry, options: action.options },
260
265
  action
261
266
  );
@@ -6,20 +6,93 @@ import { get } from "lodash";
6
6
 
7
7
  import { onMaxTagsReached } from "./StorageActions";
8
8
  import { ScreenMultiSelectProvider } from "../storage/ScreenStateMultiSelectProvider";
9
+ import { ScreenSingleValueProvider } from "../storage/ScreenSingleValueProvider";
10
+ import { useScreenStateStore } from "../reactHooks/navigation/useScreenStateStore";
9
11
 
10
- export const screenSetVariable = (
11
- _screenRoute: string,
12
- _key: string,
13
- _value: string
14
- ) => {};
12
+ export const screenSetVariable = async (
13
+ screenRoute: string,
14
+ screenStateStore: ReturnType<typeof useScreenStateStore>,
15
+ context: Record<string, any>,
16
+ action: ActionType
17
+ ): Promise<ActionResult> => {
18
+ if (!context) {
19
+ log_error("handleAction: screenSetVariable action missing context");
20
+
21
+ return ActionResult.Error;
22
+ }
23
+
24
+ const entry = context?.entry as ZappEntry;
25
+
26
+ if (!entry) {
27
+ log_error(
28
+ "handleAction: screenSetVariable action missing entry. Entry is required to get the value."
29
+ );
30
+
31
+ return ActionResult.Error;
32
+ }
33
+
34
+ const tag = action.options?.selector
35
+ ? get(entry, action.options.selector)
36
+ : (entry.extensions?.tag ?? entry.id);
37
+
38
+ const key = action.options?.key;
39
+
40
+ if (!key) {
41
+ log_error("handleAction: screenSetVariable action missing argument 'key'", {
42
+ key,
43
+ });
44
+
45
+ return ActionResult.Error;
46
+ }
47
+
48
+ if (!tag) {
49
+ log_error(
50
+ "handleAction: screenSetVariable action could not determine tag",
51
+ { selector: action.options?.selector, value: action.options?.value }
52
+ );
53
+
54
+ return ActionResult.Error;
55
+ }
56
+
57
+ try {
58
+ const singleValueProvider = ScreenSingleValueProvider.getProvider(
59
+ key,
60
+ screenRoute,
61
+ screenStateStore
62
+ );
63
+
64
+ const currentValue = await singleValueProvider.getValueAsync();
65
+
66
+ log_info(
67
+ `handleAction: screenSetVariable setting value: ${tag} for key: ${key}, previous value: ${currentValue}`
68
+ );
69
+
70
+ await singleValueProvider.setValue(String(tag));
71
+
72
+ log_info(
73
+ `handleAction: screenSetVariable successfully set value: ${tag} for key: ${key}`
74
+ );
75
+
76
+ return ActionResult.Success;
77
+ } catch (error) {
78
+ log_error("handleAction: screenSetVariable failed to set value", {
79
+ key,
80
+ tag,
81
+ error,
82
+ });
83
+
84
+ return ActionResult.Error;
85
+ }
86
+ };
15
87
 
16
88
  export const screenToggleFlag = async (
17
89
  screenRoute: string,
90
+ screenStateStore: ReturnType<typeof useScreenStateStore>,
18
91
  context: Record<string, any>,
19
92
  action: ActionType
20
93
  ) => {
21
94
  if (!context) {
22
- log_error("handleAction: localStorageToggleFlag action missing context");
95
+ log_error("handleAction: screenToggleFlag action missing context");
23
96
 
24
97
  return ActionResult.Error;
25
98
  }
@@ -28,7 +101,7 @@ export const screenToggleFlag = async (
28
101
 
29
102
  if (!entry) {
30
103
  log_error(
31
- "handleAction: localStorageToggleFlag action missing entry. Entry is required to get the tag."
104
+ "handleAction: screenToggleFlag action missing entry. Entry is required to get the tag."
32
105
  );
33
106
 
34
107
  return ActionResult.Error;
@@ -38,12 +111,13 @@ export const screenToggleFlag = async (
38
111
  ? get(entry, action.options.selector)
39
112
  : (entry.extensions?.tag ?? entry.id);
40
113
 
41
- const keyNamespace = action.options?.key;
114
+ const key = action.options?.key;
42
115
 
43
- if (keyNamespace && tag) {
116
+ if (key && tag) {
44
117
  const multiSelectProvider = ScreenMultiSelectProvider.getProvider(
45
- keyNamespace,
46
- screenRoute
118
+ key,
119
+ screenRoute,
120
+ screenStateStore
47
121
  );
48
122
 
49
123
  const selectedItems = await multiSelectProvider.getSelectedAsync();
@@ -52,7 +126,7 @@ export const screenToggleFlag = async (
52
126
  log_info(
53
127
  `handleAction: screenToggleFlag event will ${
54
128
  isTagInSelectedItems ? "remove" : "add"
55
- } tag: ${tag} for keyNamespace: ${keyNamespace}, current selectedItems: ${selectedItems}`
129
+ } tag: ${tag} for key: ${key}, current selectedItems: ${selectedItems}`
56
130
  );
57
131
 
58
132
  if (selectedItems.includes(tag)) {
@@ -69,7 +143,7 @@ export const screenToggleFlag = async (
69
143
  selectedItems,
70
144
  maxItems,
71
145
  tag,
72
- keyNamespace,
146
+ keyNamespace: key,
73
147
  });
74
148
 
75
149
  return ActionResult.Cancel;
@@ -78,10 +152,10 @@ export const screenToggleFlag = async (
78
152
  await multiSelectProvider.addItem(tag);
79
153
  }
80
154
  } else {
81
- log_error(
82
- "handleAction: screenToggleFlag event missing keyNamespace or tag",
83
- { keyNamespace, tag }
84
- );
155
+ log_error("handleAction: screenToggleFlag event missing key or tag", {
156
+ key,
157
+ tag,
158
+ });
85
159
 
86
160
  return ActionResult.Error;
87
161
  }
@@ -0,0 +1,171 @@
1
+ import * as R from "ramda";
2
+ import { getNamespaceAndKey } from "../appUtils/contextKeysManager/utils";
3
+ import { createLogger } from "../logger";
4
+
5
+ const { log_error, log_verbose } = createLogger({
6
+ subsystem: "FeedDecorator",
7
+ category: "General",
8
+ });
9
+
10
+ function getScope(scope) {
11
+ if (scope === "screen") {
12
+ return scope;
13
+ } else if (!scope || scope === "ctx") {
14
+ return "ctx";
15
+ } else {
16
+ log_verbose(
17
+ `decorateFeed: Unsupported scope "${scope}" provided in preference_editor_options. Defaulting to "ctx".`
18
+ );
19
+
20
+ return "ctx";
21
+ }
22
+ }
23
+
24
+ function makeMultiSelect(feed: ZappFeed, key, decoratedFeed) {
25
+ const scope = getScope(
26
+ decoratedFeed.extensions?.preference_editor_options?.scope
27
+ );
28
+
29
+ const behavior = {
30
+ ...feed.extensions?.["behavior"],
31
+ select_mode: "multi",
32
+ current_selection: `@{${scope}/${key}}`,
33
+ };
34
+
35
+ if (!feed.extensions) {
36
+ decoratedFeed.extensions = {
37
+ behavior,
38
+ };
39
+ } else {
40
+ decoratedFeed.extensions.behavior = behavior;
41
+ }
42
+
43
+ const actionType =
44
+ decoratedFeed.extensions?.preference_editor_options?.scope === "screen"
45
+ ? "screenToggleFlag"
46
+ : decoratedFeed.extensions?.preference_editor_options?.scope === "session"
47
+ ? "sessionStorageToggleFlag"
48
+ : "localStorageToggleFlag";
49
+
50
+ decoratedFeed.entry?.forEach((entry: ZappEntry) => {
51
+ entry.type.value = "action";
52
+ entry.id = entry.extensions?.tag || entry.id;
53
+
54
+ entry.extensions = {
55
+ ...(entry.extensions || {}),
56
+ tap_actions: {
57
+ actions: [
58
+ {
59
+ type: actionType,
60
+ options: {
61
+ key,
62
+ },
63
+ },
64
+ ],
65
+ },
66
+ };
67
+ });
68
+
69
+ return decoratedFeed;
70
+ }
71
+
72
+ function makeSingleSelect(feed: ZappFeed, key, decoratedFeed) {
73
+ const scope = getScope(
74
+ decoratedFeed.extensions?.preference_editor_options?.scope
75
+ );
76
+
77
+ const behavior = {
78
+ ...feed.extensions?.["behavior"],
79
+ select_mode: "single",
80
+ current_selection: `@{${scope}/${key}}`,
81
+ };
82
+
83
+ if (!feed.extensions) {
84
+ decoratedFeed.extensions = {
85
+ behavior,
86
+ };
87
+ } else {
88
+ decoratedFeed.extensions.behavior = behavior;
89
+ }
90
+
91
+ const localStorageProducer = (entry: ZappFeed) => {
92
+ entry.id = entry.extensions?.tag || entry.id;
93
+ const { key: keyName, namespace } = getNamespaceAndKey(key);
94
+
95
+ const action =
96
+ decoratedFeed.extensions?.preference_editor_options?.scope === "session"
97
+ ? "sessionStorageSet"
98
+ : "localStorageSet";
99
+
100
+ return {
101
+ type: action,
102
+ options: {
103
+ content: {
104
+ [namespace]: {
105
+ [keyName]: entry.id,
106
+ },
107
+ },
108
+ },
109
+ };
110
+ };
111
+
112
+ const screenStateProducer = (entry: ZappFeed) => {
113
+ entry.id = entry.extensions?.tag || entry.id;
114
+
115
+ return {
116
+ type: "screenSetVariable",
117
+ options: {
118
+ key,
119
+ value: entry.id,
120
+ },
121
+ };
122
+ };
123
+
124
+ const producer =
125
+ decoratedFeed.extensions?.preference_editor_options?.scope === "screen"
126
+ ? screenStateProducer
127
+ : localStorageProducer;
128
+
129
+ decoratedFeed.entry?.forEach((entry: ZappEntry) => {
130
+ entry.type.value = "action";
131
+
132
+ entry.extensions = {
133
+ ...(entry.extensions || {}),
134
+ tap_actions: {
135
+ actions: [producer(entry)],
136
+ },
137
+ };
138
+ });
139
+
140
+ return decoratedFeed;
141
+ }
142
+
143
+ export const decorateFeed = (feed: ZappFeed) => {
144
+ if (!(feed.extensions?.["role"] === "preference_editor")) {
145
+ return feed;
146
+ }
147
+
148
+ const key = feed.extensions?.["preference_editor_options"]?.["key"];
149
+
150
+ if (!key) {
151
+ log_error(
152
+ `decorateFeed: No 'key' provided in feed titled "${feed.title}" with id "${feed.id}" in preference_editor_options. The preference_editor role requires a key for local storage.`
153
+ );
154
+
155
+ throw new Error(
156
+ "decorateFeed: No 'key' provided in feed preference_editor_options"
157
+ );
158
+ }
159
+
160
+ const decoratedFeed = R.clone(feed);
161
+
162
+ const isSingleSelect =
163
+ (feed.extensions?.["preference_editor_options"]?.select_mode ||
164
+ feed.extensions?.["behavior"]?.select_mode) === "single";
165
+
166
+ if (isSingleSelect) {
167
+ return makeSingleSelect(feed, key, decoratedFeed);
168
+ } else {
169
+ return makeMultiSelect(feed, key, decoratedFeed);
170
+ }
171
+ };
@@ -1,8 +1,11 @@
1
1
  import { useMemo } from "react";
2
2
  import { ScreenStateResolver } from "../appUtils/contextKeysManager/contextResolver";
3
+ import { useScreenStateStore } from "../reactHooks/navigation/useScreenStateStore";
4
+
5
+ export const useScreenResolvers = () => {
6
+ const screenStateStore = useScreenStateStore();
3
7
 
4
- export const useScreenResolvers = (screenRoute: string) => {
5
8
  return useMemo(() => {
6
- return { screen: new ScreenStateResolver(screenRoute) };
7
- }, [screenRoute]);
9
+ return { screen: new ScreenStateResolver(screenStateStore) };
10
+ }, [screenStateStore]);
8
11
  };
@@ -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-utils/storage/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
 
@@ -4,7 +4,7 @@ import { postAnalyticEvent } from "../manager";
4
4
  import { ANALYTICS_CORE_EVENTS } from "../events";
5
5
 
6
6
  type SendHeaderClickEventProps = {
7
- extraProps: ExtraProps;
7
+ extraProps: Record<string, any>;
8
8
  component?: ZappUIComponent;
9
9
  zappPipesData?: ZappPipesData;
10
10
  item?: ZappEntry;
@@ -1,10 +1,11 @@
1
+ /// <reference types="../../" />
1
2
  import { log_error, log_debug } from "../logger";
2
3
  import { replaceAnalyticsPropsNils } from "./helper";
3
4
  import { postAnalyticEvent } from "../manager";
4
5
 
5
6
  import { ANALYTICS_CORE_EVENTS } from "../events";
6
7
 
7
- declare type AnalyticsDefaultHelperProperties = {
8
+ type AnalyticsDefaultHelperProperties = {
8
9
  analyticsScreenData: AnalyticsScreenProperties;
9
10
  extraProps: any;
10
11
  props;
@@ -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,6 +3,9 @@ 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
 
8
11
  const mock_postAnalyticEvent = 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,4 +1,3 @@
1
- /// <reference types="@applicaster/zapp-react-native-utils" />
2
1
  import * as R from "ramda";
3
2
  import * as React from "react";
4
3
  import { isWeb } from "@applicaster/zapp-react-native-utils/reactUtils";
@@ -31,7 +30,7 @@ import { ANALYTICS_CORE_EVENTS } from "./events";
31
30
  import { noop } from "../functionUtils";
32
31
 
33
32
  type ComponentWithChildrenProps = {
34
- children: React.ReactChildren;
33
+ children: React.ReactElement;
35
34
  };
36
35
 
37
36
  export function sendSelectCellEvent(item, component, headerTitle, itemIndex) {
@@ -120,11 +119,11 @@ export function getAnalyticsFunctions({
120
119
  export const AnalyticsContext =
121
120
  React.createContext<GetAnalyticsFunctions>(noop);
122
121
 
123
- export function AnalyticsProvider(props: ComponentWithChildrenProps) {
122
+ export function AnalyticsProvider({ children }: ComponentWithChildrenProps) {
124
123
  return (
125
124
  // @ts-ignore - this is a valid context provider
126
125
  <AnalyticsContext.Provider value={getAnalyticsFunctions}>
127
- {props?.children}
126
+ {children}
128
127
  </AnalyticsContext.Provider>
129
128
  );
130
129
  }
@@ -1,7 +1,7 @@
1
1
  /* eslint-disable @typescript-eslint/no-use-before-define */
2
2
  import * as R from "ramda";
3
3
  import { NativeModules } from "react-native";
4
- import { ANALYTICS_CORE_EVENTS } from "@applicaster/zapp-react-native-utils/analyticsUtils/events";
4
+ import { ANALYTICS_CORE_EVENTS } from "./events";
5
5
 
6
6
  import { analyticsUtilsLogger } from "./logger";
7
7
 
@@ -105,7 +105,8 @@ export class PlayerAnalyticsTracker implements PlayerAnalyticsTrackerI {
105
105
  return this.getDateTimestamp();
106
106
  }
107
107
 
108
- this.mediaTime = this.playerState?.currentTime || eventData?.currentTime;
108
+ this.mediaTime =
109
+ this.playerState?.contentPosition || eventData?.currentTime;
109
110
 
110
111
  return this.mediaTime;
111
112
  }
@@ -65,10 +65,6 @@ export class Hook implements HookInterface {
65
65
  event: (typeof HOOKS_EVENTS)[keyof typeof HOOKS_EVENTS],
66
66
  ...args
67
67
  ) {
68
- if (this.state === hookState(HOOKS_EVENTS.CANCEL)) {
69
- return;
70
- }
71
-
72
68
  this.state = hookState(event);
73
69
  this.manager.subscriber.invokeHandler(event, ...args);
74
70
  }
@@ -198,4 +194,8 @@ export class Hook implements HookInterface {
198
194
  R.eqProps("weight", nextHook, this)
199
195
  );
200
196
  }
197
+
198
+ isCancelled(): boolean {
199
+ return this.state === hookState(HOOKS_EVENTS.CANCEL);
200
+ }
201
201
  }
@@ -255,7 +255,7 @@ export function HooksManager({
255
255
  * @param {Array<Hook>} restOfHooks to run
256
256
  * @returns {function} callback function
257
257
  */
258
- function hookCallback(hookPlugin, restOfHooks, initialPayload) {
258
+ function hookCallback(hookPlugin: Hook, restOfHooks: Hook[], initialPayload) {
259
259
  /**
260
260
  * callback invoked after a hook is executed
261
261
  * @param {object} options
@@ -273,6 +273,16 @@ export function HooksManager({
273
273
  }) {
274
274
  let callback = callbackArg;
275
275
 
276
+ if (hookPlugin.isCancelled()) {
277
+ logHookEvent(
278
+ hooksManagerLogger.info,
279
+ `hookCallback: hook was cancelled: ${hookPlugin["identifier"]}`,
280
+ {}
281
+ );
282
+
283
+ return;
284
+ }
285
+
276
286
  if (error) {
277
287
  logHookEvent(
278
288
  hooksManagerLogger.error,