@applicaster/zapp-react-native-utils 15.0.0-rc.12 → 15.0.0-rc.120

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 (159) hide show
  1. package/README.md +0 -6
  2. package/actionsExecutor/ActionExecutorContext.tsx +3 -6
  3. package/actionsExecutor/feedDecorator.ts +6 -6
  4. package/adsUtils/__tests__/createVMAP.test.ts +419 -0
  5. package/adsUtils/index.ts +2 -2
  6. package/analyticsUtils/README.md +1 -1
  7. package/analyticsUtils/analyticsMapper.ts +10 -2
  8. package/appDataUtils/__tests__/urlScheme.test.ts +678 -0
  9. package/appUtils/HooksManager/__tests__/__snapshots__/hooksManager.test.js.snap +0 -188
  10. package/appUtils/HooksManager/__tests__/hooksManager.test.js +16 -2
  11. package/appUtils/HooksManager/index.ts +10 -10
  12. package/appUtils/RiverFocusManager/{index.js → index.ts} +25 -18
  13. package/appUtils/accessibilityManager/__tests__/utils.test.ts +360 -0
  14. package/appUtils/accessibilityManager/const.ts +4 -0
  15. package/appUtils/accessibilityManager/hooks.ts +20 -13
  16. package/appUtils/accessibilityManager/index.ts +28 -1
  17. package/appUtils/accessibilityManager/utils.ts +59 -8
  18. package/appUtils/contextKeysManager/__tests__/getKeys/failure.test.ts +7 -2
  19. package/appUtils/contextKeysManager/__tests__/getKeys/success.test.ts +48 -0
  20. package/appUtils/contextKeysManager/contextResolver.ts +51 -22
  21. package/appUtils/contextKeysManager/index.ts +65 -10
  22. package/appUtils/focusManager/__tests__/__snapshots__/focusManager.test.js.snap +4 -0
  23. package/appUtils/focusManager/index.ios.ts +59 -3
  24. package/appUtils/focusManager/treeDataStructure/Tree/__tests__/Tree.test.js +46 -0
  25. package/appUtils/focusManager/treeDataStructure/Tree/index.js +18 -18
  26. package/appUtils/focusManagerAux/utils/index.ios.ts +122 -0
  27. package/appUtils/focusManagerAux/utils/index.ts +21 -5
  28. package/appUtils/focusManagerAux/utils/utils.ios.ts +234 -0
  29. package/appUtils/keyCodes/keys/keys.web.ts +1 -4
  30. package/appUtils/localizationsHelper.ts +4 -0
  31. package/appUtils/orientationHelper.ts +2 -4
  32. package/appUtils/platform/platformUtils.ts +117 -18
  33. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +94 -4
  34. package/appUtils/playerManager/OverlayObserver/utils.ts +32 -20
  35. package/appUtils/playerManager/player.ts +4 -0
  36. package/appUtils/playerManager/playerNative.ts +31 -17
  37. package/appUtils/playerManager/usePlayerState.tsx +14 -2
  38. package/arrayUtils/__tests__/allTruthy.test.ts +24 -0
  39. package/arrayUtils/__tests__/anyThruthy.test.ts +24 -0
  40. package/arrayUtils/index.ts +5 -0
  41. package/cellUtils/index.ts +32 -0
  42. package/cloudEventsUtils/__tests__/index.test.ts +529 -0
  43. package/cloudEventsUtils/index.ts +65 -1
  44. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +38 -0
  45. package/configurationUtils/__tests__/manifestKeyParser.test.ts +26 -26
  46. package/configurationUtils/index.ts +17 -11
  47. package/dateUtils/__tests__/dayjs.test.ts +330 -0
  48. package/enumUtils/__tests__/getEnumKeyByEnumValue.test.ts +207 -0
  49. package/errorUtils/__tests__/GeneralError.test.ts +97 -0
  50. package/errorUtils/__tests__/HttpStatusCode.test.ts +344 -0
  51. package/errorUtils/__tests__/MissingPluginError.test.ts +113 -0
  52. package/errorUtils/__tests__/NetworkError.test.ts +202 -0
  53. package/errorUtils/__tests__/getParsedResponse.test.ts +188 -0
  54. package/errorUtils/__tests__/invariant.test.ts +112 -0
  55. package/focusManager/aux/index.ts +1 -1
  56. package/headersUtils/__tests__/headersUtils.test.js +11 -1
  57. package/headersUtils/index.ts +2 -1
  58. package/manifestUtils/defaultManifestConfigurations/player.js +115 -11
  59. package/manifestUtils/keys.js +21 -0
  60. package/manifestUtils/platformIsTV.js +13 -0
  61. package/manifestUtils/sharedConfiguration/screenPicker/utils.js +1 -0
  62. package/manifestUtils/tvAction/container/index.js +1 -1
  63. package/navigationUtils/index.ts +15 -5
  64. package/numberUtils/__tests__/toNumber.test.ts +27 -0
  65. package/numberUtils/__tests__/toPositiveNumber.test.ts +193 -0
  66. package/numberUtils/index.ts +23 -1
  67. package/package.json +4 -4
  68. package/playerUtils/usePlayerTTS.ts +8 -3
  69. package/pluginUtils/index.ts +4 -0
  70. package/reactHooks/advertising/index.ts +2 -2
  71. package/reactHooks/analytics/__tests__/useSendAnalyticsOnPress.test.ts +537 -0
  72. package/reactHooks/app/__tests__/useAppState.test.ts +1 -1
  73. package/reactHooks/autoscrolling/__tests__/useTrackCurrentAutoScrollingElement.test.ts +1 -1
  74. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +1 -2
  75. package/reactHooks/cell-click/__tests__/index.test.js +1 -3
  76. package/reactHooks/configuration/__tests__/index.test.tsx +1 -1
  77. package/reactHooks/connection/__tests__/index.test.js +1 -1
  78. package/reactHooks/debugging/__tests__/index.test.js +4 -4
  79. package/reactHooks/dev/__tests__/useReRenderLog.test.ts +188 -0
  80. package/reactHooks/device/useIsTablet.tsx +14 -19
  81. package/reactHooks/device/useMemoizedIsTablet.ts +3 -3
  82. package/reactHooks/events/index.ts +20 -0
  83. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +32 -23
  84. package/reactHooks/feed/__tests__/useBuildPipesUrl.test.tsx +19 -19
  85. package/reactHooks/feed/__tests__/useEntryScreenId.test.tsx +4 -1
  86. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +42 -30
  87. package/reactHooks/feed/__tests__/{useInflatedUrl.test.ts → useInflatedUrl.test.tsx} +62 -7
  88. package/reactHooks/feed/index.ts +0 -2
  89. package/reactHooks/feed/useBatchLoading.ts +7 -1
  90. package/reactHooks/feed/useEntryScreenId.ts +2 -2
  91. package/reactHooks/feed/useInflatedUrl.ts +44 -18
  92. package/reactHooks/feed/usePipesCacheReset.ts +3 -1
  93. package/reactHooks/flatList/useLoadNextPageIfNeeded.ts +13 -16
  94. package/reactHooks/hookModal/hooks/useHookModalScreenData.ts +12 -8
  95. package/reactHooks/index.ts +2 -0
  96. package/reactHooks/layout/__tests__/index.test.tsx +1 -1
  97. package/reactHooks/layout/__tests__/useLayoutVersion.test.tsx +1 -1
  98. package/reactHooks/layout/index.ts +1 -1
  99. package/reactHooks/layout/useDimensions/__tests__/{useDimensions.test.ts → useDimensions.test.tsx} +105 -25
  100. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -2
  101. package/reactHooks/navigation/__tests__/index.test.tsx +40 -9
  102. package/reactHooks/navigation/index.ts +27 -11
  103. package/reactHooks/navigation/useRoute.ts +11 -7
  104. package/reactHooks/player/TVSeekControlller/TVSeekController.ts +27 -10
  105. package/reactHooks/player/__tests__/useAutoSeek._test.tsx +1 -1
  106. package/reactHooks/player/__tests__/useTapSeek._test.ts +1 -1
  107. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +1 -1
  108. package/reactHooks/resolvers/__tests__/useComponentResolver.test.tsx +1 -1
  109. package/reactHooks/resolvers/useCellResolver.ts +6 -2
  110. package/reactHooks/resolvers/useComponentResolver.ts +19 -3
  111. package/reactHooks/screen/__tests__/useCurrentScreenData.test.tsx +2 -2
  112. package/reactHooks/screen/__tests__/useScreenBackgroundColor.test.tsx +1 -1
  113. package/reactHooks/screen/__tests__/useScreenData.test.tsx +1 -1
  114. package/reactHooks/screen/__tests__/useTargetScreenData.test.tsx +12 -4
  115. package/reactHooks/screen/useTargetScreenData.ts +4 -2
  116. package/reactHooks/state/__tests__/useComponentScreenState.test.ts +246 -0
  117. package/reactHooks/state/index.ts +2 -0
  118. package/reactHooks/state/useComponentScreenState.ts +45 -0
  119. package/reactHooks/state/useRefWithInitialValue.ts +10 -0
  120. package/reactHooks/state/useRivers.ts +1 -1
  121. package/reactHooks/ui/__tests__/useFadeOutWhenBlurred.test.ts +580 -0
  122. package/reactHooks/usePluginConfiguration.ts +2 -2
  123. package/reactHooks/utils/__tests__/index.test.js +1 -1
  124. package/rectUtils/__tests__/index.test.ts +549 -0
  125. package/rectUtils/index.ts +2 -2
  126. package/refreshUtils/RefreshCoordinator/__tests__/refreshCoordinator.test.ts +161 -0
  127. package/refreshUtils/RefreshCoordinator/index.ts +216 -0
  128. package/refreshUtils/RefreshCoordinator/utils/__tests__/getDataRefreshConfig.test.ts +104 -0
  129. package/refreshUtils/RefreshCoordinator/utils/index.ts +29 -0
  130. package/screenPickerUtils/__tests__/index.test.ts +333 -0
  131. package/screenPickerUtils/index.ts +5 -0
  132. package/screenState/__tests__/index.test.ts +1 -1
  133. package/screenUtils/index.ts +3 -0
  134. package/searchUtils/const.ts +7 -0
  135. package/searchUtils/index.ts +3 -0
  136. package/services/storageServiceSync.web.ts +1 -1
  137. package/stringUtils/index.ts +1 -1
  138. package/testUtils/index.tsx +30 -21
  139. package/time/__tests__/BackgroundTimer.test.ts +156 -0
  140. package/time/__tests__/Timer.test.ts +236 -0
  141. package/typeGuards/__tests__/isString.test.ts +21 -0
  142. package/typeGuards/index.ts +4 -0
  143. package/utils/__tests__/clone.test.ts +158 -0
  144. package/utils/__tests__/mapAccum.test.ts +73 -0
  145. package/utils/__tests__/mergeRight.test.ts +48 -0
  146. package/utils/__tests__/path.test.ts +7 -0
  147. package/utils/__tests__/selectors.test.ts +124 -0
  148. package/utils/clone.ts +7 -0
  149. package/utils/index.ts +22 -1
  150. package/utils/mapAccum.ts +23 -0
  151. package/utils/mergeRight.ts +5 -0
  152. package/utils/path.ts +6 -3
  153. package/utils/pathOr.ts +5 -1
  154. package/utils/selectors.ts +46 -0
  155. package/zappFrameworkUtils/HookCallback/callbackNavigationAction.ts +49 -12
  156. package/zappFrameworkUtils/HookCallback/hookCallbackManifestExtensions.config.js +1 -1
  157. package/reactHooks/componentsMap/index.ts +0 -55
  158. package/reactHooks/feed/__tests__/useFeedRefresh.test.tsx +0 -75
  159. package/reactHooks/feed/useFeedRefresh.tsx +0 -65
