frigate/web/src/components/classification/ClassificationModelWizardDialog.tsx

197 lines
5.0 KiB
TypeScript
Raw Normal View History

import { useTranslation } from "react-i18next";
import StepIndicator from "../indicators/StepIndicator";
import {
Dialog,
DialogContent,
DialogDescription,
DialogHeader,
DialogTitle,
} from "../ui/dialog";
2025-10-22 18:44:30 +03:00
import { useReducer, useMemo } from "react";
2025-10-22 18:24:50 +03:00
import Step1NameAndDefine, { Step1FormData } from "./wizard/Step1NameAndDefine";
import Step2StateArea, { Step2FormData } from "./wizard/Step2StateArea";
2025-10-22 22:38:07 +03:00
import Step3ChooseExamples, {
Step3FormData,
} from "./wizard/Step3ChooseExamples";
import { cn } from "@/lib/utils";
import { isDesktop } from "react-device-detect";
2025-10-22 18:44:30 +03:00
const OBJECT_STEPS = [
"wizard.steps.nameAndDefine",
"wizard.steps.chooseExamples",
"wizard.steps.train",
];
const STATE_STEPS = [
"wizard.steps.nameAndDefine",
"wizard.steps.stateArea",
"wizard.steps.chooseExamples",
"wizard.steps.train",
];
type ClassificationModelWizardDialogProps = {
open: boolean;
onClose: () => void;
};
2025-10-22 18:24:50 +03:00
type WizardState = {
currentStep: number;
step1Data?: Step1FormData;
step2Data?: Step2FormData;
2025-10-22 22:38:07 +03:00
step3Data?: Step3FormData;
2025-10-22 18:24:50 +03:00
};
type WizardAction =
| { type: "NEXT_STEP"; payload?: Partial<WizardState> }
| { type: "PREVIOUS_STEP" }
| { type: "SET_STEP_1"; payload: Step1FormData }
| { type: "SET_STEP_2"; payload: Step2FormData }
2025-10-22 22:38:07 +03:00
| { type: "SET_STEP_3"; payload: Step3FormData }
2025-10-22 18:24:50 +03:00
| { type: "RESET" };
const initialState: WizardState = {
currentStep: 0,
};
function wizardReducer(state: WizardState, action: WizardAction): WizardState {
switch (action.type) {
case "SET_STEP_1":
return {
...state,
step1Data: action.payload,
currentStep: 1,
};
case "SET_STEP_2":
return {
...state,
step2Data: action.payload,
currentStep: 2,
};
2025-10-22 22:38:07 +03:00
case "SET_STEP_3":
return {
...state,
step3Data: action.payload,
currentStep: 3,
};
2025-10-22 18:24:50 +03:00
case "NEXT_STEP":
return {
...state,
...action.payload,
currentStep: state.currentStep + 1,
};
case "PREVIOUS_STEP":
return {
...state,
currentStep: Math.max(0, state.currentStep - 1),
};
case "RESET":
return initialState;
default:
return state;
}
}
export default function ClassificationModelWizardDialog({
open,
onClose,
}: ClassificationModelWizardDialogProps) {
const { t } = useTranslation(["views/classificationModel"]);
2025-10-22 18:24:50 +03:00
const [wizardState, dispatch] = useReducer(wizardReducer, initialState);
2025-10-22 18:44:30 +03:00
const steps = useMemo(() => {
if (!wizardState.step1Data) {
return OBJECT_STEPS;
}
return wizardState.step1Data.modelType === "state"
? STATE_STEPS
: OBJECT_STEPS;
}, [wizardState.step1Data]);
2025-10-22 18:24:50 +03:00
const handleStep1Next = (data: Step1FormData) => {
dispatch({ type: "SET_STEP_1", payload: data });
};
const handleStep2Next = (data: Step2FormData) => {
dispatch({ type: "SET_STEP_2", payload: data });
};
2025-10-22 22:38:07 +03:00
const handleStep3Next = (data: Step3FormData) => {
dispatch({ type: "SET_STEP_3", payload: data });
};
const handleBack = () => {
dispatch({ type: "PREVIOUS_STEP" });
};
2025-10-22 18:24:50 +03:00
const handleCancel = () => {
dispatch({ type: "RESET" });
onClose();
};
return (
<Dialog
open={open}
onOpenChange={(open) => {
if (!open) {
2025-10-22 18:24:50 +03:00
handleCancel();
}
}}
>
<DialogContent
className={cn(
"",
isDesktop && "max-h-[75dvh] max-w-6xl overflow-y-auto",
)}
onInteractOutside={(e) => {
e.preventDefault();
}}
>
<StepIndicator
2025-10-22 18:44:30 +03:00
steps={steps}
2025-10-22 18:24:50 +03:00
currentStep={wizardState.currentStep}
variant="dots"
className="mb-4 justify-start"
/>
<DialogHeader>
<DialogTitle>{t("wizard.title")}</DialogTitle>
2025-10-22 18:24:50 +03:00
{wizardState.currentStep === 0 && (
<DialogDescription>{t("wizard.description")}</DialogDescription>
)}
</DialogHeader>
<div className="pb-4">
2025-10-22 18:24:50 +03:00
{wizardState.currentStep === 0 && (
<Step1NameAndDefine
initialData={wizardState.step1Data}
onNext={handleStep1Next}
onCancel={handleCancel}
/>
)}
{wizardState.currentStep === 1 &&
wizardState.step1Data?.modelType === "state" && (
<Step2StateArea
initialData={wizardState.step2Data}
onNext={handleStep2Next}
onBack={handleBack}
/>
)}
2025-10-22 22:38:07 +03:00
{((wizardState.currentStep === 2 &&
wizardState.step1Data?.modelType === "state") ||
(wizardState.currentStep === 1 &&
wizardState.step1Data?.modelType === "object")) &&
wizardState.step1Data && (
<Step3ChooseExamples
step1Data={wizardState.step1Data}
step2Data={wizardState.step2Data}
initialData={wizardState.step3Data}
onNext={handleStep3Next}
onBack={handleBack}
/>
)}
</div>
</DialogContent>
</Dialog>
);
}