2023-01-08 12:15:32 +00:00
|
|
|
import React, {
|
|
|
|
createContext,
|
|
|
|
MutableRefObject,
|
2023-01-08 14:37:16 +00:00
|
|
|
useContext,
|
2023-01-08 12:15:32 +00:00
|
|
|
useEffect,
|
|
|
|
useReducer,
|
|
|
|
} from "react";
|
2023-01-08 14:37:16 +00:00
|
|
|
import {
|
|
|
|
initialPlayerState,
|
2023-01-10 18:53:55 +00:00
|
|
|
PlayerContext,
|
2023-01-08 14:37:16 +00:00
|
|
|
useVideoPlayer,
|
|
|
|
} from "./hooks/useVideoPlayer";
|
2023-01-08 12:15:32 +00:00
|
|
|
|
|
|
|
interface VideoPlayerContextType {
|
2023-01-08 14:37:16 +00:00
|
|
|
source: string | null;
|
2023-01-10 18:53:55 +00:00
|
|
|
sourceType: "m3u8" | "mp4";
|
|
|
|
state: PlayerContext;
|
2023-01-08 12:15:32 +00:00
|
|
|
}
|
2023-01-08 14:37:16 +00:00
|
|
|
const initial: VideoPlayerContextType = {
|
2023-01-08 12:15:32 +00:00
|
|
|
source: null,
|
2023-01-10 18:53:55 +00:00
|
|
|
sourceType: "mp4",
|
2023-01-08 14:37:16 +00:00
|
|
|
state: initialPlayerState,
|
|
|
|
};
|
2023-01-08 12:15:32 +00:00
|
|
|
|
|
|
|
type VideoPlayerContextAction =
|
2023-01-10 18:53:55 +00:00
|
|
|
| { type: "SET_SOURCE"; url: string; sourceType: "m3u8" | "mp4" }
|
2023-01-08 12:15:32 +00:00
|
|
|
| {
|
|
|
|
type: "UPDATE_PLAYER";
|
2023-01-10 18:53:55 +00:00
|
|
|
state: PlayerContext;
|
2023-01-08 12:15:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
function videoPlayerContextReducer(
|
|
|
|
original: VideoPlayerContextType,
|
|
|
|
action: VideoPlayerContextAction
|
|
|
|
): VideoPlayerContextType {
|
|
|
|
const video = { ...original };
|
|
|
|
if (action.type === "SET_SOURCE") {
|
|
|
|
video.source = action.url;
|
2023-01-10 18:53:55 +00:00
|
|
|
video.sourceType = action.sourceType;
|
2023-01-08 12:15:32 +00:00
|
|
|
return video;
|
|
|
|
}
|
|
|
|
if (action.type === "UPDATE_PLAYER") {
|
2023-01-08 14:37:16 +00:00
|
|
|
video.state = action.state;
|
2023-01-08 12:15:32 +00:00
|
|
|
return video;
|
|
|
|
}
|
|
|
|
|
|
|
|
return original;
|
|
|
|
}
|
|
|
|
|
2023-01-08 14:37:16 +00:00
|
|
|
export const VideoPlayerContext =
|
|
|
|
createContext<VideoPlayerContextType>(initial);
|
2023-01-08 12:15:32 +00:00
|
|
|
export const VideoPlayerDispatchContext = createContext<
|
|
|
|
React.Dispatch<VideoPlayerContextAction>
|
|
|
|
>(null as any);
|
|
|
|
|
|
|
|
export function VideoPlayerContextProvider(props: {
|
|
|
|
children: React.ReactNode;
|
|
|
|
player: MutableRefObject<HTMLVideoElement | null>;
|
2023-01-08 15:23:42 +00:00
|
|
|
wrapper: MutableRefObject<HTMLDivElement | null>;
|
2023-01-08 12:15:32 +00:00
|
|
|
}) {
|
2023-01-08 15:23:42 +00:00
|
|
|
const { playerState } = useVideoPlayer(props.player, props.wrapper);
|
2023-01-08 12:15:32 +00:00
|
|
|
const [videoData, dispatch] = useReducer<typeof videoPlayerContextReducer>(
|
|
|
|
videoPlayerContextReducer,
|
2023-01-08 14:37:16 +00:00
|
|
|
initial
|
2023-01-08 12:15:32 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
dispatch({
|
|
|
|
type: "UPDATE_PLAYER",
|
2023-01-08 14:37:16 +00:00
|
|
|
state: playerState,
|
2023-01-08 12:15:32 +00:00
|
|
|
});
|
2023-01-08 14:37:16 +00:00
|
|
|
}, [playerState]);
|
2023-01-08 12:15:32 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<VideoPlayerContext.Provider value={videoData}>
|
|
|
|
<VideoPlayerDispatchContext.Provider value={dispatch}>
|
|
|
|
{props.children}
|
|
|
|
</VideoPlayerDispatchContext.Provider>
|
|
|
|
</VideoPlayerContext.Provider>
|
|
|
|
);
|
|
|
|
}
|
2023-01-08 14:37:16 +00:00
|
|
|
|
|
|
|
export function useVideoPlayerState() {
|
|
|
|
const { state } = useContext(VideoPlayerContext);
|
|
|
|
|
|
|
|
return {
|
|
|
|
videoState: state,
|
|
|
|
};
|
|
|
|
}
|