import { CheckCircleIcon } from "@heroicons/react/16/solid"; import { useCallback, useEffect, useState } from "react"; import MouseIcon from "@/assets/mouse-icon.svg"; import PointingFinger from "@/assets/pointing-finger.svg"; import { GridCard } from "@/components/Card"; import { Checkbox } from "@/components/Checkbox"; import { useSettingsStore } from "@/hooks/stores"; import { JsonRpcResponse, useJsonRpc } from "@/hooks/useJsonRpc"; import { SettingsPageHeader } from "@components/SettingsPageheader"; import { SelectMenuBasic } from "@components/SelectMenuBasic"; import { JigglerSetting } from "@components/JigglerSetting"; import { cx } from "../cva.config"; import notifications from "../notifications"; import SettingsNestedSection from "../components/SettingsNestedSection"; import { SettingsItem } from "./devices.$id.settings"; export interface JigglerConfig { inactivity_limit_seconds: number; jitter_percentage: number; schedule_cron_tab: string; timezone?: string; } const jigglerOptions = [ { value: "disabled", label: "Disabled", config: null }, { value: "frequent", label: "Frequent - 30s", config: { inactivity_limit_seconds: 30, jitter_percentage: 25, schedule_cron_tab: "*/30 * * * * *", // We don't care about the timezone for this preset // timezone: "UTC", }, }, { value: "standard", label: "Standard - 1m", config: { inactivity_limit_seconds: 60, jitter_percentage: 25, schedule_cron_tab: "0 * * * * *", // We don't care about the timezone for this preset // timezone: "UTC", }, }, { value: "light", label: "Light - 5m", config: { inactivity_limit_seconds: 300, jitter_percentage: 25, schedule_cron_tab: "0 */5 * * * *", // We don't care about the timezone for this preset // timezone: "UTC", }, }, ] as const; type JigglerValues = (typeof jigglerOptions)[number]["value"] | "custom"; export default function SettingsMouseRoute() { const hideCursor = useSettingsStore(state => state.isCursorHidden); const setHideCursor = useSettingsStore(state => state.setCursorVisibility); const mouseMode = useSettingsStore(state => state.mouseMode); const setMouseMode = useSettingsStore(state => state.setMouseMode); const scrollThrottling = useSettingsStore(state => state.scrollThrottling); const setScrollThrottling = useSettingsStore(state => state.setScrollThrottling); const [selectedJigglerOption, setSelectedJigglerOption] = useState(null); const [currentJigglerConfig, setCurrentJigglerConfig] = useState( null, ); const scrollThrottlingOptions = [ { value: "0", label: "Off" }, { value: "10", label: "Low" }, { value: "25", label: "Medium" }, { value: "50", label: "High" }, { value: "100", label: "Very High" }, ]; const { send } = useJsonRpc(); const syncJigglerSettings = useCallback(() => { send("getJigglerState", {}, (resp: JsonRpcResponse) => { if ("error" in resp) return; const isEnabled = resp.result as boolean; // If the jiggler is disabled, set the selected option to "disabled" and nothing else if (!isEnabled) return setSelectedJigglerOption("disabled"); send("getJigglerConfig", {}, (resp: JsonRpcResponse) => { if ("error" in resp) return; const result = resp.result as JigglerConfig; setCurrentJigglerConfig(result); const value = jigglerOptions.find( o => o?.config?.inactivity_limit_seconds === result.inactivity_limit_seconds && o?.config?.jitter_percentage === result.jitter_percentage && o?.config?.schedule_cron_tab === result.schedule_cron_tab, )?.value; setSelectedJigglerOption(value || "custom"); }); }); }, [send]); useEffect(() => { syncJigglerSettings(); }, [syncJigglerSettings]); const saveJigglerConfig = useCallback( (jigglerConfig: JigglerConfig) => { // We assume the jiggler should be set to enabled if the config is being updated send("setJigglerState", { enabled: true }, (resp: JsonRpcResponse) => { if ("error" in resp) { return notifications.error( `Failed to set jiggler state: ${resp.error.data || "Unknown error"}`, ); } }); send("setJigglerConfig", { jigglerConfig }, (resp: JsonRpcResponse) => { if ("error" in resp) { const errorMsg = resp.error.data || "Unknown error"; // Check for cron syntax errors and provide user-friendly message if ( errorMsg.includes("invalid syntax") || errorMsg.includes("parse failure") || errorMsg.includes("invalid cron") ) { return notifications.error( "Invalid cron expression. Please check your schedule format (e.g., '0 * * * * *' for every minute).", ); } return notifications.error(`Failed to set jiggler config: ${errorMsg}`); } notifications.success(`Jiggler Config successfully updated`); syncJigglerSettings(); }); }, [send, syncJigglerSettings], ); const handleJigglerChange = (option: JigglerValues) => { if (option === "custom") { setSelectedJigglerOption("custom"); // We don't need to sync the jiggler settings when the option is "custom". The user will press "Save" to save the custom settings. return; } // We don't need to update the device jiggler state when the option is "disabled" if (option === "disabled") { send("setJigglerState", { enabled: false }, (resp: JsonRpcResponse) => { if ("error" in resp) { return notifications.error( `Failed to set jiggler state: ${resp.error.data || "Unknown error"}`, ); } }); notifications.success(`Jiggler Config successfully updated`); return setSelectedJigglerOption("disabled"); } const jigglerConfig = jigglerOptions.find(o => o.value === option)?.config; if (!jigglerConfig) { return notifications.error("There was an error setting the jiggler config"); } saveJigglerConfig(jigglerConfig); }; return (
setHideCursor(e.target.checked)} /> setScrollThrottling(parseInt(e.target.value))} options={scrollThrottlingOptions} /> ({ value: option.value, label: option.label, })), { value: "custom", label: "Custom" }, ]} onChange={e => { handleJigglerChange( e.target.value as (typeof jigglerOptions)[number]["value"], ); }} /> {selectedJigglerOption === "custom" && ( )}
); }