@applicaster/zapp-react-native-utils 15.0.0-rc.1 → 15.0.0-rc.100

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 (145) hide show
  1. package/actionsExecutor/ActionExecutorContext.tsx +3 -6
  2. package/actionsExecutor/feedDecorator.ts +6 -6
  3. package/adsUtils/__tests__/createVMAP.test.ts +419 -0
  4. package/adsUtils/index.ts +2 -2
  5. package/analyticsUtils/AnalyticPlayerListener.ts +5 -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/focusManagerAux/utils/index.ios.ts +122 -0
  25. package/appUtils/focusManagerAux/utils/index.ts +19 -1
  26. package/appUtils/focusManagerAux/utils/utils.ios.ts +231 -0
  27. package/appUtils/keyCodes/keys/keys.web.ts +1 -4
  28. package/appUtils/orientationHelper.ts +2 -4
  29. package/appUtils/platform/platformUtils.ts +117 -18
  30. package/appUtils/playerManager/OverlayObserver/OverlaysObserver.ts +94 -4
  31. package/appUtils/playerManager/OverlayObserver/utils.ts +32 -20
  32. package/appUtils/playerManager/conts.ts +21 -0
  33. package/appUtils/playerManager/player.ts +4 -0
  34. package/appUtils/playerManager/playerNative.ts +29 -16
  35. package/appUtils/playerManager/usePlayerState.tsx +14 -2
  36. package/arrayUtils/__tests__/allTruthy.test.ts +24 -0
  37. package/arrayUtils/__tests__/anyThruthy.test.ts +24 -0
  38. package/arrayUtils/index.ts +5 -0
  39. package/cellUtils/index.ts +32 -0
  40. package/cloudEventsUtils/__tests__/index.test.ts +529 -0
  41. package/cloudEventsUtils/index.ts +65 -1
  42. package/configurationUtils/__tests__/imageSrcFromMediaItem.test.ts +38 -0
  43. package/configurationUtils/__tests__/manifestKeyParser.test.ts +26 -26
  44. package/configurationUtils/index.ts +17 -11
  45. package/dateUtils/__tests__/dayjs.test.ts +330 -0
  46. package/enumUtils/__tests__/getEnumKeyByEnumValue.test.ts +207 -0
  47. package/errorUtils/__tests__/GeneralError.test.ts +97 -0
  48. package/errorUtils/__tests__/HttpStatusCode.test.ts +344 -0
  49. package/errorUtils/__tests__/MissingPluginError.test.ts +113 -0
  50. package/errorUtils/__tests__/NetworkError.test.ts +202 -0
  51. package/errorUtils/__tests__/getParsedResponse.test.ts +188 -0
  52. package/errorUtils/__tests__/invariant.test.ts +112 -0
  53. package/focusManager/aux/index.ts +1 -1
  54. package/headersUtils/__tests__/headersUtils.test.js +11 -1
  55. package/headersUtils/index.ts +2 -1
  56. package/manifestUtils/defaultManifestConfigurations/player.js +125 -11
  57. package/manifestUtils/keys.js +21 -0
  58. package/manifestUtils/platformIsTV.js +13 -0
  59. package/manifestUtils/sharedConfiguration/screenPicker/utils.js +1 -0
  60. package/manifestUtils/tvAction/container/index.js +1 -1
  61. package/navigationUtils/index.ts +34 -21
  62. package/numberUtils/__tests__/toNumber.test.ts +12 -0
  63. package/numberUtils/__tests__/toPositiveNumber.test.ts +165 -0
  64. package/numberUtils/index.ts +19 -1
  65. package/package.json +4 -4
  66. package/playerUtils/usePlayerTTS.ts +8 -3
  67. package/pluginUtils/index.ts +4 -0
  68. package/reactHooks/advertising/index.ts +2 -2
  69. package/reactHooks/analytics/__tests__/useSendAnalyticsOnPress.test.ts +537 -0
  70. package/reactHooks/app/__tests__/useAppState.test.ts +1 -1
  71. package/reactHooks/autoscrolling/__tests__/useTrackCurrentAutoScrollingElement.test.ts +1 -1
  72. package/reactHooks/autoscrolling/__tests__/useTrackedView.test.tsx +1 -2
  73. package/reactHooks/cell-click/__tests__/index.test.js +1 -3
  74. package/reactHooks/configuration/__tests__/index.test.tsx +1 -1
  75. package/reactHooks/connection/__tests__/index.test.js +1 -1
  76. package/reactHooks/debugging/__tests__/index.test.js +4 -4
  77. package/reactHooks/dev/__tests__/useReRenderLog.test.ts +188 -0
  78. package/reactHooks/device/useIsTablet.tsx +14 -19
  79. package/reactHooks/device/useMemoizedIsTablet.ts +3 -3
  80. package/reactHooks/events/index.ts +20 -0
  81. package/reactHooks/feed/__tests__/useBatchLoading.test.tsx +32 -23
  82. package/reactHooks/feed/__tests__/useBuildPipesUrl.test.tsx +19 -19
  83. package/reactHooks/feed/__tests__/useEntryScreenId.test.tsx +4 -1
  84. package/reactHooks/feed/__tests__/useFeedLoader.test.tsx +42 -30
  85. package/reactHooks/feed/__tests__/useFeedRefresh.test.tsx +1 -1
  86. package/reactHooks/feed/__tests__/{useInflatedUrl.test.ts → useInflatedUrl.test.tsx} +62 -7
  87. package/reactHooks/feed/useBatchLoading.ts +7 -1
  88. package/reactHooks/feed/useEntryScreenId.ts +2 -2
  89. package/reactHooks/feed/useFeedLoader.tsx +0 -9
  90. package/reactHooks/feed/useInflatedUrl.ts +57 -37
  91. package/reactHooks/feed/usePipesCacheReset.ts +3 -1
  92. package/reactHooks/flatList/useLoadNextPageIfNeeded.ts +13 -16
  93. package/reactHooks/hookModal/hooks/useHookModalScreenData.ts +12 -8
  94. package/reactHooks/index.ts +2 -0
  95. package/reactHooks/layout/__tests__/index.test.tsx +1 -1
  96. package/reactHooks/layout/__tests__/useLayoutVersion.test.tsx +1 -1
  97. package/reactHooks/layout/index.ts +1 -1
  98. package/reactHooks/layout/useDimensions/__tests__/{useDimensions.test.ts → useDimensions.test.tsx} +105 -25
  99. package/reactHooks/layout/useDimensions/useDimensions.ts +2 -2
  100. package/reactHooks/navigation/__tests__/index.test.tsx +40 -9
  101. package/reactHooks/navigation/index.ts +27 -11
  102. package/reactHooks/navigation/useRoute.ts +11 -7
  103. package/reactHooks/player/TVSeekControlller/TVSeekController.ts +27 -10
  104. package/reactHooks/player/__tests__/useAutoSeek._test.tsx +1 -1
  105. package/reactHooks/player/__tests__/useTapSeek._test.ts +1 -1
  106. package/reactHooks/resolvers/__tests__/useCellResolver.test.tsx +1 -1
  107. package/reactHooks/resolvers/__tests__/useComponentResolver.test.tsx +1 -1
  108. package/reactHooks/resolvers/useCellResolver.ts +6 -2
  109. package/reactHooks/resolvers/useComponentResolver.ts +8 -2
  110. package/reactHooks/screen/__tests__/useCurrentScreenData.test.tsx +2 -2
  111. package/reactHooks/screen/__tests__/useScreenBackgroundColor.test.tsx +1 -1
  112. package/reactHooks/screen/__tests__/useScreenData.test.tsx +1 -1
  113. package/reactHooks/screen/__tests__/useTargetScreenData.test.tsx +12 -4
  114. package/reactHooks/screen/useTargetScreenData.ts +4 -2
  115. package/reactHooks/state/useRefWithInitialValue.ts +10 -0
  116. package/reactHooks/state/useRivers.ts +1 -1
  117. package/reactHooks/ui/__tests__/useFadeOutWhenBlurred.test.ts +580 -0
  118. package/reactHooks/usePluginConfiguration.ts +2 -2
  119. package/reactHooks/utils/__tests__/index.test.js +1 -1
  120. package/rectUtils/__tests__/index.test.ts +549 -0
  121. package/rectUtils/index.ts +2 -2
  122. package/screenPickerUtils/__tests__/index.test.ts +333 -0
  123. package/screenState/__tests__/index.test.ts +1 -1
  124. package/searchUtils/const.ts +7 -0
  125. package/searchUtils/index.ts +3 -0
  126. package/services/storageServiceSync.web.ts +1 -1
  127. package/stringUtils/index.ts +1 -1
  128. package/testUtils/index.tsx +30 -21
  129. package/time/__tests__/BackgroundTimer.test.ts +156 -0
  130. package/time/__tests__/Timer.test.ts +236 -0
  131. package/typeGuards/__tests__/isString.test.ts +21 -0
  132. package/typeGuards/index.ts +4 -0
  133. package/utils/__tests__/mapAccum.test.ts +73 -0
  134. package/utils/__tests__/mergeRight.test.ts +48 -0
  135. package/utils/__tests__/selectors.test.ts +124 -0
  136. package/utils/index.ts +20 -0
  137. package/utils/mapAccum.ts +23 -0
  138. package/utils/mergeRight.ts +5 -0
  139. package/utils/path.ts +6 -3
  140. package/utils/pathOr.ts +5 -1
  141. package/utils/selectors.ts +46 -0
  142. package/zappFrameworkUtils/HookCallback/callbackNavigationAction.ts +128 -38
  143. package/zappFrameworkUtils/HookCallback/hookCallbackManifestExtensions.config.js +26 -10
  144. package/zappFrameworkUtils/HookCallback/useCallbackActions.ts +6 -9
  145. package/reactHooks/componentsMap/index.ts +0 -55