package/README.md CHANGED
@@ -245,12 +245,6 @@ const connectionType = useConnectionInfo(true);
245
245
 
246
246
  `@applicaster/zapp-react-native/reactHooks`
247
247
 
248
- - `useFeedRefresh: ({ reloadData: function, component: { id: boolean | string, rules: {enable_data_refreshing: boolean, refreshing_interval: number} } }) => void` - Hook will call `reloadData` function, in the specified intervals if `enable_data_refreshing` is set to true;
249
-
250
- ```javascript
251
- useFeedRefresh({ reloadData, component });
252
- ```
253
-
254
248
  - `useFeedLoader: ({ feedUrl: string, pipesOptions?: { clearCache?: boolean, loadLocalFavourites?: boolean, silentRefresh?: boolean} }) => ({data: ?ApplicasterFeed, loading: boolean, url: string, error: Error,reloadData: (silentRefresh?: boolean) => void, loadNext: () => void})` - Hook will load data to the redux store and return a feed for the provided DSP URL. If the data for the provided url was already loaded, it will return that value
255
249
 
256
250
  ```javascript
@@ -23,12 +23,9 @@ import {
23
23
  EntryResolver,
24
24
  resolveObjectValues,
25
25
  } from "../appUtils/contextKeysManager/contextResolver";
26
- import { useNavigation } from "../reactHooks";
26
+ import { useNavigation, useRivers } from "../reactHooks";
27
27
 
28
- import {
29
- useContentTypes,
30
- usePickFromState,
31
- } from "@applicaster/zapp-react-native-redux/hooks";
28
+ import { useContentTypes } from "@applicaster/zapp-react-native-redux/hooks";
32
29
  import { useSubscriberFor } from "../reactHooks/useSubscriberFor";
33
30
  import { APP_EVENTS } from "../appUtils/events";
34
31
  import {
@@ -278,7 +275,7 @@ export function withActionExecutor(Component) {
278
275
 
279
276
  return function ActionExecutorComponent(props: Props) {
280
277
  const navigator = useNavigation();
281
- const { rivers } = usePickFromState(["rivers"]);
278
+ const rivers = useRivers();
282
279
  const contentTypes = useContentTypes();
283
280
 
284
281
  const handlers = useMemo(() => {
@@ -27,7 +27,7 @@ function makeMultiSelect(feed: ZappFeed, key, decoratedFeed) {
27
27
  );
28
28
 
29
29
  const behavior = {
30
- ...feed.extensions?.["behavior"],
30
+ ...feed.extensions?.behavior,
31
31
  select_mode: "multi",
32
32
  current_selection: `@{${scope}/${key}}`,
33
33
  };
@@ -75,7 +75,7 @@ function makeSingleSelect(feed: ZappFeed, key, decoratedFeed) {
75
75
  );
76
76
 
77
77
  const behavior = {
78
- ...feed.extensions?.["behavior"],
78
+ ...feed.extensions?.behavior,
79
79
  select_mode: "single",
80
80
  current_selection: `@{${scope}/${key}}`,
81
81
  };
@@ -141,11 +141,11 @@ function makeSingleSelect(feed: ZappFeed, key, decoratedFeed) {
141
141
  }
142
142
 
143
143
  export const decorateFeed = (feed: ZappFeed) => {
144
- if (!(feed.extensions?.["role"] === "preference_editor")) {
144
+ if (!(feed.extensions?.role === "preference_editor")) {
145
145
  return feed;
146
146
  }
147
147
 
148
- const key = feed.extensions?.["preference_editor_options"]?.["key"];
148
+ const key = feed.extensions?.preference_editor_options?.key;
149
149
 
150
150
  if (!key) {
151
151
  log_error(
@@ -160,8 +160,8 @@ export const decorateFeed = (feed: ZappFeed) => {
160
160
  const decoratedFeed = R.clone(feed);
161
161
 
162
162
  const isSingleSelect =
163
- (feed.extensions?.["preference_editor_options"]?.select_mode ||
164
- feed.extensions?.["behavior"]?.select_mode) === "single";
163
+ (feed.extensions?.preference_editor_options?.select_mode ||
164
+ feed.extensions?.behavior?.select_mode) === "single";
165
165
 
166
166
  if (isSingleSelect) {
167
167
  return makeSingleSelect(feed, key, decoratedFeed);
@@ -0,0 +1,419 @@
1
+ import { createVMAP } from "../index";
2
+
3
+ describe("createVMAP", () => {
4
+ describe("basic functionality", () => {
5
+ it("should create VMAP XML with single ad", () => {
6
+ const ads = [
7
+ {
8
+ offset: 0,
9
+ ad_url: "https://example.com/ad1.xml",
10
+ },
11
+ ];
12
+
13
+ const result = createVMAP(ads);
14
+
15
+ expect(result).toContain(
16
+ '<vmap:VMAP xmlns:vmap="http://www.iab.net/videosuite/vmap" version="1.0">'
17
+ );
18
+
19
+ expect(result).toContain("</vmap:VMAP>");
20
+ expect(result).toContain('timeOffset="00:00:00.000"');
21
+ expect(result).toContain("https://example.com/ad1.xml");
22
+ });
23
+
24
+ it("should create VMAP XML with multiple ads", () => {
25
+ const ads = [
26
+ {
27
+ offset: 0,
28
+ ad_url: "https://example.com/preroll.xml",
29
+ },
30
+ {
31
+ offset: 300,
32
+ ad_url: "https://example.com/midroll.xml",
33
+ },
34
+ {
35
+ offset: "postroll",
36
+ ad_url: "https://example.com/postroll.xml",
37
+ },
38
+ ];
39
+
40
+ const result = createVMAP(ads);
41
+
42
+ expect(result).toContain('timeOffset="00:00:00.000"');
43
+ expect(result).toContain('timeOffset="00:05:00.000"');
44
+ expect(result).toContain('timeOffset="end"');
45
+ expect(result).toContain("https://example.com/preroll.xml");
46
+ expect(result).toContain("https://example.com/midroll.xml");
47
+ expect(result).toContain("https://example.com/postroll.xml");
48
+ });
49
+
50
+ it("should create empty VMAP with no ads", () => {
51
+ const ads: Array<{ [key: string]: any }> = [];
52
+
53
+ const result = createVMAP(ads);
54
+
55
+ expect(result).toBe(
56
+ '<vmap:VMAP xmlns:vmap="http://www.iab.net/videosuite/vmap" version="1.0">\n</vmap:VMAP>'
57
+ );
58
+ });
59
+ });
60
+
61
+ describe("time offset conversion", () => {
62
+ it("should convert 0 seconds to 00:00:00.000", () => {
63
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
64
+ const result = createVMAP(ads);
65
+
66
+ expect(result).toContain('timeOffset="00:00:00.000"');
67
+ });
68
+
69
+ it("should convert seconds to HH:MM:SS.000 format", () => {
70
+ const ads = [
71
+ { offset: 60, ad_url: "https://example.com/ad.xml" }, // 1 minute
72
+ ];
73
+
74
+ const result = createVMAP(ads);
75
+
76
+ expect(result).toContain('timeOffset="00:01:00.000"');
77
+ });
78
+
79
+ it("should convert minutes to HH:MM:SS.000 format", () => {
80
+ const ads = [
81
+ { offset: 300, ad_url: "https://example.com/ad.xml" }, // 5 minutes
82
+ ];
83
+
84
+ const result = createVMAP(ads);
85
+
86
+ expect(result).toContain('timeOffset="00:05:00.000"');
87
+ });
88
+
89
+ it("should convert hours to HH:MM:SS.000 format", () => {
90
+ const ads = [
91
+ { offset: 3661, ad_url: "https://example.com/ad.xml" }, // 1:01:01
92
+ ];
93
+
94
+ const result = createVMAP(ads);
95
+
96
+ expect(result).toContain('timeOffset="01:01:01.000"');
97
+ });
98
+
99
+ it("should handle large time offsets", () => {
100
+ const ads = [
101
+ { offset: 36000, ad_url: "https://example.com/ad.xml" }, // 10 hours
102
+ ];
103
+
104
+ const result = createVMAP(ads);
105
+
106
+ expect(result).toContain('timeOffset="10:00:00.000"');
107
+ });
108
+
109
+ it("should convert postroll string to end", () => {
110
+ const ads = [
111
+ { offset: "postroll", ad_url: "https://example.com/ad.xml" },
112
+ ];
113
+
114
+ const result = createVMAP(ads);
115
+
116
+ expect(result).toContain('timeOffset="end"');
117
+ });
118
+
119
+ it("should convert numeric string to HH:MM:SS.000", () => {
120
+ const ads = [
121
+ { offset: "120", ad_url: "https://example.com/ad.xml" }, // 2 minutes
122
+ ];
123
+
124
+ const result = createVMAP(ads);
125
+
126
+ expect(result).toContain('timeOffset="00:02:00.000"');
127
+ });
128
+
129
+ it("should keep non-numeric string as-is", () => {
130
+ const ads = [{ offset: "start", ad_url: "https://example.com/ad.xml" }];
131
+
132
+ const result = createVMAP(ads);
133
+
134
+ expect(result).toContain('timeOffset="start"');
135
+ });
136
+
137
+ it("should handle fractional seconds", () => {
138
+ const ads = [{ offset: "90.5", ad_url: "https://example.com/ad.xml" }];
139
+
140
+ const result = createVMAP(ads);
141
+
142
+ // parseFloat("90.5") = 90.5, seconds = 90.5 % 60 = 30.5
143
+ // Note: The toHHMMSS function doesn't floor seconds, so fractional seconds appear as-is
144
+ expect(result).toContain('timeOffset="00:01:30.5.000"');
145
+ });
146
+
147
+ it("should pad single digit hours, minutes, and seconds", () => {
148
+ const ads = [
149
+ { offset: 3661, ad_url: "https://example.com/ad.xml" }, // 1:01:01
150
+ ];
151
+
152
+ const result = createVMAP(ads);
153
+
154
+ expect(result).toContain('timeOffset="01:01:01.000"');
155
+ });
156
+
157
+ it("should handle offset 0 from various types", () => {
158
+ const testCases = [
159
+ { offset: 0, expected: "00:00:00.000" },
160
+ { offset: "0", expected: "00:00:00.000" },
161
+ // Note: null and undefined will cause errors when calling offset.toString()
162
+ // so they are not valid inputs for createVMAP
163
+ ];
164
+
165
+ testCases.forEach(({ offset, expected }) => {
166
+ const ads = [{ offset, ad_url: "https://example.com/ad.xml" }];
167
+ const result = createVMAP(ads);
168
+ expect(result).toContain(`timeOffset="${expected}"`);
169
+ });
170
+ });
171
+ });
172
+
173
+ describe("XML structure", () => {
174
+ it("should include proper VMAP namespace", () => {
175
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
176
+ const result = createVMAP(ads);
177
+
178
+ expect(result).toContain(
179
+ 'xmlns:vmap="http://www.iab.net/videosuite/vmap"'
180
+ );
181
+
182
+ expect(result).toContain('version="1.0"');
183
+ });
184
+
185
+ it("should create AdBreak with breakType linear", () => {
186
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
187
+ const result = createVMAP(ads);
188
+
189
+ expect(result).toContain('breakType="linear"');
190
+ });
191
+
192
+ it("should create AdBreak with unique breakId based on offset", () => {
193
+ const ads = [
194
+ { offset: 0, ad_url: "https://example.com/ad1.xml" },
195
+ { offset: 300, ad_url: "https://example.com/ad2.xml" },
196
+ ];
197
+
198
+ const result = createVMAP(ads);
199
+
200
+ expect(result).toContain('breakId="break-0"');
201
+ expect(result).toContain('breakId="break-300"');
202
+ });
203
+
204
+ it("should create AdSource with proper attributes", () => {
205
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
206
+ const result = createVMAP(ads);
207
+
208
+ expect(result).toContain('allowMultipleAds="true"');
209
+ expect(result).toContain('followRedirects="true"');
210
+ });
211
+
212
+ it("should use VAST3 template type", () => {
213
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
214
+ const result = createVMAP(ads);
215
+
216
+ expect(result).toContain('templateType="vast3"');
217
+ });
218
+
219
+ it("should wrap URL in CDATA", () => {
220
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
221
+ const result = createVMAP(ads);
222
+
223
+ expect(result).toContain("<![CDATA[ https://example.com/ad.xml ]]>");
224
+ });
225
+
226
+ it("should create unique ad IDs based on offset", () => {
227
+ const ads = [
228
+ { offset: 0, ad_url: "https://example.com/ad1.xml" },
229
+ { offset: 100, ad_url: "https://example.com/ad2.xml" },
230
+ ];
231
+
232
+ const result = createVMAP(ads);
233
+
234
+ expect(result).toContain('id="ad-0-ad-1"');
235
+ expect(result).toContain('id="ad-100-ad-1"');
236
+ });
237
+ });
238
+
239
+ describe("URL handling", () => {
240
+ it("should trim whitespace from URLs", () => {
241
+ const ads = [{ offset: 0, ad_url: " https://example.com/ad.xml " }];
242
+
243
+ const result = createVMAP(ads);
244
+
245
+ expect(result).toContain("<![CDATA[ https://example.com/ad.xml ]]>");
246
+ });
247
+
248
+ it("should handle URLs with query parameters", () => {
249
+ const ads = [
250
+ {
251
+ offset: 0,
252
+ ad_url: "https://example.com/ad.xml?param1=value1&param2=value2",
253
+ },
254
+ ];
255
+
256
+ const result = createVMAP(ads);
257
+
258
+ expect(result).toContain(
259
+ "https://example.com/ad.xml?param1=value1&param2=value2"
260
+ );
261
+ });
262
+
263
+ it("should handle URLs with special characters", () => {
264
+ const ads = [
265
+ {
266
+ offset: 0,
267
+ ad_url: "https://example.com/ad.xml?url=https%3A%2F%2Ftest.com",
268
+ },
269
+ ];
270
+
271
+ const result = createVMAP(ads);
272
+
273
+ expect(result).toContain(
274
+ "https://example.com/ad.xml?url=https%3A%2F%2Ftest.com"
275
+ );
276
+ });
277
+
278
+ it("should handle URLs with ampersands", () => {
279
+ const ads = [
280
+ { offset: 0, ad_url: "https://example.com/ad.xml?a=1&b=2&c=3" },
281
+ ];
282
+
283
+ const result = createVMAP(ads);
284
+
285
+ expect(result).toContain(
286
+ "<![CDATA[ https://example.com/ad.xml?a=1&b=2&c=3 ]]>"
287
+ );
288
+ });
289
+ });
290
+
291
+ describe("complex scenarios", () => {
292
+ it("should create VMAP with preroll, midroll, and postroll", () => {
293
+ const ads = [
294
+ { offset: 0, ad_url: "https://example.com/preroll.xml" },
295
+ { offset: 300, ad_url: "https://example.com/midroll1.xml" },
296
+ { offset: 600, ad_url: "https://example.com/midroll2.xml" },
297
+ { offset: "postroll", ad_url: "https://example.com/postroll.xml" },
298
+ ];
299
+
300
+ const result = createVMAP(ads);
301
+
302
+ expect(result).toContain('timeOffset="00:00:00.000"');
303
+ expect(result).toContain('timeOffset="00:05:00.000"');
304
+ expect(result).toContain('timeOffset="00:10:00.000"');
305
+ expect(result).toContain('timeOffset="end"');
306
+ });
307
+
308
+ it("should handle ads in any order", () => {
309
+ const ads = [
310
+ { offset: "postroll", ad_url: "https://example.com/postroll.xml" },
311
+ { offset: 0, ad_url: "https://example.com/preroll.xml" },
312
+ { offset: 300, ad_url: "https://example.com/midroll.xml" },
313
+ ];
314
+
315
+ const result = createVMAP(ads);
316
+
317
+ // Should preserve order from input array
318
+ const postrollIndex = result.indexOf('timeOffset="end"');
319
+ const prerollIndex = result.indexOf('timeOffset="00:00:00.000"');
320
+ const midrollIndex = result.indexOf('timeOffset="00:05:00.000"');
321
+
322
+ expect(postrollIndex).toBeLessThan(prerollIndex);
323
+ expect(prerollIndex).toBeLessThan(midrollIndex);
324
+ });
325
+
326
+ it("should handle very long video with multiple ads", () => {
327
+ const ads = Array.from({ length: 10 }, (_, i) => ({
328
+ offset: i * 600, // Every 10 minutes
329
+ ad_url: `https://example.com/ad${i}.xml`,
330
+ }));
331
+
332
+ const result = createVMAP(ads);
333
+
334
+ expect(result).toContain('timeOffset="00:00:00.000"'); // 0 min
335
+ expect(result).toContain('timeOffset="00:10:00.000"'); // 10 min
336
+ expect(result).toContain('timeOffset="01:30:00.000"'); // 90 min
337
+ });
338
+
339
+ it("should create valid XML structure", () => {
340
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml" }];
341
+
342
+ const result = createVMAP(ads);
343
+
344
+ // Check for proper XML nesting
345
+ expect(result.split("<vmap:VMAP").length - 1).toBe(1);
346
+ expect(result.split("</vmap:VMAP>").length - 1).toBe(1);
347
+ expect(result.split("<vmap:AdBreak").length - 1).toBe(1);
348
+ expect(result.split("</vmap:AdBreak>").length - 1).toBe(1);
349
+ expect(result.split("<vmap:AdSource").length - 1).toBe(1);
350
+ expect(result.split("</vmap:AdSource>").length - 1).toBe(1);
351
+ });
352
+ });
353
+
354
+ describe("edge cases", () => {
355
+ it("should preserve negative offset in breakId and timeOffset", () => {
356
+ const ads = [{ offset: -100, ad_url: "https://example.com/ad.xml" }];
357
+
358
+ const result = createVMAP(ads);
359
+
360
+ expect(result).toContain('breakId="break--100"');
361
+ expect(result).toContain('timeOffset="-1:-2:-40.000"');
362
+ });
363
+
364
+ it("should handle very large offset", () => {
365
+ const ads = [{ offset: 999999, ad_url: "https://example.com/ad.xml" }];
366
+
367
+ const result = createVMAP(ads);
368
+
369
+ expect(result).toContain('timeOffset="277:46:39.000"');
370
+ });
371
+
372
+ it("should handle empty string offset", () => {
373
+ const ads = [{ offset: "", ad_url: "https://example.com/ad.xml" }];
374
+
375
+ const result = createVMAP(ads);
376
+
377
+ // Empty string is not a valid number, so it should be kept as-is or converted to 0
378
+ expect(result).toContain("timeOffset=");
379
+ });
380
+
381
+ it("should handle mixed offset types", () => {
382
+ const ads = [
383
+ { offset: 0, ad_url: "https://example.com/ad1.xml" },
384
+ { offset: "60", ad_url: "https://example.com/ad2.xml" },
385
+ { offset: 120, ad_url: "https://example.com/ad3.xml" },
386
+ { offset: "postroll", ad_url: "https://example.com/ad4.xml" },
387
+ ];
388
+
389
+ const result = createVMAP(ads);
390
+
391
+ expect(result).toContain('timeOffset="00:00:00.000"');
392
+ expect(result).toContain('timeOffset="00:01:00.000"');
393
+ expect(result).toContain('timeOffset="00:02:00.000"');
394
+ expect(result).toContain('timeOffset="end"');
395
+ });
396
+
397
+ it("should handle ads with same offset", () => {
398
+ const ads = [
399
+ { offset: 0, ad_url: "https://example.com/ad1.xml" },
400
+ { offset: 0, ad_url: "https://example.com/ad2.xml" },
401
+ ];
402
+
403
+ const result = createVMAP(ads);
404
+
405
+ // Both should have the same timeOffset but different ad IDs (though ID is also based on offset)
406
+ const matches = result.match(/timeOffset="00:00:00.000"/g);
407
+ expect(matches?.length).toBe(2);
408
+ });
409
+
410
+ it("should handle URL with newlines and tabs", () => {
411
+ const ads = [{ offset: 0, ad_url: "https://example.com/ad.xml\n\t" }];
412
+
413
+ const result = createVMAP(ads);
414
+
415
+ // trim() should remove the whitespace
416
+ expect(result).toContain("<![CDATA[ https://example.com/ad.xml ]]>");
417
+ });
418
+ });
419
+ });
package/adsUtils/index.ts CHANGED
@@ -33,10 +33,10 @@ function convertOffset(offset: any): string {
33
33
  }
