import QtQuick import Quickshell import Quickshell.Services.UPower import QtQuick.Layouts import qs.Components import qs.Settings Item { id: batteryWidget property var battery: UPower.displayDevice property bool isReady: battery && battery.ready && battery.isLaptopBattery && battery.isPresent property real percent: isReady ? (battery.percentage * 100) : 0 property bool charging: isReady ? battery.state === UPowerDeviceState.Charging : false property bool show: isReady && percent > 0 // Choose icon based on charge and charging state function batteryIcon() { if (!show) return ""; if (charging) return "battery_android_bolt"; if (percent >= 95) return "battery_android_full"; var step = Math.round(percent / (100 / 6)); return "battery_android_" + step } visible: isReady && battery.isLaptopBattery width: pill.width height: pill.height PillIndicator { id: pill icon: batteryIcon() text: Math.round(batteryWidget.percent) + "%" pillColor: Theme.surfaceVariant iconCircleColor: Theme.accentPrimary textColor: charging ? Theme.accentPrimary : Theme.textPrimary MouseArea { anchors.fill: parent hoverEnabled: true onEntered: { pill.show(); batteryTooltip.tooltipVisible = true; } onExited: { pill.hide(); batteryTooltip.tooltipVisible = false; } } StyledTooltip { id: batteryTooltip text: { let lines = []; if (batteryWidget.isReady) { lines.push(batteryWidget.charging ? "Charging" : "Discharging"); lines.push(Math.round(batteryWidget.percent) + "%"); if (batteryWidget.battery.changeRate !== undefined) lines.push("Rate: " + batteryWidget.battery.changeRate.toFixed(2) + " W"); if (batteryWidget.battery.timeToEmpty > 0) lines.push("Time left: " + Math.floor(batteryWidget.battery.timeToEmpty / 60) + " min"); if (batteryWidget.battery.timeToFull > 0) lines.push("Time to full: " + Math.floor(batteryWidget.battery.timeToFull / 60) + " min"); if (batteryWidget.battery.healthPercentage !== undefined) lines.push("Health: " + Math.round(batteryWidget.battery.healthPercentage) + "%"); } return lines.join("\n"); } tooltipVisible: false targetItem: pill delay: 1500 } } }