@@ -2,6 +2,27 @@ export const userPreferencesNamespace = "user_preferences";
2
2
 
3
3
  export const skipActionType = "show_skip";
4
4
 
5
+ export class PlayerError
6
+ extends Error
7
+ implements QuickBrickPlayer.PlayerErrorI
8
+ {
9
+ description: string;
10
+
11
+ constructor(message: string, description: string) {
12
+ super(message);
13
+ this.description = description;
14
+
15
+ Object.setPrototypeOf(this, PlayerError.prototype);
16
+ }
17
+
18
+ toObject() {
19
+ return {
20
+ error: this.message,
21
+ message: this.description,
22
+ };
23
+ }
24
+ }
25
+
5
26
  export enum SharedPlayerCallBacksKeys {
6
27
  OnPlayerResume = "onPlayerResume",
7
28
  OnPlayerPause = "onPlayerPause",
@@ -250,6 +250,10 @@ export class Player {
250
250
  return false;
251
251
  }
252
252
 
253
+ if (!Number.isFinite(duration)) {
254
+ return this.getSeekableDuration() > 0;
255
+ }
256
+
253
257
  return duration > 0;
254
258
  };
255
259
 
@@ -15,6 +15,22 @@ const logger = createLogger({
15
15
 
16
16
  const { log_warning } = logger;
17
17
 
18
+ interface PlayerMethods {
19
+ seeking?: (position: number) => void;
20
+ forward?: (deltaTime: number) => void;
21
+ rewind?: (deltaTime: number) => void;
22
+ startSleepTimer?: (sleepTimestamp: number) => void;
23
+ selectTrack?: (
24
+ selected: QuickBrickPlayer.TextTrack | QuickBrickPlayer.AudioTrack
25
+ ) => void;
26
+ setPlaybackRate?: (rate: number) => void;
27
+ appStateChange?: (appState: string, previousAppState: string) => void;
28
+ closeNativePlayer?: () => void;
29
+ togglePlayPause?: () => void;
30
+ }
31
+
32
+ type PlayerComponent = ReactComponent<PlayerRefProps> & PlayerMethods;
33
+
18
34
  type PlayerPlugnId = "QuickBrickPlayerPlugin" | "KalturaPlayerPlugin";
19
35
 
20
36
  type Props = {
@@ -62,7 +78,7 @@ export class PlayerNative extends Player {
62
78
  return this.playerState.contentPosition;
63
79
  };
64
80
 
65
- currentPlayerComponent = (): ReactComponent<PlayerRefProps> =>
81
+ currentPlayerComponent = (): PlayerComponent | null =>
66
82
  this.playerComponent?.current;
67
83
 
68
84
  getState = () => this.playerState;
@@ -121,7 +137,7 @@ export class PlayerNative extends Player {
121
137
 
122
138
  this.playerState.seekPosition = newPosition;
123
139
 
124
- this.currentPlayerComponent()?.["seeking"](this.playerState.seekPosition);
140
+ this.currentPlayerComponent()?.seeking?.(this.playerState.seekPosition);
125
141
  }
126
142
 
127
143
  this.logState(PlayerModuleFuncNames.seekTo, { position });
@@ -135,7 +151,7 @@ export class PlayerNative extends Player {
135
151
  if (!this.invokeNativeFunction(PlayerModuleFuncNames.forward, deltaTime)) {
136
152
  // Kaltura does not have yet this implementation, use legacy code
137
153
 
138
- this.currentPlayerComponent()?.["forward"](deltaTime);
154
+ this.currentPlayerComponent()?.forward?.(deltaTime);
139
155
  }
140
156
 
141
157
  this.notifyPlayHeadPositionUpdate();
@@ -151,7 +167,7 @@ export class PlayerNative extends Player {
151
167
  if (!this.invokeNativeFunction(PlayerModuleFuncNames.rewind, deltaTime)) {
152
168
  // Kaltura does not have yet this implementation, use legacy code
153
169
 
154
- this.currentPlayerComponent()?.["rewind"](deltaTime);
170
+ this.currentPlayerComponent()?.rewind?.(deltaTime);
155
171
  }
156
172
 
157
173
  this.notifyPlayHeadPositionUpdate();
@@ -178,7 +194,7 @@ export class PlayerNative extends Player {
178
194
  sleepTimestamp
179
195
  )
180
196
  ) {
181
- this.currentPlayerComponent()?.["startSleepTimer"](sleepTimestamp);
197
+ this.currentPlayerComponent()?.startSleepTimer?.(sleepTimestamp);
182
198
  }
183
199
  };
184
200
 
@@ -195,11 +211,11 @@ export class PlayerNative extends Player {
195
211
  selectTrack = (
196
212
  selected: QuickBrickPlayer.TextTrack | QuickBrickPlayer.AudioTrack
197
213
  ) => {
198
- this.currentPlayerComponent()?.["selectTrack"]?.(selected);
214
+ this.currentPlayerComponent()?.selectTrack?.(selected);
199
215
  };
200
216
 
201
217
  setPlaybackRate = (rate: number) => {
202
- this.currentPlayerComponent()?.["setPlaybackRate"]?.(rate);
218
+ this.currentPlayerComponent()?.setPlaybackRate?.(rate);
203
219
  };
204
220
 
205
221
  getPluginConfiguration = () => {
@@ -209,19 +225,16 @@ export class PlayerNative extends Player {
209
225
  getProps = () => (this.currentPlayerComponent() as any)?.props;
210
226
 
211
227
  appStateChange = (appState, previousAppState) => {
212
- this.currentPlayerComponent()?.["appStateChange"]?.(
213
- appState,
214
- previousAppState
215
- );
228
+ this.currentPlayerComponent()?.appStateChange?.(appState, previousAppState);
216
229
  };
217
230
 
218
231
  closeNativePlayer = () => {
219
232
  // TODO: Delete does not work
220
- this.currentPlayerComponent()?.["closeNativePlayer"]?.();
233
+ this.currentPlayerComponent()?.closeNativePlayer?.();
221
234
  };
222
235
 
223
236
  togglePlayPause = () => {
224
- this.currentPlayerComponent()?.["togglePlayPause"]?.();
237
+ this.currentPlayerComponent()?.togglePlayPause?.();
225
238
  };
226
239
 
227
240
  onVideoLoad = (event) => {
@@ -289,19 +302,19 @@ export class PlayerNative extends Player {
289
302
  isFullScreenSupported = (): boolean => {
290
303
  const config = this.getConfig();
291
304
 
292
- const disableFullScreen = config?.["disable_fullscreen"];
305
+ const disableFullScreen = config?.disable_fullscreen;
293
306
 
294
307
  if (disableFullScreen) {
295
308
  return false;
296
309
  }
297
310
 
298
- const isFullScreenAudioPlayer = config?.["full_screen_audio_player"];
311
+ const isFullScreenAudioPlayer = config?.full_screen_audio_player;
299
312
 
300
313
  return !(isFullScreenAudioPlayer && this.isAudioItem());
301
314
  };
302
315
 
303
316
  supportsNativeControls = (): boolean =>
304
- toBooleanWithDefaultFalse(this.getConfig()?.["supports_native_controls"]);
317
+ toBooleanWithDefaultFalse(this.getConfig()?.supports_native_controls);
305
318
 
306
319
  supportNativeCast = (): boolean =>
307
320
  toBooleanWithDefaultFalse(
@@ -2,7 +2,7 @@ import * as React from "react";
2
2
  import { Player } from "./player";
3
3
  import { usePlayer } from "./usePlayer";
4
4
 
5
- type PlayerState = {
5
+ export type PlayerState = {
6
6
  currentTime: number;
7
7
  duration: number;
8
8
  seekableDuration: number;
@@ -10,6 +10,8 @@ type PlayerState = {
10
10
  isPaused: boolean;
11
11
  isBuffering: boolean;
12
12
  isReadyToPlay: boolean;
13
+ trackState?: QuickBrickPlayer.TracksState;
14
+ isAd: boolean;
13
15
  };
14
16
 
15
17
  export const usePlayerState = (
@@ -24,6 +26,8 @@ export const usePlayerState = (
24
26
  isPaused: null,
25
27
  isBuffering: false,
26
28
  isReadyToPlay: false,
29
+ trackState: null,
30
+ isAd: false,
27
31
  });
28
32
 
29
33
  const player: Player = usePlayer(playerId);
@@ -37,6 +41,8 @@ export const usePlayerState = (
37
41
  isPaused: player.isPaused(),
38
42
  isBuffering: player.isBuffering(),
39
43
  isReadyToPlay: player.isReadyToPlay(),
44
+ trackState: player.getTracksState(),
45
+ isAd: player.isAd(),
40
46
  });
41
47
  }, [player]);
42
48
 
@@ -54,10 +60,16 @@ export const usePlayerState = (
54
60
  onPlayerPause: onPlayerChangeState,
55
61
  onPlayerResume: onPlayerChangeState,
56
62
  onPlayerSeekComplete: onPlayerChangeState,
63
+ onTracksChanged: onPlayerChangeState,
64
+ onAdBreakBegin: onPlayerChangeState,
65
+ onAdBreakEnd: onPlayerChangeState,
66
+ onAdBegin: onPlayerChangeState,
67
+ onAdEnd: onPlayerChangeState,
68
+ onAdError: onPlayerChangeState,
57
69
  },
58
70
  });
59
71
  }
60
- }, [player]);
72
+ }, [listenerId, onPlayerChangeState, player]);
61
73
 
62
74
  return state;
63
75
  };
