import QtQuick import Quickshell import Quickshell.I3 import Quickshell.Io import Quickshell.Wayland import qs.Commons import qs.Services.Keyboard Item { id: root // Properties that match the facade interface property ListModel workspaces: ListModel {} property var windows: [] property int focusedWindowIndex: -1 // Signals that match the facade interface signal workspaceChanged signal activeWindowChanged signal windowListChanged signal displayScalesChanged // I3-specific properties property bool initialized: false // Debounce timer for updates Timer { id: updateTimer interval: 50 repeat: false onTriggered: safeUpdate() } // Initialization function initialize() { if (initialized) return; try { I3.refreshWorkspaces(); I3.dispatch('(["input"])'); Qt.callLater(() => { safeUpdateWorkspaces(); safeUpdateWindows(); queryDisplayScales(); queryKeyboardLayout(); }); initialized = true; Logger.i("SwayService", "Service started"); } catch (e) { Logger.e("SwayService", "Failed to initialize:", e); } } // Query display scales function queryDisplayScales() { swayOutputsProcess.running = true; } // Sway outputs process for display scale detection Process { id: swayOutputsProcess running: false command: ["swaymsg", "-t", "get_outputs", "-r"] property string accumulatedOutput: "" stdout: SplitParser { onRead: function (line) { swayOutputsProcess.accumulatedOutput += line; } } onExited: function (exitCode) { if (exitCode !== 0 || !accumulatedOutput) { Logger.e("SwayService", "Failed to query outputs, exit code:", exitCode); accumulatedOutput = ""; return; } try { const outputsData = JSON.parse(accumulatedOutput); const scales = {}; for (const output of outputsData) { if (output.name) { scales[output.name] = { "name": output.name, "scale": output.scale || 1.0, "width": output.current_mode ? output.current_mode.width : 0, "height": output.current_mode ? output.current_mode.height : 0, "refresh_rate": output.current_mode ? output.current_mode.refresh : 0, "x": output.rect ? output.rect.x : 0, "y": output.rect ? output.rect.y : 0, "active": output.active || false, "focused": output.focused || false, "current_workspace": output.current_workspace || "" }; } } // Notify CompositorService (it will emit displayScalesChanged) if (CompositorService && CompositorService.onDisplayScalesUpdated) { CompositorService.onDisplayScalesUpdated(scales); } } catch (e) { Logger.e("SwayService", "Failed to parse outputs:", e); } finally { // Clear accumulated output for next query accumulatedOutput = ""; } } } Timer { id: keyboardLayoutUpdateTimer interval: 1000 running: true repeat: true onTriggered: { queryKeyboardLayout(); } } function queryKeyboardLayout() { swayInputsProcess.running = true; } // Sway inputs process for keyboard layout detection Process { id: swayInputsProcess running: false command: ["swaymsg", "-t", "get_inputs", "-r"] property string accumulatedOutput: "" stdout: SplitParser { onRead: function (line) { // Accumulate lines instead of parsing each one swayInputsProcess.accumulatedOutput += line; } } onExited: function (exitCode) { if (exitCode !== 0 || !accumulatedOutput) { Logger.e("SwayService", "Failed to query inputs, exit code:", exitCode); accumulatedOutput = ""; return; } try { const inputsData = JSON.parse(accumulatedOutput); for (const input of inputsData) { if (input.type == "keyboard") { const layoutName = input.xkb_active_layout_name; KeyboardLayoutService.setCurrentLayout(layoutName); Logger.d("SwayService", "Keyboard layout switched:", layoutName); break; } } } catch (e) { Logger.e("SwayService", "Failed to parse inputs:", e); } finally { // Clear accumulated output for next query accumulatedOutput = ""; } } } // Safe update wrapper function safeUpdate() { safeUpdateWindows(); safeUpdateWorkspaces(); windowListChanged(); } // Safe workspace update function safeUpdateWorkspaces() { try { workspaces.clear(); if (!I3.workspaces || !I3.workspaces.values) { return; } const hlWorkspaces = I3.workspaces.values; for (var i = 0; i < hlWorkspaces.length; i++) { const ws = hlWorkspaces[i]; if (!ws || ws.id < 1) continue; const wsData = { "id": i, "idx": ws.num, "name": ws.name || "", "output": (ws.monitor && ws.monitor.name) ? ws.monitor.name : "", "isActive": ws.active === true, "isFocused": ws.focused === true, "isUrgent": ws.urgent === true, "isOccupied": true, "handle": ws }; workspaces.append(wsData); } } catch (e) { Logger.e("SwayService", "Error updating workspaces:", e); } } // Safe window update function safeUpdateWindows() { try { const windowsList = []; if (!ToplevelManager.toplevels || !ToplevelManager.toplevels.values) { windows = []; focusedWindowIndex = -1; return; } const hlToplevels = ToplevelManager.toplevels.values; let newFocusedIndex = -1; for (var i = 0; i < hlToplevels.length; i++) { const toplevel = hlToplevels[i]; if (!toplevel) continue; const windowData = extractWindowData(toplevel); if (windowData) { windowsList.push(windowData); if (windowData.isFocused) { newFocusedIndex = windowsList.length - 1; } } } windows = windowsList; if (newFocusedIndex !== focusedWindowIndex) { focusedWindowIndex = newFocusedIndex; activeWindowChanged(); } } catch (e) { Logger.e("SwayService", "Error updating windows:", e); } } // Extract window data safely from a toplevel function extractWindowData(toplevel) { if (!toplevel) return null; try { // Safely extract properties const appId = getAppId(toplevel); const title = safeGetProperty(toplevel, "title", ""); const focused = toplevel.activated === true; return { "title": title, "appId": appId, "isFocused": focused, "handle": toplevel }; } catch (e) { return null; } } function getAppId(toplevel) { if (!toplevel) return ""; return toplevel.appId; } // Safe property getter function safeGetProperty(obj, prop, defaultValue) { try { const value = obj[prop]; if (value !== undefined && value !== null) { return String(value); } } catch (e) // Property access failed {} return defaultValue; } function handleInputEvent(ev) { try { let beforeParenthesis; const parenthesisPos = ev.lastIndexOf('('); if (parenthesisPos === -1) { beforeParenthesis = ev; } else { beforeParenthesis = ev.substring(0, parenthesisPos); } const layoutNameStart = beforeParenthesis.lastIndexOf(',') + 1; const layoutName = ev.substring(layoutNameStart); KeyboardLayoutService.setCurrentLayout(layoutName); Logger.d("HyprlandService", "Keyboard layout switched:", layoutName); } catch (e) { Logger.e("HyprlandService", "Error handling activelayout:", e); } } // Connections to I3 Connections { target: I3.workspaces enabled: initialized function onValuesChanged() { safeUpdateWorkspaces(); workspaceChanged(); } } Connections { target: ToplevelManager enabled: initialized function onActiveToplevelChanged() { updateTimer.restart(); } } Connections { target: I3 enabled: initialized function onRawEvent(event) { safeUpdateWorkspaces(); workspaceChanged(); updateTimer.restart(); if (event.type === "output") { Qt.callLater(queryDisplayScales); } if (event.type == "get_inputs") { handleInputEvent(event.data); } } } // Public functions function switchToWorkspace(workspace) { try { workspace.handle.activate(); } catch (e) { Logger.e("SwayService", "Failed to switch workspace:", e); } } function focusWindow(window) { try { window.handle.activate(); } catch (e) { Logger.e("SwayService", "Failed to switch window:", e); } } function closeWindow(window) { try { window.handle.close(); } catch (e) { Logger.e("SwayService", "Failed to close window:", e); } } function logout() { try { Quickshell.execDetached(["swaymsg", "exit"]); } catch (e) { Logger.e("SwayService", "Failed to logout:", e); } } }