import { useCallback, useEffect, useState } from "react"; import { CheckCircleIcon } from "@heroicons/react/16/solid"; import { cx } from "@/cva.config"; import MouseIcon from "@assets/mouse-icon.svg"; import PointingFinger from "@assets/pointing-finger.svg"; import { JsonRpcResponse, useJsonRpc } from "@hooks/useJsonRpc"; import { useSettingsStore } from "@hooks/stores"; import { GridCard } from "@components/Card"; import { Checkbox } from "@components/Checkbox"; import { SelectMenuBasic } from "@components/SelectMenuBasic"; import { SettingsItem } from "@components/SettingsItem"; import SettingsNestedSection from "@components/SettingsNestedSection"; import { SettingsPageHeader } from "@components/SettingsPageheader"; import { JigglerSetting } from "@components/JigglerSetting"; import notifications from "@/notifications"; import { m } from "@localizations/messages.js"; export interface JigglerConfig { inactivity_limit_seconds: number; jitter_percentage: number; schedule_cron_tab: string; timezone?: string; } const jigglerOptions = [ { value: "disabled", label: m.mouse_jiggler_disabled(), config: null }, { value: "frequent", label: m.mouse_jiggler_frequent(), 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: m.mouse_jiggler_standard(), 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: m.mouse_jiggler_light(), 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 { isCursorHidden, setCursorVisibility, mouseMode, setMouseMode, scrollThrottling, setScrollThrottling } = useSettingsStore(); const [selectedJigglerOption, setSelectedJigglerOption] = useState(null); const [currentJigglerConfig, setCurrentJigglerConfig] = useState( null, ); const scrollThrottlingOptions = [ { value: "0", label: m.mouse_scroll_off() }, { value: "10", label: m.mouse_scroll_low() }, { value: "25", label: m.mouse_scroll_medium() }, { value: "50", label: m.mouse_scroll_high() }, { value: "100", label: m.mouse_scroll_very_high() }, ]; const { send } = useJsonRpc(); const syncJigglerSettings = useCallback(() => { send("getJigglerState", {}, (resp: JsonRpcResponse) => { if ("error" in resp) return; const isEnabled = resp.result as boolean; console.log("Jiggler is enabled:", isEnabled); // 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(m.mouse_jiggler_failed_state({ error: resp.error.data || m.unknown_error() })); } }); send("setJigglerConfig", { jigglerConfig }, (resp: JsonRpcResponse) => { if ("error" in resp) { const errorMsg = resp.error.data || m.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(m.mouse_jiggler_invalid_cron()); } return notifications.error(m.mouse_jiggler_error_config({ error: errorMsg })); } notifications.success(m.mouse_jiggler_config_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(m.mouse_jiggler_failed_state({ error: resp.error.data || m.unknown_error() })); } }); notifications.success(m.mouse_jiggler_config_updated()); return setSelectedJigglerOption("disabled"); } const jigglerConfig = jigglerOptions.find(o => o.value === option)?.config; if (!jigglerConfig) { return notifications.error(m.mouse_jiggler_error_config()); } saveJigglerConfig(jigglerConfig); }; return (
setCursorVisibility(e.target.checked)} /> setScrollThrottling(parseInt(e.target.value))} options={scrollThrottlingOptions} /> ({ value: option.value, label: option.label, })), { value: "custom", label: m.mouse_jiggler_custom() }, ]} onChange={e => { handleJigglerChange( e.target.value as (typeof jigglerOptions)[number]["value"], ); }} /> {selectedJigglerOption === "custom" && ( )}
); }