Merge branch 'broulik/no_ksmserver'

Differential Revision: https://phabricator.kde.org/D28617
This commit is contained in:
Kai Uwe Broulik 2020-04-09 14:52:53 +02:00
commit 59642a4b28
8 changed files with 55 additions and 56 deletions

View file

@ -276,7 +276,7 @@ void Application::createWorkspace()
// critical startup section where x errors cause kwin to abort.
// create workspace.
(void) new Workspace(m_originalSessionKey);
(void) new Workspace();
emit workspaceCreated();
}

1
main.h
View file

@ -238,7 +238,6 @@ protected:
}
protected:
QString m_originalSessionKey;
static int crashes;
private Q_SLOTS:

View file

@ -216,11 +216,6 @@ void ApplicationX11::performStartup()
Application::setX11ScreenNumber(QX11Info::appScreen());
}
// QSessionManager for some reason triggers a very early commitDataRequest
// and updates the key - before we create the workspace and load the session
// data -> store and pass to the workspace constructor
m_originalSessionKey = sessionKey();
owner.reset(new KWinSelectionOwner(Application::x11ScreenNumber()));
connect(owner.data(), &KSelectionOwner::failedToClaimOwnership, []{
fputs(i18n("kwin: unable to claim manager selection, another wm running? (try using --replace)\n").toLocal8Bit().constData(), stderr);
@ -420,6 +415,8 @@ int main(int argc, char * argv[])
qunsetenv("QT_DEVICE_PIXEL_RATIO");
qunsetenv("QT_SCALE_FACTOR");
QCoreApplication::setAttribute(Qt::AA_DisableHighDpiScaling);
// KSMServer talks to us directly on DBus.
QCoreApplication::setAttribute(Qt::AA_DisableSessionManager);
KWin::ApplicationX11 a(argc, argv);
a.setupTranslator();

View file

@ -10,6 +10,15 @@
-->
<arg name="state" type="u" direction="in" />
</method>
<method name="loadSession">
<arg name="name" type="s" direction="in" />
</method>
<method name="aboutToSaveSession">
<arg name="name" type="s" direction="in" />
</method>
<method name="finishSaveSession">
<arg name="name" type="s" direction="in" />
</method>
</interface>
</node>

61
sm.cpp
View file

@ -81,46 +81,16 @@ static NET::WindowType txtToWindowType(const char* txt)
return static_cast< NET::WindowType >(-2); // undefined
}
void Workspace::saveState(QSessionManager &sm)
{
bool sessionManagerIsKSMServer = QDBusConnection::sessionBus().interface()->isServiceRegistered("org.kde.ksmserver").value();
// If the session manager is ksmserver, save stacking
// order, active window, active desktop etc. in phase 1,
// as ksmserver assures no interaction will be done
// before the WM finishes phase 1. Saving in phase 2 is
// too late, as possible user interaction may change some things.
// Phase2 is still needed though (ICCCM 5.2)
KConfig *config = sessionConfig(sm.sessionId(), sm.sessionKey());
if (!sm.isPhase2()) {
KConfigGroup cg(config, "Session");
cg.writeEntry("AllowsInteraction", sm.allowsInteraction());
if (sessionManagerIsKSMServer) // save stacking order etc. before "save file?" etc. dialogs change it
storeSession(config, SMSavePhase0);
config->markAsClean(); // don't write Phase #1 data to disk
sm.release(); // Qt doesn't automatically release in this case (bug?)
sm.requestPhase2();
return;
}
storeSession(config, sessionManagerIsKSMServer ? SMSavePhase2 : SMSavePhase2Full);
config->sync();
// inform the smserver on how to clean-up after us
const QString localFilePath = QStandardPaths::writableLocation(QStandardPaths::GenericConfigLocation) + QLatin1Char('/') + config->name();
if (QFile::exists(localFilePath)) { // expectable for the sync
sm.setDiscardCommand(QStringList() << QStringLiteral("rm") << localFilePath);
}
}
// Workspace
/**
* Stores the current session in the config file
*
* @see loadSessionInfo
*/
void Workspace::storeSession(KConfig* config, SMSavePhase phase)
void Workspace::storeSession(const QString &sessionName, SMSavePhase phase)
{
qCDebug(KWIN_CORE) << "storing session" << sessionName << "in phase" << phase;
KConfig *config = sessionConfig(sessionName, QString());
KConfigGroup cg(config, "Session");
int count = 0;
int active_client = -1;
@ -160,6 +130,7 @@ void Workspace::storeSession(KConfig* config, SMSavePhase phase)
cg.writeEntry("active", session_active_client);
cg.writeEntry("desktop", VirtualDesktopManager::self()->current());
}
config->sync(); // it previously did some "revert to defaults" stuff for phase1 I think
}
void Workspace::storeClient(KConfigGroup &cg, int num, X11Client *c)
@ -235,12 +206,10 @@ void Workspace::storeSubSession(const QString &name, QSet<QByteArray> sessionIds
*
* @see storeSession
*/
void Workspace::loadSessionInfo(const QString &key)
void Workspace::loadSessionInfo(const QString &sessionName)
{
// NOTICE: qApp->sessionKey() is outdated when this gets invoked
// the key parameter is cached from the application constructor.
session.clear();
KConfigGroup cg(sessionConfig(qApp->sessionId(), key), "Session");
KConfigGroup cg(sessionConfig(sessionName, QString()), "Session");
addSessionInfo(cg);
}
@ -385,6 +354,7 @@ void SessionManager::setState(uint state)
}
}
// TODO should we rethink this now that we have dedicated start end end save methods?
void SessionManager::setState(SessionState state)
{
if (state == m_sessionState) {
@ -405,5 +375,20 @@ void SessionManager::setState(SessionState state)
emit stateChanged();
}
void SessionManager::loadSession(const QString &name)
{
emit loadSessionRequested(name);
}
void SessionManager::aboutToSaveSession(const QString &name)
{
emit prepareSessionSaveRequested(name);
}
void SessionManager::finishSaveSession(const QString &name)
{
emit finishSessionSaveRequested(name);
}
} // namespace

