import { useState, useEffect } from "react"; import { DropdownItem, Field, PanelSection, PanelSectionRow, ToggleField } from "@decky/ui"; import { runInstallFGMod, runUninstallFGMod, setDefaultFsr4Variant } from "../api"; import { OperationResult } from "./ResultDisplay"; import { createAutoCleanupTimer } from "../utils"; import { TIMEOUTS, PROXY_DLL_OPTIONS, DEFAULT_PROXY_DLL, FSR4_VARIANT_OPTIONS, DEFAULT_FSR4_VARIANT } from "../utils/constants"; import { InstallationStatus } from "./InstallationStatus"; import { OptiScalerHeader } from "./OptiScalerHeader"; import { ClipboardCommands } from "./ClipboardCommands"; import { InstructionCard } from "./InstructionCard"; import { OptiScalerWiki } from "./OptiScalerWiki"; import { UninstallButton } from "./UninstallButton"; import { ManualPatchControls } from "./CustomPathOverride"; import { SteamGamePatcher } from "./SteamGamePatcher"; interface FgmodInfo { exists: boolean; version?: string | null; selected_fsr4_variant?: string | null; selected_fsr4_variant_label?: string | null; install_manifest_present?: boolean; } interface OptiScalerControlsProps { pathExists: boolean | null; setPathExists: (exists: boolean | null) => void; fgmodInfo?: FgmodInfo | null; } export function OptiScalerControls({ pathExists, setPathExists, fgmodInfo }: OptiScalerControlsProps) { const [installing, setInstalling] = useState(false); const [uninstalling, setUninstalling] = useState(false); const [installResult, setInstallResult] = useState(null); const [uninstallResult, setUninstallResult] = useState(null); const [advancedModeEnabled, setAdvancedModeEnabled] = useState(false); const [manualClipboardModeEnabled, setManualClipboardModeEnabled] = useState(false); const [dllName, setDllName] = useState(DEFAULT_PROXY_DLL); const [fsr4Variant, setFsr4Variant] = useState(DEFAULT_FSR4_VARIANT); const [fsr4VariantTouched, setFsr4VariantTouched] = useState(false); const [switchingVariant, setSwitchingVariant] = useState(false); useEffect(() => { if (installResult) { return createAutoCleanupTimer(() => setInstallResult(null), TIMEOUTS.resultDisplay); } return () => {}; // Ensure a cleanup function is always returned }, [installResult]); useEffect(() => { if (uninstallResult) { return createAutoCleanupTimer(() => setUninstallResult(null), TIMEOUTS.resultDisplay); } return () => {}; // Ensure a cleanup function is always returned }, [uninstallResult]); useEffect(() => { const installedVariant = fgmodInfo?.selected_fsr4_variant; if (!fsr4VariantTouched && installedVariant && FSR4_VARIANT_OPTIONS.some((option) => option.value === installedVariant)) { setFsr4Variant(installedVariant); } }, [fgmodInfo?.selected_fsr4_variant, fsr4VariantTouched]); const handleInstallClick = async () => { try { setInstalling(true); const result = await runInstallFGMod(fsr4Variant); setInstallResult(result); if (result.status === "success") { setPathExists(true); } } catch (e) { console.error(e); } finally { setInstalling(false); } }; const handleUninstallClick = async () => { try { setUninstalling(true); const result = await runUninstallFGMod(); setUninstallResult(result); if (result.status === "success") { setPathExists(false); } } catch (e) { console.error(e); } finally { setUninstalling(false); } }; const handleFsr4VariantChange = async (nextVariant: string) => { const previousVariant = fsr4Variant; setFsr4Variant(nextVariant); setFsr4VariantTouched(true); if (pathExists !== true) return; try { setSwitchingVariant(true); const result = await setDefaultFsr4Variant(nextVariant); if (result.status !== "success") { throw new Error(result.message || result.output || "Failed to switch default FSR4 runtime."); } setFsr4Variant(result.selected_default_variant || nextVariant); setFsr4VariantTouched(false); } catch (error) { console.error(error); setFsr4Variant(previousVariant); } finally { setSwitchingVariant(false); } }; const installedVariantLabel = fgmodInfo?.selected_fsr4_variant_label || FSR4_VARIANT_OPTIONS.find((option) => option.value === fsr4Variant)?.label; return ( option.value === fsr4Variant)?.hint} menuLabel="Default FSR4 runtime" selectedOption={fsr4Variant} rgOptions={FSR4_VARIANT_OPTIONS.map((option) => ({ data: option.value, label: option.label }))} disabled={installing || uninstalling || switchingVariant} onChange={(option) => { void handleFsr4VariantChange(String(option.data)); }} /> {pathExists === true && fgmodInfo?.version && installedVariantLabel && ( {installedVariantLabel} )} {pathExists === true && ( o.value === dllName)?.hint} menuLabel="Proxy DLL name" selectedOption={dllName} rgOptions={PROXY_DLL_OPTIONS.map((o) => ({ data: o.value, label: o.label }))} onChange={(option) => setDllName(String(option.data))} /> )} {pathExists === true && ( )} {pathExists === true && ( )} {pathExists === true && manualClipboardModeEnabled ? ( ) : null} {!advancedModeEnabled && ( )} ); }