@applicaster/quick-brick-player 15.0.0-rc.3 → 15.0.0-rc.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@applicaster/quick-brick-player",
3
- "version": "15.0.0-rc.3",
3
+ "version": "15.0.0-rc.30",
4
4
  "description": "Quick Brick Player",
5
5
  "main": "./src/index.ts",
6
6
  "types": "index.d.ts",
@@ -35,11 +35,11 @@
35
35
  },
36
36
  "homepage": "https://github.com/applicaster/quickbrick#readme",
37
37
  "dependencies": {
38
- "@applicaster/quick-brick-mobile-transport-controls": "14.0.0-rc.56",
39
- "@applicaster/quick-brick-tv-transport-controls": "14.0.0-rc.60",
40
- "@applicaster/zapp-react-native-tvos-app": "15.0.0-rc.3",
41
- "@applicaster/zapp-react-native-ui-components": "15.0.0-rc.3",
42
- "@applicaster/zapp-react-native-utils": "15.0.0-rc.3",
38
+ "@applicaster/quick-brick-mobile-transport-controls": "15.0.0-rc.27",
39
+ "@applicaster/quick-brick-tv-transport-controls": "15.0.0-rc.15",
40
+ "@applicaster/zapp-react-native-tvos-app": "15.0.0-rc.30",
41
+ "@applicaster/zapp-react-native-ui-components": "15.0.0-rc.30",
42
+ "@applicaster/zapp-react-native-utils": "15.0.0-rc.30",
43
43
  "query-string": "7.1.3",
44
44
  "shaka-player": "4.3.5",
45
45
  "typeface-montserrat": "^0.0.54",
@@ -61,6 +61,7 @@ type Props = {
61
61
  inline: boolean;
62
62
  docked: boolean;
63
63
  isModal: boolean;
64
+ pip: boolean;
64
65
  };
65
66
  playNextData: PlayNextData;
66
67
  isTabletPortrait?: boolean;
@@ -99,11 +100,7 @@ export const AudioPlayerWrapper: React.FC<Props> = (props: Props) => {
99
100
  );
100
101
 
101
102
  useEffect(() => {
102
- if (player) {
103
- setSpeedManager(new SpeedController(player));
104
- } else {
105
- setSpeedManager(null);
106
- }
103
+ setSpeedManager(player ? new SpeedController(player) : null);
107
104
  }, [player]);
108
105
 
109
106
  useEffect(() => {
@@ -217,7 +214,7 @@ export const AudioPlayerWrapper: React.FC<Props> = (props: Props) => {
217
214
  items: speedOptions,
218
215
  current_selection: playbackSpeed,
219
216
  onPress: onSpeedSelect,
220
- title: configuration["playback_speed_title"],
217
+ title: configuration.playback_speed_title,
221
218
  itemProps: getBottomSheetModalItemProps,
222
219
  iconProps: getBottomSheetModalIconProps,
223
220
  labelProps: getBottomSheetModalLabelProps,
@@ -233,7 +230,7 @@ export const AudioPlayerWrapper: React.FC<Props> = (props: Props) => {
233
230
  ],
234
231
  },
235
232
  });
236
- }, [playbackSpeed, onSpeedSelect, configuration["playback_speed_title"]]);
233
+ }, [playbackSpeed, onSpeedSelect, configuration.playback_speed_title]);
237
234
 
