// Copyright (C) 2011 ~ 2018 Deepin Technology Co., Ltd. // SPDX-FileCopyrightText: 2018 - 2023 UnionTech Software Technology Co., Ltd. // // SPDX-License-Identifier: LGPL-3.0-or-later #include "shutdownplugin.h" #include "dbus/dbusaccount.h" #include "../frame/util/utils.h" #include "../widgets/tipswidget.h" #include "./dbus/dbuspowermanager.h" #include #include #include #include #include #include #define PLUGIN_STATE_KEY "enable" #define GSETTING_SHOW_SUSPEND "showSuspend" #define GSETTING_SHOW_HIBERNATE "showHibernate" #define GSETTING_SHOW_SHUTDOWN "showShutdown" #define GSETTING_SHOW_LOCK "showLock" DCORE_USE_NAMESPACE DGUI_USE_NAMESPACE using namespace Dock; ShutdownPlugin::ShutdownPlugin(QObject *parent) : QObject(parent) , m_pluginLoaded(false) , m_shutdownWidget(nullptr) , m_tipsLabel(new TipsWidget) , m_powerManagerInter(new DBusPowerManager("org.deepin.dde.PowerManager1", "/org/deepin/dde/PowerManager1", QDBusConnection::systemBus(), this)) , m_gsettings(Utils::ModuleSettingsPtr("shutdown", QByteArray(), this)) , m_sessionShellGsettings(Utils::SettingsPtr("com.deepin.dde.session-shell", "/com/deepin/dde/session-shell/", this)) { m_tipsLabel->setVisible(false); m_tipsLabel->setAccessibleName("shutdown"); } const QString ShutdownPlugin::pluginName() const { return "shutdown"; } const QString ShutdownPlugin::pluginDisplayName() const { return tr("Power"); } QWidget *ShutdownPlugin::itemWidget(const QString &itemKey) { Q_UNUSED(itemKey); return m_shutdownWidget.data(); } QWidget *ShutdownPlugin::itemTipsWidget(const QString &itemKey) { Q_UNUSED(itemKey); // reset text every time to avoid size of LabelWidget not change after // font size be changed in ControlCenter m_tipsLabel->setText(tr("Power")); return m_tipsLabel.data(); } void ShutdownPlugin::init(PluginProxyInterface *proxyInter) { m_proxyInter = proxyInter; // transfer config QSettings settings("deepin", "dde-dock-shutdown"); if (QFile::exists(settings.fileName())) { QFile::remove(settings.fileName()); } if (!pluginIsDisable()) { loadPlugin(); } } void ShutdownPlugin::pluginStateSwitched() { m_proxyInter->saveValue(this, PLUGIN_STATE_KEY, !m_proxyInter->getValue(this, PLUGIN_STATE_KEY, true).toBool()); } bool ShutdownPlugin::pluginIsDisable() { return !m_proxyInter->getValue(this, PLUGIN_STATE_KEY, true).toBool(); } const QString ShutdownPlugin::itemCommand(const QString &itemKey) { Q_UNUSED(itemKey); return QString("dbus-send --print-reply --dest=org.deepin.dde.ShutdownFront1 /org/deepin/dde/ShutdownFront1 org.deepin.dde.ShutdownFront1.Show"); } const QString ShutdownPlugin::itemContextMenu(const QString &itemKey) { Q_UNUSED(itemKey); QList items; items.reserve(6); QMap shutdown; if (!m_gsettings || (m_gsettings->keys().contains(GSETTING_SHOW_SHUTDOWN) && m_gsettings->get(GSETTING_SHOW_SHUTDOWN).toBool())) { shutdown["itemId"] = "Shutdown"; shutdown["itemText"] = tr("Shut down"); shutdown["isActive"] = true; items.push_back(shutdown); } QMap reboot; reboot["itemId"] = "Restart"; reboot["itemText"] = tr("Reboot"); reboot["isActive"] = true; items.push_back(reboot); #ifndef DISABLE_POWER_OPTIONS QProcessEnvironment enviromentVar = QProcessEnvironment::systemEnvironment(); bool can_sleep = enviromentVar.contains("POWER_CAN_SLEEP") ? QVariant(enviromentVar.value("POWER_CAN_SLEEP")).toBool() : valueByQSettings("Power", "sleep", true) && m_powerManagerInter->CanSuspend(); ; if (can_sleep) { QMap suspend; if (!m_gsettings || (m_gsettings->keys().contains(GSETTING_SHOW_SUSPEND) && m_gsettings->get(GSETTING_SHOW_SUSPEND).toBool())) { suspend["itemId"] = "Suspend"; suspend["itemText"] = tr("Suspend"); suspend["isActive"] = true; items.push_back(suspend); } } bool can_hibernate = enviromentVar.contains("POWER_CAN_HIBERNATE") ? QVariant(enviromentVar.value("POWER_CAN_HIBERNATE")).toBool() : checkSwap() && m_powerManagerInter->CanHibernate(); if (can_hibernate) { QMap hibernate; if (!m_gsettings || (m_gsettings->keys().contains(GSETTING_SHOW_HIBERNATE) && m_gsettings->get(GSETTING_SHOW_HIBERNATE).toBool())) { hibernate["itemId"] = "Hibernate"; hibernate["itemText"] = tr("Hibernate"); hibernate["isActive"] = true; items.push_back(hibernate); } } #endif QMap lock; if (!m_gsettings || (m_gsettings->keys().contains(GSETTING_SHOW_LOCK) && m_gsettings->get(GSETTING_SHOW_LOCK).toBool())) { lock["itemId"] = "Lock"; lock["itemText"] = tr("Lock"); lock["isActive"] = true; items.push_back(lock); } QMap logout; logout["itemId"] = "Logout"; logout["itemText"] = tr("Log out"); logout["isActive"] = true; items.push_back(logout); if (!QFile::exists(ICBC_CONF_FILE)) { // com.deepin.dde.session-shell切换用户配置项 enum SwitchUserConfig { AlwaysShow = 0, OnDemand, Disabled } switchUserConfig = OnDemand; if (m_sessionShellGsettings && m_sessionShellGsettings->keys().contains("switchuser")) { switchUserConfig = SwitchUserConfig(m_sessionShellGsettings->get("switchuser").toInt()); } // 和登录锁屏界面的逻辑保持一致 if (AlwaysShow == switchUserConfig || (OnDemand == switchUserConfig && (DBusAccount().userList().count() > 1 || DSysInfo::uosType() == DSysInfo::UosType::UosServer))) { QMap switchUser; switchUser["itemId"] = "SwitchUser"; switchUser["itemText"] = tr("Switch account"); switchUser["isActive"] = true; items.push_back(switchUser); } #ifndef DISABLE_POWER_OPTIONS QMap power; power["itemId"] = "power"; power["itemText"] = tr("Power settings"); power["isActive"] = true; items.push_back(power); #endif } QMap menu; menu["items"] = items; menu["checkableMenu"] = false; menu["singleCheck"] = false; return QJsonDocument::fromVariant(menu).toJson(); } void ShutdownPlugin::invokedMenuItem(const QString &itemKey, const QString &menuId, const bool checked) { Q_UNUSED(itemKey) Q_UNUSED(checked) // 使得下面逻辑代码延迟200ms执行,保证线程不阻塞 QTime dieTime = QTime::currentTime().addMSecs(200); while (QTime::currentTime() < dieTime) QCoreApplication::processEvents(QEventLoop::AllEvents, 200); if (menuId == "power") { DDBusSender() .service("org.deepin.dde.ControlCenter1") .interface("org.deepin.dde.ControlCenter1") .path("/org/deepin/dde/ControlCenter1") .method(QString("ShowPage")) .arg(QString("power")) .call(); } else if (menuId == "Lock") { if (QFile::exists(ICBC_CONF_FILE)) { QDBusMessage send = QDBusMessage::createMethodCall("org.deepin.dde.LockFront1", "/org/deepin/dde/LockFront1", "org.deepin.dde.LockFront1", "SwitchTTYAndShow"); QDBusConnection conn = QDBusConnection::connectToBus("unix:path=/run/user/1000/bus", "unix:path=/run/user/1000/bus"); QDBusMessage reply = conn.call(send); #ifdef QT_DEBUG qInfo() << "----------" << reply; #endif } else { DDBusSender() .service("org.deepin.dde.LockFront1") .interface("org.deepin.dde.LockFront1") .path("/org/deepin/dde/LockFront1") .method(QString("Show")) .call(); } } else DDBusSender() .service("org.deepin.dde.ShutdownFront1") .interface("org.deepin.dde.ShutdownFront1") .path("/org/deepin/dde/ShutdownFront1") .method(QString(menuId)) .call(); } void ShutdownPlugin::displayModeChanged(const Dock::DisplayMode displayMode) { Q_UNUSED(displayMode); if (!pluginIsDisable()) { m_shutdownWidget->update(); } } int ShutdownPlugin::itemSortKey(const QString &itemKey) { const QString key = QString("pos_%1_%2").arg(itemKey).arg(Dock::Efficient); return m_proxyInter->getValue(this, key, 5).toInt(); } void ShutdownPlugin::setSortKey(const QString &itemKey, const int order) { const QString key = QString("pos_%1_%2").arg(itemKey).arg(Dock::Efficient); m_proxyInter->saveValue(this, key, order); } QIcon ShutdownPlugin::icon(const DockPart &dockPart, DGuiApplicationHelper::ColorType themeType) { if (dockPart == DockPart::DCCSetting) { QString iconFile(":/icons/resources/icons/dcc_shutdown.svg"); auto pixmap = ImageUtil::loadSvg(iconFile, QSize(18, 18)); if (themeType == DGuiApplicationHelper::ColorType::LightType) return pixmap; QPainter pa(&pixmap); pa.setCompositionMode(QPainter::CompositionMode_SourceIn); pa.fillRect(pixmap.rect(), Qt::white); return pixmap; } QString iconName = "system-shutdown"; if (themeType == DGuiApplicationHelper::LightType) iconName.append(PLUGIN_MIN_ICON_NAME); const auto ratio = qApp->devicePixelRatio(); QPixmap pixmap; pixmap = QIcon::fromTheme(iconName, QIcon::fromTheme(":/icons/resources/icons/system-shutdown.svg")).pixmap(QSize(PLUGIN_ICON_MAX_SIZE, PLUGIN_ICON_MAX_SIZE) * ratio); pixmap.setDevicePixelRatio(ratio); return pixmap; } PluginFlags ShutdownPlugin::flags() const { return PluginFlag::Type_System | PluginFlag::Attribute_CanSetting; } void ShutdownPlugin::loadPlugin() { if (m_pluginLoaded) { qDebug() << "shutdown plugin has been loaded! return"; return; } m_pluginLoaded = true; m_shutdownWidget.reset(new ShutdownWidget); m_proxyInter->itemAdded(this, pluginName()); displayModeChanged(displayMode()); } std::pair ShutdownPlugin::checkIsPartitionType(const QStringList &list) { std::pair result{ false, -1 }; if (list.length() != 5) { return result; } const QString type{ list[1] }; const QString size{ list[2] }; result.first = type == "partition"; result.second = size.toLongLong() * 1024.0f; return result; } qint64 ShutdownPlugin::get_power_image_size() { qint64 size{ 0 }; QFile file("/sys/power/image_size"); if (file.open(QIODevice::Text | QIODevice::ReadOnly)) { size = file.readAll().trimmed().toLongLong(); file.close(); } else{ qWarning() << "open /sys/power/image_size failed! please check permission!!!"; } return size; } bool ShutdownPlugin::checkSwap() { if (!valueByQSettings("Power", "hibernate", true)) return false; bool hasSwap = false; QFile file("/proc/swaps"); if (file.open(QIODevice::Text | QIODevice::ReadOnly)) { const QString &body = file.readAll(); QTextStream stream(body.toUtf8()); while (!stream.atEnd()) { const std::pair result = checkIsPartitionType(stream.readLine().simplified().split(" ", Qt::SkipEmptyParts)); qint64 image_size{ get_power_image_size() }; if (result.first) { hasSwap = image_size < result.second; } if (hasSwap) { break; } } file.close(); } else { qWarning() << "open /proc/swaps failed! please check permission!!!"; } return hasSwap; }