mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-07-01 01:21:14 +03:00
Some checks failed
CI / AMD64 Build (push) Has been cancelled
CI / ARM Build (push) Has been cancelled
CI / Jetson Jetpack 6 (push) Has been cancelled
CI / AMD64 Extra Build (push) Has been cancelled
CI / ARM Extra Build (push) Has been cancelled
CI / Synaptics Build (push) Has been cancelled
CI / Assemble and push default build (push) Has been cancelled
* remove redundant per-view toasters in settings * add variants to standardize dialog footer button layouts * remove text-md this class name compiles to nothing in tailwind. we used to add it to prevent iOS from zooming when focusing on an input, but that is now solved via the viewport meta in index.html * make wizard footers consistent with dialog footers * consistent destructive button style remove text-white from individual buttons and add it to the variant
172 lines
6.5 KiB
TypeScript
172 lines
6.5 KiB
TypeScript
import { useEffect } from "react";
|
|
import { Trans, useTranslation } from "react-i18next";
|
|
import { Link, useNavigate } from "react-router-dom";
|
|
import useSWR from "swr";
|
|
import { CheckCircle2, XCircle } from "lucide-react";
|
|
import { LuExternalLink } from "react-icons/lu";
|
|
import ActivityIndicator from "@/components/indicators/activity-indicator";
|
|
import { Button } from "@/components/ui/button";
|
|
import Heading from "@/components/ui/heading";
|
|
import {
|
|
SettingsGroupCard,
|
|
SplitCardRow,
|
|
} from "@/components/card/SettingsGroupCard";
|
|
import FrigatePlusCurrentModelSummary from "@/views/settings/components/FrigatePlusCurrentModelSummary";
|
|
import { useDocDomain } from "@/hooks/use-doc-domain";
|
|
import { CameraNameLabel } from "@/components/camera/FriendlyNameLabel";
|
|
import { FrigateConfig } from "@/types/frigateConfig";
|
|
import { isReplayCamera } from "@/utils/cameraUtil";
|
|
import type { SettingsPageProps } from "@/views/settings/SingleSectionPage";
|
|
|
|
export default function FrigatePlusSettingsView(_props: SettingsPageProps) {
|
|
const { t } = useTranslation("views/settings");
|
|
const { getLocaleDocUrl } = useDocDomain();
|
|
const { data: config } = useSWR<FrigateConfig>("config");
|
|
const navigate = useNavigate();
|
|
|
|
useEffect(() => {
|
|
document.title = t("documentTitle.frigatePlus");
|
|
}, [t]);
|
|
|
|
if (!config) {
|
|
return <ActivityIndicator />;
|
|
}
|
|
|
|
return (
|
|
<div className="flex size-full flex-col md:pr-2">
|
|
<div className="w-full max-w-5xl space-y-6 pt-2">
|
|
<div className="flex flex-col gap-0">
|
|
<Heading as="h4" className="mb-2">
|
|
{t("frigatePlus.title")}
|
|
</Heading>
|
|
<p className="text-sm text-muted-foreground">
|
|
{t("frigatePlus.description")}
|
|
</p>
|
|
</div>
|
|
|
|
<div className="space-y-6">
|
|
<SettingsGroupCard title={t("frigatePlus.cardTitles.api")}>
|
|
<SplitCardRow
|
|
label={t("frigatePlus.apiKey.title")}
|
|
description={
|
|
<>
|
|
<p>{t("frigatePlus.apiKey.desc")}</p>
|
|
{!config?.model.plus && (
|
|
<div className="mt-2 flex items-center text-primary-variant">
|
|
<Link
|
|
to="https://frigate.video/plus"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="inline"
|
|
>
|
|
{t("frigatePlus.apiKey.plusLink")}
|
|
<LuExternalLink className="ml-2 inline-flex size-3" />
|
|
</Link>
|
|
</div>
|
|
)}
|
|
</>
|
|
}
|
|
content={
|
|
<div className="flex items-center gap-2">
|
|
{config?.plus?.enabled ? (
|
|
<CheckCircle2 className="h-5 w-5 text-green-500" />
|
|
) : (
|
|
<XCircle className="h-5 w-5 text-red-500" />
|
|
)}
|
|
<span className="text-sm">
|
|
{config?.plus?.enabled
|
|
? t("frigatePlus.apiKey.validated")
|
|
: t("frigatePlus.apiKey.notValidated")}
|
|
</span>
|
|
</div>
|
|
}
|
|
/>
|
|
</SettingsGroupCard>
|
|
|
|
{config?.plus?.enabled && (
|
|
<FrigatePlusCurrentModelSummary
|
|
plusModel={config.model.plus}
|
|
action={
|
|
<Button
|
|
size="sm"
|
|
variant="outline"
|
|
onClick={() =>
|
|
navigate("/settings?page=systemDetectorsAndModel")
|
|
}
|
|
>
|
|
{t("frigatePlus.changeInDetectorsAndModel")}
|
|
</Button>
|
|
}
|
|
/>
|
|
)}
|
|
|
|
<SettingsGroupCard title={t("frigatePlus.cardTitles.configuration")}>
|
|
<SplitCardRow
|
|
label={t("frigatePlus.snapshotConfig.title")}
|
|
description={
|
|
<>
|
|
<p>
|
|
<Trans ns="views/settings">
|
|
frigatePlus.snapshotConfig.desc
|
|
</Trans>
|
|
</p>
|
|
<div className="mt-2 flex items-center text-primary-variant">
|
|
<Link
|
|
to={getLocaleDocUrl("plus/faq")}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="inline"
|
|
>
|
|
{t("readTheDocumentation", { ns: "common" })}
|
|
<LuExternalLink className="ml-2 inline-flex size-3" />
|
|
</Link>
|
|
</div>
|
|
</>
|
|
}
|
|
content={
|
|
<div className="space-y-3">
|
|
<div className="overflow-x-auto">
|
|
<table className="w-full text-sm">
|
|
<thead>
|
|
<tr className="border-b border-secondary">
|
|
<th className="px-4 py-2 text-left">
|
|
{t("frigatePlus.snapshotConfig.table.camera")}
|
|
</th>
|
|
<th className="px-4 py-2 text-center">
|
|
{t("frigatePlus.snapshotConfig.table.snapshots")}
|
|
</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{Object.entries(config.cameras)
|
|
.filter(([name]) => !isReplayCamera(name))
|
|
.map(([name, camera]) => (
|
|
<tr
|
|
key={name}
|
|
className="border-b border-secondary"
|
|
>
|
|
<td className="px-4 py-2">
|
|
<CameraNameLabel camera={name} />
|
|
</td>
|
|
<td className="px-4 py-2 text-center">
|
|
{camera.snapshots.enabled ? (
|
|
<CheckCircle2 className="mx-auto size-5 text-green-500" />
|
|
) : (
|
|
<XCircle className="mx-auto size-5 text-danger" />
|
|
)}
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
}
|
|
/>
|
|
</SettingsGroupCard>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|