pragma Singleton import QtQuick import Quickshell import Quickshell.Io import qs.Commons import qs.Services Singleton { id: root // Define our app directories // Default config directory: ~/.config/noctalia // Default cache directory: ~/.cache/noctalia property string shellName: "noctalia" property string configDir: Quickshell.env("NOCTALIA_CONFIG_DIR") || (Quickshell.env("XDG_CONFIG_HOME") || Quickshell.env( "HOME") + "/.config") + "/" + shellName + "/" property string cacheDir: Quickshell.env("NOCTALIA_CACHE_DIR") || (Quickshell.env("XDG_CACHE_HOME") || Quickshell.env( "HOME") + "/.cache") + "/" + shellName + "/" property string cacheDirImages: cacheDir + "images/" property string settingsFile: Quickshell.env("NOCTALIA_SETTINGS_FILE") || (configDir + "settings.json") property string defaultAvatar: Quickshell.env("HOME") + "/.face" property string defaultWallpapersDirectory: Quickshell.env("HOME") + "/Pictures/Wallpapers" property string defaultVideosDirectory: Quickshell.env("HOME") + "/Videos" property string defaultLocation: "Tokyo" property string defaultWallpaper: Quickshell.shellDir + "/Assets/Wallpaper/noctalia.png" // Used to access via Settings.data.xxx.yyy readonly property alias data: adapter property bool isLoaded: false property bool directoriesCreated: false // Signal emitted when settings are loaded after startupcale changes signal settingsLoaded // ----------------------------------------------------- // Function to validate monitor configurations function validateMonitorConfigurations() { var availableScreenNames = [] for (var i = 0; i < Quickshell.screens.length; i++) { availableScreenNames.push(Quickshell.screens[i].name) } Logger.log("Settings", "Available monitors: [" + availableScreenNames.join(", ") + "]") Logger.log("Settings", "Configured bar monitors: [" + adapter.bar.monitors.join(", ") + "]") // Check bar monitors if (adapter.bar.monitors.length > 0) { var hasValidBarMonitor = false for (var j = 0; j < adapter.bar.monitors.length; j++) { if (availableScreenNames.includes(adapter.bar.monitors[j])) { hasValidBarMonitor = true break } } if (!hasValidBarMonitor) { Logger.warn("Settings", "No configured bar monitors found on system, clearing bar monitor list to show on all screens") adapter.bar.monitors = [] } else { //Logger.log("Settings", "Found valid bar monitors, keeping configuration") } } else { //Logger.log("Settings", "Bar monitor list is empty, will show on all available screens") } } // ----------------------------------------------------- // If the settings structure has changed, ensure // backward compatibility by upgrading the settings function upgradeSettingsData() { const sections = ["left", "center", "right"] // ----------------- // 1st. check our settings are not super old, when we only had the widget type as a plain string for (var s = 0; s < sections.length; s++) { const sectionName = sections[s] for (var i = 0; i < adapter.bar.widgets[sectionName].length; i++) { var widget = adapter.bar.widgets[sectionName][i] if (typeof widget === "string") { adapter.bar.widgets[sectionName][i] = { "id": widget } } } } // ----------------- // 2nd. migrate global settings to user settings for (var s = 0; s < sections.length; s++) { const sectionName = sections[s] for (var i = 0; i < adapter.bar.widgets[sectionName].length; i++) { var widget = adapter.bar.widgets[sectionName][i] // Check if widget registry supports user settings, if it does not, then there is nothing to do const reg = BarWidgetRegistry.widgetMetadata[widget.id] if ((reg === undefined) || (reg.allowUserSettings === undefined) || !reg.allowUserSettings) { continue } // Check that the widget was not previously migrated and skip if necessary const keys = Object.keys(widget) if (keys.length > 1) { continue } migrateWidget(widget) Logger.log("Settings", JSON.stringify(widget)) } } } // ----------------------------------------------------- function migrateWidget(widget) { Logger.log("Settings", `Migrating '${widget.id}' widget`) switch (widget.id) { case "ActiveWindow": widget.showIcon = adapter.bar.showActiveWindowIcon break case "Battery": widget.alwaysShowPercentage = adapter.bar.alwaysShowBatteryPercentage break case "Brightness": widget.alwaysShowPercentage = BarWidgetRegistry.widgetMetadata[widget.id].alwaysShowPercentage break case "Clock": widget.showDate = adapter.location.showDateWithClock widget.use12HourClock = adapter.location.use12HourClock widget.reverseDayMonth = adapter.location.reverseDayMonth widget.showSeconds = BarWidgetRegistry.widgetMetadata[widget.id].showSeconds break case "MediaMini": widget.showAlbumArt = adapter.audio.showMiniplayerAlbumArt widget.showVisualizer = adapter.audio.showMiniplayerCava widget.visualizerType = BarWidgetRegistry.widgetMetadata[widget.id].visualizerType break case "NotificationHistory": widget.showUnreadBadge = BarWidgetRegistry.widgetMetadata[widget.id].showUnreadBadge widget.hideWhenZero = BarWidgetRegistry.widgetMetadata[widget.id].hideWhenZero break case "SidePanelToggle": widget.useDistroLogo = adapter.bar.useDistroLogo break case "SystemMonitor": widget.showNetworkStats = adapter.bar.showNetworkStats break case "Volume": widget.alwaysShowPercentage = BarWidgetRegistry.widgetMetadata[widget.id].alwaysShowPercentage break case "Workspace": widget.labelMode = adapter.bar.showWorkspaceLabel break } } // ----------------------------------------------------- // Kickoff essential services function kickOffServices() { // Ensure our location singleton is created as soon as possible so we start fetching weather asap LocationService.init() NightLightService.apply() ColorSchemeService.init() MatugenService.init() FontService.init() HooksService.init() } // ----------------------------------------------------- // Ensure directories exist before FileView tries to read files Component.onCompleted: { // ensure settings dir exists Quickshell.execDetached(["mkdir", "-p", configDir]) Quickshell.execDetached(["mkdir", "-p", cacheDir]) Quickshell.execDetached(["mkdir", "-p", cacheDirImages]) // Mark directories as created and trigger file loading directoriesCreated = true } // Don't write settings to disk immediately // This avoid excessive IO when a variable changes rapidly (ex: sliders) Timer { id: saveTimer running: false interval: 1000 onTriggered: settingsFileView.writeAdapter() } FileView { id: settingsFileView path: directoriesCreated ? settingsFile : "" watchChanges: true onFileChanged: reload() onAdapterUpdated: saveTimer.start() // Trigger initial load when path changes from empty to actual path onPathChanged: { if (path === settingsFile) { reload() } } onLoaded: function () { if (!isLoaded) { Logger.log("Settings", "----------------------------") Logger.log("Settings", "Settings loaded successfully") isLoaded = true upgradeSettingsData() validateMonitorConfigurations() kickOffServices() // Emit the signal root.settingsLoaded() } } onLoadFailed: function (error) { if (error.toString().includes("No such file") || error === 2) // File doesn't exist, create it with default values writeAdapter() } JsonAdapter { id: adapter property int settingsVersion: 1 // bar property JsonObject bar: JsonObject { property string position: "top" // "top" or "bottom" property real backgroundOpacity: 1.0 property list monitors: [] property bool showActiveWindowIcon: true // TODO: delete property bool alwaysShowBatteryPercentage: false // TODO: delete property bool showNetworkStats: false // TODO: delete property bool useDistroLogo: false // TODO: delete property string showWorkspaceLabel: "none" // TODO: delete // Widget configuration for modular bar system property JsonObject widgets widgets: JsonObject { property list left: [{ "id": "SystemMonitor" }, { "id": "ActiveWindow" }, { "id": "MediaMini" }] property list center: [{ "id": "Workspace" }] property list right: [{ "id": "ScreenRecorderIndicator" }, { "id": "Tray" }, { "id": "NotificationHistory" }, { "id": "WiFi" }, { "id": "Bluetooth" }, { "id": "Battery" }, { "id": "Volume" }, { "id": "Brightness" }, { "id": "NightLight" }, { "id": "Clock" }, { "id": "SidePanelToggle" }] } } // general property JsonObject general: JsonObject { property string avatarImage: defaultAvatar property bool dimDesktop: false property bool showScreenCorners: false property real radiusRatio: 1.0 // Animation speed multiplier (0.1x - 2.0x) property real animationSpeed: 1.0 } // location property JsonObject location: JsonObject { property string name: defaultLocation property bool useFahrenheit: false property bool reverseDayMonth: false // TODO: delete property bool use12HourClock: false // TODO: delete property bool showDateWithClock: false // TODO: delete } // screen recorder property JsonObject screenRecorder: JsonObject { property string directory: defaultVideosDirectory property int frameRate: 60 property string audioCodec: "opus" property string videoCodec: "h264" property string quality: "very_high" property string colorRange: "limited" property bool showCursor: true property string audioSource: "default_output" property string videoSource: "portal" } // wallpaper property JsonObject wallpaper: JsonObject { property bool enabled: true property string directory: defaultWallpapersDirectory property bool enableMultiMonitorDirectories: false property bool setWallpaperOnAllMonitors: true property string fillMode: "crop" property color fillColor: "#000000" property bool randomEnabled: false property int randomIntervalSec: 300 // 5 min property int transitionDuration: 1500 // 1500 ms property string transitionType: "random" property real transitionEdgeSmoothness: 0.05 property string defaultWallpaper: root.defaultWallpaper property list monitors: [] } // applauncher property JsonObject appLauncher: JsonObject { // When disabled, Launcher hides clipboard command and ignores cliphist property bool enableClipboardHistory: false // Position: center, top_left, top_right, bottom_left, bottom_right, bottom_center, top_center property string position: "center" property real backgroundOpacity: 1.0 property list pinnedExecs: [] property bool useApp2Unit: false } // dock property JsonObject dock: JsonObject { property bool autoHide: false property bool exclusive: false property real backgroundOpacity: 1.0 property list monitors: [] } // network property JsonObject network: JsonObject { property bool wifiEnabled: true property bool bluetoothEnabled: true } // notifications property JsonObject notifications: JsonObject { property bool doNotDisturb: false property list monitors: [] // Last time the user opened the notification history (ms since epoch) property real lastSeenTs: 0 } // audio property JsonObject audio: JsonObject { property int volumeStep: 5 property int cavaFrameRate: 60 property string visualizerType: "linear" property list mprisBlacklist: [] property string preferredPlayer: "" property bool showMiniplayerAlbumArt: false // TODO: delete property bool showMiniplayerCava: false // TODO: delete } // ui property JsonObject ui: JsonObject { property string fontDefault: "Roboto" property string fontFixed: "DejaVu Sans Mono" property string fontBillboard: "Inter" property list monitorsScaling: [] property bool idleInhibitorEnabled: false } // brightness property JsonObject brightness: JsonObject { property int brightnessStep: 5 } property JsonObject colorSchemes: JsonObject { property bool useWallpaperColors: false property string predefinedScheme: "" property bool darkMode: true } // matugen templates toggles property JsonObject matugen: JsonObject { // Per-template flags to control dynamic config generation property bool gtk4: false property bool gtk3: false property bool qt6: false property bool qt5: false property bool kitty: false property bool ghostty: false property bool foot: false property bool fuzzel: false property bool vesktop: false property bool enableUserTemplates: false } // night light property JsonObject nightLight: JsonObject { property bool enabled: false property bool autoSchedule: true property string nightTemp: "4000" property string dayTemp: "6500" property string manualSunrise: "06:30" property string manualSunset: "18:30" } // hooks property JsonObject hooks: JsonObject { property bool enabled: false property string wallpaperChange: "" property string darkModeChange: "" } } } }