2023-01-10 18:53:55 +00:00
|
|
|
import Hls from "hls.js";
|
|
|
|
import {
|
|
|
|
canChangeVolume,
|
|
|
|
canFullscreen,
|
|
|
|
canFullscreenAnyElement,
|
|
|
|
canWebkitFullscreen,
|
|
|
|
} from "@/utils/detectFeatures";
|
2023-01-13 23:12:56 +00:00
|
|
|
import { MWStreamType } from "@/backend/helpers/streams";
|
2023-01-10 00:01:51 +00:00
|
|
|
import fscreen from "fscreen";
|
2023-01-10 18:53:55 +00:00
|
|
|
import React, { RefObject } from "react";
|
|
|
|
import { PlayerState } from "./useVideoPlayer";
|
|
|
|
import { getStoredVolume, setStoredVolume } from "./volumeStore";
|
2023-01-10 00:01:51 +00:00
|
|
|
|
2023-01-21 22:45:26 +00:00
|
|
|
interface ShowData {
|
|
|
|
current?: {
|
2023-01-22 18:26:08 +00:00
|
|
|
episodeId: string;
|
|
|
|
seasonId: string;
|
2023-01-21 22:45:26 +00:00
|
|
|
};
|
|
|
|
isSeries: boolean;
|
|
|
|
}
|
|
|
|
|
2023-01-08 14:37:16 +00:00
|
|
|
export interface PlayerControls {
|
|
|
|
play(): void;
|
|
|
|
pause(): void;
|
2023-01-08 15:23:42 +00:00
|
|
|
exitFullscreen(): void;
|
|
|
|
enterFullscreen(): void;
|
2023-01-08 16:51:38 +00:00
|
|
|
setTime(time: number): void;
|
|
|
|
setVolume(volume: number): void;
|
2023-01-10 18:53:55 +00:00
|
|
|
setSeeking(active: boolean): void;
|
|
|
|
setLeftControlsHover(hovering: boolean): void;
|
2023-01-13 23:12:56 +00:00
|
|
|
initPlayer(sourceUrl: string, sourceType: MWStreamType): void;
|
2023-01-21 22:45:26 +00:00
|
|
|
setShowData(data: ShowData): void;
|
2023-01-22 19:51:58 +00:00
|
|
|
startAirplay(): void;
|
2023-01-08 14:37:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export const initialControls: PlayerControls = {
|
|
|
|
play: () => null,
|
|
|
|
pause: () => null,
|
2023-01-08 15:23:42 +00:00
|
|
|
enterFullscreen: () => null,
|
|
|
|
exitFullscreen: () => null,
|
2023-01-08 16:51:38 +00:00
|
|
|
setTime: () => null,
|
|
|
|
setVolume: () => null,
|
2023-01-10 18:53:55 +00:00
|
|
|
setSeeking: () => null,
|
|
|
|
setLeftControlsHover: () => null,
|
|
|
|
initPlayer: () => null,
|
2023-01-21 22:45:26 +00:00
|
|
|
setShowData: () => null,
|
2023-01-22 19:51:58 +00:00
|
|
|
startAirplay: () => null,
|
2023-01-08 14:37:16 +00:00
|
|
|
};
|
|
|
|
|
2023-01-08 15:23:42 +00:00
|
|
|
export function populateControls(
|
2023-01-10 18:53:55 +00:00
|
|
|
playerEl: HTMLVideoElement,
|
|
|
|
wrapperEl: HTMLDivElement,
|
|
|
|
update: (s: React.SetStateAction<PlayerState>) => void,
|
|
|
|
state: RefObject<PlayerState>
|
2023-01-08 15:23:42 +00:00
|
|
|
): PlayerControls {
|
2023-01-10 18:53:55 +00:00
|
|
|
const player = playerEl;
|
|
|
|
const wrapper = wrapperEl;
|
|
|
|
|
2023-01-08 14:37:16 +00:00
|
|
|
return {
|
|
|
|
play() {
|
|
|
|
player.play();
|
|
|
|
},
|
|
|
|
pause() {
|
|
|
|
player.pause();
|
|
|
|
},
|
2023-01-08 15:23:42 +00:00
|
|
|
enterFullscreen() {
|
2023-01-10 18:53:55 +00:00
|
|
|
if (!canFullscreen() || fscreen.fullscreenElement) return;
|
|
|
|
if (canFullscreenAnyElement()) {
|
2023-01-10 00:01:51 +00:00
|
|
|
fscreen.requestFullscreen(wrapper);
|
|
|
|
return;
|
|
|
|
}
|
2023-01-10 18:53:55 +00:00
|
|
|
if (canWebkitFullscreen()) {
|
2023-01-10 00:01:51 +00:00
|
|
|
(player as any).webkitEnterFullscreen();
|
|
|
|
}
|
2023-01-08 15:23:42 +00:00
|
|
|
},
|
|
|
|
exitFullscreen() {
|
2023-01-10 00:01:51 +00:00
|
|
|
if (!fscreen.fullscreenElement) return;
|
|
|
|
fscreen.exitFullscreen();
|
2023-01-08 15:23:42 +00:00
|
|
|
},
|
2023-01-08 16:51:38 +00:00
|
|
|
setTime(t) {
|
|
|
|
// clamp time between 0 and max duration
|
|
|
|
let time = Math.min(t, player.duration);
|
|
|
|
time = Math.max(0, time);
|
2023-01-10 18:53:55 +00:00
|
|
|
|
|
|
|
if (Number.isNaN(time)) return;
|
|
|
|
|
|
|
|
// update state
|
2023-01-08 16:51:38 +00:00
|
|
|
player.currentTime = time;
|
2023-01-10 18:53:55 +00:00
|
|
|
update((s) => ({ ...s, time }));
|
2023-01-08 16:51:38 +00:00
|
|
|
},
|
2023-01-10 18:53:55 +00:00
|
|
|
async setVolume(v) {
|
2023-01-08 16:51:38 +00:00
|
|
|
// clamp time between 0 and 1
|
|
|
|
let volume = Math.min(v, 1);
|
|
|
|
volume = Math.max(0, volume);
|
2023-01-10 18:53:55 +00:00
|
|
|
|
|
|
|
// update state
|
|
|
|
if (await canChangeVolume()) player.volume = volume;
|
|
|
|
update((s) => ({ ...s, volume }));
|
|
|
|
|
|
|
|
// update localstorage
|
|
|
|
setStoredVolume(volume);
|
|
|
|
},
|
|
|
|
setSeeking(active) {
|
|
|
|
const currentState = state.current;
|
|
|
|
if (!currentState) return;
|
|
|
|
|
|
|
|
// if it was playing when starting to seek, play again
|
|
|
|
if (!active) {
|
|
|
|
if (!currentState.pausedWhenSeeking) this.play();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// when seeking we pause the video
|
|
|
|
update((s) => ({ ...s, pausedWhenSeeking: s.isPaused }));
|
|
|
|
this.pause();
|
|
|
|
},
|
|
|
|
setLeftControlsHover(hovering) {
|
|
|
|
update((s) => ({ ...s, leftControlHovering: hovering }));
|
|
|
|
},
|
2023-01-21 22:45:26 +00:00
|
|
|
setShowData(data) {
|
|
|
|
update((s) => ({ ...s, seasonData: data }));
|
|
|
|
},
|
2023-01-22 19:51:58 +00:00
|
|
|
startAirplay() {
|
|
|
|
const videoPlayer = player as any;
|
|
|
|
if (videoPlayer.webkitShowPlaybackTargetPicker)
|
|
|
|
videoPlayer.webkitShowPlaybackTargetPicker();
|
|
|
|
},
|
2023-01-13 23:12:56 +00:00
|
|
|
initPlayer(sourceUrl: string, sourceType: MWStreamType) {
|
2023-01-10 18:53:55 +00:00
|
|
|
this.setVolume(getStoredVolume());
|
|
|
|
|
2023-01-13 23:12:56 +00:00
|
|
|
if (sourceType === MWStreamType.HLS) {
|
2023-01-10 18:53:55 +00:00
|
|
|
if (player.canPlayType("application/vnd.apple.mpegurl")) {
|
|
|
|
player.src = sourceUrl;
|
|
|
|
} else {
|
|
|
|
// HLS support
|
2023-01-15 15:51:55 +00:00
|
|
|
if (!Hls.isSupported()) {
|
|
|
|
update((s) => ({
|
|
|
|
...s,
|
|
|
|
error: {
|
|
|
|
name: `Not supported`,
|
|
|
|
description: "Your browser does not support HLS video",
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
2023-01-10 18:53:55 +00:00
|
|
|
|
|
|
|
const hls = new Hls();
|
|
|
|
|
|
|
|
hls.on(Hls.Events.ERROR, (event, data) => {
|
2023-01-15 15:51:55 +00:00
|
|
|
if (data.fatal) {
|
|
|
|
update((s) => ({
|
|
|
|
...s,
|
|
|
|
error: {
|
|
|
|
name: `error ${data.details}`,
|
|
|
|
description: data.error?.message ?? "Something went wrong",
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
console.error("HLS error", data);
|
2023-01-10 18:53:55 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
hls.attachMedia(player);
|
|
|
|
hls.loadSource(sourceUrl);
|
|
|
|
}
|
2023-01-13 23:12:56 +00:00
|
|
|
} else if (sourceType === MWStreamType.MP4) {
|
2023-01-10 18:53:55 +00:00
|
|
|
player.src = sourceUrl;
|
|
|
|
}
|
2023-01-08 16:51:38 +00:00
|
|
|
},
|
2023-01-08 14:37:16 +00:00
|
|
|
};
|
|
|
|
}
|