theshell/upowerdbus.cpp
2017-01-15 00:38:39 +11:00

405 lines
19 KiB
C++

#include "upowerdbus.h"
#include "power_adaptor.h"
extern void EndSession(EndSessionWait::shutdownType type);
UPowerDBus::UPowerDBus(NotificationDBus* notificationDBus, QObject *parent) : QObject(parent)
{
new PowerAdaptor(this);
QDBusConnection::sessionBus().registerObject("/org/thesuite/Power", "org.thesuite.Power", this);
this->notificationDBus = notificationDBus;
connect(notificationDBus, SIGNAL(ActionInvoked(uint,QString)), this, SLOT(ActionInvoked(uint,QString)));
//Inhibit logind's handling of some power events
QDBusMessage message = QDBusMessage::createMethodCall("org.freedesktop.login1", "/org/freedesktop/login1", "org.freedesktop.login1.Manager", "Inhibit");
message.setArguments(QList<QVariant>() << "handle-power-key:handle-suspend-key:handle-lid-switch" << "theShell" << "theShell Handles Hardware Power Keys" << "block");
QDBusReply<QDBusUnixFileDescriptor> inhibitReply = QDBusConnection::systemBus().call(message);
powerInhibit = inhibitReply.value();
//Set up timer to check UPower properties
checkTimer = new QTimer(this);
checkTimer->setInterval(1000);
connect(checkTimer, SIGNAL(timeout()), this, SLOT(checkUpower()));
connect(checkTimer, SIGNAL(timeout()), this, SLOT(DeviceChanged()));
checkTimer->start();
QDBusConnection::systemBus().connect("org.freedesktop.UPower", "/org/freedesktop/UPower", "org.freedesktop.UPower", "DeviceAdded", this, SLOT(devicesChanged()));
QDBusConnection::systemBus().connect("org.freedesktop.UPower", "/org/freedesktop/UPower", "org.freedesktop.UPower", "DeviceRemoved", this, SLOT(devicesChanged()));
devicesChanged();
}
void UPowerDBus::devicesChanged() {
allDevices.clear();
QDBusInterface *i = new QDBusInterface("org.freedesktop.UPower", "/org/freedesktop/UPower", "org.freedesktop.UPower", QDBusConnection::systemBus(), this);
QDBusReply<QList<QDBusObjectPath>> reply = i->call("EnumerateDevices"); //Get all devices
if (reply.isValid()) { //Check if the reply is ok
for (QDBusObjectPath device : reply.value()) {
if (device.path().contains("battery") || device.path().contains("media_player") || device.path().contains("computer") || device.path().contains("phone")) { //This is a battery or media player or tablet computer
QDBusConnection::systemBus().connect("org.freedesktop.UPower", device.path(),
"org.freedesktop.DBus.Properties", "PropertiesChanged", this,
SLOT(DeviceChanged()));
QDBusInterface *i = new QDBusInterface("org.freedesktop.UPower", device.path(), "org.freedesktop.UPower.Device", QDBusConnection::systemBus(), this);
allDevices.append(i);
}
if (device.path().contains("battery")) {
batteryPath = device;
}
}
DeviceChanged();
} else {
emit updateDisplay("Can't get battery information.");
}
}
void UPowerDBus::DeviceChanged() {
QStringList displayOutput;
if (isPowerStretchOn) {
displayOutput.append("<span style=\"background-color: orange; color: black;\">Power Stretch on</span>");
}
for (QDBusInterface *i : allDevices) {
//Get the percentage of battery remaining.
//We do the calculation ourselves because UPower can be inaccurate sometimes
double percentage;
//Check that the battery actually reports energy information
double energyFull = i->property("EnergyFull").toDouble();
double energy = i->property("Energy").toDouble();
double energyEmpty = i->property("EnergyEmpty").toDouble();
if (energyFull == 0 && energy == 0 && energyEmpty == 0) {
//The battery does not report energy information, so get the percentage from UPower.
percentage = i->property("Percentage").toDouble();
} else {
//Calculate the percentage ourself, and round it to an integer.
//Add 0.5 because C++ always rounds down.
percentage = (int) (((energy - energyEmpty) / (energyFull - energyEmpty) * 100) + 0.5);
}
if (i->path().contains("battery")) {
//PC Battery
if (i->property("IsPresent").toBool()) {
bool showRed = false;
qulonglong timeToFull = i->property("TimeToFull").toULongLong();
qulonglong timeToEmpty = i->property("TimeToEmpty").toULongLong();
//Depending on the state, do different things.
QString state;
switch (i->property("State").toUInt()) {
case 1: //Charging
state = " (Charging";
if (!isCharging) {
QString message;
if (isPowerStretchOn) {
setPowerStretch(false);
message = "The power cable has been plugged in and the battery is now being charged. Power Stretch has been turned off.";
} else {
message = "The power cable has been plugged in and the battery is now being charged.";
}
QVariantMap hints;
hints.insert("category", "battery.charging");
hints.insert("transient", true);
this->notificationDBus->Notify("theShell", 0, "", "Charging",
message, QStringList(), hints, 10000);
QSoundEffect* chargingSound = new QSoundEffect();
chargingSound->setSource(QUrl("qrc:/sounds/charging.wav"));
chargingSound->play();
connect(chargingSound, SIGNAL(playingChanged()), chargingSound, SLOT(deleteLater()));
}
isCharging = true;
isConnectedToPower = true;
if (timeToFull != 0) {
timeRemain = QDateTime::fromTime_t(timeToFull).toUTC();
state.append(" · " + timeRemain.toString("h:mm"));
} else {
timeRemain = QDateTime(QDate(0, 0, 0));
}
if (hourBatteryWarning) {
hourBatteryWarning = false;
halfHourBatteryWarning = false;
tenMinuteBatteryWarning = false;
this->notificationDBus->CloseNotification(batteryLowNotificationNumber);
batteryLowNotificationNumber = 0;
}
state += ")";
break;
case 2: //Discharging
state = " (Discharging";
if (isConnectedToPower) {
QVariantMap hints;
hints.insert("category", "battery.discharging");
hints.insert("transient", true);
this->notificationDBus->Notify("theShell", 0, "", "Discharging",
"The power cable has been removed, and your PC is now running on battery power.",
QStringList(), hints, 10000);
}
isConnectedToPower = false;
isCharging = false;
if (timeToEmpty != 0) {
timeRemain = QDateTime::fromTime_t(timeToEmpty).toUTC();
state.append(" · " + timeRemain.toString("h:mm"));
if (timeToEmpty <= 600 && tenMinuteBatteryWarning == false) { //Ten minutes left! Critical!
QVariantMap hints;
hints.insert("urgency", 2);
hints.insert("category", "battery.critical");
QStringList actions;
if (!isPowerStretchOn) {
actions.append("power-stretch-on");
actions.append("Turn on Power Stretch");
}
batteryLowNotificationNumber = this->notificationDBus->Notify("theShell", batteryLowNotificationNumber, "", "Battery Critically Low",
"You have about 10 minutes of battery remaining."
" Either plug in your PC or save your work"
" and power off the PC and change the battery.", actions, hints, 0);
tenMinuteBatteryWarning = true;
halfHourBatteryWarning = true;
hourBatteryWarning = true;
QSoundEffect* chargingSound = new QSoundEffect();
chargingSound->setSource(QUrl("qrc:/sounds/powerlow.wav"));
chargingSound->play();
connect(chargingSound, SIGNAL(playingChanged()), chargingSound, SLOT(deleteLater()));
} else if (timeToEmpty <= 1800 && halfHourBatteryWarning == false) { //Half hour left! Low!
QVariantMap hints;
hints.insert("urgency", 2);
hints.insert("category", "battery.low");
QStringList actions;
if (!isPowerStretchOn) {
actions.append("power-stretch-on");
actions.append("Turn on Power Stretch");
}
batteryLowNotificationNumber = this->notificationDBus->Notify("theShell", batteryLowNotificationNumber, "", "Battery Low",
"You have about half an hour of battery remaining."
" You should plug in your PC now.", actions, hints, 10000);
halfHourBatteryWarning = true;
hourBatteryWarning = true;
QSoundEffect* chargingSound = new QSoundEffect();
chargingSound->setSource(QUrl("qrc:/sounds/powerlow.wav"));
chargingSound->play();
connect(chargingSound, SIGNAL(playingChanged()), chargingSound, SLOT(deleteLater()));
} else if (timeToEmpty <= 3600 && hourBatteryWarning == false) { //One hour left! Warning!
QVariantMap hints;
hints.insert("urgency", 2);
hints.insert("category", "battery.low");
QStringList actions;
if (!isPowerStretchOn) {
actions.append("power-stretch-on");
actions.append("Turn on Power Stretch");
}
batteryLowNotificationNumber = this->notificationDBus->Notify("theShell", batteryLowNotificationNumber, "", "Battery Warning",
"You have about an hour of battery remaining."
" You may want to plug in your PC now.", actions, hints, 10000);
hourBatteryWarning = true;
QSoundEffect* chargingSound = new QSoundEffect();
chargingSound->setSource(QUrl("qrc:/sounds/powerlow.wav"));
chargingSound->play();
connect(chargingSound, SIGNAL(playingChanged()), chargingSound, SLOT(deleteLater()));
}
if (halfHourBatteryWarning || tenMinuteBatteryWarning) {
showRed = true;
}
} else {
timeRemain = QDateTime(QDate(0, 0, 0));
}
state += ")";
break;
case 3: //Empty
state = " (Empty)";
break;
case 4: //Charged
case 6: //Pending Discharge
state = " (Full)";
if (isCharging) {
QVariantMap hints;
hints.insert("category", "battery.charged");
hints.insert("transient", true);
this->notificationDBus->Notify("theShell", 0, "", "Battery Charged",
"The battery has been charged completely."
, QStringList(), hints, 10000);
}
isCharging = false;
isConnectedToPower = true;
timeRemain = QDateTime(QDate(0, 0, 0));
break;
case 5: //Pending Charge
state = " (Not Charging)";
break;
}
if (showRed) {
displayOutput.append("<span style=\"background-color: red; color: black;\">" + QString::number(percentage) + "% PC Battery" + state + "</span>");
} else {
displayOutput.append(QString::number(percentage) + "% PC Battery" + state);
}
batLevel = percentage;
} else {
displayOutput.append("No Battery Inserted");
}
} else if (i->path().contains("media_player") || i->path().contains("computer") || i->path().contains("phone")) {
//This is an external media player (or tablet)
//Get the model of this media player
QString model = i->property("Model").toString();
if (i->property("Serial").toString().length() == 40 && i->property("Vendor").toString().contains("Apple") && QFile("/usr/bin/idevice_id").exists()) { //This is probably an iOS device
//Get the name of the iOS device
QProcess iosName;
iosName.start("idevice_id " + i->property("Serial").toString());
iosName.waitForFinished();
QString name(iosName.readAll());
name = name.trimmed();
if (name != "" && !name.startsWith("ERROR:")) {
model = name;
}
}
if (i->property("State").toUInt() == 0) {
if (QFile("/usr/bin/thefile").exists()) {
displayOutput.append("Pair " + model + " using theFile to see battery status.");
} else {
displayOutput.append(model + " battery unavailable. Device trusted?");
}
} else {
QString batteryText;
batteryText.append(QString::number(percentage) + "% battery on " + model);
switch (i->property("State").toUInt()) {
case 1:
batteryText.append(" (Charging)");
break;
case 2:
batteryText.append(" (Discharging)");
break;
case 3:
batteryText.append(" (Empty)");
break;
case 4:
case 6:
batteryText.append(" (Full)");
break;
case 5:
batteryText.append(" (Not Charging)");
break;
}
displayOutput.append(batteryText);
}
}
}
//If KDE Connect is running, check the battery status of connected devices.
if (QDBusConnection::sessionBus().interface()->registeredServiceNames().value().contains("org.kde.kdeconnect")) {
//Get connected devices
QDBusMessage devicesMessage = QDBusMessage::createMethodCall("org.kde.kdeconnect", "/modules/kdeconnect", "org.kde.kdeconnect.daemon", "devices");
devicesMessage.setArguments(QVariantList() << true);
QDBusReply<QStringList> connectedDevices = QDBusConnection::sessionBus().call(devicesMessage, QDBus::Block, 5000);
if (connectedDevices.isValid()) {
for (QString device : connectedDevices.value()) {
QDBusInterface interface("org.kde.kdeconnect", "/modules/kdeconnect/devices/" + device, "org.kde.kdeconnect.device");
QString name = interface.property("name").toString();
QDBusInterface batteryInterface("org.kde.kdeconnect", "/modules/kdeconnect/devices/" + device, "org.kde.kdeconnect.device.battery");
if (batteryInterface.isValid()) {
QDBusReply<int> currentCharge = batteryInterface.call("charge");
QDBusReply<bool> charging = batteryInterface.call("isCharging");
if (currentCharge != -1) {
QString batteryText;
if (charging) {
if (currentCharge == 100) {
batteryText = QString::number(currentCharge) + "% battery on " + name + " (Full)";
} else {
batteryText = QString::number(currentCharge) + "% battery on " + name + " (Charging)";
}
} else {
batteryText = QString::number(currentCharge) + "% battery on " + name + " (Discharging)";
}
displayOutput.append(batteryText);
}
}
}
}
}
if (displayOutput.count() == 0) {
hasBat = false;
emit updateDisplay("");
} else {
hasBat = true;
emit updateDisplay(displayOutput.join(" · "));
}
}
bool UPowerDBus::hasBattery() {
return hasBat;
}
int UPowerDBus::currentBattery() {
return batLevel;
}
void UPowerDBus::checkUpower() {
QDBusInterface *i = new QDBusInterface("org.freedesktop.UPower", "/org/freedesktop/UPower", "org.freedesktop.UPower", QDBusConnection::systemBus(), this);
if (i->property("LidIsClosed").toBool()) { //Is the lid closed?
if (!this->isLidClosed) { //Has the value changed?
this->isLidClosed = true;
if (QApplication::screens().count() == 1) { //How many monitors do we have?
//If we only have one montior, suspend the PC.
EndSession(EndSessionWait::suspend);
}
}
} else {
if (this->isLidClosed) {
this->isLidClosed = false;
}
}
}
QDBusObjectPath UPowerDBus::defaultBattery() {
return batteryPath;
}
QDateTime UPowerDBus::batteryTimeRemaining() {
return timeRemain;
}
bool UPowerDBus::charging() {
return isCharging;
}
bool UPowerDBus::powerStretch() {
return isPowerStretchOn;
}
void UPowerDBus::setPowerStretch(bool on) {
if (isPowerStretchOn != on) {
isPowerStretchOn = on;
emit powerStretchChanged(on);
this->devicesChanged();
}
}
void UPowerDBus::ActionInvoked(uint id, QString action_key) {
if (id == batteryLowNotificationNumber) {
if (action_key == "power-stretch-on") {
setPowerStretch(true);
}
}
}