@@ -0,0 +1,24 @@
1
+ import { allTruthy } from "..";
2
+
3
+ describe("allTruthy", () => {
4
+ it("should return true when all values are true", () => {
5
+ expect(allTruthy([true, true, true])).toBe(true);
6
+ });
7
+
8
+ it("should return false when at least one value is false", () => {
9
+ expect(allTruthy([true, false, true])).toBe(false);
10
+ });
11
+
12
+ it("should return false when all values are false", () => {
13
+ expect(allTruthy([false, false, false])).toBe(false);
14
+ });
15
+
16
+ it("should return false for an empty array", () => {
17
+ expect(allTruthy([])).toBe(false);
18
+ });
19
+
20
+ it("should handle single-element arrays correctly", () => {
21
+ expect(allTruthy([true])).toBe(true);
22
+ expect(allTruthy([false])).toBe(false);
23
+ });
24
+ });
@@ -0,0 +1,24 @@
1
+ import { anyTruthy } from "..";
2
+
3
+ describe("anyTruthy", () => {
4
+ it("should return true when at least one value is true", () => {
5
+ expect(anyTruthy([false, true, false])).toBe(true);
6
+ });
7
+
8
+ it("should return false when all values are false", () => {
9
+ expect(anyTruthy([false, false, false])).toBe(false);
10
+ });
11
+
12
+ it("should return true when all values are true", () => {
13
+ expect(anyTruthy([true, true, true])).toBe(true);
14
+ });
15
+
16
+ it("should return false for an empty array", () => {
17
+ expect(anyTruthy([])).toBe(false);
18
+ });
19
+
20
+ it("should handle single-element arrays correctly", () => {
21
+ expect(anyTruthy([true])).toBe(true);
22
+ expect(anyTruthy([false])).toBe(false);
23
+ });
24
+ });
@@ -116,3 +116,8 @@ export const sample = (xs: unknown[]): unknown => {
116
116
 
117
117
  return xs[index];
118
118
  };
