2022-06-22 23:02:35 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
SPDX-FileCopyrightText: 2022 Xaver Hugl <xaver.hugl@gmail.com>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
#include "placementtracker.h"
|
2022-08-31 09:15:36 +00:00
|
|
|
#include "core/output.h"
|
2022-06-22 23:02:35 +00:00
|
|
|
#include "window.h"
|
|
|
|
#include "workspace.h"
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
PlacementTracker::PlacementTracker(Workspace *workspace)
|
|
|
|
: m_workspace(workspace)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PlacementTracker::WindowData PlacementTracker::dataForWindow(Window *window) const
|
|
|
|
{
|
|
|
|
return WindowData{
|
2022-06-27 14:19:29 +00:00
|
|
|
.outputUuid = window->moveResizeOutput()->uuid(),
|
2022-06-22 23:02:35 +00:00
|
|
|
.geometry = window->moveResizeGeometry(),
|
|
|
|
.maximize = window->requestedMaximizeMode(),
|
|
|
|
.quickTile = window->quickTileMode(),
|
|
|
|
.geometryRestore = window->geometryRestore(),
|
|
|
|
.fullscreen = window->isFullScreen(),
|
|
|
|
.fullscreenGeometryRestore = window->fullscreenGeometryRestore(),
|
|
|
|
.interactiveMoveResizeCount = window->interactiveMoveResizeCount(),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::add(Window *window)
|
|
|
|
{
|
|
|
|
if (window->isUnmanaged() || window->isAppletPopup() || window->isSpecialWindow()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
connect(window, &Window::frameGeometryChanged, this, &PlacementTracker::saveGeometry);
|
2023-02-18 11:22:34 +00:00
|
|
|
connect(window, &Window::maximizedChanged, this, &PlacementTracker::saveMaximize);
|
2022-06-22 23:02:35 +00:00
|
|
|
connect(window, &Window::quickTileModeChanged, this, &PlacementTracker::saveQuickTile);
|
|
|
|
connect(window, &Window::fullScreenChanged, this, &PlacementTracker::saveFullscreen);
|
|
|
|
connect(window, &Window::clientFinishUserMovedResized, this, &PlacementTracker::saveInteractionCounter);
|
|
|
|
WindowData data = dataForWindow(window);
|
|
|
|
m_data[m_currentKey][window] = data;
|
|
|
|
m_savedWindows.push_back(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::remove(Window *window)
|
|
|
|
{
|
|
|
|
if (m_savedWindows.contains(window)) {
|
|
|
|
disconnect(window, &Window::frameGeometryChanged, this, &PlacementTracker::saveGeometry);
|
2023-02-18 11:22:34 +00:00
|
|
|
disconnect(window, &Window::maximizedChanged, this, &PlacementTracker::saveMaximize);
|
2022-06-22 23:02:35 +00:00
|
|
|
disconnect(window, &Window::quickTileModeChanged, this, &PlacementTracker::saveQuickTile);
|
|
|
|
disconnect(window, &Window::fullScreenChanged, this, &PlacementTracker::saveFullscreen);
|
|
|
|
disconnect(window, &Window::clientFinishUserMovedResized, this, &PlacementTracker::saveInteractionCounter);
|
|
|
|
for (auto &dataMap : m_data) {
|
|
|
|
dataMap.remove(window);
|
|
|
|
}
|
|
|
|
m_savedWindows.removeOne(window);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::restore(const QString &key)
|
|
|
|
{
|
|
|
|
if (key == m_currentKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto &dataMap = m_data[key];
|
|
|
|
auto &oldDataMap = m_data[m_currentKey];
|
|
|
|
const auto outputs = m_workspace->outputs();
|
|
|
|
|
|
|
|
inhibit();
|
|
|
|
for (const auto window : std::as_const(m_savedWindows)) {
|
2022-10-28 15:43:45 +00:00
|
|
|
const auto it = dataMap.find(window);
|
|
|
|
if (it != dataMap.end()) {
|
|
|
|
const WindowData &newData = it.value();
|
|
|
|
|
2022-06-22 23:02:35 +00:00
|
|
|
// don't touch windows where the user intentionally changed their state
|
2022-10-28 15:43:45 +00:00
|
|
|
bool restore = window->interactiveMoveResizeCount() == newData.interactiveMoveResizeCount
|
|
|
|
&& window->requestedMaximizeMode() == newData.maximize
|
|
|
|
&& window->quickTileMode() == newData.quickTile
|
|
|
|
&& window->isFullScreen() == newData.fullscreen;
|
|
|
|
if (!restore) {
|
|
|
|
// the logic above can have false negatives if PlacementTracker changed the window state
|
|
|
|
// to prevent that, also restore if the window still has the same state from that
|
|
|
|
if (const auto it = m_lastRestoreData.find(window); it != m_lastRestoreData.end()) {
|
|
|
|
restore = window->interactiveMoveResizeCount() == it->interactiveMoveResizeCount
|
|
|
|
&& window->requestedMaximizeMode() == it->maximize
|
|
|
|
&& window->quickTileMode() == it->quickTile
|
|
|
|
&& window->isFullScreen() == it->fullscreen
|
|
|
|
&& window->moveResizeOutput()->uuid() == it->outputUuid;
|
|
|
|
}
|
|
|
|
}
|
2022-06-22 23:02:35 +00:00
|
|
|
if (!restore) {
|
|
|
|
// restore anyways if the output the window was on got removed
|
|
|
|
if (const auto oldData = oldDataMap.find(window); oldData != oldDataMap.end()) {
|
2022-10-28 15:43:45 +00:00
|
|
|
restore = std::none_of(outputs.begin(), outputs.end(), [&oldData](const auto output) {
|
2022-06-22 23:02:35 +00:00
|
|
|
return output->uuid() == oldData->outputUuid;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-10-28 15:43:45 +00:00
|
|
|
if (restore) {
|
2022-06-22 23:02:35 +00:00
|
|
|
window->setFullScreen(false);
|
|
|
|
window->setQuickTileMode(QuickTileFlag::None, true);
|
|
|
|
window->setMaximize(false, false);
|
|
|
|
if (newData.quickTile || newData.maximize) {
|
|
|
|
window->moveResize(newData.geometryRestore);
|
|
|
|
window->setQuickTileMode(newData.quickTile, true);
|
|
|
|
window->setMaximize(newData.maximize & MaximizeMode::MaximizeVertical, newData.maximize & MaximizeMode::MaximizeHorizontal);
|
|
|
|
}
|
|
|
|
if (newData.fullscreen) {
|
|
|
|
window->moveResize(newData.fullscreenGeometryRestore);
|
|
|
|
window->setFullScreen(newData.fullscreen);
|
|
|
|
}
|
|
|
|
if (newData.quickTile || newData.maximize || newData.fullscreen) {
|
|
|
|
// restore geometry isn't necessarily on the output the window was, so explicitly restore it
|
|
|
|
const auto outputIt = std::find_if(outputs.begin(), outputs.end(), [&newData](const auto output) {
|
|
|
|
return output->uuid() == newData.outputUuid;
|
|
|
|
});
|
|
|
|
if (outputIt != outputs.end()) {
|
|
|
|
window->sendToOutput(*outputIt);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
window->moveResize(newData.geometry);
|
|
|
|
}
|
2022-10-28 15:43:45 +00:00
|
|
|
m_lastRestoreData[window] = dataForWindow(window);
|
2022-06-22 23:02:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// ensure data in current map is always up to date
|
|
|
|
dataMap[window] = dataForWindow(window);
|
|
|
|
}
|
|
|
|
uninhibit();
|
|
|
|
m_currentKey = key;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::init(const QString &key)
|
|
|
|
{
|
|
|
|
m_currentKey = key;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::saveGeometry(Window *window)
|
|
|
|
{
|
|
|
|
if (m_inhibitCount == 0) {
|
|
|
|
auto &data = m_data[m_currentKey][window];
|
|
|
|
data.geometry = window->moveResizeGeometry();
|
2022-06-27 14:19:29 +00:00
|
|
|
data.outputUuid = window->moveResizeOutput()->uuid();
|
2022-06-22 23:02:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::saveInteractionCounter(Window *window)
|
|
|
|
{
|
|
|
|
if (m_inhibitCount == 0) {
|
|
|
|
m_data[m_currentKey][window].interactiveMoveResizeCount = window->interactiveMoveResizeCount();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::saveMaximize(Window *window, MaximizeMode mode)
|
|
|
|
{
|
|
|
|
if (m_inhibitCount == 0) {
|
|
|
|
auto &data = m_data[m_currentKey][window];
|
|
|
|
data.maximize = mode;
|
|
|
|
data.geometryRestore = window->geometryRestore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::saveQuickTile()
|
|
|
|
{
|
|
|
|
Window *window = qobject_cast<Window *>(QObject::sender());
|
|
|
|
Q_ASSERT(window);
|
|
|
|
if (m_inhibitCount == 0) {
|
|
|
|
auto &data = m_data[m_currentKey][window];
|
|
|
|
data.quickTile = window->quickTileMode();
|
|
|
|
data.geometryRestore = window->geometryRestore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::saveFullscreen()
|
|
|
|
{
|
|
|
|
Window *window = qobject_cast<Window *>(QObject::sender());
|
|
|
|
Q_ASSERT(window);
|
|
|
|
if (m_inhibitCount == 0) {
|
|
|
|
auto &data = m_data[m_currentKey][window];
|
|
|
|
data.fullscreen = window->isFullScreen();
|
|
|
|
data.fullscreenGeometryRestore = window->fullscreenGeometryRestore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::inhibit()
|
|
|
|
{
|
|
|
|
m_inhibitCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PlacementTracker::uninhibit()
|
|
|
|
{
|
|
|
|
Q_ASSERT(m_inhibitCount > 0);
|
|
|
|
m_inhibitCount--;
|
|
|
|
}
|
|
|
|
}
|