Compare commits

..

10 Commits

Author SHA1 Message Date
0x464e
10ced9195d
Defer setRecording and return true from hook for cleanup 2026-03-20 21:59:37 +02:00
0x464e
55a858783c
Bugfix: Share dialog was not receiving the player timestamp after removing key that triggered remounts 2026-03-20 21:54:07 +02:00
0x464e
dd5ad11a6e
Fixes based off PR review comments 2026-03-20 21:19:00 +02:00
Otto
738197b184
Use normal separator
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
2026-03-20 20:49:23 +02:00
0x464e
9706bcbb2e
Revert "Bugfix: guard against showing toasts twice"
This reverts commit 99fa5e1dee.
2026-03-20 20:04:01 +02:00
0x464e
8381c12920
Clamp future timestamps to now 2026-03-20 19:52:06 +02:00
0x464e
99fa5e1dee
Bugfix: guard against showing toasts twice
Because this effect ends up running multiple times
2026-03-20 18:57:24 +02:00
0x464e
eead0535a3
Make share description clearer 2026-03-20 18:46:17 +02:00
0x464e
a0956ebe0f
Add missing cancel button & separators to dialog 2026-03-20 18:44:49 +02:00
0x464e
5dad8cfb2d
Fixes/improvements based off PR review comments 2026-03-20 18:30:29 +02:00
8 changed files with 164 additions and 65 deletions

View File

@ -131,8 +131,6 @@
"close": "Close", "close": "Close",
"copy": "Copy", "copy": "Copy",
"copiedToClipboard": "Copied to clipboard", "copiedToClipboard": "Copied to clipboard",
"shareTimestamp": "Share Timestamp",
"shareTimestampUrl": "Share Timestamp URL",
"back": "Back", "back": "Back",
"history": "History", "history": "History",
"fullscreen": "Fullscreen", "fullscreen": "Fullscreen",

View File

