import QtQuick import QtQuick.Controls import QtQuick.Effects import Quickshell import Quickshell.Widgets import qs.Commons import qs.Modules.Bar.Extras import qs.Modules.Panels.Settings import qs.Services.System import qs.Services.UI import qs.Widgets NIconButton { id: root property ShellScreen screen // Widget properties passed from Bar.qml for per-instance settings property string widgetId: "" property string section: "" property int sectionWidgetIndex: -1 property int sectionWidgetsCount: 0 property var widgetMetadata: BarWidgetRegistry.widgetMetadata[widgetId] // Explicit screenName property ensures reactive binding when screen changes readonly property string screenName: screen ? screen.name : "" property var widgetSettings: { if (section && sectionWidgetIndex >= 0 && screenName) { var widgets = Settings.getBarWidgetsForScreen(screenName)[section]; if (widgets && sectionWidgetIndex < widgets.length) { return widgets[sectionWidgetIndex]; } } return {}; } readonly property string customIcon: widgetSettings.icon !== undefined ? widgetSettings.icon : widgetMetadata.icon readonly property bool useDistroLogo: widgetSettings.useDistroLogo !== undefined ? widgetSettings.useDistroLogo : widgetMetadata.useDistroLogo readonly property string customIconPath: widgetSettings.customIconPath !== undefined ? widgetSettings.customIconPath : widgetMetadata.customIconPath readonly property bool enableColorization: widgetSettings.enableColorization !== undefined ? widgetSettings.enableColorization : widgetMetadata.enableColorization readonly property string colorizeSystemIcon: widgetSettings.colorizeSystemIcon !== undefined ? widgetSettings.colorizeSystemIcon : widgetMetadata.colorizeSystemIcon readonly property color iconColor: { if (!enableColorization) return Color.mOnSurface; return Color.resolveColorKey(colorizeSystemIcon); } // If we have a custom path and not using distro logo, use the theme icon. // If using distro logo, don't use theme icon. icon: (customIconPath === "" && !useDistroLogo) ? customIcon : "" tooltipText: { if (!screen || PanelService.getPanel("controlCenterPanel", screen)?.isPanelOpen) { return ""; } else { return I18n.tr("tooltips.open-control-center"); } } tooltipDirection: BarService.getTooltipDirection(screen?.name) baseSize: Style.getCapsuleHeightForScreen(screen?.name) applyUiScale: false customRadius: Style.radiusL colorBg: Style.capsuleColor colorFg: iconColor colorBgHover: Color.mHover colorFgHover: Color.mOnHover colorBorder: Style.capsuleBorderColor colorBorderHover: Style.capsuleBorderColor NPopupContextMenu { id: contextMenu model: [ { "label": I18n.tr("actions.open-launcher"), "action": "open-launcher", "icon": "search" }, { "label": I18n.tr("actions.open-settings"), "action": "open-settings", "icon": "adjustments" }, { "label": I18n.tr("actions.widget-settings"), "action": "widget-settings", "icon": "settings" }, ] onTriggered: action => { contextMenu.close(); PanelService.closeContextMenu(screen); if (action === "open-launcher") { PanelService.toggleLauncher(screen); } else if (action === "open-settings") { var panel = PanelService.getPanel("settingsPanel", screen); panel.requestedTab = SettingsPanel.Tab.General; panel.toggle(); } else if (action === "widget-settings") { BarService.openWidgetSettings(screen, section, sectionWidgetIndex, widgetId, widgetSettings); } } } onClicked: { var controlCenterPanel = PanelService.getPanel("controlCenterPanel", screen); if (Settings.data.controlCenter.position === "close_to_bar_button") { // Will open the panel next to the bar button. controlCenterPanel?.toggle(this); } else { controlCenterPanel?.toggle(); } } onRightClicked: { PanelService.showContextMenu(contextMenu, root, screen); } onMiddleClicked: PanelService.toggleLauncher(screen) IconImage { id: customOrDistroLogo anchors.centerIn: parent width: root.buttonSize * 0.8 height: width source: { if (useDistroLogo) return HostService.osLogo; if (customIconPath !== "") return customIconPath.startsWith("file://") ? customIconPath : "file://" + customIconPath; return ""; } visible: source !== "" smooth: true asynchronous: true layer.enabled: (enableColorization) && (useDistroLogo || customIconPath !== "") layer.effect: ShaderEffect { property color targetColor: !hovering ? iconColor : Color.mOnHover property real colorizeMode: 2.0 fragmentShader: Qt.resolvedUrl(Quickshell.shellDir + "/Shaders/qsb/appicon_colorize.frag.qsb") } } }