mirror of
https://github.com/blakeblackshear/frigate.git
synced 2025-12-08 22:35:42 +03:00
* Translation module init
* Add more i18n keys
* fix: fix string wrong
* refactor: use namespace translation file
* chore: add more translation key
* fix: fix some page name error
* refactor: change Trans tag for t function
* chore: fix some key not work
* chore: fix SearchFilterDialog i18n key error
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* chore: fix en i18n file filter missing some keys
* chore: add some i18n keys
* chore: add more i18n keys again
* feat: add search page i18n
* feat: add explore model i18n keys
* Update web/src/components/menu/GeneralSettings.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/components/menu/GeneralSettings.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/components/menu/GeneralSettings.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* feat: add more live i18n keys
* feat: add more search setting i18n keys
* fix: remove some comment
* fix: fix some setting page url error
* Update web/src/views/settings/SearchSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* fix: add system missing keys
* fix: update password update i18n keys
* chore: remove outdate translation.json file
* fix: fix exploreSettings error
* chore: add object setting i18n keys
* Update web/src/views/recording/RecordingView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/public/locales/en/components/filter.json
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/components/overlay/ExportDialog.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* feat: add more i18n keys
* fix: fix motionDetectionTuner html node
* feat: add more page i18n keys
* fix: cameraStream i18n keys error
* feat: add Player i18n keys
* feat: add more toast i18n keys
* feat: change explore setting name
* feat: add more document title i18n keys
* feat: add more search i18n keys
* fix: fix accessDenied i18n keys error
* chore: add objectType i18n
* chore: add inputWithTags i18n
* chore: add SearchFilterDialog i18n
* Update web/src/views/settings/ObjectSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/views/settings/ObjectSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/views/settings/ObjectSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/views/settings/ObjectSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* Update web/src/views/settings/ObjectSettingsView.tsx
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
* chore: add some missing i18n keys
* chore: remove most import { t } from "i18next";
---------
Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
143 lines
4.6 KiB
TypeScript
143 lines
4.6 KiB
TypeScript
import useSWR from "swr";
|
|
import { FrigateStats } from "@/types/stats";
|
|
import { useEffect, useMemo, useState } from "react";
|
|
import TimeAgo from "@/components/dynamic/TimeAgo";
|
|
import { ToggleGroup, ToggleGroupItem } from "@/components/ui/toggle-group";
|
|
import { isDesktop, isMobile } from "react-device-detect";
|
|
import GeneralMetrics from "@/views/system/GeneralMetrics";
|
|
import StorageMetrics from "@/views/system/StorageMetrics";
|
|
import { LuActivity, LuHardDrive, LuSearchCode } from "react-icons/lu";
|
|
import { FaVideo } from "react-icons/fa";
|
|
import Logo from "@/components/Logo";
|
|
import useOptimisticState from "@/hooks/use-optimistic-state";
|
|
import CameraMetrics from "@/views/system/CameraMetrics";
|
|
import { useHashState } from "@/hooks/use-overlay-state";
|
|
import { Toaster } from "@/components/ui/sonner";
|
|
import { FrigateConfig } from "@/types/frigateConfig";
|
|
import FeatureMetrics from "@/views/system/FeatureMetrics";
|
|
import { useTranslation } from "react-i18next";
|
|
|
|
const allMetrics = ["general", "features", "storage", "cameras"] as const;
|
|
type SystemMetric = (typeof allMetrics)[number];
|
|
|
|
function System() {
|
|
const { t } = useTranslation(["views/system"]);
|
|
const { data: config } = useSWR<FrigateConfig>("config", {
|
|
revalidateOnFocus: false,
|
|
});
|
|
|
|
const metrics = useMemo(() => {
|
|
const metrics = [...allMetrics];
|
|
|
|
if (
|
|
!config?.semantic_search.enabled &&
|
|
!config?.lpr.enabled &&
|
|
!config?.face_recognition.enabled
|
|
) {
|
|
const index = metrics.indexOf("features");
|
|
metrics.splice(index, 1);
|
|
}
|
|
|
|
return metrics;
|
|
}, [config]);
|
|
|
|
// stats page
|
|
|
|
const [page, setPage] = useHashState<SystemMetric>();
|
|
const [pageToggle, setPageToggle] = useOptimisticState(
|
|
page ?? "general",
|
|
setPage,
|
|
100,
|
|
);
|
|
const [lastUpdated, setLastUpdated] = useState<number>(Date.now() / 1000);
|
|
|
|
useEffect(() => {
|
|
if (pageToggle) {
|
|
document.title = t("documentTitle." + pageToggle);
|
|
}
|
|
}, [pageToggle, t]);
|
|
|
|
// stats collection
|
|
|
|
const { data: statsSnapshot } = useSWR<FrigateStats>("stats", {
|
|
revalidateOnFocus: false,
|
|
});
|
|
|
|
return (
|
|
<div className="flex size-full flex-col p-2">
|
|
<Toaster position="top-center" />
|
|
<div className="relative flex h-11 w-full items-center justify-between">
|
|
{isMobile && (
|
|
<Logo className="absolute inset-x-1/2 h-8 -translate-x-1/2" />
|
|
)}
|
|
<ToggleGroup
|
|
className="*:rounded-md *:px-3 *:py-4"
|
|
type="single"
|
|
size="sm"
|
|
value={pageToggle}
|
|
onValueChange={(value: SystemMetric) => {
|
|
if (value) {
|
|
setPageToggle(value);
|
|
}
|
|
}} // don't allow the severity to be unselected
|
|
>
|
|
{Object.values(metrics).map((item) => (
|
|
<ToggleGroupItem
|
|
key={item}
|
|
className={`flex items-center justify-between gap-2 ${pageToggle == item ? "" : "*:text-muted-foreground"}`}
|
|
value={item}
|
|
aria-label={`Select ${item}`}
|
|
>
|
|
{item == "general" && <LuActivity className="size-4" />}
|
|
{item == "features" && <LuSearchCode className="size-4" />}
|
|
{item == "storage" && <LuHardDrive className="size-4" />}
|
|
{item == "cameras" && <FaVideo className="size-4" />}
|
|
{isDesktop && (
|
|
<div className="capitalize">{t(item + ".title")}</div>
|
|
)}
|
|
</ToggleGroupItem>
|
|
))}
|
|
</ToggleGroup>
|
|
|
|
<div className="flex h-full items-center">
|
|
{lastUpdated && (
|
|
<div className="h-full content-center text-sm text-muted-foreground">
|
|
{t("lastRefreshed")}
|
|
<TimeAgo time={lastUpdated * 1000} dense />
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
<div className="mt-2 flex items-end gap-2">
|
|
<div className="h-full content-center font-medium">{t("title")}</div>
|
|
{statsSnapshot && (
|
|
<div className="h-full content-center text-sm text-muted-foreground">
|
|
{statsSnapshot.service.version}
|
|
</div>
|
|
)}
|
|
</div>
|
|
{page == "general" && (
|
|
<GeneralMetrics
|
|
lastUpdated={lastUpdated}
|
|
setLastUpdated={setLastUpdated}
|
|
/>
|
|
)}
|
|
{page == "features" && (
|
|
<FeatureMetrics
|
|
lastUpdated={lastUpdated}
|
|
setLastUpdated={setLastUpdated}
|
|
/>
|
|
)}
|
|
{page == "storage" && <StorageMetrics setLastUpdated={setLastUpdated} />}
|
|
{page == "cameras" && (
|
|
<CameraMetrics
|
|
lastUpdated={lastUpdated}
|
|
setLastUpdated={setLastUpdated}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default System;
|