119
+
120
+ export const allTruthy = (xs: boolean[]) =>
121
+ isFilledArray(xs) && xs.every(Boolean);
122
+
123
+ export const anyTruthy = (xs: boolean[]) => xs.some(Boolean);
@@ -505,3 +505,35 @@ export const getImageContainerMarginStyles = ({ value }: { value: any }) => {
505
505
  marginRight: value("image_margin_right"),
506
506
  };
507
507
  };
508
+
509
+ export const isTextLabel = (key: string): boolean =>
510
+ key.includes("text_label_") && key.endsWith("_switch");
511
+
512
+ export const hasTextLabelsEnabled = (
513
+ configuration: Record<string, any>
514
+ ): boolean => {
515
+ const textLabelsKeys = Object.keys(configuration).filter(isTextLabel);
516
+
517
+ const picked = textLabelsKeys.reduce(
518
+ (acc, key) => {
519
+ acc[key] = configuration[key];
520
+
521
+ return acc;
522
+ },
523
+ {} as Record<string, any>
524
+ );
525
+
526
+ const pickedValues = Object.values(picked);
527
+
528
+ return pickedValues.some((value) => {
529
+ if (typeof value === "boolean") {
530
+ return value === true;
531
+ }
532
+
533
+ if (typeof value === "string") {
534
+ return value !== "" && value.toLowerCase() !== "false";
535
+ }
536
+
537
+ return Boolean(value);
538
+ });
539
+ };