mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2025-12-05 21:15:38 -05:00
980 lines
32 KiB
QML
980 lines
32 KiB
QML
pragma Singleton
|
|
|
|
pragma ComponentBehavior: Bound
|
|
|
|
import QtQuick
|
|
import Quickshell
|
|
import Quickshell.Io
|
|
import qs.Common
|
|
|
|
Singleton {
|
|
id: root
|
|
|
|
property bool brightnessAvailable: devices.length > 0
|
|
property var devices: []
|
|
property var ddcDevices: []
|
|
property var deviceBrightness: ({})
|
|
property var ddcPendingInit: ({})
|
|
property string currentDevice: ""
|
|
property string lastIpcDevice: ""
|
|
property bool ddcAvailable: false
|
|
property var ddcInitQueue: []
|
|
property bool skipDdcRead: false
|
|
property int brightnessLevel: {
|
|
const deviceToUse = lastIpcDevice === "" ? getDefaultDevice() : (lastIpcDevice || currentDevice)
|
|
if (!deviceToUse) {
|
|
return 50
|
|
}
|
|
|
|
return getDeviceBrightness(deviceToUse)
|
|
}
|
|
property int maxBrightness: 100
|
|
property bool brightnessInitialized: false
|
|
|
|
signal brightnessChanged
|
|
signal deviceSwitched
|
|
|
|
property bool nightModeActive: nightModeEnabled
|
|
|
|
property bool nightModeEnabled: false
|
|
property bool automationAvailable: false
|
|
property bool geoclueAvailable: false
|
|
property bool isAutomaticNightTime: false
|
|
|
|
function buildGammastepCommand(gammastepArgs) {
|
|
const commandStr = "pkill gammastep; " + ["gammastep"].concat(gammastepArgs).join(" ")
|
|
return ["sh", "-c", commandStr]
|
|
}
|
|
|
|
function setBrightnessInternal(percentage, device) {
|
|
const clampedValue = Math.max(1, Math.min(100, percentage))
|
|
const actualDevice = device === "" ? getDefaultDevice() : (device || currentDevice || getDefaultDevice())
|
|
|
|
if (actualDevice) {
|
|
const newBrightness = Object.assign({}, deviceBrightness)
|
|
newBrightness[actualDevice] = clampedValue
|
|
deviceBrightness = newBrightness
|
|
}
|
|
|
|
const deviceInfo = getCurrentDeviceInfoByName(actualDevice)
|
|
|
|
if (deviceInfo && deviceInfo.class === "ddc") {
|
|
ddcBrightnessSetProcess.command = ["ddcutil", "setvcp", "-d", String(deviceInfo.ddcDisplay), "10", String(clampedValue)]
|
|
ddcBrightnessSetProcess.running = true
|
|
} else {
|
|
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, saveToSession = false) {
|
|
if (currentDevice === deviceName) {
|
|
return
|
|
}
|
|
|
|
currentDevice = deviceName
|
|
lastIpcDevice = deviceName
|
|
|
|
if (saveToSession) {
|
|
SessionData.setLastBrightnessDevice(deviceName)
|
|
}
|
|
|
|
deviceSwitched()
|
|
|
|
const deviceInfo = getCurrentDeviceInfoByName(deviceName)
|
|
if (deviceInfo && deviceInfo.class === "ddc") {
|
|
return
|
|
} else {
|
|
brightnessGetProcess.command = ["brightnessctl", "-m", "-d", deviceName, "get"]
|
|
brightnessGetProcess.running = true
|
|
}
|
|
}
|
|
|
|
function refreshDevices() {
|
|
deviceListProcess.running = true
|
|
}
|
|
|
|
function refreshDevicesInternal() {
|
|
const allDevices = [...devices, ...ddcDevices]
|
|
|
|
allDevices.sort((a, b) => {
|
|
if (a.class === "backlight" && b.class !== "backlight") {
|
|
return -1
|
|
}
|
|
if (a.class !== "backlight" && b.class === "backlight") {
|
|
return 1
|
|
}
|
|
|
|
if (a.class === "ddc" && b.class !== "ddc" && b.class !== "backlight") {
|
|
return -1
|
|
}
|
|
if (a.class !== "ddc" && b.class === "ddc" && a.class !== "backlight") {
|
|
return 1
|
|
}
|
|
|
|
return a.name.localeCompare(b.name)
|
|
})
|
|
|
|
devices = allDevices
|
|
|
|
if (devices.length > 0 && !currentDevice) {
|
|
const lastDevice = SessionData.lastBrightnessDevice || ""
|
|
const deviceExists = devices.some(d => d.name === lastDevice)
|
|
if (deviceExists) {
|
|
setCurrentDevice(lastDevice, false)
|
|
} else {
|
|
const nonKbdDevice = devices.find(d => !d.name.includes("kbd")) || devices[0]
|
|
setCurrentDevice(nonKbdDevice.name, false)
|
|
}
|
|
}
|
|
}
|
|
|
|
function getDeviceBrightness(deviceName) {
|
|
if (!deviceName) {
|
|
return
|
|
} 50
|
|
|
|
const deviceInfo = getCurrentDeviceInfoByName(deviceName)
|
|
if (!deviceInfo) {
|
|
return 50
|
|
}
|
|
|
|
if (deviceInfo.class === "ddc") {
|
|
return deviceBrightness[deviceName] || 50
|
|
}
|
|
|
|
return deviceBrightness[deviceName] || deviceInfo.percentage || 50
|
|
}
|
|
|
|
function getDefaultDevice() {
|
|
for (const device of devices) {
|
|
if (device.class === "backlight") {
|
|
return device.name
|
|
}
|
|
}
|
|
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 isCurrentDeviceReady() {
|
|
const deviceToUse = lastIpcDevice === "" ? getDefaultDevice() : (lastIpcDevice || currentDevice)
|
|
if (!deviceToUse) {
|
|
return false
|
|
}
|
|
|
|
if (ddcPendingInit[deviceToUse]) {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
function getCurrentDeviceInfoByName(deviceName) {
|
|
if (!deviceName) {
|
|
return null
|
|
}
|
|
|
|
for (const device of devices) {
|
|
if (device.name === deviceName) {
|
|
return device
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
function processNextDdcInit() {
|
|
if (ddcInitQueue.length === 0 || ddcInitialBrightnessProcess.running) {
|
|
return
|
|
}
|
|
|
|
const displayId = ddcInitQueue.shift()
|
|
ddcInitialBrightnessProcess.command = ["ddcutil", "getvcp", "-d", String(displayId), "10", "--brief"]
|
|
ddcInitialBrightnessProcess.running = true
|
|
}
|
|
|
|
// Night Mode Functions - Simplified
|
|
function enableNightMode() {
|
|
if (!automationAvailable) {
|
|
gammaStepTestProcess.running = true
|
|
return
|
|
}
|
|
|
|
nightModeEnabled = true
|
|
SessionData.setNightModeEnabled(true)
|
|
|
|
// Apply immediately or start automation
|
|
if (SessionData.nightModeAutoEnabled) {
|
|
startAutomation()
|
|
} else {
|
|
applyNightModeDirectly()
|
|
}
|
|
}
|
|
|
|
function disableNightMode() {
|
|
nightModeEnabled = false
|
|
SessionData.setNightModeEnabled(false)
|
|
stopAutomation()
|
|
// Nuclear approach - kill ALL gammastep processes multiple times
|
|
Quickshell.execDetached(["pkill", "-f", "gammastep"])
|
|
Quickshell.execDetached(["pkill", "-9", "gammastep"])
|
|
Quickshell.execDetached(["killall", "gammastep"])
|
|
// Also stop all related processes
|
|
gammaStepProcess.running = false
|
|
automationProcess.running = false
|
|
gammaStepTestProcess.running = false
|
|
}
|
|
|
|
function toggleNightMode() {
|
|
if (nightModeEnabled) {
|
|
disableNightMode()
|
|
} else {
|
|
enableNightMode()
|
|
}
|
|
}
|
|
|
|
function applyNightModeDirectly() {
|
|
const temperature = SessionData.nightModeTemperature || 4500
|
|
gammaStepProcess.command = buildGammastepCommand(["-m", "wayland", "-O", String(temperature)])
|
|
gammaStepProcess.running = true
|
|
}
|
|
|
|
function resetToNormalMode() {
|
|
// Just kill gammastep to return to normal display temperature
|
|
Quickshell.execDetached(["pkill", "gammastep"])
|
|
}
|
|
|
|
function startAutomation() {
|
|
if (!automationAvailable) {
|
|
return
|
|
}
|
|
|
|
const mode = SessionData.nightModeAutoMode || "time"
|
|
|
|
switch (mode) {
|
|
case "time":
|
|
startTimeBasedMode()
|
|
break
|
|
case "location":
|
|
startLocationBasedMode()
|
|
break
|
|
}
|
|
}
|
|
|
|
function stopAutomation() {
|
|
automationProcess.running = false
|
|
gammaStepProcess.running = false
|
|
isAutomaticNightTime = false
|
|
// Nuclear approach - kill ALL gammastep processes multiple times
|
|
Quickshell.execDetached(["pkill", "-f", "gammastep"])
|
|
Quickshell.execDetached(["pkill", "-9", "gammastep"])
|
|
Quickshell.execDetached(["killall", "gammastep"])
|
|
}
|
|
|
|
function startTimeBasedMode() {
|
|
checkTimeBasedMode()
|
|
}
|
|
|
|
function startLocationBasedMode() {
|
|
const temperature = SessionData.nightModeTemperature || 4500
|
|
const dayTemp = 6500
|
|
|
|
if (SessionData.latitude !== 0.0 && SessionData.longitude !== 0.0) {
|
|
automationProcess.command = buildGammastepCommand(["-m", "wayland", "-l", `${SessionData.latitude.toFixed(6)}:${SessionData.longitude.toFixed(6)}`, "-t", `${dayTemp}:${temperature}`, "-v"])
|
|
automationProcess.running = true
|
|
return
|
|
}
|
|
|
|
if (SessionData.nightModeLocationProvider === "geoclue2") {
|
|
automationProcess.command = buildGammastepCommand(["-m", "wayland", "-l", "geoclue2", "-t", `${dayTemp}:${temperature}`, "-v"])
|
|
automationProcess.running = true
|
|
return
|
|
}
|
|
|
|
console.warn("DisplayService: Location mode selected but no coordinates or geoclue provider set")
|
|
}
|
|
|
|
function checkTimeBasedMode() {
|
|
if (!nightModeEnabled || !SessionData.nightModeAutoEnabled || SessionData.nightModeAutoMode !== "time") {
|
|
return
|
|
}
|
|
|
|
const currentTime = systemClock.hours * 60 + systemClock.minutes
|
|
|
|
const startMinutes = SessionData.nightModeStartHour * 60 + SessionData.nightModeStartMinute
|
|
const endMinutes = SessionData.nightModeEndHour * 60 + SessionData.nightModeEndMinute
|
|
|
|
let shouldBeNight = false
|
|
|
|
if (startMinutes > endMinutes) {
|
|
shouldBeNight = (currentTime >= startMinutes) || (currentTime < endMinutes)
|
|
} else {
|
|
shouldBeNight = (currentTime >= startMinutes) && (currentTime < endMinutes)
|
|
}
|
|
|
|
if (shouldBeNight !== isAutomaticNightTime) {
|
|
isAutomaticNightTime = shouldBeNight
|
|
|
|
if (shouldBeNight) {
|
|
applyNightModeDirectly()
|
|
} else {
|
|
resetToNormalMode()
|
|
}
|
|
}
|
|
}
|
|
|
|
function detectLocationProviders() {
|
|
geoclueDetectionProcess.running = true
|
|
}
|
|
|
|
function setNightModeAutomationMode(mode) {
|
|
SessionData.setNightModeAutoMode(mode)
|
|
}
|
|
|
|
function evaluateNightMode() {
|
|
// Always stop all processes first to clean slate
|
|
stopAutomation()
|
|
|
|
if (!nightModeEnabled) {
|
|
return
|
|
}
|
|
|
|
if (SessionData.nightModeAutoEnabled) {
|
|
restartTimer.nextAction = "automation"
|
|
restartTimer.start()
|
|
} else {
|
|
restartTimer.nextAction = "direct"
|
|
restartTimer.start()
|
|
}
|
|
}
|
|
|
|
function checkNightModeAvailability() {
|
|
gammastepAvailabilityProcess.running = true
|
|
}
|
|
|
|
Timer {
|
|
id: restartTimer
|
|
property string nextAction: ""
|
|
interval: 100
|
|
repeat: false
|
|
|
|
onTriggered: {
|
|
if (nextAction === "automation") {
|
|
startAutomation()
|
|
} else if (nextAction === "direct") {
|
|
applyNightModeDirectly()
|
|
}
|
|
nextAction = ""
|
|
}
|
|
}
|
|
|
|
Component.onCompleted: {
|
|
ddcDetectionProcess.running = true
|
|
refreshDevices()
|
|
checkNightModeAvailability()
|
|
|
|
// Initialize night mode state from session
|
|
nightModeEnabled = SessionData.nightModeEnabled
|
|
}
|
|
|
|
Component.onDestruction: {
|
|
gammaStepProcess.running = false
|
|
automationProcess.running = false
|
|
}
|
|
|
|
SystemClock {
|
|
id: systemClock
|
|
precision: SystemClock.Minutes
|
|
onDateChanged: {
|
|
if (nightModeEnabled && SessionData.nightModeAutoEnabled && SessionData.nightModeAutoMode === "time") {
|
|
checkTimeBasedMode()
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: ddcDetectionProcess
|
|
|
|
command: ["which", "ddcutil"]
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
ddcAvailable = (exitCode === 0)
|
|
if (ddcAvailable) {
|
|
ddcDisplayDetectionProcess.running = true
|
|
} else {
|
|
console.log("DisplayService: ddcutil not available")
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: ddcDisplayDetectionProcess
|
|
|
|
command: ["bash", "-c", "ddcutil detect --brief 2>/dev/null | grep '^Display [0-9]' | awk '{print \"{\\\"display\\\":\" $2 \",\\\"name\\\":\\\"ddc-\" $2 \"\\\",\\\"class\\\":\\\"ddc\\\"}\"}' | tr '\\n' ',' | sed 's/,$//' | sed 's/^/[/' | sed 's/$/]/' || echo '[]'"]
|
|
running: false
|
|
|
|
stdout: StdioCollector {
|
|
onStreamFinished: {
|
|
if (!text.trim()) {
|
|
ddcDevices = []
|
|
return
|
|
}
|
|
|
|
try {
|
|
const parsedDevices = JSON.parse(text.trim())
|
|
const newDdcDevices = []
|
|
|
|
for (const device of parsedDevices) {
|
|
if (device.display && device.class === "ddc") {
|
|
newDdcDevices.push({
|
|
"name": device.name,
|
|
"class": "ddc",
|
|
"current": 50,
|
|
"percentage": 50,
|
|
"max": 100,
|
|
"ddcDisplay": device.display
|
|
})
|
|
}
|
|
}
|
|
|
|
ddcDevices = newDdcDevices
|
|
console.log("DisplayService: Found", ddcDevices.length, "DDC displays")
|
|
|
|
// Queue initial brightness readings for DDC devices
|
|
ddcInitQueue = []
|
|
for (const device of ddcDevices) {
|
|
ddcInitQueue.push(device.ddcDisplay)
|
|
// Mark DDC device as pending initialization
|
|
ddcPendingInit[device.name] = true
|
|
}
|
|
|
|
// Start processing the queue
|
|
processNextDdcInit()
|
|
|
|
// Refresh device list to include DDC devices
|
|
refreshDevicesInternal()
|
|
|
|
// Retry setting last device now that DDC devices are available
|
|
const lastDevice = SessionData.lastBrightnessDevice || ""
|
|
if (lastDevice) {
|
|
const deviceExists = devices.some(d => d.name === lastDevice)
|
|
if (deviceExists && (!currentDevice || currentDevice !== lastDevice)) {
|
|
setCurrentDevice(lastDevice, false)
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.warn("DisplayService: Failed to parse DDC devices:", error)
|
|
ddcDevices = []
|
|
}
|
|
}
|
|
}
|
|
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to detect DDC displays:", exitCode)
|
|
ddcDevices = []
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: deviceListProcess
|
|
|
|
command: ["brightnessctl", "-m", "-l"]
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to list devices:", exitCode)
|
|
brightnessAvailable = false
|
|
}
|
|
}
|
|
|
|
stdout: StdioCollector {
|
|
onStreamFinished: {
|
|
if (!text.trim()) {
|
|
console.warn("DisplayService: 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])
|
|
})
|
|
}
|
|
}
|
|
// Store brightnessctl devices separately
|
|
devices = newDevices
|
|
|
|
// Always refresh to combine with DDC devices and set up device selection
|
|
refreshDevicesInternal()
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: brightnessSetProcess
|
|
|
|
running: false
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to set brightness:", exitCode)
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: ddcBrightnessSetProcess
|
|
|
|
running: false
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to set DDC brightness:", exitCode)
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: ddcInitialBrightnessProcess
|
|
|
|
running: false
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to get initial DDC brightness:", exitCode)
|
|
}
|
|
|
|
processNextDdcInit()
|
|
}
|
|
|
|
stdout: StdioCollector {
|
|
onStreamFinished: {
|
|
if (!text.trim())
|
|
return
|
|
|
|
const parts = text.trim().split(" ")
|
|
if (parts.length >= 5) {
|
|
const current = parseInt(parts[3]) || 50
|
|
const max = parseInt(parts[4]) || 100
|
|
const brightness = Math.round((current / max) * 100)
|
|
|
|
const commandParts = ddcInitialBrightnessProcess.command
|
|
if (commandParts && commandParts.length >= 4) {
|
|
const displayId = commandParts[3]
|
|
const deviceName = "ddc-" + displayId
|
|
|
|
var newBrightness = Object.assign({}, deviceBrightness)
|
|
newBrightness[deviceName] = brightness
|
|
deviceBrightness = newBrightness
|
|
|
|
var newPending = Object.assign({}, ddcPendingInit)
|
|
delete newPending[deviceName]
|
|
ddcPendingInit = newPending
|
|
|
|
console.log("DisplayService: Initial DDC Device", deviceName, "brightness:", brightness + "%")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: brightnessGetProcess
|
|
|
|
running: false
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: 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 = Object.assign({}, deviceBrightness)
|
|
newBrightness[currentDevice] = brightness
|
|
deviceBrightness = newBrightness
|
|
}
|
|
|
|
brightnessInitialized = true
|
|
console.log("DisplayService: Device", currentDevice, "brightness:", brightness + "%")
|
|
brightnessChanged()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: ddcBrightnessGetProcess
|
|
|
|
running: false
|
|
onExited: function (exitCode) {
|
|
if (exitCode !== 0) {
|
|
console.warn("DisplayService: Failed to get DDC brightness:", exitCode)
|
|
}
|
|
}
|
|
|
|
stdout: StdioCollector {
|
|
onStreamFinished: {
|
|
if (!text.trim())
|
|
return
|
|
|
|
// Parse ddcutil getvcp output format: "VCP 10 C 50 100"
|
|
const parts = text.trim().split(" ")
|
|
if (parts.length >= 5) {
|
|
const current = parseInt(parts[3]) || 50
|
|
const max = parseInt(parts[4]) || 100
|
|
maxBrightness = max
|
|
const brightness = Math.round((current / max) * 100)
|
|
|
|
// Update the device brightness cache
|
|
if (currentDevice) {
|
|
var newBrightness = Object.assign({}, deviceBrightness)
|
|
newBrightness[currentDevice] = brightness
|
|
deviceBrightness = newBrightness
|
|
}
|
|
|
|
brightnessInitialized = true
|
|
console.log("DisplayService: DDC Device", currentDevice, "brightness:", brightness + "%")
|
|
brightnessChanged()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: gammastepAvailabilityProcess
|
|
command: ["which", "gammastep"]
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
automationAvailable = (exitCode === 0)
|
|
if (automationAvailable) {
|
|
detectLocationProviders()
|
|
|
|
// If night mode should be enabled on startup
|
|
if (nightModeEnabled && SessionData.nightModeAutoEnabled) {
|
|
startAutomation()
|
|
} else if (nightModeEnabled) {
|
|
applyNightModeDirectly()
|
|
}
|
|
} else {
|
|
console.log("DisplayService: gammastep not available")
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: geoclueDetectionProcess
|
|
command: ["sh", "-c", "busctl --system list | grep -qF org.freedesktop.GeoClue2"]
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
geoclueAvailable = (exitCode === 0)
|
|
console.log("DisplayService: geoclue available:", geoclueAvailable)
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: gammaStepTestProcess
|
|
command: ["which", "gammastep"]
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
if (exitCode === 0) {
|
|
automationAvailable = true
|
|
nightModeEnabled = true
|
|
SessionData.setNightModeEnabled(true)
|
|
|
|
if (SessionData.nightModeAutoEnabled) {
|
|
startAutomation()
|
|
} else {
|
|
applyNightModeDirectly()
|
|
}
|
|
} else {
|
|
console.warn("DisplayService: gammastep not found")
|
|
ToastService.showWarning("Night mode failed: gammastep not found")
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: gammaStepProcess
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
if (nightModeEnabled && exitCode !== 0 && exitCode !== 15) {
|
|
console.warn("DisplayService: Night mode process failed:", exitCode)
|
|
}
|
|
}
|
|
}
|
|
|
|
Process {
|
|
id: automationProcess
|
|
running: false
|
|
property string processType: "automation"
|
|
|
|
onExited: function (exitCode) {
|
|
if (nightModeEnabled && SessionData.nightModeAutoEnabled && exitCode !== 0 && exitCode !== 15) {
|
|
console.warn("DisplayService: Night mode automation failed:", exitCode)
|
|
// Location mode failed
|
|
console.warn("DisplayService: Location-based night mode failed")
|
|
}
|
|
}
|
|
}
|
|
|
|
// Session Data Connections
|
|
Connections {
|
|
target: SessionData
|
|
|
|
function onNightModeEnabledChanged() {
|
|
nightModeEnabled = SessionData.nightModeEnabled
|
|
evaluateNightMode()
|
|
}
|
|
|
|
function onNightModeAutoEnabledChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeAutoModeChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeStartHourChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeStartMinuteChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeEndHourChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeEndMinuteChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeTemperatureChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onLatitudeChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onLongitudeChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
function onNightModeLocationProviderChanged() {
|
|
evaluateNightMode()
|
|
}
|
|
}
|
|
|
|
// 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)
|
|
if (isNaN(value)) {
|
|
return "Invalid brightness value: " + percentage
|
|
}
|
|
|
|
const clampedValue = Math.max(1, Math.min(100, value))
|
|
const targetDevice = device || ""
|
|
|
|
// Ensure device exists if specified
|
|
if (targetDevice && !root.devices.some(d => d.name === targetDevice)) {
|
|
return "Device not found: " + targetDevice
|
|
}
|
|
|
|
root.lastIpcDevice = targetDevice
|
|
if (targetDevice && targetDevice !== root.currentDevice) {
|
|
root.setCurrentDevice(targetDevice, false)
|
|
}
|
|
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
|
|
|
|
// Ensure device exists
|
|
if (actualDevice && !root.devices.some(d => d.name === actualDevice)) {
|
|
return "Device not found: " + actualDevice
|
|
}
|
|
|
|
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, false)
|
|
}
|
|
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
|
|
|
|
// Ensure device exists
|
|
if (actualDevice && !root.devices.some(d => d.name === actualDevice)) {
|
|
return "Device not found: " + actualDevice
|
|
}
|
|
|
|
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, false)
|
|
}
|
|
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.nightModeEnabled ? "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.nightModeEnabled ? "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)
|
|
|
|
// Restart night mode with new temperature if active
|
|
if (root.nightModeEnabled) {
|
|
if (SessionData.nightModeAutoEnabled) {
|
|
root.startAutomation()
|
|
} else {
|
|
root.applyNightModeDirectly()
|
|
}
|
|
}
|
|
|
|
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"
|
|
}
|
|
}
|