238
235
  const onPlayerSeek = React.useCallback(
239
236
  (targetTime: number) => {
@@ -250,13 +247,28 @@ export const AudioPlayerWrapper: React.FC<Props> = (props: Props) => {
250
247
  player?.seekTo(0);
251
248
  }, [player]);
252
249
 
250
+ const layoutStateMemo = React.useMemo(
251
+ () => ({
252
+ inline: layoutState.inline,
253
+ docked: layoutState.docked,
254
+ isModal: layoutState.isModal,
255
+ pip: layoutState.pip,
256
+ }),
257
+ [
258
+ layoutState.inline,
259
+ layoutState.docked,
260
+ layoutState.isModal,
261
+ layoutState.pip,
262
+ ]
263
+ );
264
+
253
265
  return (
254
266
  <SafeAreaView style={[style, Styles.playerContainer]} edges={edges}>
255
267
  <View style={Styles.playerContainer} onLayout={onLayout}>
256
268
  <Layout
257
269
  entry={entry}
258
270
  playerState={playerState}
259
- layoutState={layoutState}
271
+ layoutState={layoutStateMemo}
260
272
  value={value}
261
273
  onPlaybackButtonPress={onPlaybackButtonPress}
262
274
  onRewindButtonPress={onRewindButtonPress}
@@ -1,6 +1,5 @@
1
1
  import * as React from "react";
2
- import { StyleSheet, View, ViewStyle } from "react-native";
3
- import * as R from "ramda";
2
+ import { StyleSheet, View } from "react-native";
4
3
  import {
5
4
  useIsTablet,
6
5
  useZStore,
@@ -16,7 +15,6 @@ import {
16
15
 
17
16
  import { PlayNextData } from "@applicaster/zapp-react-native-ui-components/Components/PlayerContainer/PlayerContainer";
18
17
 
19
- import { toNumberWithDefault } from "@applicaster/zapp-react-native-utils/numberUtils";
20
18
  import { useSafeAreaInsets } from "react-native-safe-area-context";
21
19
  import { useNavigation } from "@applicaster/zapp-react-native-utils/reactHooks/navigation/useNavigation";
22
20
  import { getTabBarHeight } from "@applicaster/zapp-react-native-utils/reactHooks/navigation/getTabBarHeight";
@@ -27,22 +25,17 @@ import { noop } from "@applicaster/zapp-react-native-utils/functionUtils";
27
25
  import { PROGRESS_BAR_HEIGHT } from "@applicaster/zapp-react-native-ui-components/Components/VideoModal/ModalAnimation/utils";
28
26
  import { isMenuVisible } from "@applicaster/zapp-react-native-ui-components/Components/Screen/navigationHandler";
29
27
  import { usePlugins } from "@applicaster/zapp-react-native-redux";
28
+ import { partial } from "@applicaster/zapp-react-native-utils/utils";
30
29
 
31
30
  const bottomTabBarHeight = getTabBarHeight();
32
31
 
33
32
  type Props = {
34
33
  entry: ZappEntry;
35
34
  layoutState: QuickBrickPlayer.LayoutState;
36
- playerState: QuickBrickPlayer.PlayerState;
37
- value: GetValue;
38
- onCloseVideoModal: () => void;
39
- onPlayerSeek: (targetTime: number) => void;
35
+ onPlayerSeek?: (targetTime: number) => void;
40
36
  playNextData: PlayNextData;
41
- title: string | number;
42
- summary: string | number;
43
37
  ImageComponent?: React.ReactNode;
44
38
  isActiveGesture?: boolean;
45
- minimisedHeight: number;
46
39
  aspectRatio?: number;
47
40
  };
48
41
 
@@ -51,7 +44,6 @@ const styles = StyleSheet.create({
51
44
  flex: 1,
52
45
  flexDirection: "column",
53
46
  alignItems: "stretch",
54
- marginTop: PROGRESS_BAR_HEIGHT,
55
47
  },
56
48
  content: {
57
49
  flex: 1,
@@ -94,14 +86,12 @@ const styles = StyleSheet.create({
94
86
  buttons: {
95
87
  marginVertical: 10,
96
88
  },
89
+ liveMarginTop: { marginTop: 0 },
90
+ marginTop: { marginTop: PROGRESS_BAR_HEIGHT },
97
91
  });
98
92
 
99
- const playerDimensionsStyle = (
100
- minimisedHeight: number,
101
- addPadding: boolean
102
- ): ViewStyle => ({
103
- width: addPadding ? toNumberWithDefault(100, minimisedHeight) : 0,
104
- });
93
+ const getValue = (key: string, stylesObj: Record<string, any>) =>
94
+ stylesObj?.[key] ?? null;
105
95
 
106
96
  const controlButtons = (
107
97
  live: boolean,
@@ -167,7 +157,6 @@ export const DockedControls = (props: Props) => {
167
157
  onPlayerSeek = noop,
168
158
  playNextData,
169
159
  ImageComponent,
170
- isActiveGesture = true,
171
160
  aspectRatio,
172
161
  } = props;
173
162
 
@@ -185,11 +174,6 @@ export const DockedControls = (props: Props) => {
185
174
  const configuration = useZStore("playerConfiguration")();
186
175
  const playerState = usePlayerState("shared-player-wrapper-docked");
187
176
 
188
- const getValue = React.useCallback(
189
- (key: string, stylesObj: {}) => R.propOr(null, key, stylesObj),
190
- []
191
- );
192
-
193
177
  const value = React.useCallback(
194
178
  (key: string) => {
195
179
  const retVal = getValue(key, configuration);
@@ -208,7 +192,7 @@ export const DockedControls = (props: Props) => {
208
192
 
209
193
  return retVal;
210
194
  },
211
- [...Object.values(configuration), playerState.seekableDuration, getValue]
195
+ [...Object.values(configuration), playerState.seekableDuration]
212
196
  );
213
197
 
214
198
  const { title, summary } = useTitleSummaryOverlay(
@@ -225,15 +209,65 @@ export const DockedControls = (props: Props) => {
225
209
 
226
210
  const insets = useSafeAreaInsets();
227
211
 
212
+ const progressBarLayoutState = React.useMemo(
213
+ () => ({
214
+ inline: layoutState.inline,
215
+ docked: true,
216
+ isModal: layoutState.isModal,
217
+ pip: layoutState.pip,
218
+ }),
219
+ [layoutState.inline, layoutState.isModal, layoutState.pip]
220
+ );
221
+
222
+ const _controlButtons = React.useMemo(
223
+ () => partial(controlButtons, playerState.isLive && liveButton),
224
+ [playerState.isLive, liveButton]
225
+ );
226
+
227
+ const containerStyles = React.useMemo(
228
+ () => ({
229
+ height: minimisedHeight,
230
+ maxHeight: minimisedHeight,
231
+ }),
232
+ [minimisedHeight]
233
+ );
234
+
235
+ const imageComponentWrapperStyles = React.useMemo(
236
+ () => ({
237
+ aspectRatio: aspectRatio,
238
+ height: minimisedHeight,
239
+ }),
240
+ [aspectRatio, minimisedHeight]
241
+ );
242
+
243
+ const backgroundColorStyle = React.useMemo(
244
+ () => ({
245
+ backgroundColor: dockedBackgroundColor,
246
+ }),
247
+ [dockedBackgroundColor]
248
+ );
249
+
250
+ const bottomSpacerStyle = React.useMemo(() => {
251
+ const tabBarHeight = menuVisible ? bottomTabBarHeight : 0;
252
+ const safeAreaBottomInset = insets.bottom || 0;
253
+
254
+ return {
255
+ height: safeAreaBottomInset + tabBarHeight,
256
+ };
257
+ }, [menuVisible, insets.bottom]);
258
+
259
+ const contentInfoContent = React.useMemo(
260
+ () => ({ ...entry, title, summary }),
261
+ [title, summary, entry]
262
+ );
263
+
228
264
  return (
229
265
  <>
230
266
  <View
231
267
  style={[
232
268
  styles.container,
233
- {
234
- height: minimisedHeight,
235
- maxHeight: minimisedHeight,
236
- },
269
+ containerStyles,
270
+ playerState.isLive ? styles.liveMarginTop : styles.marginTop,
237
271
  ]}
238
272
  >
239
273
  {!playerState.isLive ? (
@@ -242,42 +276,23 @@ export const DockedControls = (props: Props) => {
242
276
  content={entry}
243
277
  value={value}
244
278
  visible
245
- layoutState={layoutState}
279
+ layoutState={progressBarLayoutState}
246
280
  onPlayerSeek={onPlayerSeek}
247
281
  playerState={playerState}
248
- startComponentsAnimation={noop}
249
282
  docked
250
283
  />
251
284
  ) : null}
252
285
 
253
286
  <View style={styles.content}>
254
287
  <View style={styles.touchableArea}>
255
- {isActiveGesture || layoutState.docked ? (
256
- <View
257
- style={{ aspectRatio: aspectRatio, height: minimisedHeight }}
258
- >
259
- {ImageComponent || null}
260
- </View>
261
- ) : (
262
- <View
263
- style={playerDimensionsStyle(
264
- minimisedHeight,
265
- isActiveGesture || layoutState.docked
266
- )}
267
- />
268
- )}
288
+ <View style={imageComponentWrapperStyles}>
289
+ {ImageComponent || null}
290
+ </View>
269
291
  <View
270
292
  testID="content-info"
271
- style={[
272
- styles.contentInfo,
273
- { backgroundColor: dockedBackgroundColor },
274
- ]}
293
+ style={[styles.contentInfo, backgroundColorStyle]}
275
294
  >
276
- <ContentInfo
277
- content={{ ...entry, title, summary }}
278
- value={value}
279
- docked
280
- />
295
+ <ContentInfo content={contentInfoContent} value={value} docked />
281
296
  </View>
282
297
  </View>
283
298
  <View
@@ -285,15 +300,13 @@ export const DockedControls = (props: Props) => {
285
300
  isTablet
286
301
  ? styles.controlsContainerForTablet
287
302
  : styles.controlsContainerForMobile,
288
- { backgroundColor: dockedBackgroundColor },
303
+ backgroundColorStyle,
289
304
  ]}
290
305
  >
291
306
  <Controls
292
307
  value={value}
293
308
  visible
294
- buttons={R.partial(controlButtons, [
295
- playerState.isLive && liveButton,
296
- ])}
309
+ buttons={_controlButtons}
297
310
  style={styles.playPauseButton}
298
311
  playNextData={playNextData}
299
312
  playerState={playerState}
@@ -308,12 +321,7 @@ export const DockedControls = (props: Props) => {
308
321
  </View>
309
322
  </View>
310
323
  </View>
311
- <View
312
- style={{
313
- height: insets.bottom + (menuVisible ? bottomTabBarHeight : 0),
314
- backgroundColor: dockedBackgroundColor,
315
- }}
316
- />
324
+ <View style={[bottomSpacerStyle, backgroundColorStyle]} />
317
325
  </>
318
326
  );
319
327
  };
@@ -151,12 +151,7 @@ export function MobileLayout({
151
151
  },
152
152
  ]}
153
153
  >
154
- <PlayerImage
155
- entry={entry}
156
- docked={false}
157
- imageWidth={imageSize}
158
- configuration={configuration}
159
- />
154
+ <PlayerImage entry={entry} configuration={configuration} />
160
155
  </View>
161
156
  {/* PlayerImage */}
162
157
 
@@ -4,7 +4,6 @@ import { LayoutChangeEvent, View } from "react-native";
4
4
  import { noop } from "@applicaster/zapp-react-native-utils/functionUtils";
5
5
  import { toBooleanWithDefaultTrue } from "@applicaster/zapp-react-native-utils/booleanUtils";
6
6
 
7
- import { AnimatedImage } from "./AnimatedImage";
8
7
  import { FlexImage } from "./FlexImage";
9
8
 
10
9
  import { styles } from "./styles";
@@ -13,19 +12,12 @@ import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/nu
13
12
  type Props = {
14
13
  entry: ZappEntry;
15
14
  configuration: QuickBrickPlayer.AudioPlayerProps["configuration"];
16
- docked: boolean;
17
- imageWidth?: Option<number>;
15
+ docked?: boolean;
18
16
  onLayoutImage?: (event: LayoutChangeEvent) => void;
19
17
  };
20
18
 
21
19
  export const PlayerImage = (props: Props) => {
22
- const {
23
- entry,
24
- docked,
25
- imageWidth,
26
- configuration,
27
- onLayoutImage = noop,
28
- } = props;
20
+ const { entry, docked = false, configuration, onLayoutImage = noop } = props;
29
21
 
30
22
  const isShadowEnabled = toBooleanWithDefaultTrue(
31
23
  configuration?.audio_player_artwork_shadow_enabled
@@ -43,19 +35,9 @@ export const PlayerImage = (props: Props) => {
43
35
  onLayout={onLayoutImage}
44
36
  style={[shouldShowShadow ? styles.shadow : undefined, { borderRadius }]}
45
37
  >
46
- {!imageWidth ? (
47
- <View style={[styles.defaultImageWrapperView, { borderRadius }]}>
48
- <FlexImage entry={entry} style={styles.backgroundImageContainer} />
49
- </View>
50
- ) : (
51
- // we have imageSize here and now we could animate resizing of this image
52
- <AnimatedImage
53
- entry={entry}
54
- style={[styles.backgroundImageContainer, { borderRadius }]}
55
- imageWidth={imageWidth}
56
- docked={docked}
57
- />
58
- )}
38
+ <View style={[styles.defaultImageWrapperView, { borderRadius }]}>
39
+ <FlexImage entry={entry} style={styles.backgroundImageContainer} />
40
+ </View>
59
41
  </View>
60
42
  </View>
61
43
  );
@@ -117,14 +117,11 @@ export function TabletLandscapeLayout({
117
117
  <View style={[styles.flex, styles.audioPlayerContainerHeightLimit]}>
118
118
  {/* PlayerImage non-docked */}
119
119
  <View style={[styles.flex, withDebug(styles.debugYellow)]}>
120
- {!docked ? (
121
- <PlayerImage
122
- docked={false}
123
- entry={entry}
124
- configuration={configuration}
125
- onLayoutImage={onLayoutImage}
126
- />
127
- ) : null}
120
+ <PlayerImage
121
+ entry={entry}
122
+ configuration={configuration}
123
+ onLayoutImage={onLayoutImage}
124
+ />
128
125
  </View>
129
126
  {/* PlayerImage */}
130
127
 
@@ -144,12 +144,7 @@ export function TabletPortraitLayout({
144
144
  ]}
145
145
  onLayout={onLayout}
146
146
  >
147
- <PlayerImage
148
- entry={entry}
149
- configuration={configuration}
150
- docked={docked}
151
- imageWidth={imageSize}
152
- />
147
+ <PlayerImage entry={entry} configuration={configuration} />
153
148
  </View>
154
149
 
155
150
  <View
@@ -1,10 +1,6 @@
1
1
  import { localStorage } from "@applicaster/zapp-react-native-bridge/ZappStorage/LocalStorage";
2
- import {
3
- PlayerAnimationStateEnum,
4
- PlayerAnimationStateT,
5
- } from "@applicaster/zapp-react-native-ui-components/Components/VideoModal/ModalAnimation/ModalAnimationContext";
2
+
6
3
  import { parseJsonIfNeeded } from "@applicaster/zapp-react-native-utils/functionUtils";
7
- import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/numberUtils";
8
4
 
9
5
  export const NAMESPACE = "audio-player";
10
6
 
@@ -36,25 +32,3 @@ export const formatSpeed = (value: number) => {
36
32
  // Return the formatted string with "speed_" prefix
37
33
  return `speed_${formattedValue}`;
38
34
  };
39
-
40
- export const calculateBorderRadius = (
41
- docked: boolean,
42
- startComponentsAnimation: boolean,
43
- isActiveGesture: boolean,
44
- playerAnimationState: PlayerAnimationStateT,
45
- configuration: Record<string, any>
46
- ): number => {
47
- const shouldRemoveBorderRadius =
48
- (startComponentsAnimation &&
49
- isActiveGesture &&
50
- playerAnimationState !== PlayerAnimationStateEnum.maximize) ||
51
- docked;
52
-
53
- if (shouldRemoveBorderRadius) {
54
- return 0;
55
- }
56
-
57
- return toNumberWithDefaultZero(
58
- configuration.audio_player_artwork_border_radius
59
- );
60
- };
@@ -4,8 +4,9 @@ import { Animated, Pressable } from "react-native";
4
4
  import { PanGestureHandler } from "react-native-gesture-handler";
5
5
  import { useStyles } from "./styles";
6
6
  import { useVideoModalState } from "./hooks";
7
- import { MODAL_COLLAPSE_START, MODAL_RADIUS, SCREEN_HEIGHT } from "./consts";
7
+ import { MODAL_COLLAPSE_RATIO, MODAL_RADIUS } from "./consts";
8
8
  import { useModalAnimationContext } from "@applicaster/zapp-react-native-ui-components/Components/VideoModal/ModalAnimation";
9
+ import { getWindowHeight } from "./utils";
9
10
 
10
11
  type AnimatedModalProps = {
11
12
  content?: ReactElement;
@@ -28,21 +29,26 @@ export function AnimatedModal({
28
29
 
29
30
  const styles = useStyles({ height: collapsedHeight, type: "audio" });
30
31
 
32
+ const height = getWindowHeight();
33
+ const heightAboveMinimised = height - collapsedHeight;
34
+
35
+ const MODAL_COLLAPSE_START = height * MODAL_COLLAPSE_RATIO;
36
+
31
37
  // Interpolated opacities for smooth cross-fade
32
38
  const collapsedOpacity = translateY.interpolate({
33
- inputRange: [MODAL_COLLAPSE_START, SCREEN_HEIGHT - collapsedHeight],
39
+ inputRange: [MODAL_COLLAPSE_START, heightAboveMinimised],
34
40
  outputRange: [0, 1],
35
41
  extrapolate: "clamp",
36
42
  });
37
43
 
38
44
  const expandedOpacity = translateY.interpolate({
39
- inputRange: [0, SCREEN_HEIGHT - collapsedHeight],
45
+ inputRange: [0, heightAboveMinimised],
40
46
  outputRange: [1, 0],
41
47
  extrapolate: "clamp",
42
48
  });
43
49
 
44
50
  const borderTopRadiusAnimated = translateY.interpolate({
45
- inputRange: [0, SCREEN_HEIGHT - collapsedHeight],
51
+ inputRange: [0, heightAboveMinimised],
46
52
  outputRange: [MODAL_RADIUS, 0],
47
53
  extrapolate: "clamp",
48
54
  });
@@ -57,7 +63,7 @@ export function AnimatedModal({
57
63
  borderTopLeftRadius: borderTopRadiusAnimated,
58
64
  borderTopRightRadius: borderTopRadiusAnimated,
59
65
  transform: [{ translateY }],
60
- height: SCREEN_HEIGHT,
66
+ height: height,
61
67
  },
62
68
  ]}
63
69
  >
@@ -67,7 +73,6 @@ export function AnimatedModal({
67
73
  {
68
74
  borderTopLeftRadius: borderTopRadiusAnimated,
69
75
  borderTopRightRadius: borderTopRadiusAnimated,
70
- height: SCREEN_HEIGHT,
71
76
  },
72
77
  ]}
73
78
  >