| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | import React, { | 
					
						
							|  |  |  |   forwardRef, | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useEffect, | 
					
						
							|  |  |  |   useImperativeHandle, | 
					
						
							|  |  |  |   useRef, | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |   type RefObject, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | } from 'react'; | 
					
						
							| 
									
										
										
										
											2024-10-13 02:06:12 -06:00
										 |  |  | import shaka from 'shaka-player'; | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  | import type {VideoRef, ReactVideoProps, VideoMetadata} from './types'; | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | const Video = forwardRef<VideoRef, ReactVideoProps>( | 
					
						
							|  |  |  |   ( | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       source, | 
					
						
							|  |  |  |       paused, | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       muted, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |       volume, | 
					
						
							| 
									
										
										
										
											2024-06-30 11:38:52 +00:00
										 |  |  |       rate, | 
					
						
							|  |  |  |       repeat, | 
					
						
							|  |  |  |       controls, | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       showNotificationControls = false, | 
					
						
							| 
									
										
										
										
											2024-06-30 11:38:52 +00:00
										 |  |  |       poster, | 
					
						
							| 
									
										
										
										
											2024-07-10 17:18:21 +07:00
										 |  |  |       fullscreen, | 
					
						
							|  |  |  |       fullscreenAutorotate, | 
					
						
							|  |  |  |       fullscreenOrientation, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |       onBuffer, | 
					
						
							|  |  |  |       onLoad, | 
					
						
							|  |  |  |       onProgress, | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       onPlaybackRateChange, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |       onError, | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |       onReadyForDisplay, | 
					
						
							|  |  |  |       onSeek, | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       onVolumeChange, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |       onEnd, | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       onPlaybackStateChanged, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     }, | 
					
						
							|  |  |  |     ref, | 
					
						
							|  |  |  |   ) => { | 
					
						
							|  |  |  |     const nativeRef = useRef<HTMLVideoElement>(null); | 
					
						
							| 
									
										
										
										
											2024-10-12 23:48:55 -06:00
										 |  |  |     const shakaPlayerRef = useRef<shaka.Player | null>(null); | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |     const isSeeking = useRef(false); | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |     const seek = useCallback( | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       async (time: number, _tolerance?: number) => { | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |         if (isNaN(time)) { | 
					
						
							|  |  |  |           throw new Error('Specified time is not a number'); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         if (!nativeRef.current) { | 
					
						
							|  |  |  |           console.warn('Video Component is not mounted'); | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         time = Math.max(0, Math.min(time, nativeRef.current.duration)); | 
					
						
							|  |  |  |         nativeRef.current.currentTime = time; | 
					
						
							|  |  |  |         onSeek?.({seekTime: time, currentTime: nativeRef.current.currentTime}); | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |       }, | 
					
						
							|  |  |  |       [onSeek], | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const pause = useCallback(() => { | 
					
						
							|  |  |  |       if (!nativeRef.current) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       nativeRef.current.pause(); | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const resume = useCallback(() => { | 
					
						
							|  |  |  |       if (!nativeRef.current) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       nativeRef.current.play(); | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |     const setVolume = useCallback((vol: number) => { | 
					
						
							|  |  |  |       if (!nativeRef.current) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       nativeRef.current.volume = Math.max(0, Math.min(vol, 100)) / 100; | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const getCurrentPosition = useCallback(async () => { | 
					
						
							|  |  |  |       if (!nativeRef.current) { | 
					
						
							|  |  |  |         throw new Error('Video Component is not mounted'); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return nativeRef.current.currentTime; | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     const unsupported = useCallback(() => { | 
					
						
							|  |  |  |       throw new Error('This is unsupported on the web'); | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-10 17:18:21 +07:00
										 |  |  |     // Stock this in a ref to not invalidate memoization when those changes.
 | 
					
						
							|  |  |  |     const fsPrefs = useRef({ | 
					
						
							|  |  |  |       fullscreenAutorotate, | 
					
						
							|  |  |  |       fullscreenOrientation, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     fsPrefs.current = { | 
					
						
							|  |  |  |       fullscreenOrientation, | 
					
						
							|  |  |  |       fullscreenAutorotate, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     const setFullScreen = useCallback( | 
					
						
							|  |  |  |       ( | 
					
						
							|  |  |  |         newVal: boolean, | 
					
						
							|  |  |  |         orientation?: ReactVideoProps['fullscreenOrientation'], | 
					
						
							|  |  |  |         autorotate?: boolean, | 
					
						
							|  |  |  |       ) => { | 
					
						
							|  |  |  |         orientation ??= fsPrefs.current.fullscreenOrientation; | 
					
						
							|  |  |  |         autorotate ??= fsPrefs.current.fullscreenAutorotate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const run = async () => { | 
					
						
							|  |  |  |           try { | 
					
						
							|  |  |  |             if (newVal) { | 
					
						
							|  |  |  |               await nativeRef.current?.requestFullscreen({ | 
					
						
							|  |  |  |                 navigationUI: 'hide', | 
					
						
							|  |  |  |               }); | 
					
						
							|  |  |  |               if (orientation === 'all' || !orientation || autorotate) { | 
					
						
							|  |  |  |                 screen.orientation.unlock(); | 
					
						
							|  |  |  |               } else { | 
					
						
							|  |  |  |                 await screen.orientation.lock(orientation); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |               if (document.fullscreenElement) { | 
					
						
							|  |  |  |                 await document.exitFullscreen(); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               screen.orientation.unlock(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } catch (e) { | 
					
						
							|  |  |  |             // Changing fullscreen status without a button click is not allowed so it throws.
 | 
					
						
							|  |  |  |             // Some browsers also used to throw when locking screen orientation was not supported.
 | 
					
						
							|  |  |  |             console.error('Could not toggle fullscreen/screen lock status', e); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |         run(); | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       [], | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     useEffect(() => { | 
					
						
							|  |  |  |       setFullScreen( | 
					
						
							|  |  |  |         fullscreen || false, | 
					
						
							|  |  |  |         fullscreenOrientation, | 
					
						
							|  |  |  |         fullscreenAutorotate, | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     }, [ | 
					
						
							|  |  |  |       setFullScreen, | 
					
						
							|  |  |  |       fullscreen, | 
					
						
							|  |  |  |       fullscreenAutorotate, | 
					
						
							|  |  |  |       fullscreenOrientation, | 
					
						
							|  |  |  |     ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const presentFullscreenPlayer = useCallback( | 
					
						
							|  |  |  |       () => setFullScreen(true), | 
					
						
							|  |  |  |       [setFullScreen], | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     const dismissFullscreenPlayer = useCallback( | 
					
						
							|  |  |  |       () => setFullScreen(false), | 
					
						
							|  |  |  |       [setFullScreen], | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     useImperativeHandle( | 
					
						
							|  |  |  |       ref, | 
					
						
							|  |  |  |       () => ({ | 
					
						
							|  |  |  |         seek, | 
					
						
							|  |  |  |         pause, | 
					
						
							|  |  |  |         resume, | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |         setVolume, | 
					
						
							|  |  |  |         getCurrentPosition, | 
					
						
							| 
									
										
										
										
											2024-07-10 17:18:21 +07:00
										 |  |  |         presentFullscreenPlayer, | 
					
						
							|  |  |  |         dismissFullscreenPlayer, | 
					
						
							|  |  |  |         setFullScreen, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |         save: unsupported, | 
					
						
							|  |  |  |         restoreUserInterfaceForPictureInPictureStopCompleted: unsupported, | 
					
						
							| 
									
										
										
										
											2024-07-09 12:44:01 +07:00
										 |  |  |         nativeHtmlVideoRef: nativeRef, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |       }), | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       [ | 
					
						
							|  |  |  |         seek, | 
					
						
							|  |  |  |         pause, | 
					
						
							|  |  |  |         resume, | 
					
						
							|  |  |  |         unsupported, | 
					
						
							|  |  |  |         setVolume, | 
					
						
							|  |  |  |         getCurrentPosition, | 
					
						
							|  |  |  |         nativeRef, | 
					
						
							| 
									
										
										
										
											2024-07-10 17:18:21 +07:00
										 |  |  |         presentFullscreenPlayer, | 
					
						
							|  |  |  |         dismissFullscreenPlayer, | 
					
						
							|  |  |  |         setFullScreen, | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       ], | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     useEffect(() => { | 
					
						
							|  |  |  |       if (paused) { | 
					
						
							|  |  |  |         pause(); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         resume(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, [paused, pause, resume]); | 
					
						
							|  |  |  |     useEffect(() => { | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |       if (volume === undefined) { | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-06-30 11:25:43 +00:00
										 |  |  |       setVolume(volume); | 
					
						
							|  |  |  |     }, [volume, setVolume]); | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |     // we use a ref to prevent triggerring the useEffect when the component rerender with a non-stable `onPlaybackStateChanged`.
 | 
					
						
							|  |  |  |     const playbackStateRef = useRef(onPlaybackStateChanged); | 
					
						
							|  |  |  |     playbackStateRef.current = onPlaybackStateChanged; | 
					
						
							|  |  |  |     useEffect(() => { | 
					
						
							|  |  |  |       // Not sure about how to do this but we want to wait for nativeRef to be initialized
 | 
					
						
							|  |  |  |       setTimeout(() => { | 
					
						
							|  |  |  |         if (!nativeRef.current) { | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Set play state to the player's value (if autoplay is denied)
 | 
					
						
							|  |  |  |         // This is useful if our UI is in a play state but autoplay got denied so
 | 
					
						
							|  |  |  |         // the video is actaully in a paused state.
 | 
					
						
							|  |  |  |         playbackStateRef.current?.({ | 
					
						
							|  |  |  |           isPlaying: !nativeRef.current.paused, | 
					
						
							|  |  |  |           isSeeking: isSeeking.current, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }, 500); | 
					
						
							|  |  |  |     }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 11:38:52 +00:00
										 |  |  |     useEffect(() => { | 
					
						
							|  |  |  |       if (!nativeRef.current || rate === undefined) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       nativeRef.current.playbackRate = rate; | 
					
						
							|  |  |  |     }, [rate]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-12 23:48:55 -06:00
										 |  |  |     useEffect(() => { | 
					
						
							| 
									
										
										
										
											2024-10-13 20:45:25 -06:00
										 |  |  |       if (!nativeRef.current) { | 
					
						
							| 
									
										
										
										
											2024-10-13 15:14:48 -06:00
										 |  |  |         console.log("Not starting shaka yet bc undefined") | 
					
						
							| 
									
										
										
										
											2024-10-12 23:48:55 -06:00
										 |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       if (shakaPlayerRef.current) { | 
					
						
							|  |  |  |         shakaPlayerRef.current.unload() | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       shakaPlayerRef.current = new shaka.Player(); | 
					
						
							|  |  |  |       shakaPlayerRef.current.addEventListener("error", (event) => { | 
					
						
							| 
									
										
										
										
											2024-10-13 01:07:48 -06:00
										 |  |  |         //@ts-ignore
 | 
					
						
							| 
									
										
										
										
											2024-10-12 23:48:55 -06:00
										 |  |  |         const shakaError = event.detail; | 
					
						
							|  |  |  |         console.error('Shaka Player Error', shakaError); | 
					
						
							|  |  |  |         onError?.({ | 
					
						
							|  |  |  |           error: { | 
					
						
							|  |  |  |             errorString: shakaError.message, | 
					
						
							|  |  |  |             code: shakaError.code, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2024-10-13 15:14:48 -06:00
										 |  |  |       console.log("Initializing and attaching shaka") | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |       shakaPlayerRef.current.attach(nativeRef.current, true); | 
					
						
							| 
									
										
										
										
											2024-10-13 15:14:48 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       //@ts-ignore
 | 
					
						
							|  |  |  |       shakaPlayerRef.current.load(source?.uri).then( | 
					
						
							|  |  |  |         () => console.log(`${source?.uri} finished loading`) | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2024-10-13 19:17:01 -06:00
										 |  |  |       console.log("Started shaka loading"); | 
					
						
							|  |  |  |     }, [source, nativeRef.current]) | 
					
						
							| 
									
										
										
										
											2024-10-12 23:48:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |     useMediaSession(source?.metadata, nativeRef, showNotificationControls); | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       <video | 
					
						
							|  |  |  |         ref={nativeRef} | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         muted={muted} | 
					
						
							|  |  |  |         autoPlay={!paused} | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |         controls={controls} | 
					
						
							|  |  |  |         loop={repeat} | 
					
						
							|  |  |  |         playsInline | 
					
						
							| 
									
										
										
										
											2024-10-12 22:33:11 -06:00
										 |  |  |         //@ts-ignore
 | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |         poster={poster} | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         onCanPlay={() => onBuffer?.({isBuffering: false})} | 
					
						
							|  |  |  |         onWaiting={() => onBuffer?.({isBuffering: true})} | 
					
						
							|  |  |  |         onRateChange={() => { | 
					
						
							|  |  |  |           if (!nativeRef.current) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           onPlaybackRateChange?.({ | 
					
						
							|  |  |  |             playbackRate: nativeRef.current?.playbackRate, | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         }} | 
					
						
							|  |  |  |         onDurationChange={() => { | 
					
						
							|  |  |  |           if (!nativeRef.current) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           onLoad?.({ | 
					
						
							|  |  |  |             currentTime: nativeRef.current.currentTime, | 
					
						
							|  |  |  |             duration: nativeRef.current.duration, | 
					
						
							|  |  |  |             videoTracks: [], | 
					
						
							|  |  |  |             textTracks: [], | 
					
						
							|  |  |  |             audioTracks: [], | 
					
						
							|  |  |  |             naturalSize: { | 
					
						
							|  |  |  |               width: nativeRef.current.videoWidth, | 
					
						
							|  |  |  |               height: nativeRef.current.videoHeight, | 
					
						
							|  |  |  |               orientation: 'landscape', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         }} | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |         onTimeUpdate={() => { | 
					
						
							|  |  |  |           if (!nativeRef.current) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           onProgress?.({ | 
					
						
							|  |  |  |             currentTime: nativeRef.current.currentTime, | 
					
						
							|  |  |  |             playableDuration: nativeRef.current.buffered.length | 
					
						
							|  |  |  |               ? nativeRef.current.buffered.end( | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |                   nativeRef.current.buffered.length - 1, | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |                 ) | 
					
						
							|  |  |  |               : 0, | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |             seekableDuration: 0, | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |           }); | 
					
						
							|  |  |  |         }} | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         onLoadedData={() => onReadyForDisplay?.()} | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |         onError={() => { | 
					
						
							|  |  |  |           if (!nativeRef.current?.error) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           onError?.({ | 
					
						
							|  |  |  |             error: { | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |               errorString: nativeRef.current.error.message ?? 'Unknown error', | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |               code: nativeRef.current.error.code, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         }} | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |         onLoadedMetadata={() => { | 
					
						
							|  |  |  |           if (source?.startPosition) { | 
					
						
							|  |  |  |             seek(source.startPosition / 1000); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }} | 
					
						
							|  |  |  |         onPlay={() => | 
					
						
							|  |  |  |           onPlaybackStateChanged?.({ | 
					
						
							|  |  |  |             isPlaying: true, | 
					
						
							|  |  |  |             isSeeking: isSeeking.current, | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         onPause={() => | 
					
						
							|  |  |  |           onPlaybackStateChanged?.({ | 
					
						
							|  |  |  |             isPlaying: false, | 
					
						
							|  |  |  |             isSeeking: isSeeking.current, | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         onSeeking={() => (isSeeking.current = true)} | 
					
						
							|  |  |  |         onSeeked={() => (isSeeking.current = false)} | 
					
						
							|  |  |  |         onVolumeChange={() => { | 
					
						
							|  |  |  |           if (!nativeRef.current) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           onVolumeChange?.({volume: nativeRef.current.volume}); | 
					
						
							|  |  |  |         }} | 
					
						
							|  |  |  |         onEnded={onEnd} | 
					
						
							| 
									
										
										
										
											2024-07-09 12:43:10 +07:00
										 |  |  |         style={videoStyle} | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       /> | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2024-10-13 15:04:44 -06:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-09 12:43:10 +07:00
										 |  |  | const videoStyle = { | 
					
						
							|  |  |  |   position: 'absolute', | 
					
						
							|  |  |  |   inset: 0, | 
					
						
							|  |  |  |   objectFit: 'contain', | 
					
						
							|  |  |  |   width: '100%', | 
					
						
							|  |  |  |   height: '100%', | 
					
						
							|  |  |  | } satisfies React.CSSProperties; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  | const useMediaSession = ( | 
					
						
							|  |  |  |   metadata: VideoMetadata | undefined, | 
					
						
							|  |  |  |   nativeRef: RefObject<HTMLVideoElement>, | 
					
						
							|  |  |  |   showNotification: boolean, | 
					
						
							|  |  |  | ) => { | 
					
						
							|  |  |  |   const isPlaying = !nativeRef.current?.paused ?? false; | 
					
						
							|  |  |  |   const progress = nativeRef.current?.currentTime ?? 0; | 
					
						
							| 
									
										
										
										
											2024-07-01 03:32:39 +00:00
										 |  |  |   const duration = Number.isFinite(nativeRef.current?.duration) | 
					
						
							|  |  |  |     ? nativeRef.current?.duration | 
					
						
							|  |  |  |     : undefined; | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |   const playbackRate = nativeRef.current?.playbackRate ?? 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const enabled = 'mediaSession' in navigator && showNotification; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (enabled) { | 
					
						
							|  |  |  |       navigator.mediaSession.metadata = new MediaMetadata({ | 
					
						
							|  |  |  |         title: metadata?.title, | 
					
						
							|  |  |  |         artist: metadata?.artist, | 
					
						
							|  |  |  |         artwork: metadata?.imageUri ? [{src: metadata.imageUri}] : undefined, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [enabled, metadata]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (!enabled) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |     const seekTo = (time: number) => { | 
					
						
							|  |  |  |       if (nativeRef.current) { | 
					
						
							|  |  |  |         nativeRef.current.currentTime = time; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |     const seekRelative = (offset: number) => { | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |       if (nativeRef.current) { | 
					
						
							|  |  |  |         nativeRef.current.currentTime = nativeRef.current.currentTime + offset; | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const mediaActions: [ | 
					
						
							|  |  |  |       MediaSessionAction, | 
					
						
							|  |  |  |       MediaSessionActionHandler | null, | 
					
						
							|  |  |  |     ][] = [ | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |       ['play', () => nativeRef.current?.play()], | 
					
						
							|  |  |  |       ['pause', () => nativeRef.current?.pause()], | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |       [ | 
					
						
							|  |  |  |         'seekbackward', | 
					
						
							|  |  |  |         (evt: MediaSessionActionDetails) => | 
					
						
							|  |  |  |           seekRelative(evt.seekOffset ? -evt.seekOffset : -10), | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |       [ | 
					
						
							|  |  |  |         'seekforward', | 
					
						
							|  |  |  |         (evt: MediaSessionActionDetails) => | 
					
						
							|  |  |  |           seekRelative(evt.seekOffset ? evt.seekOffset : 10), | 
					
						
							|  |  |  |       ], | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |       ['seekto', (evt: MediaSessionActionDetails) => seekTo(evt.seekTime!)], | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  |     ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const [action, handler] of mediaActions) { | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         navigator.mediaSession.setActionHandler(action, handler); | 
					
						
							|  |  |  |       } catch { | 
					
						
							|  |  |  |         // ignored
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-07-01 03:45:36 +00:00
										 |  |  |   }, [enabled, nativeRef]); | 
					
						
							| 
									
										
										
										
											2024-06-30 13:25:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (enabled) { | 
					
						
							|  |  |  |       navigator.mediaSession.playbackState = isPlaying ? 'playing' : 'paused'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [isPlaying, enabled]); | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (enabled && duration !== undefined) { | 
					
						
							|  |  |  |       navigator.mediaSession.setPositionState({ | 
					
						
							|  |  |  |         position: Math.min(progress, duration), | 
					
						
							|  |  |  |         duration, | 
					
						
							|  |  |  |         playbackRate: playbackRate, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [progress, duration, playbackRate, enabled]); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 10:21:34 +00:00
										 |  |  | Video.displayName = 'Video'; | 
					
						
							|  |  |  | export default Video; |