blakeblackshear.frigate/web/src/components/HistoryViewer/HistoryVideo.tsx

146 lines
3.5 KiB
TypeScript
Raw Normal View History

2022-02-27 15:04:12 +01:00
import { h } from 'preact';
import { useCallback, useEffect, useRef, useState } from 'preact/hooks';
import { useApiHost } from '../../api';
import { isNullOrUndefined } from '../../utils/objectUtils';
interface OnTimeUpdateEvent {
timestamp: number;
isPlaying: boolean;
}
interface VideoProperties {
posterUrl: string;
videoUrl: string;
}
interface HistoryVideoProps {
2022-03-06 05:16:31 +01:00
id?: string;
2022-02-27 15:04:12 +01:00
isPlaying: boolean;
currentTime: number;
onTimeUpdate?: (event: OnTimeUpdateEvent) => void;
onPause: () => void;
onPlay: () => void;
}
export const HistoryVideo = ({
id,
isPlaying: videoIsPlaying,
currentTime,
onTimeUpdate,
onPause,
onPlay,
}: HistoryVideoProps) => {
const apiHost = useApiHost();
const videoRef = useRef<HTMLVideoElement | null>(null);
const [posterLoaded, setPosterLoaded] = useState(false);
const [videoHeight, setVideoHeight] = useState<number | undefined>(undefined);
2022-03-06 05:16:31 +01:00
const [videoProperties, setVideoProperties] = useState<VideoProperties>({
posterUrl: '',
videoUrl: '',
});
2022-02-27 15:04:12 +01:00
const videoCallback = useCallback(
(domNode: any) => {
videoRef.current = domNode;
if (posterLoaded) {
setVideoHeight(videoRef.current?.offsetHeight);
}
},
[posterLoaded]
);
2022-02-27 15:04:12 +01:00
useEffect(() => {
const idExists = !isNullOrUndefined(id);
if (idExists) {
if (videoRef.current && !videoRef.current.paused) {
2022-03-06 05:16:31 +01:00
videoRef.current = null;
2022-02-27 15:04:12 +01:00
}
const posterUrl = `${apiHost}/api/events/${id}/snapshot.jpg`;
const poster = new Image();
poster.src = posterUrl;
poster.onload = () => {
setPosterLoaded(true);
};
2022-02-27 15:04:12 +01:00
setVideoProperties({
posterUrl,
2022-02-27 15:04:12 +01:00
videoUrl: `${apiHost}/vod/event/${id}/index.m3u8`,
});
} else {
2022-03-06 05:16:31 +01:00
setVideoProperties({
posterUrl: '',
videoUrl: '',
});
2022-02-27 15:04:12 +01:00
}
}, [id, videoHeight, videoRef, apiHost]);
useEffect(() => {
const playVideo = (video: HTMLMediaElement) => video.play();
const attemptPlayVideo = (video: HTMLMediaElement) => {
const videoHasNotLoaded = video.readyState <= 1;
if (videoHasNotLoaded) {
video.oncanplay = () => {
playVideo(video);
};
video.load();
} else {
playVideo(video);
}
};
const video = videoRef.current;
const videoExists = !isNullOrUndefined(video);
2022-03-06 05:16:31 +01:00
if (video && videoExists) {
2022-02-27 15:04:12 +01:00
if (videoIsPlaying) {
attemptPlayVideo(video);
} else {
video.pause();
}
}
}, [videoIsPlaying, videoRef]);
useEffect(() => {
const video = videoRef.current;
const videoExists = !isNullOrUndefined(video);
const hasSeeked = currentTime >= 0;
2022-03-06 05:16:31 +01:00
if (video && videoExists && hasSeeked) {
2022-02-27 15:04:12 +01:00
video.currentTime = currentTime;
}
}, [currentTime, videoRef]);
const onTimeUpdateHandler = useCallback(
(event: Event) => {
const target = event.target as HTMLMediaElement;
const timeUpdateEvent = {
isPlaying: videoIsPlaying,
timestamp: target.currentTime,
};
onTimeUpdate && onTimeUpdate(timeUpdateEvent);
},
[videoIsPlaying, onTimeUpdate]
);
const { posterUrl, videoUrl } = videoProperties;
2022-02-27 15:04:12 +01:00
return (
<video
ref={videoCallback}
2022-02-27 15:04:12 +01:00
key={posterUrl}
onTimeUpdate={onTimeUpdateHandler}
onPause={onPause}
onPlay={onPlay}
poster={posterUrl}
preload="metadata"
2022-02-27 15:04:12 +01:00
controls
style={videoHeight ? { minHeight: `${videoHeight}px` } : {}}
2022-02-27 15:04:12 +01:00
playsInline
>
<source type="application/vnd.apple.mpegurl" src={videoUrl} />
2022-02-27 15:04:12 +01:00
</video>
);
};