Separate mobile and desktop views and don't rerender

This commit is contained in:
Nick Mowen 2023-12-31 11:40:44 -07:00
parent e3eadf5cee
commit 6c5615ca43
3 changed files with 328 additions and 273 deletions

View File

@ -25,6 +25,8 @@ import { IoMdArrowBack } from "react-icons/io";
import useOverlayState from "@/hooks/use-overlay-state"; import useOverlayState from "@/hooks/use-overlay-state";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { Dialog, DialogContent } from "@/components/ui/dialog"; import { Dialog, DialogContent } from "@/components/ui/dialog";
import MobileTimelineView from "@/views/history/MobileTimelineView";
import DesktopTimelineView from "@/views/history/DesktopTimelineView";
const API_LIMIT = 200; const API_LIMIT = 200;
@ -245,14 +247,7 @@ function TimelineViewer({
if (isMobile) { if (isMobile) {
return playback != undefined ? ( return playback != undefined ? (
<div className="w-screen absolute left-0 top-20 bottom-0 bg-background z-50"> <div className="w-screen absolute left-0 top-20 bottom-0 bg-background z-50">
{timelineData && ( {timelineData && <MobileTimelineView playback={playback} />}
<HistoryTimelineView
timelineData={timelineData}
allPreviews={allPreviews}
initialPlayback={playback}
isMobile={isMobile}
/>
)}
</div> </div>
) : null; ) : null;
} }
@ -261,11 +256,10 @@ function TimelineViewer({
<Dialog open={playback != undefined} onOpenChange={(_) => onClose()}> <Dialog open={playback != undefined} onOpenChange={(_) => onClose()}>
<DialogContent className="md:max-w-2xl lg:max-w-4xl xl:max-w-6xl 2xl:max-w-7xl 3xl:max-w-[1720px]"> <DialogContent className="md:max-w-2xl lg:max-w-4xl xl:max-w-6xl 2xl:max-w-7xl 3xl:max-w-[1720px]">
{timelineData && playback && ( {timelineData && playback && (
<HistoryTimelineView <DesktopTimelineView
timelineData={timelineData} timelineData={timelineData}
allPreviews={allPreviews} allPreviews={allPreviews}
initialPlayback={playback} initialPlayback={playback}
isMobile={isMobile}
/> />
)} )}
</DialogContent> </DialogContent>

View File

@ -1,41 +1,27 @@
import { useApiHost } from "@/api"; import { useApiHost } from "@/api";
import TimelineEventOverlay from "@/components/overlay/TimelineDataOverlay"; import TimelineEventOverlay from "@/components/overlay/TimelineDataOverlay";
import VideoPlayer from "@/components/player/VideoPlayer"; import VideoPlayer from "@/components/player/VideoPlayer";
import ActivityScrubber, { import ActivityScrubber from "@/components/scrubber/ActivityScrubber";
ScrubberItem,
} from "@/components/scrubber/ActivityScrubber";
import ActivityIndicator from "@/components/ui/activity-indicator"; import ActivityIndicator from "@/components/ui/activity-indicator";
import { FrigateConfig } from "@/types/frigateConfig"; import { FrigateConfig } from "@/types/frigateConfig";
import {
getTimelineDetectionIcon, import { useCallback, useEffect, useMemo, useRef, useState } from "react";
getTimelineIcon,
} from "@/utils/timelineUtil";
import { renderToStaticMarkup } from "react-dom/server";
import React, {
useCallback,
useEffect,
useMemo,
useRef,
useState,
} from "react";
import useSWR from "swr"; import useSWR from "swr";
import Player from "video.js/dist/types/player"; import Player from "video.js/dist/types/player";
import TimelineItemCard from "@/components/card/TimelineItemCard"; import TimelineItemCard from "@/components/card/TimelineItemCard";
import { getTimelineHoursForDay } from "@/utils/historyUtil"; import { getTimelineHoursForDay } from "@/utils/historyUtil";
type HistoryTimelineViewProps = { type DesktopTimelineViewProps = {
timelineData: CardsData; timelineData: CardsData;
allPreviews: Preview[]; allPreviews: Preview[];
initialPlayback: TimelinePlayback; initialPlayback: TimelinePlayback;
isMobile: boolean;
}; };
export default function HistoryTimelineView({ export default function DesktopTimelineView({
timelineData, timelineData,
allPreviews, allPreviews,
initialPlayback, initialPlayback,
isMobile, }: DesktopTimelineViewProps) {
}: HistoryTimelineViewProps) {
const apiHost = useApiHost(); const apiHost = useApiHost();
const { data: config } = useSWR<FrigateConfig>("config"); const { data: config } = useSWR<FrigateConfig>("config");
const timezone = useMemo( const timezone = useMemo(
@ -185,84 +171,18 @@ export default function HistoryTimelineView({
} }
}, [timeToSeek, seeking]); }, [timeToSeek, seeking]);
if (!config || !recordings) { // handle loading main playback when selected hour changes
return <ActivityIndicator />; useEffect(() => {
} if (!playerRef.current) {
return;
}
if (isMobile) { playerRef.current.src({
return ( src: playbackUri,
<MobileView type: "application/vnd.apple.mpegurl",
config={config} });
playerRef={playerRef} }, [playerRef, selectedPlayback]);
previewRef={previewRef}
playback={selectedPlayback}
playbackUri={playbackUri}
timelineTime={timelineTime}
scrubbing={scrubbing}
focusedItem={focusedItem}
setSeeking={setSeeking}
onSelectItem={onSelectItem}
onScrubTime={onScrubTime}
onStopScrubbing={onStopScrubbing}
/>
);
}
return (
<DesktopView
config={config}
playerRef={playerRef}
previewRef={previewRef}
timelineData={timelineData}
allPreviews={allPreviews}
selectedPlayback={selectedPlayback}
setSelectedPlayback={setSelectedPlayback}
playbackUri={playbackUri}
timelineTime={timelineTime}
scrubbing={scrubbing}
focusedItem={focusedItem}
setSeeking={setSeeking}
onSelectItem={onSelectItem}
onScrubTime={onScrubTime}
onStopScrubbing={onStopScrubbing}
/>
);
}
type DesktopViewProps = {
config: FrigateConfig;
playerRef: React.MutableRefObject<Player | undefined>;
previewRef: React.MutableRefObject<Player | undefined>;
timelineData: CardsData;
allPreviews: Preview[];
selectedPlayback: TimelinePlayback;
setSelectedPlayback: (timeline: TimelinePlayback) => void;
playbackUri: string;
timelineTime: number;
scrubbing: boolean;
focusedItem: Timeline | undefined;
setSeeking: (seeking: boolean) => void;
onSelectItem: (timeline: Timeline | undefined) => void;
onScrubTime: ({ time }: { time: Date }) => void;
onStopScrubbing: ({ time }: { time: Date }) => void;
};
function DesktopView({
config,
playerRef,
previewRef,
timelineData,
allPreviews,
selectedPlayback,
setSelectedPlayback,
playbackUri,
timelineTime,
scrubbing,
focusedItem,
setSeeking,
onSelectItem,
onScrubTime,
onStopScrubbing,
}: DesktopViewProps) {
const timelineStack = useMemo( const timelineStack = useMemo(
() => () =>
getTimelineHoursForDay( getTimelineHoursForDay(
@ -274,6 +194,10 @@ function DesktopView({
[] []
); );
if (!config) {
return <ActivityIndicator />;
}
return ( return (
<div className="w-full"> <div className="w-full">
<div className="flex"> <div className="flex">
@ -290,12 +214,6 @@ function DesktopView({
options={{ options={{
preload: "auto", preload: "auto",
autoplay: true, autoplay: true,
sources: [
{
src: playbackUri,
type: "application/vnd.apple.mpegurl",
},
],
}} }}
seekOptions={{ forward: 10, backward: 5 }} seekOptions={{ forward: 10, backward: 5 }}
onReady={(player) => { onReady={(player) => {
@ -311,12 +229,12 @@ function DesktopView({
playerRef.current = undefined; playerRef.current = undefined;
}} }}
> >
{config && focusedItem ? ( {focusedItem && (
<TimelineEventOverlay <TimelineEventOverlay
timeline={focusedItem} timeline={focusedItem}
cameraConfig={config.cameras[selectedPlayback.camera]} cameraConfig={config.cameras[selectedPlayback.camera]}
/> />
) : undefined} )}
</VideoPlayer> </VideoPlayer>
</div> </div>
{selectedPlayback.relevantPreview && ( {selectedPlayback.relevantPreview && (
@ -367,9 +285,11 @@ function DesktopView({
timeline.range.start == selectedPlayback.range.start; timeline.range.start == selectedPlayback.range.start;
return ( return (
<div className={`${isSelected ? "border border-primary" : ""}`}> <div
key={timeline.range.start}
className={`${isSelected ? "border border-primary" : ""}`}
>
<ActivityScrubber <ActivityScrubber
key={timeline.range.start}
items={[]} items={[]}
timeBars={ timeBars={
isSelected && selectedPlayback.relevantPreview isSelected && selectedPlayback.relevantPreview
@ -405,160 +325,3 @@ function DesktopView({
</div> </div>
); );
} }
type MobileViewProps = {
config: FrigateConfig;
playerRef: React.MutableRefObject<Player | undefined>;
previewRef: React.MutableRefObject<Player | undefined>;
playback: TimelinePlayback;
playbackUri: string;
timelineTime: number;
scrubbing: boolean;
focusedItem: Timeline | undefined;
setSeeking: (seeking: boolean) => void;
onSelectItem: (timeline: Timeline | undefined) => void;
onScrubTime: ({ time }: { time: Date }) => void;
onStopScrubbing: ({ time }: { time: Date }) => void;
};
function MobileView({
config,
playerRef,
previewRef,
playback,
playbackUri,
timelineTime,
scrubbing,
focusedItem,
setSeeking,
onSelectItem,
onScrubTime,
onStopScrubbing,
}: MobileViewProps) {
return (
<div className="w-full">
<>
<div
className={`relative ${
playback.relevantPreview && scrubbing ? "hidden" : "visible"
}`}
>
<VideoPlayer
options={{
preload: "auto",
autoplay: true,
sources: [
{
src: playbackUri,
type: "application/vnd.apple.mpegurl",
},
],
}}
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: false,
controls: false,
muted: true,
loadingSpinner: false,
sources: [
{
src: `${playback.relevantPreview?.src}`,
type: "video/mp4",
},
],
}}
seekOptions={{}}
onReady={(player) => {
previewRef.current = player;
player.on("seeked", () => setSeeking(false));
}}
onDispose={() => {
previewRef.current = undefined;
}}
/>
</div>
)}
</>
<div className="m-1">
{playback != undefined && (
<ActivityScrubber
items={timelineItemsToScrubber(playback.timelineItems)}
timeBars={
playback.relevantPreview
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
: []
}
options={{
start: new Date(
Math.max(playback.range.start, timelineTime - 300) * 1000
),
end: new Date(
Math.min(playback.range.end, timelineTime + 300) * 1000
),
snap: null,
min: new Date(playback.range.start * 1000),
max: new Date(playback.range.end * 1000),
timeAxis: { scale: "minute", step: 5 },
}}
timechangeHandler={onScrubTime}
timechangedHandler={onStopScrubbing}
selectHandler={(data) => {
if (data.items.length > 0) {
const selected = data.items[0];
onSelectItem(
playback.timelineItems.find(
(timeline) => timeline.timestamp == selected
)
);
}
}}
/>
)}
</div>
</div>
);
}
function timelineItemsToScrubber(items: Timeline[]): ScrubberItem[] {
return items.map((item) => {
return {
id: item.timestamp,
content: getTimelineContentElement(item),
start: new Date(item.timestamp * 1000),
end: new Date(item.timestamp * 1000),
type: "box",
};
});
}
function getTimelineContentElement(item: Timeline): HTMLElement {
const output = document.createElement(`div-${item.timestamp}`);
output.innerHTML = renderToStaticMarkup(
<div className="flex items-center">
{getTimelineDetectionIcon(item)} : {getTimelineIcon(item)}
</div>
);
return output;
}

View File

@ -0,0 +1,298 @@
import { useApiHost } from "@/api";
import TimelineEventOverlay from "@/components/overlay/TimelineDataOverlay";
import VideoPlayer from "@/components/player/VideoPlayer";
import ActivityScrubber, {
ScrubberItem,
} from "@/components/scrubber/ActivityScrubber";
import ActivityIndicator from "@/components/ui/activity-indicator";
import { FrigateConfig } from "@/types/frigateConfig";
import {
getTimelineDetectionIcon,
getTimelineIcon,
} from "@/utils/timelineUtil";
import { renderToStaticMarkup } from "react-dom/server";
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
import useSWR from "swr";
import Player from "video.js/dist/types/player";
type MobileTimelineViewProps = {
playback: TimelinePlayback;
};
export default function MobileTimelineView({
playback,
}: MobileTimelineViewProps) {
const apiHost = useApiHost();
const { data: config } = useSWR<FrigateConfig>("config");
const timezone = useMemo(
() =>
config?.ui?.timezone || Intl.DateTimeFormat().resolvedOptions().timeZone,
[config]
);
const playerRef = useRef<Player | 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 = useMemo(() => {
if (!playback || playback.timelineItems.length == 0) {
return 0;
}
return playback.timelineItems.at(0)!!.timestamp;
}, [playback]);
const recordingParams = useMemo(() => {
return {
before: playback.range.end,
after: playback.range.start,
};
}, [playback]);
const { data: recordings } = useSWR<Recording[]>(
playback ? [`${playback.camera}/recordings`, recordingParams] : null,
{ 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;
console.log(
"seeking to " +
seekTime +
" comparing " +
new Date(seekTimestamp * 1000) +
" - " +
new Date(playback.relevantPreview.start * 1000)
);
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) {
return <ActivityIndicator />;
}
return (
<div className="w-full">
<>
<div
className={`relative ${
playback.relevantPreview && scrubbing ? "hidden" : "visible"
}`}
>
<VideoPlayer
options={{
preload: "auto",
autoplay: true,
sources: [
{
src: playbackUri,
type: "application/vnd.apple.mpegurl",
},
],
}}
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: false,
controls: false,
muted: true,
loadingSpinner: false,
sources: [
{
src: `${playback.relevantPreview?.src}`,
type: "video/mp4",
},
],
}}
seekOptions={{}}
onReady={(player) => {
previewRef.current = player;
player.on("seeked", () => setSeeking(false));
}}
onDispose={() => {
previewRef.current = undefined;
}}
/>
</div>
)}
</>
<div className="m-1">
{playback != undefined && (
<ActivityScrubber
items={timelineItemsToScrubber(playback.timelineItems)}
timeBars={
playback.relevantPreview
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
: []
}
options={{
start: new Date(
Math.max(playback.range.start, timelineTime - 300) * 1000
),
end: new Date(
Math.min(playback.range.end, timelineTime + 300) * 1000
),
snap: null,
min: new Date(playback.range.start * 1000),
max: new Date(playback.range.end * 1000),
timeAxis: { scale: "minute", step: 5 },
}}
timechangeHandler={onScrubTime}
timechangedHandler={onStopScrubbing}
selectHandler={(data) => {
if (data.items.length > 0) {
const selected = data.items[0];
onSelectItem(
playback.timelineItems.find(
(timeline) => timeline.timestamp == selected
)
);
}
}}
/>
)}
</div>
</div>
);
}
function timelineItemsToScrubber(items: Timeline[]): ScrubberItem[] {
return items.map((item) => {
return {
id: item.timestamp,
content: getTimelineContentElement(item),
start: new Date(item.timestamp * 1000),
end: new Date(item.timestamp * 1000),
type: "box",
};
});
}
function getTimelineContentElement(item: Timeline): HTMLElement {
const output = document.createElement(`div-${item.timestamp}`);
output.innerHTML = renderToStaticMarkup(
<div className="flex items-center">
{getTimelineDetectionIcon(item)} : {getTimelineIcon(item)}
</div>
);
return output;
}