mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-02-08 04:05:26 +03:00
Extend improvements to mobile as well
This commit is contained in:
parent
b1d4ea9fa8
commit
7d5709a818
@ -125,7 +125,6 @@ export function getTimelineHoursForDay(
|
|||||||
const data: TimelinePlayback[] = [];
|
const data: TimelinePlayback[] = [];
|
||||||
const startDay = new Date(timestamp * 1000);
|
const startDay = new Date(timestamp * 1000);
|
||||||
startDay.setHours(23, 59, 59, 999);
|
startDay.setHours(23, 59, 59, 999);
|
||||||
const dayEnd = startDay.getTime() / 1000;
|
|
||||||
startDay.setHours(0, 0, 0, 0);
|
startDay.setHours(0, 0, 0, 0);
|
||||||
const startTimestamp = startDay.getTime() / 1000;
|
const startTimestamp = startDay.getTime() / 1000;
|
||||||
let start = startDay.getTime() / 1000;
|
let start = startDay.getTime() / 1000;
|
||||||
|
|||||||
@ -30,12 +30,11 @@ export default function DesktopTimelineView({
|
|||||||
config?.ui?.timezone || Intl.DateTimeFormat().resolvedOptions().timeZone,
|
config?.ui?.timezone || Intl.DateTimeFormat().resolvedOptions().timeZone,
|
||||||
[config]
|
[config]
|
||||||
);
|
);
|
||||||
|
|
||||||
const controllerRef = useRef<DynamicVideoController | undefined>(undefined);
|
const controllerRef = useRef<DynamicVideoController | undefined>(undefined);
|
||||||
|
|
||||||
const [selectedPlayback, setSelectedPlayback] = useState(initialPlayback);
|
|
||||||
|
|
||||||
const initialScrollRef = useRef<HTMLDivElement | null>(null);
|
const initialScrollRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
|
||||||
|
const [selectedPlayback, setSelectedPlayback] = useState(initialPlayback);
|
||||||
const [timelineTime, setTimelineTime] = useState(0);
|
const [timelineTime, setTimelineTime] = useState(0);
|
||||||
|
|
||||||
// handle scrolling to initial timeline item
|
// handle scrolling to initial timeline item
|
||||||
@ -73,6 +72,7 @@ export default function DesktopTimelineView({
|
|||||||
],
|
],
|
||||||
{ revalidateOnFocus: false }
|
{ revalidateOnFocus: false }
|
||||||
);
|
);
|
||||||
|
|
||||||
const timelineGraphData = useMemo(() => {
|
const timelineGraphData = useMemo(() => {
|
||||||
if (!activity) {
|
if (!activity) {
|
||||||
return {};
|
return {};
|
||||||
|
|||||||
@ -1,6 +1,3 @@
|
|||||||
import { useApiHost } from "@/api";
|
|
||||||
import TimelineEventOverlay from "@/components/overlay/TimelineDataOverlay";
|
|
||||||
import VideoPlayer from "@/components/player/VideoPlayer";
|
|
||||||
import ActivityScrubber, {
|
import ActivityScrubber, {
|
||||||
ScrubberItem,
|
ScrubberItem,
|
||||||
} from "@/components/scrubber/ActivityScrubber";
|
} from "@/components/scrubber/ActivityScrubber";
|
||||||
@ -11,9 +8,11 @@ import {
|
|||||||
getTimelineIcon,
|
getTimelineIcon,
|
||||||
} from "@/utils/timelineUtil";
|
} from "@/utils/timelineUtil";
|
||||||
import { renderToStaticMarkup } from "react-dom/server";
|
import { renderToStaticMarkup } from "react-dom/server";
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
import { useMemo, useRef, useState } from "react";
|
||||||
import useSWR from "swr";
|
import useSWR from "swr";
|
||||||
import Player from "video.js/dist/types/player";
|
import DynamicVideoPlayer, {
|
||||||
|
DynamicVideoController,
|
||||||
|
} from "@/components/player/DynamicVideoPlayer";
|
||||||
|
|
||||||
type MobileTimelineViewProps = {
|
type MobileTimelineViewProps = {
|
||||||
playback: TimelinePlayback;
|
playback: TimelinePlayback;
|
||||||
@ -22,34 +21,9 @@ type MobileTimelineViewProps = {
|
|||||||
export default function MobileTimelineView({
|
export default function MobileTimelineView({
|
||||||
playback,
|
playback,
|
||||||
}: MobileTimelineViewProps) {
|
}: MobileTimelineViewProps) {
|
||||||
const apiHost = useApiHost();
|
|
||||||
const { data: config } = useSWR<FrigateConfig>("config");
|
const { data: config } = useSWR<FrigateConfig>("config");
|
||||||
const timezone = useMemo(
|
|
||||||
() =>
|
|
||||||
config?.ui?.timezone || Intl.DateTimeFormat().resolvedOptions().timeZone,
|
|
||||||
[config]
|
|
||||||
);
|
|
||||||
|
|
||||||
const playerRef = useRef<Player | undefined>(undefined);
|
const controllerRef = useRef<DynamicVideoController | undefined>(undefined);
|
||||||
const previewRef = useRef<Player | undefined>(undefined);
|
|
||||||
|
|
||||||
const [scrubbing, setScrubbing] = useState(false);
|
|
||||||
const [focusedItem, setFocusedItem] = useState<Timeline | undefined>(
|
|
||||||
undefined
|
|
||||||
);
|
|
||||||
|
|
||||||
const [seeking, setSeeking] = useState(false);
|
|
||||||
const [timeToSeek, setTimeToSeek] = useState<number | undefined>(undefined);
|
|
||||||
|
|
||||||
const annotationOffset = useMemo(() => {
|
|
||||||
if (!config) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
(config.cameras[playback.camera]?.detect?.annotation_offset || 0) / 1000
|
|
||||||
);
|
|
||||||
}, [config]);
|
|
||||||
|
|
||||||
const [timelineTime, setTimelineTime] = useState(
|
const [timelineTime, setTimelineTime] = useState(
|
||||||
playback.timelineItems.length > 0
|
playback.timelineItems.length > 0
|
||||||
@ -68,165 +42,32 @@ export default function MobileTimelineView({
|
|||||||
{ revalidateOnFocus: false }
|
{ revalidateOnFocus: false }
|
||||||
);
|
);
|
||||||
|
|
||||||
const playbackUri = useMemo(() => {
|
|
||||||
if (!playback) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
const date = new Date(playback.range.start * 1000);
|
|
||||||
return `${apiHost}vod/${date.getFullYear()}-${
|
|
||||||
date.getMonth() + 1
|
|
||||||
}/${date.getDate()}/${date.getHours()}/${
|
|
||||||
playback.camera
|
|
||||||
}/${timezone.replaceAll("/", ",")}/master.m3u8`;
|
|
||||||
}, [playback]);
|
|
||||||
|
|
||||||
const onSelectItem = useCallback(
|
|
||||||
(timeline: Timeline | undefined) => {
|
|
||||||
if (timeline) {
|
|
||||||
setFocusedItem(timeline);
|
|
||||||
const selected = timeline.timestamp;
|
|
||||||
playerRef.current?.pause();
|
|
||||||
|
|
||||||
let seekSeconds = 0;
|
|
||||||
(recordings || []).every((segment) => {
|
|
||||||
// if the next segment is past the desired time, stop calculating
|
|
||||||
if (segment.start_time > selected) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (segment.end_time < selected) {
|
|
||||||
seekSeconds += segment.end_time - segment.start_time;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
seekSeconds +=
|
|
||||||
segment.end_time -
|
|
||||||
segment.start_time -
|
|
||||||
(segment.end_time - selected);
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
playerRef.current?.currentTime(seekSeconds);
|
|
||||||
} else {
|
|
||||||
setFocusedItem(undefined);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[annotationOffset, recordings, playerRef]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onScrubTime = useCallback(
|
|
||||||
(data: { time: Date }) => {
|
|
||||||
if (!playback.relevantPreview) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (playerRef.current?.paused() == false) {
|
|
||||||
setScrubbing(true);
|
|
||||||
playerRef.current?.pause();
|
|
||||||
}
|
|
||||||
|
|
||||||
const seekTimestamp = data.time.getTime() / 1000;
|
|
||||||
const seekTime = seekTimestamp - playback.relevantPreview.start;
|
|
||||||
setTimelineTime(seekTimestamp);
|
|
||||||
setTimeToSeek(Math.round(seekTime));
|
|
||||||
},
|
|
||||||
[scrubbing, playerRef, playback]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onStopScrubbing = useCallback(
|
|
||||||
(data: { time: Date }) => {
|
|
||||||
const playbackTime = data.time.getTime() / 1000;
|
|
||||||
playerRef.current?.currentTime(playbackTime - playback.range.start);
|
|
||||||
setScrubbing(false);
|
|
||||||
playerRef.current?.play();
|
|
||||||
},
|
|
||||||
[playback, playerRef]
|
|
||||||
);
|
|
||||||
|
|
||||||
// handle seeking to next frame when seek is finished
|
|
||||||
useEffect(() => {
|
|
||||||
if (seeking) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (timeToSeek && timeToSeek != previewRef.current?.currentTime()) {
|
|
||||||
setSeeking(true);
|
|
||||||
previewRef.current?.currentTime(timeToSeek);
|
|
||||||
}
|
|
||||||
}, [timeToSeek, seeking]);
|
|
||||||
|
|
||||||
if (!config || !recordings) {
|
if (!config || !recordings) {
|
||||||
return <ActivityIndicator />;
|
return <ActivityIndicator />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<>
|
<DynamicVideoPlayer
|
||||||
<div
|
camera={playback.camera}
|
||||||
className={`relative ${
|
timeRange={playback.range}
|
||||||
playback.relevantPreview && scrubbing ? "hidden" : "visible"
|
cameraPreviews={
|
||||||
}`}
|
playback.relevantPreview ? [playback.relevantPreview] : []
|
||||||
>
|
}
|
||||||
<VideoPlayer
|
onControllerReady={(controller) => {
|
||||||
options={{
|
controllerRef.current = controller;
|
||||||
preload: "auto",
|
controllerRef.current.onPlayerTimeUpdate((timestamp: number) => {
|
||||||
autoplay: true,
|
setTimelineTime(timestamp);
|
||||||
sources: [
|
});
|
||||||
{
|
|
||||||
src: playbackUri,
|
if (playback.timelineItems.length > 0) {
|
||||||
type: "application/vnd.apple.mpegurl",
|
controllerRef.current?.seekToTimestamp(
|
||||||
},
|
playback.timelineItems[0].timestamp,
|
||||||
],
|
true
|
||||||
}}
|
);
|
||||||
seekOptions={{ forward: 10, backward: 5 }}
|
}
|
||||||
onReady={(player) => {
|
}}
|
||||||
playerRef.current = player;
|
/>
|
||||||
player.currentTime(timelineTime - playback.range.start);
|
|
||||||
player.on("playing", () => {
|
|
||||||
onSelectItem(undefined);
|
|
||||||
});
|
|
||||||
}}
|
|
||||||
onDispose={() => {
|
|
||||||
playerRef.current = undefined;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{config && focusedItem ? (
|
|
||||||
<TimelineEventOverlay
|
|
||||||
timeline={focusedItem}
|
|
||||||
cameraConfig={config.cameras[playback.camera]}
|
|
||||||
/>
|
|
||||||
) : undefined}
|
|
||||||
</VideoPlayer>
|
|
||||||
</div>
|
|
||||||
{playback.relevantPreview && (
|
|
||||||
<div className={`${scrubbing ? "visible" : "hidden"}`}>
|
|
||||||
<VideoPlayer
|
|
||||||
options={{
|
|
||||||
preload: "auto",
|
|
||||||
autoplay: true,
|
|
||||||
controls: false,
|
|
||||||
muted: true,
|
|
||||||
loadingSpinner: false,
|
|
||||||
sources: [
|
|
||||||
{
|
|
||||||
src: `${playback.relevantPreview?.src}`,
|
|
||||||
type: "video/mp4",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}}
|
|
||||||
seekOptions={{}}
|
|
||||||
onReady={(player) => {
|
|
||||||
previewRef.current = player;
|
|
||||||
player.pause();
|
|
||||||
player.on("seeked", () => setSeeking(false));
|
|
||||||
}}
|
|
||||||
onDispose={() => {
|
|
||||||
previewRef.current = undefined;
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
<div className="m-1">
|
<div className="m-1">
|
||||||
{playback != undefined && (
|
{playback != undefined && (
|
||||||
<ActivityScrubber
|
<ActivityScrubber
|
||||||
@ -248,17 +89,29 @@ export default function MobileTimelineView({
|
|||||||
max: new Date(playback.range.end * 1000),
|
max: new Date(playback.range.end * 1000),
|
||||||
timeAxis: { scale: "minute", step: 5 },
|
timeAxis: { scale: "minute", step: 5 },
|
||||||
}}
|
}}
|
||||||
timechangeHandler={onScrubTime}
|
timechangeHandler={(data) => {
|
||||||
timechangedHandler={onStopScrubbing}
|
controllerRef.current?.scrubToTimestamp(
|
||||||
|
data.time.getTime() / 1000
|
||||||
|
);
|
||||||
|
setTimelineTime(data.time.getTime() / 1000);
|
||||||
|
}}
|
||||||
|
timechangedHandler={(data) => {
|
||||||
|
controllerRef.current?.seekToTimestamp(
|
||||||
|
data.time.getTime() / 1000,
|
||||||
|
true
|
||||||
|
);
|
||||||
|
}}
|
||||||
selectHandler={(data) => {
|
selectHandler={(data) => {
|
||||||
if (data.items.length > 0) {
|
if (data.items.length > 0) {
|
||||||
const selected = parseFloat(data.items[0].split("-")[0]);
|
const selected = parseFloat(data.items[0].split("-")[0]);
|
||||||
|
|
||||||
onSelectItem(
|
const timeline = playback.timelineItems.find(
|
||||||
playback.timelineItems.find(
|
(timeline) => timeline.timestamp == selected
|
||||||
(timeline) => timeline.timestamp == selected
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (timeline) {
|
||||||
|
controllerRef.current?.seekToTimelineItem(timeline);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@ -12,24 +12,24 @@ export default defineConfig({
|
|||||||
server: {
|
server: {
|
||||||
proxy: {
|
proxy: {
|
||||||
'/api': {
|
'/api': {
|
||||||
target: 'http://192.168.50.106:5000',
|
target: 'http://localhost:5000',
|
||||||
ws: true,
|
ws: true,
|
||||||
},
|
},
|
||||||
'/vod': {
|
'/vod': {
|
||||||
target: 'http://192.168.50.106:5000'
|
target: 'http://localhost:5000'
|
||||||
},
|
},
|
||||||
'/clips': {
|
'/clips': {
|
||||||
target: 'http://192.168.50.106:5000'
|
target: 'http://localhost:5000'
|
||||||
},
|
},
|
||||||
'/exports': {
|
'/exports': {
|
||||||
target: 'http://192.168.50.106:5000'
|
target: 'http://localhost:5000'
|
||||||
},
|
},
|
||||||
'/ws': {
|
'/ws': {
|
||||||
target: 'ws://192.168.50.106:5000',
|
target: 'ws://localhost:5000',
|
||||||
ws: true,
|
ws: true,
|
||||||
},
|
},
|
||||||
'/live': {
|
'/live': {
|
||||||
target: 'ws://192.168.50.106:5000',
|
target: 'ws://localhost:5000',
|
||||||
changeOrigin: true,
|
changeOrigin: true,
|
||||||
ws: true,
|
ws: true,
|
||||||
},
|
},
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user