Files

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"