34
34
 
35
35
  function createAdBreak(ad: AdMap): string {
36
- const offset = ad["offset"];
36
+ const offset = ad.offset;
37
37
  const id = offset.toString();
38
38
  const timestamp = convertOffset(offset);
39
- const url = ad["ad_url"].toString().trim();
39
+ const url = ad.ad_url.toString().trim();
40
40
 
41
41
  return `
42
42
  <vmap:AdBreak timeOffset="${timestamp}" breakType="linear" breakId="break-${id}">
@@ -388,7 +388,7 @@ export function AnalyticsProvider(props: ComponentWithChildrenProps) {
388
388
 
389
389
  ```ts
390
390
  export function useAnalytics(props: any): any {
391
- const { appData } = usePickFromState(["appData"]);
391
+ const appData = useAppData();
392
392
  const getAnalyticsFunctions = React.useContext(AnalyticsContext);
393
393
 
394
394
  const analyticsFunctions = React.useMemo(
@@ -63,10 +63,14 @@ class StateHolder {
63
63
  return value as string;
64
64
  }
65
65
 
66
+ public get analyticsCustomPropertiesConsumed(): boolean {
67
+ return this._cap !== undefined;
68
+ }
69
+
66
70
  private acp(prefix: string, suffix: string): string | null {
67
71
  if (this._cap === undefined) {
68
- this._cap = this.pluckedParams[prefix]
69
- ? JSON.parse(this.pluckedParams[prefix])
72
+ this._cap = this.event.params[prefix]
73
+ ? JSON.parse(this.event.params[prefix])
70
74
  : null;
71
75
 
72
76
  delete this.pluckedParams[prefix];
@@ -254,6 +258,10 @@ export class Mapper {
254
258
 
255
259
  let params = resultParams.size ? Object.fromEntries(resultParams) : {};
256
260
 
261
+ if (state.analyticsCustomPropertiesConsumed) {
262
+ delete params[sourceType.analyticsCustomProperties];
263
+ }
264
+
257
265
  // deal with the remaining params
258
266
  if (!isEmptyOrNil(state.pluckedParams)) {
259
267
  if (rule.strategy) {