You've already forked ukui-session-manager
mirror of
https://github.com/ukui/ukui-session-manager.git
synced 2026-03-09 09:22:17 -07:00
615 lines
20 KiB
C++
615 lines
20 KiB
C++
/*
|
|
* Copyright (C) 2019 Tianjin KYLIN Information Technology Co., Ltd.
|
|
* 2010-2016 LXQt team
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
* 02110-1301, USA.
|
|
**/
|
|
#include "modulemanager.h"
|
|
#include "ukuimodule.h"
|
|
#include "idlewatcher.h"
|
|
#include "ukuismserver.h"
|
|
#include "ukuisessiondebug.h"
|
|
|
|
#include <QCoreApplication>
|
|
#include "xdgautostart.h"
|
|
#include "xdgdesktopfile.h"
|
|
#include "xdgdirs.h"
|
|
#include <QFileInfo>
|
|
#include <QStringList>
|
|
#include <QSettings>
|
|
#include <QStandardPaths>
|
|
#include <QDebug>
|
|
#include <QGSettings/QGSettings>
|
|
// #include <QSoundEffect>
|
|
#include <QDBusInterface>
|
|
#include <QDir>
|
|
/* qt会将glib里的signals成员识别为宏,所以取消该宏
|
|
* 后面如果用到signals时,使用Q_SIGNALS代替即可
|
|
**/
|
|
#ifdef signals
|
|
#undef signals
|
|
#endif
|
|
|
|
#define SESSION_REQUIRED_COMPONENTS "org.ukui.session.required-components"
|
|
#define SESSION_REQUIRED_COMPONENTS_PATH "/org/ukui/desktop/session/required-components/"
|
|
|
|
extern UKUISMServer*& getGlobalServer();
|
|
|
|
//kylin-kmre-manager在ukui-session之前已经被启动,在执行恢复会话的时候,其还未注册到ukui-session中,所以需要对其进行特殊处理
|
|
std::map<QString, int> ModuleManager::m_startupMap = {{QString("/usr/bin/kylin-kmre-manager"), 1}};
|
|
|
|
void ModuleManager::playBootMusic(bool arg)
|
|
{
|
|
//set default value of whether boot-music is opened
|
|
bool play_music = true;
|
|
if (QGSettings::isSchemaInstalled("org.ukui.session")) {
|
|
QGSettings *gset = new QGSettings("org.ukui.session", "/org/ukui/desktop/session/", this);
|
|
if (gset == NULL) {
|
|
qDebug() << "QGSettings init error";
|
|
free(gset);
|
|
return;
|
|
}
|
|
QString xdg_session_type = qgetenv("XDG_SESSION_TYPE");
|
|
if (arg) {
|
|
play_music = gset->get("startup-music").toBool();
|
|
if (play_music) {
|
|
if (xdg_session_type == "wayland") {
|
|
QProcess::startDetached("paplay --volume=23456 /usr/share/ukui/ukui-session-manager/startup.wav");
|
|
} else {
|
|
QProcess::startDetached("aplay /usr/share/ukui/ukui-session-manager/startup.wav");
|
|
}
|
|
}
|
|
} else {
|
|
play_music = gset->get("weakup-music").toBool();
|
|
if (play_music) {
|
|
if (xdg_session_type == "wayland") {
|
|
QProcess::startDetached("paplay --volume=23456 /usr/share/ukui/ukui-session-manager/weakup.wav");
|
|
} else {
|
|
QProcess::startDetached("aplay /usr/share/ukui/ukui-session-manager/weakup.wav");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//睡眠转休眠判断
|
|
bool isSuspendToHibernate()
|
|
{
|
|
QString path = "/sys/power/wakeup_reason";
|
|
if (!QFile::exists(path)) {
|
|
return false;
|
|
}
|
|
QFile wakeup_reason_file(path);
|
|
wakeup_reason_file.open(QIODevice::ReadOnly | QIODevice::Text);
|
|
QString wakeupReason = wakeup_reason_file.readAll().simplified();
|
|
wakeup_reason_file.close();
|
|
if (QString::compare(wakeupReason,"0xaa") == 0) {
|
|
qDebug()<< "Suspend to Hibernate, wakeup_reason is "<<wakeupReason;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
ModuleManager::ModuleManager( QObject* parent) : QObject(parent)
|
|
{
|
|
QDBusConnection::systemBus().connect(QString("org.freedesktop.login1"),
|
|
QString("/org/freedesktop/login1"),
|
|
QString("org.freedesktop.login1.Manager"),
|
|
QString("PrepareForSleep"), this, SLOT(weakup(bool)));
|
|
constructStartupList();
|
|
}
|
|
|
|
void ModuleManager::weakup(bool arg)
|
|
{
|
|
if (arg) {
|
|
qDebug() << "准备执行睡眠休眠";
|
|
} else {
|
|
qDebug() << "从睡眠休眠中唤醒";
|
|
if (!isSuspendToHibernate()) {
|
|
qDebug() << "播放唤醒音乐";
|
|
playBootMusic(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
ModuleManager::~ModuleManager()
|
|
{
|
|
ModulesMapIterator i(mNameMap);
|
|
while (i.hasNext()) {
|
|
i.next();
|
|
|
|
auto p = i.value();
|
|
disconnect(p, SIGNAL(finished(int, QProcess::ExitStatus)), nullptr, nullptr);
|
|
|
|
delete p;
|
|
mNameMap[i.key()] = nullptr;
|
|
}
|
|
}
|
|
|
|
void ModuleManager::constructStartupList()
|
|
{
|
|
const QByteArray id(SESSION_REQUIRED_COMPONENTS);
|
|
QString window_manager;
|
|
QString panel;
|
|
QString file_manager;
|
|
QString wm_notfound;
|
|
if (QGSettings::isSchemaInstalled(id)) {
|
|
const QGSettings *gs = new QGSettings(SESSION_REQUIRED_COMPONENTS, SESSION_REQUIRED_COMPONENTS_PATH, this);
|
|
if (gs == NULL) {
|
|
qDebug() << "QGSettings init error";
|
|
return;
|
|
}
|
|
window_manager = gs->get("windowmanager").toString() + ".desktop";
|
|
panel = gs->get("panel").toString() + ".desktop";
|
|
file_manager = gs->get("filemanager").toString() + ".desktop";
|
|
wm_notfound = gs->get("windowmanager").toString();
|
|
} else {
|
|
//gsetting安装失败,或无法获取,设置默认值
|
|
qDebug() << "从gsettings 中或取值失败,设置默认值";
|
|
window_manager = "ukwm.desktop";
|
|
panel = "ukui-panel.desktop";
|
|
file_manager = "peony-qt-desktop.desktop";
|
|
}
|
|
|
|
QString xdg_session_type = qgetenv("XDG_SESSION_TYPE");
|
|
if (xdg_session_type == "wayland") {
|
|
isWayland = true;
|
|
}
|
|
|
|
QStringList desktop_paths;
|
|
desktop_paths << "/usr/share/applications";
|
|
desktop_paths << "/etc/xdg/autostart";
|
|
bool panel_found = false;
|
|
bool fm_found = false;
|
|
bool wm_found = false;
|
|
|
|
//const auto files = XdgAutoStart::desktopFileList(desktop_paths, false);
|
|
for (const QString &dirName : const_cast<const QStringList&>(desktop_paths)) {
|
|
QDir dir(dirName);
|
|
if (!dir.exists()) continue;
|
|
const QFileInfoList files = dir.entryInfoList(QStringList(QLatin1String("*.desktop")), QDir::Files | QDir::Readable);
|
|
for (const QFileInfo &fi : files) {
|
|
if (fi.fileName() == panel) {
|
|
mPanel.load(fi.absoluteFilePath());
|
|
panel_found = true;
|
|
qDebug() << "panel has been found";
|
|
}
|
|
if (fi.fileName() == file_manager) {
|
|
mFileManager.load(fi.absoluteFilePath());
|
|
fm_found = true;
|
|
qDebug() << "filemanager has been found";
|
|
}
|
|
if (fi.fileName() == window_manager) {
|
|
mWindowManager.load(fi.absoluteFilePath());
|
|
wm_found = true;
|
|
qDebug() << "windowmanager has been found";
|
|
}
|
|
|
|
if (fm_found && panel_found && wm_found)
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!panel_found || !fm_found || !wm_found) isDirectInstall = true;
|
|
if (isDirectInstall) {
|
|
wmFound = wm_found;
|
|
}
|
|
|
|
if (wm_found == false) {
|
|
QFileInfo check_ukwm("/usr/share/applications/ukwm.desktop");
|
|
QFileInfo check_ukuikwin("/usr/share/applications/ukui-kwin.desktop");
|
|
if (check_ukwm.exists()) {
|
|
window_manager = "ukwm.desktop";
|
|
mWindowManager.load("/usr/share/applications/ukwm.desktop");
|
|
} else if (check_ukuikwin.exists()) {
|
|
window_manager = "ukui-kwin.desktop";
|
|
mWindowManager.load("/usr/share/applications/ukui-kwin.desktop");
|
|
}
|
|
wm_found = true;
|
|
}
|
|
|
|
//配置文件所给的窗口管理器找不到.desktop文件时,将所给QString设为可执行命令,创建一个desktop文件赋给mWindowManager
|
|
// if (wm_found == false) {
|
|
// mWindowManager = XdgDesktopFile(XdgDesktopFile::ApplicationType,"window-manager", wm_notfound);
|
|
// qDebug() << "windowmanager has been created";
|
|
// }
|
|
|
|
QString desktop_phase = "X-UKUI-Autostart-Phase";
|
|
QString desktop_type = "Type";
|
|
//设置excludeHidden为true,判断所有desktop文件的Hidden值,若为true,则将其从自启列表中去掉
|
|
const XdgDesktopFileList all_file_list = XdgAutoStart::desktopFileList(true);
|
|
for (XdgDesktopFileList::const_iterator i = all_file_list.constBegin(); i != all_file_list.constEnd(); ++i) {
|
|
QString filename = QFileInfo(i->fileName()).fileName();
|
|
if (filename == panel || filename == file_manager || filename == window_manager) {
|
|
continue;
|
|
}
|
|
const XdgDesktopFile file = *i;
|
|
if (i->contains(desktop_phase)) {
|
|
QStringList s1 = file.value(desktop_phase).toString().split(QLatin1Char(';'));
|
|
if (s1.contains("Initialization")) {
|
|
mInitialization << file;
|
|
} else if (s1.contains("Desktop")) {
|
|
mDesktop << file;
|
|
} else if (s1.contains("Application")) {
|
|
mApplication << file;
|
|
}
|
|
} else if (i->contains(desktop_type)) {
|
|
QStringList s2 = file.value(desktop_type).toString().split(QLatin1Char(';'));
|
|
if (s2.contains("Application")) {
|
|
mApplication << file;
|
|
}
|
|
}
|
|
}
|
|
|
|
QStringList force_app_paths;
|
|
force_app_paths << "/usr/share/ukui/applications";
|
|
const XdgDesktopFileList force_file_list = XdgAutoStart::desktopFileList(force_app_paths, true);
|
|
for (XdgDesktopFileList::const_iterator i = force_file_list.constBegin(); i != force_file_list.constEnd(); ++i) {
|
|
qDebug() << (*i).fileName();
|
|
mForceApplication << *i;
|
|
}
|
|
}
|
|
|
|
/* Startup Phare:
|
|
* Initialization
|
|
* WindowManager
|
|
* Panel
|
|
* FileManager
|
|
* Desktop
|
|
* Application
|
|
*
|
|
*/
|
|
|
|
bool ModuleManager::startModuleTimer(QTimer *timer, int i)
|
|
{
|
|
timer->setSingleShot(true);
|
|
connect(timer, SIGNAL(timeout()), this, SLOT(timeup()));
|
|
timer->start(i * 1000);
|
|
return true;
|
|
}
|
|
|
|
void ModuleManager::startupfinished(const QString &appName, const QString &string)
|
|
{
|
|
qDebug() << "moudle :" + appName + " startup finished, and it want to say " + string;
|
|
if (appName == "ukui-settings-daemon") {
|
|
tusd->stop();
|
|
emit usdfinished();
|
|
return;
|
|
}
|
|
if (appName == "ukui-kwin") {
|
|
twm->stop();
|
|
isWMStarted = true;
|
|
emit wmfinished();
|
|
return;
|
|
}
|
|
if (appName == "ukui-panel") {
|
|
tpanel->stop();
|
|
isPanelStarted = true;
|
|
emit panelfinished();
|
|
return;
|
|
}
|
|
if (appName == "peony-qt-desktop") {
|
|
tdesktop->stop();
|
|
isDesktopStarted = true;
|
|
emit desktopfinished();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void ModuleManager::timeup()
|
|
{
|
|
QTimer *time_out = qobject_cast<QTimer*>(sender());
|
|
if (time_out == tusd) {
|
|
qDebug() << "usd超时";
|
|
emit usdfinished();
|
|
return;
|
|
}
|
|
if (time_out == twm) {
|
|
qDebug() <<"wm超时";
|
|
isWMStarted = true;
|
|
emit wmfinished();
|
|
return;
|
|
}
|
|
if (time_out == tpanel) {
|
|
qDebug() << "panel超时";
|
|
isPanelStarted = true;
|
|
emit panelfinished();
|
|
return;
|
|
}
|
|
if (time_out == tdesktop) {
|
|
qDebug() << "peony-qt-desktop超时";
|
|
isDesktopStarted = true;
|
|
emit desktopfinished();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void ModuleManager::startCompsite()
|
|
{
|
|
if (isWayland) return;
|
|
|
|
qDebug() << "Enter:: startCompsite";
|
|
if (!isPanelStarted || !isDesktopStarted || !isWMStarted) return;// || !isWMStarted
|
|
|
|
if (isCompsiteStarted) return;
|
|
isCompsiteStarted = true;
|
|
|
|
// start composite
|
|
QDBusInterface dbus("org.ukui.KWin", "/Compositor", "org.ukui.kwin.Compositing", QDBusConnection::sessionBus());
|
|
|
|
if (!dbus.isValid()) {
|
|
qWarning() << "dbusCall: QDBusInterface is invalid ; kwin do not exit!";
|
|
timerUpdate();
|
|
} else {
|
|
qDebug() << "Start composite";
|
|
dbus.call("resume");
|
|
|
|
timerUpdate();
|
|
}
|
|
}
|
|
|
|
void ModuleManager::startup()
|
|
{
|
|
//直接安装进入的流程
|
|
const QFile file_installer("/etc/xdg/autostart/kylin-os-installer.desktop");
|
|
if (file_installer.exists() && isDirectInstall) {
|
|
if (wmFound) {
|
|
isPanelStarted = true;
|
|
isDesktopStarted = true;
|
|
connect(this, &ModuleManager::wmfinished, [=](){ startCompsite(); });
|
|
|
|
startProcess(mWindowManager, false);
|
|
startModuleTimer(twm, 18);
|
|
} else {
|
|
timerUpdate();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (isWayland) {
|
|
startProcess("hwaudioservice", true);
|
|
}
|
|
|
|
if (!isWayland) {
|
|
connect(this, &ModuleManager::panelfinished, [=](){ startCompsite(); });
|
|
connect(this, &ModuleManager::wmfinished, [=](){ startCompsite(); });
|
|
connect(this, &ModuleManager::desktopfinished, [=](){ startCompsite(); });
|
|
|
|
dostartwm();
|
|
} else {
|
|
QTimer::singleShot(500, this, SLOT(timerUpdate()));
|
|
}
|
|
|
|
qDebug() << "Start Initialization app: ";
|
|
for (XdgDesktopFileList::const_iterator i = mInitialization.constBegin(); i != mInitialization.constEnd(); ++i) {
|
|
startProcess(*i, true);
|
|
}
|
|
startModuleTimer(tusd,5);
|
|
|
|
startProcess(mPanel, true);
|
|
startModuleTimer(tpanel,5);
|
|
|
|
startProcess(mFileManager, true);
|
|
startModuleTimer(tdesktop,5);
|
|
|
|
if (!isWayland) {
|
|
qDebug() << "Start desktop: ";
|
|
for (XdgDesktopFileList::const_iterator i = mDesktop.constBegin(); i != mDesktop.constEnd(); ++i) {
|
|
startProcess(*i, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ModuleManager::dostartwm()
|
|
{
|
|
if (mWindowManager.name() != "UKUI-KWin") {
|
|
qDebug() << "Start window manager: " << mWindowManager.name();
|
|
startProcess(mWindowManager, false);
|
|
startModuleTimer(twm, 18);
|
|
} else {
|
|
startProcess(mWindowManager, false);
|
|
isWMStarted = true;
|
|
}
|
|
}
|
|
|
|
void ModuleManager::timerUpdate()
|
|
{
|
|
playBootMusic(true);
|
|
QTimer::singleShot(500, this, [&](){
|
|
emit finished();
|
|
});
|
|
|
|
if (isWayland) {
|
|
qDebug() << "Start desktop: ";
|
|
for (XdgDesktopFileList::const_iterator i = mDesktop.constBegin(); i != mDesktop.constEnd(); ++i) {
|
|
startProcess(*i, true);
|
|
}
|
|
}
|
|
|
|
qDebug() << "Start application: ";
|
|
QFile file_nm("/etc/xdg/autostart/kylin-nm.desktop");
|
|
QFile file_sogou("/usr/bin/sogouImeService");
|
|
for (XdgDesktopFileList::const_iterator i = mApplication.constBegin(); i != mApplication.constEnd(); ++i) {
|
|
qDebug() << i->fileName();
|
|
if (i->fileName() == "/etc/xdg/autostart/nm-applet.desktop" && file_nm.exists()) {
|
|
qDebug() << "the kylin-nm exist so the nm-applet will not start";
|
|
continue;
|
|
}
|
|
if (i->fileName() == "/etc/xdg/autostart/fcitx-qimpanel-autostart.desktop" && file_sogou.exists()) {
|
|
qDebug() << "the sogouImeService exist so the fcitx-ui-qimpanel will not start";
|
|
continue;
|
|
}
|
|
startProcess(*i, false);
|
|
}
|
|
|
|
qDebug() << "Start force application: ";
|
|
for (XdgDesktopFileList::const_iterator i = mForceApplication.constBegin(); i != mForceApplication.constEnd(); ++i) {
|
|
startProcess(*i, true);
|
|
}
|
|
|
|
if (QGSettings::isSchemaInstalled("org.ukui.session")) {
|
|
QGSettings *gset = new QGSettings("org.ukui.session", "/org/ukui/desktop/session/", this);
|
|
QStringList keyList = gset->keys();
|
|
//keyList中的值是restoreSession而不是restore-session
|
|
if (keyList.contains("restoreSession")) {
|
|
QVariant res = gset->get("restore-session");
|
|
bool restoreSession = res.toBool();
|
|
if (restoreSession) {
|
|
//此处加上一个半秒的延迟可以更快的加速恢复会话,实验发现如果不加延迟,在此处直接开始恢复会话,反而会更慢
|
|
QTimer::singleShot(500, [](){
|
|
qDebug(UKUI_SESSION) << "began restore session";
|
|
getGlobalServer()->restoreSession();
|
|
});
|
|
}
|
|
} else {
|
|
qDebug() << "lack of required QGsettings";
|
|
}
|
|
}
|
|
}
|
|
|
|
void ModuleManager::startProcess(const XdgDesktopFile &file, bool required)
|
|
{
|
|
QStringList args = file.expandExecString();
|
|
if (args.isEmpty()) {
|
|
qWarning() << "Wrong desktop file: " << file.fileName();
|
|
return;
|
|
}
|
|
|
|
QString name = QFileInfo(file.fileName()).fileName();
|
|
if (!mNameMap.contains(name)) {
|
|
UkuiModule *proc = new UkuiModule(file, this);
|
|
connect(proc, &UkuiModule::moduleStateChanged, this, &ModuleManager::moduleStateChanged);
|
|
proc->start();
|
|
|
|
mNameMap[name] = proc;
|
|
|
|
if (required || autoRestart(file)) {
|
|
connect(proc, SIGNAL(finished(int, QProcess::ExitStatus)),
|
|
this, SLOT(restartModules(int, QProcess::ExitStatus)));
|
|
}
|
|
}
|
|
}
|
|
|
|
void ModuleManager::startProcess(const QString& name, bool required)
|
|
{
|
|
QString desktop_name = name + ".desktop";
|
|
QStringList desktop_paths;
|
|
desktop_paths << "/usr/share/applications";
|
|
|
|
const auto files = XdgAutoStart::desktopFileList(desktop_paths, false);
|
|
for (const XdgDesktopFile& file : files) {
|
|
if (QFileInfo(file.fileName()).fileName() == desktop_name) {
|
|
startProcess(file, required);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ModuleManager::stopProcess(const QString& name)
|
|
{
|
|
if (mNameMap.contains(name)) {
|
|
mNameMap[name]->terminate();
|
|
}
|
|
}
|
|
|
|
bool ModuleManager::nativeEventFilter(const QByteArray &eventType, void *message, long *result)
|
|
{
|
|
if (eventType != "xcb_generic_event_t") // We only want to handle XCB events
|
|
return false;
|
|
|
|
return false;
|
|
}
|
|
|
|
void ModuleManager::insertStartupList(QString &&str)
|
|
{
|
|
if (!str.isEmpty()) {
|
|
m_startupMap[std::forward<QString>(str)] = 1;
|
|
}
|
|
}
|
|
|
|
bool ModuleManager::isProgramStarted(QString &&str)
|
|
{
|
|
auto it = m_startupMap.find(std::forward<QString>(str));
|
|
|
|
if (it != m_startupMap.end()) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool ModuleManager::autoRestart(const XdgDesktopFile &file)
|
|
{
|
|
QString auto_restart = "X-UKUI-AutoRestart";
|
|
return file.value(auto_restart).toBool();
|
|
}
|
|
|
|
void ModuleManager::restartModules(int /*exitCode*/, QProcess::ExitStatus exitStatus)
|
|
{
|
|
if (getGlobalServer()->prepareForShutdown()) {
|
|
qDebug() << "Logout phase, don't Restart";
|
|
return;
|
|
}
|
|
|
|
UkuiModule *proc = qobject_cast<UkuiModule*>(sender());
|
|
|
|
if (nullptr == proc) {
|
|
qWarning() << "Got an invalid (null) module to restart, Ignoring it";
|
|
return;
|
|
}
|
|
|
|
//需要做出修改
|
|
if (proc->restartNum > 10) {
|
|
mNameMap.remove(proc->fileName);
|
|
disconnect(proc, SIGNAL(finished(int, QProcess::ExitStatus)), nullptr, nullptr);
|
|
proc->deleteLater();
|
|
return;
|
|
}
|
|
|
|
if (!proc->isTerminating()) {
|
|
//根据退出码来判断程序是否属于异常退出。
|
|
QString procName = proc->file.name();
|
|
if (proc->exitCode() == 0) {
|
|
qDebug() << "Process" << procName << "(" << proc << ") exited correctly. " << "With the exitcode = " << proc->exitCode() << ",exitStatus = " << exitStatus;
|
|
} else {
|
|
qDebug() << "Process" << procName << "(" << proc << ") has to be restarted. " << "With the exitcode = " << proc->exitCode() << ",exitStatus = " << exitStatus;
|
|
proc->start();
|
|
proc->restartNum++;
|
|
return;
|
|
}
|
|
}
|
|
mNameMap.remove(proc->fileName);
|
|
proc->deleteLater();
|
|
}
|
|
|
|
void ModuleManager::logout(bool doExit)
|
|
{
|
|
// /org/freedesktop/login1/session/self 和 /org/freedesktop/login1/session/auto
|
|
//有什么区别
|
|
QDBusInterface face("org.freedesktop.login1",\
|
|
"/org/freedesktop/login1/session/self",\
|
|
"org.freedesktop.login1.Session",\
|
|
QDBusConnection::systemBus());
|
|
|
|
if (doExit) {
|
|
face.call("Terminate");
|
|
exit(0);
|
|
}
|
|
}
|
|
|
|
#include "modulemanager.moc"
|