7
sm.h
View file

@ -45,8 +45,15 @@ public:
Q_SIGNALS:
void stateChanged();
void loadSessionRequested(const QString &name);
void prepareSessionSaveRequested(const QString &name);
void finishSessionSaveRequested(const QString &name);
public Q_SLOTS: // DBus API
void setState(uint state);
void loadSession(const QString &name);
void aboutToSaveSession(const QString &name);
void finishSaveSession(const QString &name);
private:
void setState(SessionState state);

View file

@ -102,7 +102,7 @@ void ColorMapper::update()
Workspace* Workspace::_self = nullptr;
Workspace::Workspace(const QString &sessionKey)
Workspace::Workspace()
: QObject(nullptr)
, m_compositor(nullptr)
// Unsorted
@ -153,10 +153,6 @@ Workspace::Workspace(const QString &sessionKey)
delayFocusTimer = nullptr;
if (!sessionKey.isEmpty())
loadSessionInfo(sessionKey);
connect(qApp, &QGuiApplication::saveStateRequest, this, &Workspace::saveState);
RuleBook::create(this)->load();
kwinApp()->createScreens();
@ -187,6 +183,15 @@ Workspace::Workspace(const QString &sessionKey)
decorationBridge->init();
connect(this, &Workspace::configChanged, decorationBridge, &Decoration::DecorationBridge::reconfigure);
connect(m_sessionManager, &SessionManager::loadSessionRequested, this, &Workspace::loadSessionInfo);
connect(m_sessionManager, &SessionManager::prepareSessionSaveRequested, this, [this](const QString &name) {
storeSession(name, SMSavePhase0);
});
connect(m_sessionManager, &SessionManager::finishSessionSaveRequested, this, [this](const QString &name) {
storeSession(name, SMSavePhase2);
});
new DBusInterface(this);
Outline::create(this);

View file

@ -69,7 +69,7 @@ class KWIN_EXPORT Workspace : public QObject
{
Q_OBJECT
public:
explicit Workspace(const QString &sessionKey = QString());
explicit Workspace();
~Workspace() override;
static Workspace* self() {
@ -309,7 +309,7 @@ public:
void updateOnAllDesktopsOfTransients(AbstractClient*);
void checkTransients(xcb_window_t w);
void storeSession(KConfig* config, SMSavePhase phase);
void storeSession(const QString &sessionName, SMSavePhase phase);
void storeClient(KConfigGroup &cg, int num, X11Client *c);
void storeSubSession(const QString &name, QSet<QByteArray> sessionIds);
void loadSubSessionInfo(const QString &name);
@ -494,9 +494,6 @@ private Q_SLOTS:
void slotDesktopCountChanged(uint previousCount, uint newCount);
void slotCurrentDesktopChanged(uint oldDesktop, uint newDesktop);
// session management
void saveState(QSessionManager &sm);
Q_SIGNALS:
/**
* Emitted after the Workspace has setup the complete initialization process.
@ -581,7 +578,7 @@ private:
AbstractClient* active_popup_client;
int m_initialDesktop;
void loadSessionInfo(const QString &key);
void loadSessionInfo(const QString &sessionName);
void addSessionInfo(KConfigGroup &cg);
QList<SessionInfo*> session;