1
0
mirror of https://github.com/AvengeMedia/DankMaterialShell.git synced 2025-12-09 15:05:39 -05:00
Files
DankMaterialShell/Services/BrightnessService.qml
2025-08-13 09:56:37 -04:00

399 lines
13 KiB
QML

import QtQuick
import Quickshell
import Quickshell.Io
import qs.Common
pragma Singleton
pragma ComponentBehavior
Singleton {
id: root
property bool brightnessAvailable: devices.length > 0
property var devices: []
property var deviceBrightness: ({})
property string currentDevice: ""
property string lastIpcDevice: ""
property int brightnessLevel: {
const deviceToUse = lastIpcDevice === "" ? getDefaultDevice() : (lastIpcDevice || currentDevice);
return deviceToUse ? (deviceBrightness[deviceToUse] || 50) : 50;
}
property int maxBrightness: 100
property bool brightnessInitialized: false
signal brightnessChanged()
signal deviceSwitched()
property bool nightModeActive: false
function setBrightnessInternal(percentage, device) {
const clampedValue = Math.max(1, Math.min(100, percentage));
const actualDevice = device === "" ? getDefaultDevice() : (device || currentDevice || getDefaultDevice());
// Update the device brightness cache
if (actualDevice) {
var newBrightness = deviceBrightness;
newBrightness[actualDevice] = clampedValue;
deviceBrightness = newBrightness;
}
if (device)
brightnessSetProcess.command = ["brightnessctl", "-d", device, "set", clampedValue + "%"];
else
brightnessSetProcess.command = ["brightnessctl", "set", clampedValue + "%"];
brightnessSetProcess.running = true;
}
function setBrightness(percentage, device) {
setBrightnessInternal(percentage, device);
brightnessChanged();
}
function setCurrentDevice(deviceName) {
if (currentDevice === deviceName)
return ;
currentDevice = deviceName;
lastIpcDevice = deviceName;
deviceSwitched();
brightnessGetProcess.command = ["brightnessctl", "-m", "-d", deviceName, "get"];
brightnessGetProcess.running = true;
}
function refreshDevices() {
deviceListProcess.running = true;
}
function getDeviceBrightness(deviceName) {
return deviceBrightness[deviceName] || 50;
}
function getDefaultDevice() {
// Find first backlight device
for (const device of devices) {
if (device.class === "backlight") {
return device.name;
}
}
// Fallback to first device if no backlight found
return devices.length > 0 ? devices[0].name : "";
}
function getCurrentDeviceInfo() {
const deviceToUse = lastIpcDevice === "" ? getDefaultDevice() : (lastIpcDevice || currentDevice);
if (!deviceToUse) return null;
for (const device of devices) {
if (device.name === deviceToUse) {
return device;
}
}
return null;
}
function enableNightMode() {
if (nightModeActive) return;
nightModeActive = true;
SessionData.setNightModeEnabled(true);
}
function updateNightModeTemperature(temperature) {
SessionData.setNightModeTemperature(temperature);
// If night mode is active, restart it with new temperature
if (nightModeActive) {
// Temporarily disable and re-enable to restart with new temp
nightModeActive = false;
Qt.callLater(() => {
if (SessionData.nightModeEnabled) {
nightModeActive = true;
}
});
}
}
function disableNightMode() {
nightModeActive = false;
SessionData.setNightModeEnabled(false);
// Also kill any stray gammastep processes
Quickshell.execDetached(["pkill", "gammastep"]);
}
function toggleNightMode() {
if (nightModeActive) {
disableNightMode();
} else {
enableNightMode();
}
}
Component.onCompleted: {
refreshDevices();
// Check if night mode was enabled on startup
if (SessionData.nightModeEnabled) {
enableNightMode();
}
}
Process {
id: deviceListProcess
command: ["brightnessctl", "-m", "-l"]
onExited: function(exitCode) {
if (exitCode !== 0) {
console.warn("BrightnessService: Failed to list devices:", exitCode);
brightnessAvailable = false;
}
}
stdout: StdioCollector {
onStreamFinished: {
if (!text.trim()) {
console.warn("BrightnessService: No devices found");
return ;
}
const lines = text.trim().split("\n");
const newDevices = [];
for (const line of lines) {
const parts = line.split(",");
if (parts.length >= 5)
newDevices.push({
"name": parts[0],
"class": parts[1],
"current": parseInt(parts[2]),
"percentage": parseInt(parts[3]),
"max": parseInt(parts[4])
});
}
newDevices.sort((a, b) => {
if (a.class === "backlight" && b.class !== "backlight")
return -1;
if (a.class !== "backlight" && b.class === "backlight")
return 1;
return a.name.localeCompare(b.name);
});
devices = newDevices;
if (devices.length > 0 && !currentDevice)
setCurrentDevice(devices[0].name);
}
}
}
Process {
id: brightnessSetProcess
running: false
onExited: function(exitCode) {
if (exitCode !== 0)
console.warn("BrightnessService: Failed to set brightness:", exitCode);
}
}
Process {
id: brightnessGetProcess
running: false
onExited: function(exitCode) {
if (exitCode !== 0)
console.warn("BrightnessService: Failed to get brightness:", exitCode);
}
stdout: StdioCollector {
onStreamFinished: {
if (!text.trim())
return ;
const parts = text.trim().split(",");
if (parts.length >= 5) {
const current = parseInt(parts[2]);
const max = parseInt(parts[4]);
maxBrightness = max;
const brightness = Math.round((current / max) * 100);
// Update the device brightness cache
if (currentDevice) {
var newBrightness = deviceBrightness;
newBrightness[currentDevice] = brightness;
deviceBrightness = newBrightness;
}
brightnessInitialized = true;
console.log("BrightnessService: Device", currentDevice, "brightness:", brightness + "%");
brightnessChanged();
}
}
}
}
Process {
id: gammaStepProcess
command: {
const temperature = SessionData.nightModeTemperature || 4500;
return ["gammastep", "-m", "wayland", "-O", String(temperature)];
}
running: nightModeActive
onExited: function(exitCode) {
// Only show error if we're still supposed to be active (not manually disabled)
if (exitCode !== 0 && nightModeActive) {
console.warn("BrightnessService: Failed to enable night mode (gammastep not found or error)");
nightModeActive = false;
SessionData.setNightModeEnabled(false);
ToastService.showWarning("Night mode failed: gammastep not found or error occurred");
}
}
}
// IPC Handler for external control
IpcHandler {
function set(percentage: string, device: string) : string {
if (!root.brightnessAvailable)
return "Brightness control not available";
const value = parseInt(percentage);
const clampedValue = Math.max(1, Math.min(100, value));
const targetDevice = device || "";
root.lastIpcDevice = targetDevice;
if (targetDevice && targetDevice !== root.currentDevice) {
root.setCurrentDevice(targetDevice);
}
root.setBrightness(clampedValue, targetDevice);
if (targetDevice)
return "Brightness set to " + clampedValue + "% on " + targetDevice;
else
return "Brightness set to " + clampedValue + "%";
}
function increment(step: string, device: string) : string {
if (!root.brightnessAvailable)
return "Brightness control not available";
const targetDevice = device || "";
const actualDevice = targetDevice === "" ? root.getDefaultDevice() : targetDevice;
const currentLevel = actualDevice ? root.getDeviceBrightness(actualDevice) : root.brightnessLevel;
const stepValue = parseInt(step || "10");
const newLevel = Math.max(1, Math.min(100, currentLevel + stepValue));
root.lastIpcDevice = targetDevice;
if (targetDevice && targetDevice !== root.currentDevice) {
root.setCurrentDevice(targetDevice);
}
root.setBrightness(newLevel, targetDevice);
if (targetDevice)
return "Brightness increased to " + newLevel + "% on " + targetDevice;
else
return "Brightness increased to " + newLevel + "%";
}
function decrement(step: string, device: string) : string {
if (!root.brightnessAvailable)
return "Brightness control not available";
const targetDevice = device || "";
const actualDevice = targetDevice === "" ? root.getDefaultDevice() : targetDevice;
const currentLevel = actualDevice ? root.getDeviceBrightness(actualDevice) : root.brightnessLevel;
const stepValue = parseInt(step || "10");
const newLevel = Math.max(1, Math.min(100, currentLevel - stepValue));
root.lastIpcDevice = targetDevice;
if (targetDevice && targetDevice !== root.currentDevice) {
root.setCurrentDevice(targetDevice);
}
root.setBrightness(newLevel, targetDevice);
if (targetDevice)
return "Brightness decreased to " + newLevel + "% on " + targetDevice;
else
return "Brightness decreased to " + newLevel + "%";
}
function status() : string {
if (!root.brightnessAvailable)
return "Brightness control not available";
return "Device: " + root.currentDevice + " - Brightness: " + root.brightnessLevel + "%";
}
function list() : string {
if (!root.brightnessAvailable)
return "No brightness devices available";
let result = "Available devices:\n";
for (const device of root.devices) {
result += device.name + " (" + device.class + ")\n";
}
return result;
}
target: "brightness"
}
// IPC Handler for night mode control
IpcHandler {
function toggle() : string {
root.toggleNightMode();
return root.nightModeActive ? "Night mode enabled" : "Night mode disabled";
}
function enable() : string {
root.enableNightMode();
return "Night mode enabled";
}
function disable() : string {
root.disableNightMode();
return "Night mode disabled";
}
function status() : string {
return root.nightModeActive ? "Night mode is enabled" : "Night mode is disabled";
}
function temperature(value: string) : string {
if (!value) {
return "Current temperature: " + SessionData.nightModeTemperature + "K";
}
const temp = parseInt(value);
if (isNaN(temp)) {
return "Invalid temperature. Use a value between 2500 and 6000 (in steps of 500)";
}
// Validate temperature is in valid range and steps
if (temp < 2500 || temp > 6000) {
return "Temperature must be between 2500K and 6000K";
}
// Round to nearest 500
const rounded = Math.round(temp / 500) * 500;
SessionData.setNightModeTemperature(rounded);
// If night mode is active, restart it with new temperature
if (root.nightModeActive) {
root.nightModeActive = false;
Qt.callLater(() => {
root.nightModeActive = true;
});
}
if (rounded !== temp) {
return "Night mode temperature set to " + rounded + "K (rounded from " + temp + "K)";
} else {
return "Night mode temperature set to " + rounded + "K";
}
}
target: "night"
}
}