2022-09-07 09:00:28 +02:00
|
|
|
import React, { FC } from 'react';
|
2023-05-06 17:40:42 -07:00
|
|
|
import videojs from 'video.js';
|
|
|
|
import type VideoJsPlayer from 'video.js/dist/types/player';
|
|
|
|
|
2022-09-07 09:00:28 +02:00
|
|
|
import styles from './VideoJS.module.scss';
|
2022-04-26 19:29:13 -07:00
|
|
|
|
|
|
|
require('video.js/dist/video-js.css');
|
|
|
|
|
2022-09-07 09:00:28 +02:00
|
|
|
export type VideoJSProps = {
|
2023-05-06 17:40:42 -07:00
|
|
|
options: any;
|
|
|
|
onReady: (player: VideoJsPlayer, vjsInstance: typeof videojs) => void;
|
2022-09-07 09:00:28 +02:00
|
|
|
};
|
2022-04-26 19:29:13 -07:00
|
|
|
|
2022-09-07 09:00:28 +02:00
|
|
|
export const VideoJS: FC<VideoJSProps> = ({ options, onReady }) => {
|
2023-01-01 01:08:54 +01:00
|
|
|
const videoRef = React.useRef<HTMLVideoElement | null>(null);
|
|
|
|
const playerRef = React.useRef<VideoJsPlayer | null>(null);
|
2022-04-26 19:29:13 -07:00
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
// Make sure Video.js player is only initialized once
|
|
|
|
if (!playerRef.current) {
|
|
|
|
const videoElement = videoRef.current;
|
|
|
|
|
2022-05-11 23:31:31 -07:00
|
|
|
// eslint-disable-next-line no-multi-assign
|
2023-01-01 01:08:54 +01:00
|
|
|
const player: VideoJsPlayer = (playerRef.current = videojs(videoElement, options, () => {
|
2022-11-12 23:35:53 -08:00
|
|
|
console.debug('player is ready');
|
2022-06-02 14:23:51 -07:00
|
|
|
return onReady && onReady(player, videojs);
|
2022-04-26 19:29:13 -07:00
|
|
|
}));
|
|
|
|
|
|
|
|
player.autoplay(options.autoplay);
|
|
|
|
player.src(options.sources);
|
|
|
|
}
|
2022-11-17 22:10:48 -08:00
|
|
|
|
|
|
|
// Add a cachebuster param to playlist URLs.
|
2023-05-06 20:06:13 -07:00
|
|
|
if (
|
|
|
|
(videojs.getPlayer(videoRef.current).tech({ IWillNotUseThisInPlugins: true }) as any)?.vhs
|
|
|
|
) {
|
|
|
|
(
|
|
|
|
videojs.getPlayer(videoRef.current).tech({ IWillNotUseThisInPlugins: true }) as any
|
|
|
|
).vhs.xhr.beforeRequest = o => {
|
2023-05-06 17:40:42 -07:00
|
|
|
if (o.uri.match('m3u8')) {
|
|
|
|
const cachebuster = Math.random().toString(16).substr(2, 8);
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
|
|
o.uri = `${o.uri}?cachebust=${cachebuster}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
};
|
|
|
|
}
|
2022-04-26 19:29:13 -07:00
|
|
|
}, [options, videoRef]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div data-vjs-player>
|
2022-05-11 23:31:31 -07:00
|
|
|
{/* eslint-disable-next-line jsx-a11y/media-has-caption */}
|
2022-09-07 09:00:28 +02:00
|
|
|
<video
|
|
|
|
ref={videoRef}
|
2022-10-12 19:24:37 -07:00
|
|
|
className={`video-js vjs-big-play-centered vjs-show-big-play-button-on-pause ${styles.player} vjs-owncast`}
|
2022-09-07 09:00:28 +02:00
|
|
|
/>
|
2022-04-26 19:29:13 -07:00
|
|
|
</div>
|
|
|
|
);
|
2022-09-07 09:00:28 +02:00
|
|
|
};
|