mirror of https://github.com/jetkvm/kvm.git
229 lines
8.1 KiB
TypeScript
229 lines
8.1 KiB
TypeScript
import { useEffect, useState } from "react";
|
|
|
|
import { SettingsItem } from "@components/SettingsItem";
|
|
import { SettingsPageHeader } from "@components/SettingsPageheader";
|
|
import { SettingsSectionHeader } from "@components/SettingsSectionHeader";
|
|
import { BacklightSettings, useSettingsStore } from "@/hooks/stores";
|
|
import { JsonRpcResponse, useJsonRpc } from "@/hooks/useJsonRpc";
|
|
import { SelectMenuBasic } from "@components/SelectMenuBasic";
|
|
import { UsbDeviceSetting } from "@components/UsbDeviceSetting";
|
|
import { Checkbox } from "@components/Checkbox";
|
|
|
|
import notifications from "../notifications";
|
|
import { UsbInfoSetting } from "../components/UsbInfoSetting";
|
|
import { FeatureFlag } from "../components/FeatureFlag";
|
|
|
|
export default function SettingsHardwareRoute() {
|
|
const { send } = useJsonRpc();
|
|
const settings = useSettingsStore();
|
|
const { setDisplayRotation } = useSettingsStore();
|
|
const [powerSavingEnabled, setPowerSavingEnabled] = useState(false);
|
|
|
|
const handleDisplayRotationChange = (rotation: string) => {
|
|
setDisplayRotation(rotation);
|
|
handleDisplayRotationSave();
|
|
};
|
|
|
|
const handleDisplayRotationSave = () => {
|
|
send("setDisplayRotation", { params: { rotation: settings.displayRotation } }, (resp: JsonRpcResponse) => {
|
|
if ("error" in resp) {
|
|
notifications.error(
|
|
`Failed to set display orientation: ${resp.error.data || "Unknown error"}`,
|
|
);
|
|
return;
|
|
}
|
|
notifications.success("Display orientation updated successfully");
|
|
});
|
|
};
|
|
|
|
const { setBacklightSettings } = useSettingsStore();
|
|
|
|
const handleBacklightSettingsChange = (settings: BacklightSettings) => {
|
|
// If the user has set the display to dim after it turns off, set the dim_after
|
|
// value to never.
|
|
if (settings.dim_after > settings.off_after && settings.off_after != 0) {
|
|
settings.dim_after = 0;
|
|
}
|
|
|
|
setBacklightSettings(settings);
|
|
handleBacklightSettingsSave();
|
|
};
|
|
|
|
const handleBacklightSettingsSave = () => {
|
|
send("setBacklightSettings", { params: settings.backlightSettings }, (resp: JsonRpcResponse) => {
|
|
if ("error" in resp) {
|
|
notifications.error(
|
|
`Failed to set backlight settings: ${resp.error.data || "Unknown error"}`,
|
|
);
|
|
return;
|
|
}
|
|
notifications.success("Backlight settings updated successfully");
|
|
});
|
|
};
|
|
|
|
const handlePowerSavingChange = (enabled: boolean) => {
|
|
setPowerSavingEnabled(enabled);
|
|
const duration = enabled ? 90 : -1;
|
|
send("setVideoSleepMode", { duration }, (resp: JsonRpcResponse) => {
|
|
if ("error" in resp) {
|
|
notifications.error(
|
|
`Failed to set power saving mode: ${resp.error.data || "Unknown error"}`,
|
|
);
|
|
setPowerSavingEnabled(!enabled); // Revert on error
|
|
return;
|
|
}
|
|
notifications.success(`Power saving mode ${enabled ? "enabled" : "disabled"}`);
|
|
});
|
|
};
|
|
|
|
useEffect(() => {
|
|
send("getBacklightSettings", {}, (resp: JsonRpcResponse) => {
|
|
if ("error" in resp) {
|
|
return notifications.error(
|
|
`Failed to get backlight settings: ${resp.error.data || "Unknown error"}`,
|
|
);
|
|
}
|
|
const result = resp.result as BacklightSettings;
|
|
setBacklightSettings(result);
|
|
});
|
|
}, [send, setBacklightSettings]);
|
|
|
|
useEffect(() => {
|
|
send("getVideoSleepMode", {}, (resp: JsonRpcResponse) => {
|
|
if ("error" in resp) {
|
|
console.error("Failed to get power saving mode:", resp.error);
|
|
return;
|
|
}
|
|
const result = resp.result as { enabled: boolean; duration: number };
|
|
setPowerSavingEnabled(result.duration >= 0);
|
|
});
|
|
}, [send]);
|
|
|
|
return (
|
|
<div className="space-y-4">
|
|
<SettingsPageHeader
|
|
title="Hardware"
|
|
description="Configure display settings and hardware options for your JetKVM device"
|
|
/>
|
|
<div className="space-y-4">
|
|
<SettingsItem
|
|
title="Display Orientation"
|
|
description="Set the orientation of the display"
|
|
>
|
|
<SelectMenuBasic
|
|
size="SM"
|
|
label=""
|
|
value={settings.displayRotation.toString()}
|
|
options={[
|
|
{ value: "270", label: "Normal" },
|
|
{ value: "90", label: "Inverted" },
|
|
]}
|
|
onChange={e => {
|
|
settings.displayRotation = e.target.value;
|
|
handleDisplayRotationChange(settings.displayRotation);
|
|
}}
|
|
/>
|
|
</SettingsItem>
|
|
<SettingsItem
|
|
title="Display Brightness"
|
|
description="Set the brightness of the display"
|
|
>
|
|
<SelectMenuBasic
|
|
size="SM"
|
|
label=""
|
|
value={settings.backlightSettings.max_brightness.toString()}
|
|
options={[
|
|
{ value: "0", label: "Off" },
|
|
{ value: "10", label: "Low" },
|
|
{ value: "35", label: "Medium" },
|
|
{ value: "64", label: "High" },
|
|
]}
|
|
onChange={e => {
|
|
settings.backlightSettings.max_brightness = parseInt(e.target.value);
|
|
handleBacklightSettingsChange(settings.backlightSettings);
|
|
}}
|
|
/>
|
|
</SettingsItem>
|
|
{settings.backlightSettings.max_brightness != 0 && (
|
|
<>
|
|
<SettingsItem
|
|
title="Dim Display After"
|
|
description="Set how long to wait before dimming the display"
|
|
>
|
|
<SelectMenuBasic
|
|
size="SM"
|
|
label=""
|
|
value={settings.backlightSettings.dim_after.toString()}
|
|
options={[
|
|
{ value: "0", label: "Never" },
|
|
{ value: "60", label: "1 Minute" },
|
|
{ value: "300", label: "5 Minutes" },
|
|
{ value: "600", label: "10 Minutes" },
|
|
{ value: "1800", label: "30 Minutes" },
|
|
{ value: "3600", label: "1 Hour" },
|
|
]}
|
|
onChange={e => {
|
|
settings.backlightSettings.dim_after = parseInt(e.target.value);
|
|
handleBacklightSettingsChange(settings.backlightSettings);
|
|
}}
|
|
/>
|
|
</SettingsItem>
|
|
<SettingsItem
|
|
title="Turn off Display After"
|
|
description="Period of inactivity before display automatically turns off"
|
|
>
|
|
<SelectMenuBasic
|
|
size="SM"
|
|
label=""
|
|
value={settings.backlightSettings.off_after.toString()}
|
|
options={[
|
|
{ value: "0", label: "Never" },
|
|
{ value: "300", label: "5 Minutes" },
|
|
{ value: "600", label: "10 Minutes" },
|
|
{ value: "1800", label: "30 Minutes" },
|
|
{ value: "3600", label: "1 Hour" },
|
|
]}
|
|
onChange={e => {
|
|
settings.backlightSettings.off_after = parseInt(e.target.value);
|
|
handleBacklightSettingsChange(settings.backlightSettings);
|
|
}}
|
|
/>
|
|
</SettingsItem>
|
|
</>
|
|
)}
|
|
<p className="text-xs text-slate-600 dark:text-slate-400">
|
|
The display will wake up when the connection state changes, or when touched.
|
|
</p>
|
|
</div>
|
|
|
|
<FeatureFlag minAppVersion="0.4.9">
|
|
<div className="space-y-4">
|
|
<div className="h-px w-full bg-slate-800/10 dark:bg-slate-300/20" />
|
|
<SettingsSectionHeader
|
|
title="Power Saving"
|
|
description="Reduce power consumption when not in use"
|
|
/>
|
|
<SettingsItem
|
|
badge="Experimental"
|
|
title="HDMI Sleep Mode"
|
|
description="Turn off capture after 90 seconds of inactivity"
|
|
>
|
|
<Checkbox
|
|
checked={powerSavingEnabled}
|
|
onChange={(e) => handlePowerSavingChange(e.target.checked)}
|
|
/>
|
|
</SettingsItem>
|
|
</div>
|
|
</FeatureFlag>
|
|
|
|
<FeatureFlag minAppVersion="0.3.8">
|
|
<UsbDeviceSetting />
|
|
</FeatureFlag>
|
|
|
|
<FeatureFlag minAppVersion="0.3.8">
|
|
<UsbInfoSetting />
|
|
</FeatureFlag>
|
|
</div>
|
|
);
|
|
}
|