@ -100,11 +100,13 @@
}, },
"recording": { "recording": {
"shareTimestamp": { "shareTimestamp": {
"label": "Share Timestamp",
"title": "Share Review Timestamp", "title": "Share Review Timestamp",
"description": "Share the current player position or choose a custom timestamp.", "description": "Share a timestamped URL of current player position or choose a custom timestamp. Note that this is not a public share URL and is only accessible to users with access to Frigate and this camera.",
"current": "Current Player Timestamp", "current": "Current Player Timestamp",
"custom": "Custom Timestamp", "custom": "Custom Timestamp",
"customDescription": "Pick a specific point in time to share.", "customDescription": "Pick a specific point in time to share.",
"button": "Share Timestamp URL",
"shareTitle": "Frigate Review Timestamp: {{camera}}" "shareTitle": "Frigate Review Timestamp: {{camera}}"
}, },
"confirmDelete": { "confirmDelete": {

View File

@ -40,7 +40,7 @@ export default function ActionsDropdown({
{t("menu.export", { ns: "common" })} {t("menu.export", { ns: "common" })}
</DropdownMenuItem> </DropdownMenuItem>
<DropdownMenuItem onClick={onShareTimestampClick}> <DropdownMenuItem onClick={onShareTimestampClick}>
{t("button.shareTimestamp", { ns: "common" })} {t("recording.shareTimestamp.label", { ns: "components/dialog" })}
</DropdownMenuItem> </DropdownMenuItem>
<DropdownMenuItem onClick={onDebugReplayClick}> <DropdownMenuItem onClick={onDebugReplayClick}>
{t("title", { ns: "views/replay" })} {t("title", { ns: "views/replay" })}

View File

@ -26,6 +26,7 @@ import SaveExportOverlay from "./SaveExportOverlay";
import { isIOS, isMobile } from "react-device-detect"; import { isIOS, isMobile } from "react-device-detect";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { ShareTimestampContent } from "./ShareTimestampDialog";
type DrawerMode = type DrawerMode =
| "none" | "none"
@ -70,7 +71,7 @@ type MobileReviewSettingsDrawerProps = {
debugReplayRange?: TimeRange; debugReplayRange?: TimeRange;
setDebugReplayMode?: (mode: ExportMode) => void; setDebugReplayMode?: (mode: ExportMode) => void;
setDebugReplayRange?: (range: TimeRange | undefined) => void; setDebugReplayRange?: (range: TimeRange | undefined) => void;
onShareTimestampClick?: () => void; onShareTimestamp?: (timestamp: number) => void;
onUpdateFilter: (filter: ReviewFilter) => void; onUpdateFilter: (filter: ReviewFilter) => void;
setRange: (range: TimeRange | undefined) => void; setRange: (range: TimeRange | undefined) => void;
setMode: (mode: ExportMode) => void; setMode: (mode: ExportMode) => void;
@ -94,7 +95,7 @@ export default function MobileReviewSettingsDrawer({
debugReplayRange, debugReplayRange,
setDebugReplayMode = () => {}, setDebugReplayMode = () => {},
setDebugReplayRange = () => {}, setDebugReplayRange = () => {},
onShareTimestampClick = () => {}, onShareTimestamp = () => {},
onUpdateFilter, onUpdateFilter,
setRange, setRange,
setMode, setMode,
@ -112,6 +113,15 @@ export default function MobileReviewSettingsDrawer({
"1" | "5" | "custom" | "timeline" "1" | "5" | "custom" | "timeline"
>("1"); >("1");
const [isDebugReplayStarting, setIsDebugReplayStarting] = useState(false); const [isDebugReplayStarting, setIsDebugReplayStarting] = useState(false);
const [selectedShareOption, setSelectedShareOption] = useState<
"current" | "custom"
>("current");
const [shareTimestampAtOpen, setShareTimestampAtOpen] = useState(
Math.floor(currentTime),
);
const [customShareTimestamp, setCustomShareTimestamp] = useState(
Math.floor(currentTime),
);
// exports // exports
@ -284,14 +294,22 @@ export default function MobileReviewSettingsDrawer({
{features.includes("share-timestamp") && ( {features.includes("share-timestamp") && (
<Button <Button
className="flex w-full items-center justify-center gap-2" className="flex w-full items-center justify-center gap-2"
aria-label={t("button.shareTimestamp", { ns: "common" })} aria-label={t("recording.shareTimestamp.label", {
ns: "components/dialog",
})}
onClick={() => { onClick={() => {
setDrawerMode("none"); const initialTimestamp = Math.floor(currentTime);
onShareTimestampClick();
setShareTimestampAtOpen(initialTimestamp);
setCustomShareTimestamp(initialTimestamp);
setSelectedShareOption("current");
setDrawerMode("share-timestamp");
}} }}
> >
<LuShare2 className="size-5 rounded-md bg-secondary-foreground stroke-secondary p-1" /> <LuShare2 className="size-5 rounded-md bg-secondary-foreground stroke-secondary p-1" />
{t("button.shareTimestamp", { ns: "common" })} {t("recording.shareTimestamp.label", {
ns: "components/dialog",
})}
</Button> </Button>
)} )}
{features.includes("calendar") && ( {features.includes("calendar") && (
@ -496,6 +514,34 @@ export default function MobileReviewSettingsDrawer({
}} }}
/> />
); );
} else if (drawerMode == "share-timestamp") {
content = (
<div className="w-full">
<div className="relative h-8 w-full">
<div
className="absolute left-0 text-selected"
onClick={() => setDrawerMode("select")}
>
{t("button.back", { ns: "common" })}
</div>
<div className="absolute left-1/2 -translate-x-1/2 text-muted-foreground">
{t("recording.shareTimestamp.title", { ns: "components/dialog" })}
</div>
</div>
<ShareTimestampContent
currentTime={shareTimestampAtOpen}
selectedOption={selectedShareOption}
setSelectedOption={setSelectedShareOption}
customTimestamp={customShareTimestamp}
setCustomTimestamp={setCustomShareTimestamp}
onShareTimestamp={(timestamp) => {
onShareTimestamp(timestamp);
setDrawerMode("none");
}}
onCancel={() => setDrawerMode("select")}
/>
</div>
);
} }
return ( return (

View File

@ -3,6 +3,7 @@ import {
Dialog, Dialog,
DialogContent, DialogContent,
DialogDescription, DialogDescription,
DialogFooter,
DialogHeader, DialogHeader,
DialogTitle, DialogTitle,
} from "@/components/ui/dialog"; } from "@/components/ui/dialog";
@ -14,7 +15,9 @@ import {
PopoverTrigger, PopoverTrigger,
} from "@/components/ui/popover"; } from "@/components/ui/popover";
import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group"; import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group";
import { Separator } from "@/components/ui/separator";
import { useFormattedTimestamp } from "@/hooks/use-date-utils"; import { useFormattedTimestamp } from "@/hooks/use-date-utils";
import { cn } from "@/lib/utils";
import { FrigateConfig } from "@/types/frigateConfig"; import { FrigateConfig } from "@/types/frigateConfig";
import { getUTCOffset } from "@/utils/dateUtil"; import { getUTCOffset } from "@/utils/dateUtil";
import { useCallback, useMemo, useState } from "react"; import { useCallback, useMemo, useState } from "react";
@ -22,13 +25,16 @@ import useSWR from "swr";
import { TimezoneAwareCalendar } from "./ReviewActivityCalendar"; import { TimezoneAwareCalendar } from "./ReviewActivityCalendar";
import { FaCalendarAlt } from "react-icons/fa"; import { FaCalendarAlt } from "react-icons/fa";
import { isDesktop, isIOS, isMobile } from "react-device-detect"; import { isDesktop, isIOS, isMobile } from "react-device-detect";
import { LuShare2 } from "react-icons/lu";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
type ShareTimestampDialogProps = { type ShareTimestampDialogProps = {
currentTime: number; currentTime: number;
open: boolean; open: boolean;
onOpenChange: (open: boolean) => void; onOpenChange: (open: boolean) => void;
selectedOption: "current" | "custom";
setSelectedOption: (option: "current" | "custom") => void;
customTimestamp: number;
setCustomTimestamp: (timestamp: number) => void;
onShareTimestamp: (timestamp: number) => void; onShareTimestamp: (timestamp: number) => void;
}; };
@ -36,26 +42,17 @@ export default function ShareTimestampDialog({
currentTime, currentTime,
open, open,
onOpenChange, onOpenChange,
selectedOption,
setSelectedOption,
customTimestamp,
setCustomTimestamp,
onShareTimestamp, onShareTimestamp,
}: Readonly<ShareTimestampDialogProps>) { }: Readonly<ShareTimestampDialogProps>) {
const { t } = useTranslation(["components/dialog"]); const { t } = useTranslation(["components/dialog"]);
const [selectedOption, setSelectedOption] = useState<"current" | "custom">(
"current",
);
const [customTimestamp, setCustomTimestamp] = useState(
Math.floor(currentTime),
);
const handleOpenChange = useCallback( const handleOpenChange = useCallback(
(nextOpen: boolean) => { (nextOpen: boolean) => onOpenChange(nextOpen),
if (nextOpen) { [onOpenChange],
setSelectedOption("current");
setCustomTimestamp(Math.floor(currentTime));
}
onOpenChange(nextOpen);
},
[currentTime, onOpenChange],
); );
const content = ( const content = (
@ -69,6 +66,7 @@ export default function ShareTimestampDialog({
onShareTimestamp(timestamp); onShareTimestamp(timestamp);
onOpenChange(false); onOpenChange(false);
}} }}
onCancel={() => onOpenChange(false)}
/> />
); );
@ -108,15 +106,17 @@ type ShareTimestampContentProps = {
customTimestamp: number; customTimestamp: number;
setCustomTimestamp: (timestamp: number) => void; setCustomTimestamp: (timestamp: number) => void;
onShareTimestamp: (timestamp: number) => void; onShareTimestamp: (timestamp: number) => void;
onCancel?: () => void;
}; };
function ShareTimestampContent({ export function ShareTimestampContent({
currentTime, currentTime,
selectedOption, selectedOption,
setSelectedOption, setSelectedOption,
customTimestamp, customTimestamp,
setCustomTimestamp, setCustomTimestamp,
onShareTimestamp, onShareTimestamp,
onCancel,
}: Readonly<ShareTimestampContentProps>) { }: Readonly<ShareTimestampContentProps>) {
const { t } = useTranslation(["common", "components/dialog"]); const { t } = useTranslation(["common", "components/dialog"]);
const { data: config } = useSWR<FrigateConfig>("config"); const { data: config } = useSWR<FrigateConfig>("config");
@ -140,6 +140,8 @@ function ShareTimestampContent({
</div> </div>
</div> </div>
{isDesktop && <Separator className="my-4 bg-secondary" />}
<RadioGroup <RadioGroup
className="mt-4 flex flex-col gap-4" className="mt-4 flex flex-col gap-4"
value={selectedOption} value={selectedOption}
@ -191,16 +193,32 @@ function ShareTimestampContent({
</div> </div>
</RadioGroup> </RadioGroup>
<div className="mt-4"> {isDesktop && <Separator className="my-4 bg-secondary" />}
<DialogFooter
className={cn("mt-4", !isDesktop && "flex flex-col-reverse gap-4")}
>
{onCancel && (
<Button
type="button"
className={cn(
"cursor-pointer p-2 text-center",
!isDesktop && "w-full",
)}
onClick={onCancel}
>
{t("button.cancel", { ns: "common" })}
</Button>
)}
<Button <Button
className="w-full justify-between gap-3" className={cn(!isDesktop && "w-full")}
variant="select" variant="select"
size="sm"
onClick={() => onShareTimestamp(Math.floor(selectedTimestamp))} onClick={() => onShareTimestamp(Math.floor(selectedTimestamp))}
> >
<span>{t("button.shareTimestampUrl", { ns: "common" })}</span> {t("recording.shareTimestamp.button", { ns: "components/dialog" })}
<LuShare2 className="size-4" />
</Button> </Button>
</div> </DialogFooter>
</div> </div>
); );
} }
@ -275,7 +293,10 @@ function CustomTimestampSelector({
return ( return (
<div <div
className={`flex items-center rounded-lg bg-secondary text-secondary-foreground ${isDesktop ? "gap-2 px-2" : "pl-2"}`} className={cn(
"flex items-center rounded-lg bg-secondary text-secondary-foreground",
isDesktop ? "gap-2 px-2" : "pl-2",
)}
> >
<FaCalendarAlt /> <FaCalendarAlt />
<div className="flex flex-wrap items-center"> <div className="flex flex-wrap items-center">
@ -289,7 +310,7 @@ function CustomTimestampSelector({
> >
<PopoverTrigger asChild> <PopoverTrigger asChild>
<Button <Button
className={`text-primary ${isDesktop ? "" : "text-xs"}`} className={cn("text-primary", !isDesktop && "text-xs")}
aria-label={label} aria-label={label}
variant={selectorOpen ? "select" : "default"} variant={selectorOpen ? "select" : "default"}
size="sm" size="sm"
@ -298,7 +319,7 @@ function CustomTimestampSelector({
{formattedTimestamp} {formattedTimestamp}
</Button> </Button>
</PopoverTrigger> </PopoverTrigger>
<PopoverContent className="flex flex-col items-center" disablePortal> <PopoverContent className="flex flex-col items-center">
<TimezoneAwareCalendar <TimezoneAwareCalendar
timezone={config?.ui.timezone} timezone={config?.ui.timezone}
selectedDay={new Date(displayTimestamp * 1000)} selectedDay={new Date(displayTimestamp * 1000)}

View File

@ -272,19 +272,21 @@ export default function Events() {
...reviewFilter, ...reviewFilter,
...getReviewDayBounds(new Date(reviewLink.timestamp * 1000)), ...getReviewDayBounds(new Date(reviewLink.timestamp * 1000)),
}); });
setRecording( globalThis.setTimeout(() => {
{ setRecording(
camera: reviewLink.camera, {
startTime: reviewLink.timestamp, camera: reviewLink.camera,
// severity not actually applicable here, but the type requires it startTime: reviewLink.timestamp,
// this pattern is also used LiveCameraView to enter recording view // severity not actually applicable here, but the type requires it
severity: "alert", // this pattern is also used LiveCameraView to enter recording view
timelineType: notificationTab, severity: "alert",
}, timelineType: notificationTab,
true, },
); true,
);
}, 0);
return false; return true;
}); });
// review paging // review paging

View File

@ -1,3 +1,5 @@
import { baseUrl } from "@/api/baseUrl.ts";
export const RECORDING_REVIEW_LINK_PARAM = "timestamp"; export const RECORDING_REVIEW_LINK_PARAM = "timestamp";
export type RecordingReviewLinkState = { export type RecordingReviewLinkState = {
@ -12,21 +14,30 @@ export function parseRecordingReviewLink(
return undefined; return undefined;
} }
const [camera, timestamp] = value.split("|"); const separatorIndex = value.lastIndexOf("_");
if (separatorIndex <= 0 || separatorIndex == value.length - 1) {
return undefined;
}
const camera = value.slice(0, separatorIndex);
const timestamp = value.slice(separatorIndex + 1);
if (!camera || !timestamp) { if (!camera || !timestamp) {
return undefined; return undefined;
} }
const parsedTimestamp = Number(timestamp); const parsedTimestamp = Number(timestamp);
const now = Math.floor(Date.now() / 1000);
if (!Number.isFinite(parsedTimestamp)) { if (!Number.isFinite(parsedTimestamp) || parsedTimestamp <= 0) {
return undefined; return undefined;
} }
return { return {
camera, camera,
timestamp: Math.floor(parsedTimestamp), // clamp future timestamps to now
timestamp: Math.min(Math.floor(parsedTimestamp), now),
}; };
} }
@ -34,11 +45,12 @@ export function createRecordingReviewUrl(
pathname: string, pathname: string,
state: RecordingReviewLinkState, state: RecordingReviewLinkState,
): string { ): string {
const url = new URL(globalThis.location.href); const url = new URL(baseUrl);
const normalizedPathname = pathname.startsWith("/") url.pathname = pathname.startsWith("/") ? pathname : `/${pathname}`;
? pathname url.searchParams.set(
: `/${pathname}`; RECORDING_REVIEW_LINK_PARAM,
const reviewLink = `${state.camera}|${Math.floor(state.timestamp)}`; `${state.camera}_${Math.floor(state.timestamp)}`,
);
return `${url.origin}${normalizedPathname}?${RECORDING_REVIEW_LINK_PARAM}=${reviewLink}`; return url.toString();
} }

View File

@ -210,6 +210,15 @@ export function RecordingView({
const [debugReplayMode, setDebugReplayMode] = useState<ExportMode>("none"); const [debugReplayMode, setDebugReplayMode] = useState<ExportMode>("none");
const [debugReplayRange, setDebugReplayRange] = useState<TimeRange>(); const [debugReplayRange, setDebugReplayRange] = useState<TimeRange>();
const [shareTimestampOpen, setShareTimestampOpen] = useState(false); const [shareTimestampOpen, setShareTimestampOpen] = useState(false);
const [shareTimestampAtOpen, setShareTimestampAtOpen] = useState(
Math.floor(startTime),
);
const [shareTimestampOption, setShareTimestampOption] = useState<
"current" | "custom"
>("current");
const [customShareTimestamp, setCustomShareTimestamp] = useState(
Math.floor(startTime),
);
// move to next clip // move to next clip
@ -686,15 +695,26 @@ export function RecordingView({
setMotionOnly={() => {}} setMotionOnly={() => {}}
/> />
)} )}
<ShareTimestampDialog {isDesktop && (
currentTime={currentTime} <ShareTimestampDialog
open={shareTimestampOpen} currentTime={shareTimestampAtOpen}
onOpenChange={setShareTimestampOpen} open={shareTimestampOpen}
onShareTimestamp={onShareReviewLink} onOpenChange={setShareTimestampOpen}
/> selectedOption={shareTimestampOption}
setSelectedOption={setShareTimestampOption}
customTimestamp={customShareTimestamp}
setCustomTimestamp={setCustomShareTimestamp}
onShareTimestamp={onShareReviewLink}
/>
)}
{isDesktop && ( {isDesktop && (
<ActionsDropdown <ActionsDropdown
onShareTimestampClick={() => { onShareTimestampClick={() => {
const initialTimestamp = Math.floor(currentTime);
setShareTimestampAtOpen(initialTimestamp);
setShareTimestampOption("current");
setCustomShareTimestamp(initialTimestamp);
setShareTimestampOpen(true); setShareTimestampOpen(true);
}} }}
onDebugReplayClick={() => { onDebugReplayClick={() => {
@ -776,9 +796,7 @@ export function RecordingView({
mainControllerRef.current?.pause(); mainControllerRef.current?.pause();
} }
}} }}
onShareTimestampClick={() => { onShareTimestamp={onShareReviewLink}
setShareTimestampOpen(true);
}}
onUpdateFilter={updateFilter} onUpdateFilter={updateFilter}
setRange={setExportRange} setRange={setExportRange}
setMode={setExportMode} setMode={setExportMode}