mirror of
https://github.com/noctalia-dev/noctalia-shell.git
synced 2026-05-11 17:08:27 +08:00
201 lines
6.7 KiB
QML
201 lines
6.7 KiB
QML
import QtQuick
|
|
import QtQuick.Controls
|
|
import Quickshell
|
|
import qs.Commons
|
|
import qs.Modules.Bar.Extras
|
|
import qs.Modules.Panels.Settings
|
|
import qs.Services.Hardware
|
|
import qs.Services.UI
|
|
import qs.Widgets
|
|
|
|
Item {
|
|
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 barPosition: Settings.getBarPositionForScreen(screenName)
|
|
readonly property bool isBarVertical: barPosition === "left" || barPosition === "right"
|
|
readonly property string displayMode: (widgetSettings.displayMode !== undefined) ? widgetSettings.displayMode : widgetMetadata.displayMode
|
|
readonly property string iconColorKey: widgetSettings.iconColor !== undefined ? widgetSettings.iconColor : widgetMetadata.iconColor
|
|
readonly property string textColorKey: widgetSettings.textColor !== undefined ? widgetSettings.textColor : widgetMetadata.textColor
|
|
readonly property bool applyToAllMonitors: widgetSettings.applyToAllMonitors !== undefined ? widgetSettings.applyToAllMonitors : (Settings.data.brightness.syncAllMonitors !== undefined ? Settings.data.brightness.syncAllMonitors : widgetMetadata.applyToAllMonitors)
|
|
readonly property bool reverseScroll: Settings.data.general.reverseScroll
|
|
|
|
// Used to avoid opening the pill on Quickshell startup
|
|
property bool firstBrightnessReceived: false
|
|
|
|
implicitWidth: pill.width
|
|
implicitHeight: pill.height
|
|
|
|
// Track the brightness monitor reactively via declarative binding so it
|
|
// updates atomically when monitors change, avoiding a transient undefined
|
|
// state that occurs when Monitor QtObjects are destroyed before the
|
|
// imperative updateMonitor() call would run.
|
|
property var brightnessMonitor: {
|
|
var _ = BrightnessService.monitors; // reactive dependency
|
|
var __ = BrightnessService.ddcMonitors; // reactive dependency
|
|
if (!screen)
|
|
return null;
|
|
return BrightnessService.getMonitorForScreen(screen) ?? null;
|
|
}
|
|
|
|
function getControllableMonitorCount() {
|
|
var monitors = BrightnessService.monitors || [];
|
|
var count = 0;
|
|
for (var i = 0; i < monitors.length; i++) {
|
|
if (monitors[i] && monitors[i].brightnessControlAvailable)
|
|
count++;
|
|
}
|
|
return count;
|
|
}
|
|
|
|
visible: brightnessMonitor !== null
|
|
opacity: brightnessMonitor !== null ? 1.0 : 0.0
|
|
|
|
function getIcon() {
|
|
var monitor = brightnessMonitor;
|
|
if (!monitor || !monitor.brightnessControlAvailable || isNaN(monitor.brightness))
|
|
return "sun-off";
|
|
var brightness = monitor.brightness;
|
|
if (brightness <= 0.001)
|
|
return "sun-off";
|
|
return brightness <= 0.5 ? "brightness-low" : "brightness-high";
|
|
}
|
|
|
|
// Connection used to open the pill when brightness changes
|
|
Connections {
|
|
target: brightnessMonitor
|
|
ignoreUnknownSignals: true
|
|
function onBrightnessUpdated() {
|
|
// Ignore if this is the first time we receive an update.
|
|
// Most likely service just kicked off.
|
|
if (!firstBrightnessReceived) {
|
|
firstBrightnessReceived = true;
|
|
return;
|
|
}
|
|
|
|
pill.show();
|
|
hideTimerAfterChange.restart();
|
|
}
|
|
}
|
|
|
|
Timer {
|
|
id: hideTimerAfterChange
|
|
interval: 2500
|
|
running: false
|
|
repeat: false
|
|
onTriggered: pill.hide()
|
|
}
|
|
|
|
NPopupContextMenu {
|
|
id: contextMenu
|
|
|
|
model: [
|
|
{
|
|
"label": I18n.tr("actions.open-display-settings"),
|
|
"action": "open-display-settings",
|
|
"icon": "sun"
|
|
},
|
|
{
|
|
"label": I18n.tr("actions.widget-settings"),
|
|
"action": "widget-settings",
|
|
"icon": "settings"
|
|
},
|
|
]
|
|
|
|
onTriggered: action => {
|
|
contextMenu.close();
|
|
PanelService.closeContextMenu(screen);
|
|
|
|
if (action === "open-display-settings") {
|
|
var settingsPanel = PanelService.getPanel("settingsPanel", screen);
|
|
settingsPanel.requestedTab = SettingsPanel.Tab.Display;
|
|
settingsPanel.open();
|
|
} else if (action === "widget-settings") {
|
|
BarService.openWidgetSettings(screen, section, sectionWidgetIndex, widgetId, widgetSettings);
|
|
}
|
|
}
|
|
}
|
|
|
|
BarPill {
|
|
id: pill
|
|
|
|
screen: root.screen
|
|
oppositeDirection: BarService.getPillDirection(root)
|
|
customIconColor: Color.resolveColorKeyOptional(root.iconColorKey)
|
|
customTextColor: Color.resolveColorKeyOptional(root.textColorKey)
|
|
icon: getIcon()
|
|
autoHide: false // Important to be false so we can hover as long as we want
|
|
text: {
|
|
var monitor = brightnessMonitor;
|
|
if (!monitor || !monitor.brightnessControlAvailable || isNaN(monitor.brightness))
|
|
return "";
|
|
return Math.round(monitor.brightness * 100);
|
|
}
|
|
suffix: text.length > 0 ? "%" : "-"
|
|
forceOpen: displayMode === "alwaysShow"
|
|
forceClose: displayMode === "alwaysHide"
|
|
tooltipText: {
|
|
var monitor = brightnessMonitor;
|
|
var panel = PanelService.getPanel("brightnessPanel", screen);
|
|
if (panel?.isPanelOpen || !monitor || !monitor.brightnessControlAvailable || isNaN(monitor.brightness))
|
|
return "";
|
|
return I18n.tr("tooltips.brightness-at", {
|
|
"brightness": Math.round(monitor.brightness * 100)
|
|
});
|
|
}
|
|
|
|
onWheel: function (angle) {
|
|
var monitor = brightnessMonitor;
|
|
if (!monitor || !monitor.brightnessControlAvailable)
|
|
return;
|
|
|
|
if (root.reverseScroll)
|
|
angle *= -1;
|
|
|
|
if (angle === 0)
|
|
return;
|
|
|
|
var shouldApplyToAll = root.applyToAllMonitors && root.getControllableMonitorCount() > 1;
|
|
if (angle > 0) {
|
|
if (shouldApplyToAll) {
|
|
BrightnessService.increaseBrightness();
|
|
} else {
|
|
monitor.increaseBrightness();
|
|
}
|
|
} else if (angle < 0) {
|
|
if (shouldApplyToAll) {
|
|
BrightnessService.decreaseBrightness();
|
|
} else {
|
|
monitor.decreaseBrightness();
|
|
}
|
|
}
|
|
}
|
|
|
|
onClicked: PanelService.getPanel("brightnessPanel", screen)?.toggle(this)
|
|
|
|
onRightClicked: {
|
|
PanelService.showContextMenu(contextMenu, pill, screen);
|
|
}
|
|
}
|
|
}
|