mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-05-09 23:15:28 +03:00
Compare commits
10 Commits
6fa64a8db3
...
10ced9195d
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
10ced9195d | ||
|
|
55a858783c | ||
|
|
dd5ad11a6e | ||
|
|
738197b184 | ||
|
|
9706bcbb2e | ||
|
|
8381c12920 | ||
|
|
99fa5e1dee | ||
|
|
eead0535a3 | ||
|
|
a0956ebe0f | ||
|
|
5dad8cfb2d |
@ -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",
|
||||||
|
|||||||
@ -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": {
|
||||||
|
|||||||
@ -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" })}
|
||||||
|
|||||||
@ -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 (
|
||||||
|
|||||||
@ -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)}
|
||||||
|
|||||||
@ -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
|
||||||
|
|||||||
@ -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();
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user