dotfiles/.config/quickshell/services/ConfigLoader.qml
2025-12-15 18:20:18 +01:00

146 lines
No EOL
5.1 KiB
QML

pragma Singleton
pragma ComponentBehavior: Bound
import "root:/modules/common"
import "root:/modules/common/functions/file_utils.js" as FileUtils
import "root:/modules/common/functions/string_utils.js" as StringUtils
import "root:/modules/common/functions/object_utils.js" as ObjectUtils
import QtQuick
import Quickshell
import Quickshell.Io
import Quickshell.Hyprland
import Qt.labs.platform
/**
* Loads and manages the shell configuration file.
* The config file is by default at XDG_CONFIG_HOME/quickshell/config.json.
* Automatically reloaded when the file changes, but does not provide a way to save changes.
*/
Singleton {
id: root
property string filePath: Directories.shellConfigPath
property bool firstLoad: true
function loadConfig() {
configFileView.reload()
}
function applyConfig(fileContent) {
try {
const json = JSON.parse(fileContent);
// Extract font configuration if it exists
let fontConfig = null;
let configForOptions = {};
// Copy all properties except font to configForOptions
for (let key in json) {
if (key !== "font") {
configForOptions[key] = json[key];
} else {
fontConfig = json[key];
}
}
// Apply the non-font configuration to ConfigOptions
ObjectUtils.applyToQtObject(ConfigOptions, configForOptions);
// Apply font configuration to Appearance if it exists
if (fontConfig && typeof Appearance !== 'undefined') {
if (fontConfig.family && Appearance.font && Appearance.font.family) {
ObjectUtils.applyToQtObject(Appearance.font.family, fontConfig.family);
}
if (fontConfig.pixelSize && Appearance.font && Appearance.font.pixelSize) {
ObjectUtils.applyToQtObject(Appearance.font.pixelSize, fontConfig.pixelSize);
}
}
if (root.firstLoad) {
root.firstLoad = false;
} else {
Hyprland.dispatch(`exec notify-send "${qsTr("Shell configuration reloaded")}" "${root.filePath}"`)
}
} catch (e) {
console.error("[ConfigLoader] Error reading file:", e);
Hyprland.dispatch(`exec notify-send "${qsTr("Shell configuration failed to load")}" "${root.filePath}"`)
return;
}
}
function setLiveConfigValue(nestedKey, value) {
let keys = nestedKey.split(".");
let targetObject = ConfigOptions;
// Check if this is a font-related configuration
if (keys[0] === "font") {
targetObject = Appearance;
}
let obj = targetObject;
let parents = [obj];
// Traverse and collect parent objects
for (let i = 0; i < keys.length - 1; ++i) {
if (!obj[keys[i]] || typeof obj[keys[i]] !== "object") {
obj[keys[i]] = {};
}
obj = obj[keys[i]];
parents.push(obj);
}
// Convert value to correct type using JSON.parse when safe
let convertedValue = value;
if (typeof value === "string") {
let trimmed = value.trim();
if (trimmed === "true" || trimmed === "false" || !isNaN(Number(trimmed))) {
try {
convertedValue = JSON.parse(trimmed);
} catch (e) {
convertedValue = value;
}
}
}
console.log(`[ConfigLoader] Setting live config value: ${nestedKey} = ${convertedValue}`);
obj[keys[keys.length - 1]] = convertedValue;
}
function saveConfig() {
const plainConfig = ObjectUtils.toPlainObject(ConfigOptions);
Hyprland.dispatch(`exec echo '${StringUtils.shellSingleQuoteEscape(JSON.stringify(plainConfig, null, 2))}' > '${root.filePath}'`)
}
Timer {
id: delayedFileRead
interval: ConfigOptions.hacks.arbitraryRaceConditionDelay
repeat: false
running: false
onTriggered: {
root.applyConfig(configFileView.text())
}
}
FileView {
id: configFileView
path: Qt.resolvedUrl(root.filePath)
watchChanges: true
onFileChanged: {
console.log("[ConfigLoader] File changed, reloading...")
this.reload()
delayedFileRead.start()
}
onLoadedChanged: {
const fileContent = configFileView.text()
root.applyConfig(fileContent)
}
onLoadFailed: (error) => {
if(error == FileViewError.FileNotFound) {
console.log("[ConfigLoader] File not found, creating new file.")
root.saveConfig()
Hyprland.dispatch(`exec notify-send "${qsTr("Shell configuration created")}" "${root.filePath}"`)
} else {
Hyprland.dispatch(`exec notify-send "${qsTr("Shell configuration failed to load")}" "${root.filePath}"`)
}
}
}
}