mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-02-07 03:35:26 +03:00
Improve hooks
This commit is contained in:
parent
51011dcd87
commit
7be8ecd109
@ -218,6 +218,7 @@ function History() {
|
|||||||
/>
|
/>
|
||||||
<TimelineViewer
|
<TimelineViewer
|
||||||
timelineData={timelineCards}
|
timelineData={timelineCards}
|
||||||
|
allPreviews={allPreviews || []}
|
||||||
playback={viewingPlayback ? playback : undefined}
|
playback={viewingPlayback ? playback : undefined}
|
||||||
isMobile={isMobile}
|
isMobile={isMobile}
|
||||||
onClose={() => setPlaybackState(undefined)}
|
onClose={() => setPlaybackState(undefined)}
|
||||||
@ -228,6 +229,7 @@ function History() {
|
|||||||
|
|
||||||
type TimelineViewerProps = {
|
type TimelineViewerProps = {
|
||||||
timelineData: CardsData | undefined;
|
timelineData: CardsData | undefined;
|
||||||
|
allPreviews: Preview[];
|
||||||
playback: TimelinePlayback | undefined;
|
playback: TimelinePlayback | undefined;
|
||||||
isMobile: boolean;
|
isMobile: boolean;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
@ -235,6 +237,7 @@ type TimelineViewerProps = {
|
|||||||
|
|
||||||
function TimelineViewer({
|
function TimelineViewer({
|
||||||
timelineData,
|
timelineData,
|
||||||
|
allPreviews,
|
||||||
playback,
|
playback,
|
||||||
isMobile,
|
isMobile,
|
||||||
onClose,
|
onClose,
|
||||||
@ -245,6 +248,7 @@ function TimelineViewer({
|
|||||||
{timelineData && (
|
{timelineData && (
|
||||||
<HistoryTimelineView
|
<HistoryTimelineView
|
||||||
timelineData={timelineData}
|
timelineData={timelineData}
|
||||||
|
allPreviews={allPreviews}
|
||||||
initialPlayback={playback}
|
initialPlayback={playback}
|
||||||
isMobile={isMobile}
|
isMobile={isMobile}
|
||||||
/>
|
/>
|
||||||
@ -259,6 +263,7 @@ function TimelineViewer({
|
|||||||
{timelineData && playback && (
|
{timelineData && playback && (
|
||||||
<HistoryTimelineView
|
<HistoryTimelineView
|
||||||
timelineData={timelineData}
|
timelineData={timelineData}
|
||||||
|
allPreviews={allPreviews}
|
||||||
initialPlayback={playback}
|
initialPlayback={playback}
|
||||||
isMobile={isMobile}
|
isMobile={isMobile}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@ -105,6 +105,7 @@ export function getHourlyTimelineData(
|
|||||||
export function getTimelineHoursForDay(
|
export function getTimelineHoursForDay(
|
||||||
camera: string,
|
camera: string,
|
||||||
cards: CardsData,
|
cards: CardsData,
|
||||||
|
allPreviews: Preview[],
|
||||||
timestamp: number
|
timestamp: number
|
||||||
): TimelinePlayback[] {
|
): TimelinePlayback[] {
|
||||||
const now = new Date();
|
const now = new Date();
|
||||||
@ -155,11 +156,15 @@ export function getTimelineHoursForDay(
|
|||||||
return [];
|
return [];
|
||||||
})
|
})
|
||||||
: [];
|
: [];
|
||||||
|
const relevantPreview = Object.values(allPreviews || []).find(
|
||||||
|
(preview) =>
|
||||||
|
preview.camera == camera && preview.start < start && preview.end > start
|
||||||
|
);
|
||||||
data.push({
|
data.push({
|
||||||
camera,
|
camera,
|
||||||
range: { start, end },
|
range: { start, end },
|
||||||
timelineItems,
|
timelineItems,
|
||||||
relevantPreview: undefined,
|
relevantPreview,
|
||||||
});
|
});
|
||||||
start = startDay.getTime() / 1000;
|
start = startDay.getTime() / 1000;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -25,12 +25,14 @@ import { getTimelineHoursForDay } from "@/utils/historyUtil";
|
|||||||
|
|
||||||
type HistoryTimelineViewProps = {
|
type HistoryTimelineViewProps = {
|
||||||
timelineData: CardsData;
|
timelineData: CardsData;
|
||||||
|
allPreviews: Preview[];
|
||||||
initialPlayback: TimelinePlayback;
|
initialPlayback: TimelinePlayback;
|
||||||
isMobile: boolean;
|
isMobile: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function HistoryTimelineView({
|
export default function HistoryTimelineView({
|
||||||
timelineData,
|
timelineData,
|
||||||
|
allPreviews,
|
||||||
initialPlayback,
|
initialPlayback,
|
||||||
isMobile,
|
isMobile,
|
||||||
}: HistoryTimelineViewProps) {
|
}: HistoryTimelineViewProps) {
|
||||||
@ -43,10 +45,6 @@ export default function HistoryTimelineView({
|
|||||||
);
|
);
|
||||||
|
|
||||||
const [selectedPlayback, setSelectedPlayback] = useState(initialPlayback);
|
const [selectedPlayback, setSelectedPlayback] = useState(initialPlayback);
|
||||||
const hasRelevantPreview = useMemo(
|
|
||||||
() => selectedPlayback.relevantPreview != undefined,
|
|
||||||
[selectedPlayback]
|
|
||||||
);
|
|
||||||
|
|
||||||
const playerRef = useRef<Player | undefined>(undefined);
|
const playerRef = useRef<Player | undefined>(undefined);
|
||||||
const previewRef = useRef<Player | undefined>(undefined);
|
const previewRef = useRef<Player | undefined>(undefined);
|
||||||
@ -65,10 +63,10 @@ export default function HistoryTimelineView({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
(config.cameras[selectedPlayback.camera]?.detect?.annotation_offset ||
|
(config.cameras[initialPlayback.camera]?.detect?.annotation_offset || 0) /
|
||||||
0) / 1000
|
1000
|
||||||
);
|
);
|
||||||
}, [config, selectedPlayback]);
|
}, [config]);
|
||||||
|
|
||||||
const timelineTime = useMemo(() => {
|
const timelineTime = useMemo(() => {
|
||||||
if (!selectedPlayback || selectedPlayback.timelineItems.length == 0) {
|
if (!selectedPlayback || selectedPlayback.timelineItems.length == 0) {
|
||||||
@ -139,7 +137,7 @@ export default function HistoryTimelineView({
|
|||||||
|
|
||||||
const onScrubTime = useCallback(
|
const onScrubTime = useCallback(
|
||||||
(data: { time: Date }) => {
|
(data: { time: Date }) => {
|
||||||
if (!hasRelevantPreview) {
|
if (!selectedPlayback.relevantPreview) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,7 +190,6 @@ export default function HistoryTimelineView({
|
|||||||
playback={selectedPlayback}
|
playback={selectedPlayback}
|
||||||
playbackUri={playbackUri}
|
playbackUri={playbackUri}
|
||||||
timelineTime={timelineTime}
|
timelineTime={timelineTime}
|
||||||
hasRelevantPreview={hasRelevantPreview}
|
|
||||||
scrubbing={scrubbing}
|
scrubbing={scrubbing}
|
||||||
focusedItem={focusedItem}
|
focusedItem={focusedItem}
|
||||||
setSeeking={setSeeking}
|
setSeeking={setSeeking}
|
||||||
@ -209,11 +206,11 @@ export default function HistoryTimelineView({
|
|||||||
playerRef={playerRef}
|
playerRef={playerRef}
|
||||||
previewRef={previewRef}
|
previewRef={previewRef}
|
||||||
timelineData={timelineData}
|
timelineData={timelineData}
|
||||||
|
allPreviews={allPreviews}
|
||||||
selectedPlayback={selectedPlayback}
|
selectedPlayback={selectedPlayback}
|
||||||
setSelectedPlayback={setSelectedPlayback}
|
setSelectedPlayback={setSelectedPlayback}
|
||||||
playbackUri={playbackUri}
|
playbackUri={playbackUri}
|
||||||
timelineTime={timelineTime}
|
timelineTime={timelineTime}
|
||||||
hasRelevantPreview={hasRelevantPreview}
|
|
||||||
scrubbing={scrubbing}
|
scrubbing={scrubbing}
|
||||||
focusedItem={focusedItem}
|
focusedItem={focusedItem}
|
||||||
setSeeking={setSeeking}
|
setSeeking={setSeeking}
|
||||||
@ -229,11 +226,11 @@ type DesktopViewProps = {
|
|||||||
playerRef: React.MutableRefObject<Player | undefined>;
|
playerRef: React.MutableRefObject<Player | undefined>;
|
||||||
previewRef: React.MutableRefObject<Player | undefined>;
|
previewRef: React.MutableRefObject<Player | undefined>;
|
||||||
timelineData: CardsData;
|
timelineData: CardsData;
|
||||||
|
allPreviews: Preview[];
|
||||||
selectedPlayback: TimelinePlayback;
|
selectedPlayback: TimelinePlayback;
|
||||||
setSelectedPlayback: (timeline: TimelinePlayback) => void;
|
setSelectedPlayback: (timeline: TimelinePlayback) => void;
|
||||||
playbackUri: string;
|
playbackUri: string;
|
||||||
timelineTime: number;
|
timelineTime: number;
|
||||||
hasRelevantPreview: boolean;
|
|
||||||
scrubbing: boolean;
|
scrubbing: boolean;
|
||||||
focusedItem: Timeline | undefined;
|
focusedItem: Timeline | undefined;
|
||||||
setSeeking: (seeking: boolean) => void;
|
setSeeking: (seeking: boolean) => void;
|
||||||
@ -246,11 +243,11 @@ function DesktopView({
|
|||||||
playerRef,
|
playerRef,
|
||||||
previewRef,
|
previewRef,
|
||||||
timelineData,
|
timelineData,
|
||||||
|
allPreviews,
|
||||||
selectedPlayback,
|
selectedPlayback,
|
||||||
setSelectedPlayback,
|
setSelectedPlayback,
|
||||||
playbackUri,
|
playbackUri,
|
||||||
timelineTime,
|
timelineTime,
|
||||||
hasRelevantPreview,
|
|
||||||
scrubbing,
|
scrubbing,
|
||||||
focusedItem,
|
focusedItem,
|
||||||
setSeeking,
|
setSeeking,
|
||||||
@ -258,14 +255,16 @@ function DesktopView({
|
|||||||
onScrubTime,
|
onScrubTime,
|
||||||
onStopScrubbing,
|
onStopScrubbing,
|
||||||
}: DesktopViewProps) {
|
}: DesktopViewProps) {
|
||||||
const timelineStack =
|
const timelineStack = useMemo(
|
||||||
selectedPlayback == undefined
|
() =>
|
||||||
? []
|
getTimelineHoursForDay(
|
||||||
: getTimelineHoursForDay(
|
selectedPlayback.camera,
|
||||||
selectedPlayback.camera,
|
timelineData,
|
||||||
timelineData,
|
allPreviews,
|
||||||
timelineTime
|
timelineTime
|
||||||
);
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
@ -274,7 +273,9 @@ function DesktopView({
|
|||||||
<div className="w-2/3 bg-black flex justify-center items-center">
|
<div className="w-2/3 bg-black flex justify-center items-center">
|
||||||
<div
|
<div
|
||||||
className={`w-full relative ${
|
className={`w-full relative ${
|
||||||
hasRelevantPreview && scrubbing ? "hidden" : "visible"
|
selectedPlayback.relevantPreview != undefined && scrubbing
|
||||||
|
? "hidden"
|
||||||
|
: "visible"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<VideoPlayer
|
<VideoPlayer
|
||||||
@ -310,7 +311,7 @@ function DesktopView({
|
|||||||
) : undefined}
|
) : undefined}
|
||||||
</VideoPlayer>
|
</VideoPlayer>
|
||||||
</div>
|
</div>
|
||||||
{hasRelevantPreview && (
|
{selectedPlayback.relevantPreview && (
|
||||||
<div className={`w-full ${scrubbing ? "visible" : "hidden"}`}>
|
<div className={`w-full ${scrubbing ? "visible" : "hidden"}`}>
|
||||||
<VideoPlayer
|
<VideoPlayer
|
||||||
options={{
|
options={{
|
||||||
@ -352,7 +353,7 @@ function DesktopView({
|
|||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="m-1 max-h-96 overflow-auto">
|
<div className="m-1 max-h-72 2xl:max-h-80 3xl:max-h-96 overflow-auto">
|
||||||
{timelineStack.map((timeline) => {
|
{timelineStack.map((timeline) => {
|
||||||
const isSelected =
|
const isSelected =
|
||||||
timeline.range.start == selectedPlayback.range.start;
|
timeline.range.start == selectedPlayback.range.start;
|
||||||
@ -363,7 +364,7 @@ function DesktopView({
|
|||||||
key={timeline.range.start}
|
key={timeline.range.start}
|
||||||
items={[]}
|
items={[]}
|
||||||
timeBars={
|
timeBars={
|
||||||
hasRelevantPreview
|
isSelected && selectedPlayback.relevantPreview
|
||||||
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
|
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
|
||||||
: []
|
: []
|
||||||
}
|
}
|
||||||
@ -375,7 +376,7 @@ function DesktopView({
|
|||||||
}}
|
}}
|
||||||
timechangeHandler={onScrubTime}
|
timechangeHandler={onScrubTime}
|
||||||
timechangedHandler={onStopScrubbing}
|
timechangedHandler={onStopScrubbing}
|
||||||
doubleClickHandler={(data) => {
|
doubleClickHandler={() => {
|
||||||
setSelectedPlayback(timeline);
|
setSelectedPlayback(timeline);
|
||||||
}}
|
}}
|
||||||
selectHandler={(data) => {
|
selectHandler={(data) => {
|
||||||
@ -404,7 +405,6 @@ type MobileViewProps = {
|
|||||||
playback: TimelinePlayback;
|
playback: TimelinePlayback;
|
||||||
playbackUri: string;
|
playbackUri: string;
|
||||||
timelineTime: number;
|
timelineTime: number;
|
||||||
hasRelevantPreview: boolean;
|
|
||||||
scrubbing: boolean;
|
scrubbing: boolean;
|
||||||
focusedItem: Timeline | undefined;
|
focusedItem: Timeline | undefined;
|
||||||
setSeeking: (seeking: boolean) => void;
|
setSeeking: (seeking: boolean) => void;
|
||||||
@ -419,7 +419,6 @@ function MobileView({
|
|||||||
playback,
|
playback,
|
||||||
playbackUri,
|
playbackUri,
|
||||||
timelineTime,
|
timelineTime,
|
||||||
hasRelevantPreview,
|
|
||||||
scrubbing,
|
scrubbing,
|
||||||
focusedItem,
|
focusedItem,
|
||||||
setSeeking,
|
setSeeking,
|
||||||
@ -432,7 +431,7 @@ function MobileView({
|
|||||||
<>
|
<>
|
||||||
<div
|
<div
|
||||||
className={`relative ${
|
className={`relative ${
|
||||||
hasRelevantPreview && scrubbing ? "hidden" : "visible"
|
playback.relevantPreview && scrubbing ? "hidden" : "visible"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<VideoPlayer
|
<VideoPlayer
|
||||||
@ -466,7 +465,7 @@ function MobileView({
|
|||||||
) : undefined}
|
) : undefined}
|
||||||
</VideoPlayer>
|
</VideoPlayer>
|
||||||
</div>
|
</div>
|
||||||
{hasRelevantPreview && (
|
{playback.relevantPreview && (
|
||||||
<div className={`${scrubbing ? "visible" : "hidden"}`}>
|
<div className={`${scrubbing ? "visible" : "hidden"}`}>
|
||||||
<VideoPlayer
|
<VideoPlayer
|
||||||
options={{
|
options={{
|
||||||
@ -499,7 +498,7 @@ function MobileView({
|
|||||||
<ActivityScrubber
|
<ActivityScrubber
|
||||||
items={timelineItemsToScrubber(playback.timelineItems)}
|
items={timelineItemsToScrubber(playback.timelineItems)}
|
||||||
timeBars={
|
timeBars={
|
||||||
hasRelevantPreview
|
playback.relevantPreview
|
||||||
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
|
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
|
||||||
: []
|
: []
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user