1124 lines
36 KiB
C++
1124 lines
36 KiB
C++
/*
|
|
This file is part of KDDockWidgets.
|
|
|
|
SPDX-FileCopyrightText: 2019-2022 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
|
|
Author: Sérgio Martins <sergio.martins@kdab.com>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only
|
|
|
|
Contact KDAB at <info@kdab.com> for commercial licensing options.
|
|
*/
|
|
|
|
/**
|
|
* @file
|
|
* @brief Class to save and restore dock widget layouts.
|
|
*
|
|
* @author Sérgio Martins \<sergio.martins@kdab.com\>
|
|
*/
|
|
|
|
#include "LayoutSaver.h"
|
|
#include "Config.h"
|
|
#include "ViewFactory.h"
|
|
#include "nlohmann_qt_helpers.h"
|
|
|
|
#include "private/multisplitter/Item_p.h"
|
|
#include "private/LayoutSaver_p.h"
|
|
#include "private/DockRegistry_p.h"
|
|
#include "private/Logging_p.h"
|
|
#include "private/Position_p.h"
|
|
#include "private/Utils_p.h"
|
|
|
|
#include "controllers/Layout.h"
|
|
#include "controllers/Frame.h"
|
|
#include "controllers/FloatingWindow.h"
|
|
#include "controllers/DockWidget.h"
|
|
#include "controllers/DockWidget_p.h"
|
|
#include "controllers/MainWindow.h"
|
|
|
|
|
|
#include <qmath.h>
|
|
#include <QDebug>
|
|
#include <QFile>
|
|
|
|
/**
|
|
* Some implementation details:
|
|
*
|
|
* Restoring is done in two phases. From the JSON, we construct an intermediate representation,
|
|
* which doesn't have any GUI types. Finally we then construct the GUI from the intermediate
|
|
* representation.
|
|
*
|
|
* JSON <-> Intermediate rep (bunch non-gui structs) <-> GUI classes
|
|
*
|
|
* This is in contrast to many other dock widget frameworks which just do:
|
|
* serialized <-> GUI
|
|
*
|
|
* The advantage of having the intermediate structs is that we can do validations on them and if
|
|
* we find some corruption we don't even start messing with the GUI.
|
|
*
|
|
* See the LayoutSaver::* structs in LayoutSaver_p.h, those are the intermediate structs.
|
|
* They have methods to convert to/from JSON.
|
|
* All other gui classes have methods to convert to/from these structs. For example
|
|
* FloatingWindow::serialize()/deserialize()
|
|
*/
|
|
using namespace KDDockWidgets;
|
|
using namespace KDDockWidgets::Controllers;
|
|
|
|
QHash<QString, LayoutSaver::DockWidget::Ptr> LayoutSaver::DockWidget::s_dockWidgets;
|
|
LayoutSaver::Layout *LayoutSaver::Layout::s_currentLayoutBeingRestored = nullptr;
|
|
|
|
|
|
inline InternalRestoreOptions internalRestoreOptions(RestoreOptions options)
|
|
{
|
|
if (options == RestoreOption_None) {
|
|
return InternalRestoreOption::None;
|
|
} else if (options == RestoreOption_RelativeToMainWindow) {
|
|
return InternalRestoreOptions(InternalRestoreOption::SkipMainWindowGeometry)
|
|
| InternalRestoreOption::RelativeFloatingWindowGeometry;
|
|
} else {
|
|
qWarning() << Q_FUNC_INFO << "Unknown options" << options;
|
|
return {};
|
|
}
|
|
}
|
|
|
|
bool LayoutSaver::Private::s_restoreInProgress = false;
|
|
|
|
namespace KDDockWidgets {
|
|
template<typename Type>
|
|
void to_json(nlohmann::json &json, const typename Type::List &list)
|
|
{
|
|
for (const auto &l : list) {
|
|
json.push_back(l);
|
|
}
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::Frame &f)
|
|
{
|
|
json["id"] = f.id;
|
|
json["isNull"] = f.isNull;
|
|
json["objectName"] = f.objectName;
|
|
json["geometry"] = f.geometry;
|
|
json["options"] = f.options;
|
|
json["currentTabIndex"] = f.currentTabIndex;
|
|
json["mainWindowUniqueName"] = f.mainWindowUniqueName;
|
|
json["dockWidgets"] = dockWidgetNames(f.dockWidgets);
|
|
}
|
|
void from_json(const nlohmann::json &json, LayoutSaver::Frame &f)
|
|
{
|
|
f.id = json.value("id", QString());
|
|
f.isNull = json.value("isNull", true);
|
|
f.objectName = json.value("objectName", QString());
|
|
f.geometry = json.value("geometry", QRect());
|
|
f.options = json.value("options", QFlags<FrameOption>::Int {});
|
|
f.currentTabIndex = json.value("currentTabIndex", 0);
|
|
f.mainWindowUniqueName = json.value("mainWindowUniqueName", QString());
|
|
|
|
auto it = json.find("dockWidgets");
|
|
if (it == json.end())
|
|
return;
|
|
|
|
auto &dockWidgets = *it;
|
|
f.dockWidgets.reserve(( int )dockWidgets.size());
|
|
for (const auto &d : dockWidgets) {
|
|
LayoutSaver::DockWidget::Ptr dw = LayoutSaver::DockWidget::dockWidgetForName(d.get<QString>());
|
|
f.dockWidgets.push_back(dw);
|
|
}
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::MultiSplitter &s)
|
|
{
|
|
json["layout"] = s.layout;
|
|
auto &frames = json["frames"];
|
|
for (const auto &frame : qAsConst(s.frames)) {
|
|
frames[frame.id.toStdString()] = frame;
|
|
}
|
|
}
|
|
void from_json(const nlohmann::json &json, LayoutSaver::MultiSplitter &s)
|
|
{
|
|
s.frames.clear();
|
|
s.layout = json.value("layout", QVariantMap());
|
|
auto it = json.find("frames");
|
|
if (it == json.end())
|
|
return;
|
|
if (it->is_null())
|
|
return;
|
|
|
|
auto &frms = *it;
|
|
if (!frms.is_object())
|
|
qWarning() << Q_FUNC_INFO << "Unexpected not object";
|
|
;
|
|
|
|
for (const auto &kv : frms.items()) {
|
|
QString key = QString::fromStdString(kv.key());
|
|
auto frame = kv.value().get<LayoutSaver::Frame>();
|
|
s.frames.insert(key, frame);
|
|
}
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::MainWindow &mw)
|
|
{
|
|
json["options"] = int(mw.options);
|
|
json["multiSplitterLayout"] = mw.multiSplitterLayout;
|
|
json["uniqueName"] = mw.uniqueName;
|
|
json["geometry"] = mw.geometry;
|
|
json["normalGeometry"] = mw.normalGeometry;
|
|
json["screenIndex"] = mw.screenIndex;
|
|
json["screenSize"] = mw.screenSize;
|
|
json["isVisible"] = mw.isVisible;
|
|
json["affinities"] = mw.affinities;
|
|
json["windowState"] = mw.windowState;
|
|
|
|
for (SideBarLocation loc : { SideBarLocation::North, SideBarLocation::East, SideBarLocation::West, SideBarLocation::South }) {
|
|
const QStringList dockWidgets = mw.dockWidgetsPerSideBar.value(loc);
|
|
if (!dockWidgets.isEmpty()) {
|
|
std::string key = std::string("sidebar-") + std::to_string(( int )loc);
|
|
json[key] = dockWidgets;
|
|
}
|
|
}
|
|
}
|
|
|
|
void from_json(const nlohmann::json &json, LayoutSaver::MainWindow &mw)
|
|
{
|
|
mw.options = static_cast<decltype(mw.options)>(json.value("options", 0));
|
|
mw.multiSplitterLayout = json.value("multiSplitterLayout", LayoutSaver::MultiSplitter());
|
|
mw.uniqueName = json.value("uniqueName", QString());
|
|
mw.geometry = json.value("geometry", QRect());
|
|
mw.normalGeometry = json.value("normalGeometry", QRect());
|
|
mw.screenIndex = json.value("screenIndex", 0);
|
|
mw.screenSize = json.value("screenSize", QSize(800, 600));
|
|
mw.isVisible = json.value("isVisible", false);
|
|
mw.affinities = json.value("affinities", QStringList());
|
|
mw.windowState = ( Qt::WindowState )json.value("windowState", 0);
|
|
|
|
// Compatibility hack. Old json format had a single "affinityName" instead of an "affinities" list:
|
|
if (json.find("affinityName") != json.end()) {
|
|
QString affinityName = json["affinityName"].get<QString>();
|
|
if (!mw.affinities.contains(affinityName)) {
|
|
mw.affinities.push_back(affinityName);
|
|
}
|
|
}
|
|
|
|
for (SideBarLocation loc : { SideBarLocation::North, SideBarLocation::East, SideBarLocation::West, SideBarLocation::South }) {
|
|
std::string key = std::string("sidebar-") + std::to_string(( int )loc);
|
|
auto it = json.find(key);
|
|
if (it == json.end())
|
|
continue;
|
|
auto &val = *it;
|
|
if (val.is_array() && !val.empty()) {
|
|
mw.dockWidgetsPerSideBar[loc] = val.get<QStringList>();
|
|
}
|
|
}
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::FloatingWindow &window)
|
|
{
|
|
json["multiSplitterLayout"] = window.multiSplitterLayout;
|
|
json["parentIndex"] = window.parentIndex;
|
|
json["geometry"] = window.geometry;
|
|
json["normalGeometry"] = window.normalGeometry;
|
|
json["screenIndex"] = window.screenIndex;
|
|
json["screenSize"] = window.screenSize;
|
|
json["isVisible"] = window.isVisible;
|
|
json["windowState"] = window.windowState;
|
|
|
|
if (!window.affinities.isEmpty()) {
|
|
json["affinities"] = window.affinities;
|
|
}
|
|
}
|
|
|
|
void from_json(const nlohmann::json &json, LayoutSaver::FloatingWindow &window)
|
|
{
|
|
window.multiSplitterLayout = json.value("multiSplitterLayout", LayoutSaver::MultiSplitter());
|
|
window.parentIndex = json.value("parentIndex", -1);
|
|
window.geometry = json.value("geometry", QRect());
|
|
window.normalGeometry = json.value("normalGeometry", QRect());
|
|
window.screenIndex = json.value("screenIndex", 0);
|
|
window.screenSize = json.value("screenSize", QSize(800, 600));
|
|
window.isVisible = json.value("isVisible", false);
|
|
window.windowState = ( Qt::WindowState )json.value("windowState", 0);
|
|
window.affinities = json.value("affinities", QStringList());
|
|
|
|
// Compatibility hack. Old json format had a single "affinityName" instead of an "affinities" list:
|
|
const QString affinityName = json.value("affinityName", QString());
|
|
if (!affinityName.isEmpty() && !window.affinities.contains(affinityName)) {
|
|
window.affinities.push_back(affinityName);
|
|
}
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::ScreenInfo &screenInfo)
|
|
{
|
|
json["index"] = screenInfo.index;
|
|
json["geometry"] = screenInfo.geometry;
|
|
json["name"] = screenInfo.name;
|
|
json["devicePixelRatio"] = screenInfo.devicePixelRatio;
|
|
}
|
|
void from_json(const nlohmann::json &j, LayoutSaver::ScreenInfo &screenInfo)
|
|
{
|
|
screenInfo.index = j.value("index", 0);
|
|
screenInfo.geometry = j.value("geometry", QRect());
|
|
screenInfo.name = j.value("name", QString());
|
|
screenInfo.devicePixelRatio = j.value("devicePixelRatio", 1.0);
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::Placeholder &placeHolder)
|
|
{
|
|
json["isFloatingWindow"] = placeHolder.isFloatingWindow;
|
|
json["itemIndex"] = placeHolder.itemIndex;
|
|
if (placeHolder.isFloatingWindow)
|
|
json["indexOfFloatingWindow"] = placeHolder.indexOfFloatingWindow;
|
|
else
|
|
json["mainWindowUniqueName"] = placeHolder.mainWindowUniqueName;
|
|
}
|
|
|
|
void from_json(const nlohmann::json &json, LayoutSaver::Placeholder &placeHolder)
|
|
{
|
|
placeHolder.isFloatingWindow = json.value("isFloatingWindow", false);
|
|
placeHolder.itemIndex = json.value("itemIndex", 0);
|
|
placeHolder.indexOfFloatingWindow = json.value("indexOfFloatingWindow", -1);
|
|
placeHolder.mainWindowUniqueName = json.value("mainWindowUniqueName", QString());
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::Position &pos)
|
|
{
|
|
json["lastFloatingGeometry"] = pos.lastFloatingGeometry;
|
|
json["tabIndex"] = pos.tabIndex;
|
|
json["wasFloating"] = pos.wasFloating;
|
|
json["placeholders"] = pos.placeholders;
|
|
}
|
|
|
|
void from_json(const nlohmann::json &json, LayoutSaver::Position &pos)
|
|
{
|
|
pos.lastFloatingGeometry = json.value("lastFloatingGeometry", QRect());
|
|
pos.tabIndex = json.value("tabIndex", 0);
|
|
pos.wasFloating = json.value("wasFloating", false);
|
|
pos.placeholders = json.value("placeholders", LayoutSaver::Placeholder::List());
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const LayoutSaver::DockWidget &dw)
|
|
{
|
|
if (!dw.affinities.isEmpty())
|
|
json["affinities"] = dw.affinities;
|
|
json["uniqueName"] = dw.uniqueName;
|
|
json["lastPosition"] = dw.lastPosition;
|
|
}
|
|
void from_json(const nlohmann::json &json, LayoutSaver::DockWidget &dw)
|
|
{
|
|
auto it = json.find("affinities");
|
|
if (it != json.end())
|
|
dw.affinities = it->get<QStringList>();
|
|
|
|
dw.uniqueName = json.value("uniqueName", QString());
|
|
if (dw.uniqueName.isEmpty())
|
|
qWarning() << Q_FUNC_INFO << "Unexpected no uniqueName for dockWidget";
|
|
|
|
dw.lastPosition = json.value("lastPosition", LayoutSaver::Position());
|
|
}
|
|
|
|
void to_json(nlohmann::json &json, const typename LayoutSaver::DockWidget::List &list)
|
|
{
|
|
for (const auto &mw : list) {
|
|
json.push_back(*mw);
|
|
}
|
|
}
|
|
void from_json(const nlohmann::json &json, typename LayoutSaver::DockWidget::List &list)
|
|
{
|
|
list.clear();
|
|
for (const auto &v : json) {
|
|
auto it = v.find("uniqueName");
|
|
if (it == v.end()) {
|
|
qWarning() << Q_FUNC_INFO << "Unexpected no uniqueName";
|
|
continue;
|
|
}
|
|
QString uniqueName = it->get<QString>();
|
|
auto dw = LayoutSaver::DockWidget::dockWidgetForName(uniqueName);
|
|
from_json(v, *dw);
|
|
list.push_back(dw);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
LayoutSaver::LayoutSaver(RestoreOptions options)
|
|
: d(new Private(options))
|
|
{
|
|
}
|
|
|
|
LayoutSaver::~LayoutSaver()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
bool LayoutSaver::saveToFile(const QString &jsonFilename)
|
|
{
|
|
const QByteArray data = serializeLayout();
|
|
|
|
QFile f(jsonFilename);
|
|
if (!f.open(QIODevice::WriteOnly)) {
|
|
qWarning() << Q_FUNC_INFO << "Failed to open" << jsonFilename << f.errorString();
|
|
return false;
|
|
}
|
|
|
|
f.write(data);
|
|
return true;
|
|
}
|
|
|
|
bool LayoutSaver::restoreFromFile(const QString &jsonFilename)
|
|
{
|
|
QFile f(jsonFilename);
|
|
if (!f.open(QIODevice::ReadOnly)) {
|
|
qWarning() << Q_FUNC_INFO << "Failed to open" << jsonFilename << f.errorString();
|
|
return false;
|
|
}
|
|
|
|
const QByteArray data = f.readAll();
|
|
const bool result = restoreLayout(data);
|
|
|
|
return result;
|
|
}
|
|
|
|
QByteArray LayoutSaver::serializeLayout() const
|
|
{
|
|
if (!d->m_dockRegistry->isSane()) {
|
|
qWarning() << Q_FUNC_INFO << "Refusing to serialize this layout. Check previous warnings.";
|
|
return {};
|
|
}
|
|
|
|
LayoutSaver::Layout layout;
|
|
|
|
// Just a simplification. One less type of windows to handle.
|
|
d->m_dockRegistry->ensureAllFloatingWidgetsAreMorphed();
|
|
|
|
const auto mainWindows = d->m_dockRegistry->mainwindows();
|
|
layout.mainWindows.reserve(mainWindows.size());
|
|
for (auto mainWindow : mainWindows) {
|
|
if (d->matchesAffinity(mainWindow->affinities()))
|
|
layout.mainWindows.push_back(mainWindow->serialize());
|
|
}
|
|
|
|
const QVector<Controllers::FloatingWindow *> floatingWindows = d->m_dockRegistry->floatingWindows();
|
|
layout.floatingWindows.reserve(floatingWindows.size());
|
|
for (Controllers::FloatingWindow *floatingWindow : floatingWindows) {
|
|
if (d->matchesAffinity(floatingWindow->affinities()))
|
|
layout.floatingWindows.push_back(floatingWindow->serialize());
|
|
}
|
|
|
|
// Closed dock widgets also have interesting things to save, like geometry and placeholder info
|
|
const Controllers::DockWidget::List closedDockWidgets = d->m_dockRegistry->closedDockwidgets();
|
|
layout.closedDockWidgets.reserve(closedDockWidgets.size());
|
|
for (Controllers::DockWidget *dockWidget : closedDockWidgets) {
|
|
if (d->matchesAffinity(dockWidget->affinities()))
|
|
layout.closedDockWidgets.push_back(dockWidget->d->serialize());
|
|
}
|
|
|
|
// Save the placeholder info. We do it last, as we also restore it last, since we need all items to be created
|
|
// before restoring the placeholders
|
|
|
|
const Controllers::DockWidget::List dockWidgets = d->m_dockRegistry->dockwidgets();
|
|
layout.allDockWidgets.reserve(dockWidgets.size());
|
|
for (Controllers::DockWidget *dockWidget : dockWidgets) {
|
|
if (d->matchesAffinity(dockWidget->affinities())) {
|
|
auto dw = dockWidget->d->serialize();
|
|
dw->lastPosition = dockWidget->d->lastPosition()->serialize();
|
|
layout.allDockWidgets.push_back(dw);
|
|
}
|
|
}
|
|
|
|
return layout.toJson();
|
|
}
|
|
|
|
bool LayoutSaver::restoreLayout(const QByteArray &data)
|
|
{
|
|
d->clearRestoredProperty();
|
|
if (data.isEmpty())
|
|
return true;
|
|
|
|
struct FrameCleanup
|
|
{
|
|
FrameCleanup(LayoutSaver *saver)
|
|
: m_saver(saver)
|
|
{
|
|
}
|
|
|
|
~FrameCleanup()
|
|
{
|
|
m_saver->d->deleteEmptyFrames();
|
|
}
|
|
|
|
LayoutSaver *const m_saver;
|
|
};
|
|
|
|
FrameCleanup cleanup(this);
|
|
LayoutSaver::Layout layout;
|
|
if (!layout.fromJson(data)) {
|
|
qWarning() << Q_FUNC_INFO << "Failed to parse json data";
|
|
return false;
|
|
}
|
|
|
|
if (!layout.isValid()) {
|
|
return false;
|
|
}
|
|
|
|
layout.scaleSizes(d->m_restoreOptions);
|
|
|
|
d->floatWidgetsWhichSkipRestore(layout.mainWindowNames());
|
|
d->floatUnknownWidgets(layout);
|
|
|
|
Private::RAIIIsRestoring isRestoring;
|
|
|
|
// Hide all dockwidgets and unparent them from any layout before starting restore
|
|
// We only close the stuff that the loaded JSON knows about. Unknown widgets might be newer.
|
|
|
|
d->m_dockRegistry->clear(d->m_dockRegistry->dockWidgets(layout.dockWidgetsToClose()),
|
|
d->m_dockRegistry->mainWindows(layout.mainWindowNames()),
|
|
d->m_affinityNames);
|
|
|
|
// 1. Restore main windows
|
|
for (const LayoutSaver::MainWindow &mw : qAsConst(layout.mainWindows)) {
|
|
auto mainWindow = d->m_dockRegistry->mainWindowByName(mw.uniqueName);
|
|
if (!mainWindow) {
|
|
if (auto mwFunc = Config::self().mainWindowFactoryFunc()) {
|
|
mainWindow = mwFunc(mw.uniqueName);
|
|
} else {
|
|
qWarning() << "Failed to restore layout create MainWindow with name" << mw.uniqueName << "first";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!d->matchesAffinity(mainWindow->affinities()))
|
|
continue;
|
|
|
|
if (!(d->m_restoreOptions & InternalRestoreOption::SkipMainWindowGeometry)) {
|
|
Window::Ptr window = mainWindow->view()->window();
|
|
d->deserializeWindowGeometry(mw, window);
|
|
if (mw.windowState != Qt::WindowNoState) {
|
|
if (auto w = mainWindow->view()->window()) {
|
|
w->setWindowState(mw.windowState);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!mainWindow->deserialize(mw))
|
|
return false;
|
|
}
|
|
|
|
// 2. Restore FloatingWindows
|
|
for (LayoutSaver::FloatingWindow &fw : layout.floatingWindows) {
|
|
if (!d->matchesAffinity(fw.affinities) || fw.skipsRestore())
|
|
continue;
|
|
|
|
auto parent = fw.parentIndex == -1 ? nullptr
|
|
: DockRegistry::self()->mainwindows().at(fw.parentIndex);
|
|
|
|
auto floatingWindow = new Controllers::FloatingWindow({}, parent);
|
|
fw.floatingWindowInstance = floatingWindow;
|
|
d->deserializeWindowGeometry(fw, floatingWindow->view()->window());
|
|
if (!floatingWindow->deserialize(fw)) {
|
|
qWarning() << Q_FUNC_INFO << "Failed to deserialize floating window";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// 3. Restore closed dock widgets. They remain closed but acquire geometry and placeholder properties
|
|
for (const auto &dw : qAsConst(layout.closedDockWidgets)) {
|
|
if (d->matchesAffinity(dw->affinities)) {
|
|
Controllers::DockWidget::deserialize(dw);
|
|
}
|
|
}
|
|
|
|
// 4. Restore the placeholder info, now that the Items have been created
|
|
for (const auto &dw : qAsConst(layout.allDockWidgets)) {
|
|
if (!d->matchesAffinity(dw->affinities))
|
|
continue;
|
|
|
|
if (Controllers::DockWidget *dockWidget =
|
|
d->m_dockRegistry->dockByName(dw->uniqueName, DockRegistry::DockByNameFlag::ConsultRemapping)) {
|
|
dockWidget->d->lastPosition()->deserialize(dw->lastPosition);
|
|
} else {
|
|
qWarning() << Q_FUNC_INFO << "Couldn't find dock widget" << dw->uniqueName;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void LayoutSaver::setAffinityNames(const QStringList &affinityNames)
|
|
{
|
|
d->m_affinityNames = affinityNames;
|
|
if (affinityNames.contains(QString())) {
|
|
// Any window with empty affinity will also be subject to save/restore
|
|
d->m_affinityNames << QString();
|
|
}
|
|
}
|
|
|
|
LayoutSaver::Private *LayoutSaver::dptr() const
|
|
{
|
|
return d;
|
|
}
|
|
|
|
Controllers::DockWidget::List LayoutSaver::restoredDockWidgets() const
|
|
{
|
|
const Controllers::DockWidget::List &allDockWidgets = DockRegistry::self()->dockwidgets();
|
|
Controllers::DockWidget::List result;
|
|
result.reserve(allDockWidgets.size());
|
|
for (Controllers::DockWidget *dw : allDockWidgets) {
|
|
if (dw->property("kddockwidget_was_restored").toBool())
|
|
result.push_back(dw);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void LayoutSaver::Private::clearRestoredProperty()
|
|
{
|
|
const Controllers::DockWidget::List &allDockWidgets = DockRegistry::self()->dockwidgets();
|
|
for (Controllers::DockWidget *dw : allDockWidgets) {
|
|
dw->setProperty("kddockwidget_was_restored", QVariant());
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
void LayoutSaver::Private::deserializeWindowGeometry(const T &saved, Window::Ptr window)
|
|
{
|
|
// Not simply calling QWidget::setGeometry() here.
|
|
// For QtQuick we need to modify the QWindow's geometry.
|
|
|
|
QRect geometry = saved.geometry;
|
|
if (!isNormalWindowState(saved.windowState)) {
|
|
// The window will be maximized. We first set its geometry to normal
|
|
// Later it's maximized and will remember this value
|
|
geometry = saved.normalGeometry;
|
|
}
|
|
|
|
Controllers::FloatingWindow::ensureRectIsOnScreen(geometry);
|
|
|
|
window->setGeometry(geometry);
|
|
window->setVisible(saved.isVisible);
|
|
}
|
|
|
|
LayoutSaver::Private::Private(RestoreOptions options)
|
|
: m_dockRegistry(DockRegistry::self())
|
|
, m_restoreOptions(internalRestoreOptions(options))
|
|
{
|
|
}
|
|
|
|
bool LayoutSaver::Private::matchesAffinity(const QStringList &affinities) const
|
|
{
|
|
return m_affinityNames.isEmpty() || affinities.isEmpty()
|
|
|| DockRegistry::self()->affinitiesMatch(m_affinityNames, affinities);
|
|
}
|
|
|
|
void LayoutSaver::Private::floatWidgetsWhichSkipRestore(const QStringList &mainWindowNames)
|
|
{
|
|
// Widgets with the DockWidget::LayoutSaverOption::Skip flag skip restore completely.
|
|
// If they were visible before they need to remain visible now.
|
|
// If they were previously docked we need to float them, as the main window they were on will
|
|
// be loading a new layout.
|
|
|
|
for (auto mw : DockRegistry::self()->mainWindows(mainWindowNames)) {
|
|
const Controllers::DockWidget::List docks = mw->layout()->dockWidgets();
|
|
for (auto dw : docks) {
|
|
if (dw->skipsRestore()) {
|
|
dw->setFloating(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void LayoutSaver::Private::floatUnknownWidgets(const LayoutSaver::Layout &layout)
|
|
{
|
|
// An old *.json layout file might have not know about existing dock widgets
|
|
// When restoring such a file, we need to float any visible dock widgets which it doesn't know about
|
|
// so we can restore the MainWindow layout properly
|
|
|
|
for (auto mw : DockRegistry::self()->mainWindows(layout.mainWindowNames())) {
|
|
const Controllers::DockWidget::List docks = mw->layout()->dockWidgets();
|
|
for (Controllers::DockWidget *dw : docks) {
|
|
if (!layout.containsDockWidget(dw->uniqueName())) {
|
|
dw->setFloating(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void LayoutSaver::Private::deleteEmptyFrames()
|
|
{
|
|
// After a restore it can happen that some DockWidgets didn't exist, so weren't restored.
|
|
// Delete their frame now.
|
|
|
|
for (auto frame : m_dockRegistry->frames()) {
|
|
if (!frame->beingDeletedLater() && frame->isEmpty() && !frame->isCentralFrame())
|
|
delete frame;
|
|
}
|
|
}
|
|
|
|
std::unique_ptr<QSettings> LayoutSaver::Private::settings() const
|
|
{
|
|
auto settings = std::unique_ptr<QSettings>(new QSettings(qApp->organizationName(),
|
|
qApp->applicationName()));
|
|
settings->beginGroup(QStringLiteral("KDDockWidgets::LayoutSaver"));
|
|
|
|
return settings;
|
|
}
|
|
|
|
bool LayoutSaver::restoreInProgress()
|
|
{
|
|
return Private::s_restoreInProgress;
|
|
}
|
|
|
|
bool LayoutSaver::Layout::isValid() const
|
|
{
|
|
if (serializationVersion != KDDOCKWIDGETS_SERIALIZATION_VERSION) {
|
|
qWarning() << Q_FUNC_INFO << "Serialization format is too old"
|
|
<< serializationVersion << "current=" << KDDOCKWIDGETS_SERIALIZATION_VERSION;
|
|
return false;
|
|
}
|
|
|
|
for (auto &m : mainWindows) {
|
|
if (!m.isValid())
|
|
return false;
|
|
}
|
|
|
|
for (auto &m : floatingWindows) {
|
|
if (!m.isValid())
|
|
return false;
|
|
}
|
|
|
|
for (auto &m : allDockWidgets) {
|
|
if (!m->isValid())
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
namespace KDDockWidgets {
|
|
void to_json(nlohmann::json &j, const LayoutSaver::Layout &layout)
|
|
{
|
|
j["serializationVersion"] = layout.serializationVersion;
|
|
j["mainWindows"] = layout.mainWindows;
|
|
j["floatingWindows"] = layout.floatingWindows;
|
|
j["closedDockWidgets"] = ::dockWidgetNames(layout.closedDockWidgets);
|
|
j["allDockWidgets"] = layout.allDockWidgets;
|
|
j["screenInfo"] = layout.screenInfo;
|
|
}
|
|
|
|
void from_json(const nlohmann::json &j, LayoutSaver::Layout &layout)
|
|
{
|
|
layout.serializationVersion = j["serializationVersion"].get<int>();
|
|
layout.mainWindows = j["mainWindows"].get<LayoutSaver::MainWindow::List>();
|
|
|
|
layout.allDockWidgets = j["allDockWidgets"].get<LayoutSaver::DockWidget::List>();
|
|
|
|
layout.closedDockWidgets.clear();
|
|
const QVariantList closedDockWidgetsV = j["closedDockWidgets"].get<QVariantList>();
|
|
for (const QVariant &v : closedDockWidgetsV) {
|
|
layout.closedDockWidgets.push_back(LayoutSaver::DockWidget::dockWidgetForName(v.toString()));
|
|
}
|
|
|
|
layout.floatingWindows = j["floatingWindows"].get<LayoutSaver::FloatingWindow::List>();
|
|
layout.screenInfo = j["screenInfo"].get<LayoutSaver::ScreenInfo::List>();
|
|
}
|
|
}
|
|
|
|
QByteArray LayoutSaver::Layout::toJson() const
|
|
{
|
|
nlohmann::json json = *this;
|
|
return QByteArray::fromStdString(json.dump(4));
|
|
}
|
|
|
|
bool LayoutSaver::Layout::fromJson(const QByteArray &jsonData)
|
|
{
|
|
nlohmann::json json = nlohmann::json::parse(jsonData, nullptr, /*DisableExceptions=*/true);
|
|
if (json.is_discarded()) {
|
|
return false;
|
|
}
|
|
from_json(json, *this);
|
|
return true;
|
|
}
|
|
|
|
void LayoutSaver::Layout::scaleSizes(InternalRestoreOptions options)
|
|
{
|
|
if (mainWindows.isEmpty())
|
|
return;
|
|
|
|
const bool skipsMainWindowGeometry = options & InternalRestoreOption::SkipMainWindowGeometry;
|
|
if (!skipsMainWindowGeometry) {
|
|
// No scaling to do. All windows will be restored with the exact size specified in the
|
|
// saved JSON layouts.
|
|
return;
|
|
}
|
|
|
|
// We won't restore MainWindow's geometry, we use whatever the user has now, meaning
|
|
// we need to scale all dock widgets inside the layout, as the layout might not have
|
|
// the same size as specified in the saved JSON layout
|
|
for (auto &mw : mainWindows)
|
|
mw.scaleSizes();
|
|
|
|
|
|
// MainWindow has a different size than the one in JSON, so we also restore FloatingWindows
|
|
// relatively to the user set new MainWindow size
|
|
const bool useRelativeSizesForFloatingWidgets =
|
|
options & InternalRestoreOption::RelativeFloatingWindowGeometry;
|
|
|
|
if (useRelativeSizesForFloatingWidgets) {
|
|
for (auto &fw : floatingWindows) {
|
|
LayoutSaver::MainWindow mw = mainWindowForIndex(fw.parentIndex);
|
|
if (mw.scalingInfo.isValid())
|
|
fw.scaleSizes(mw.scalingInfo);
|
|
}
|
|
}
|
|
|
|
const ScalingInfo firstScalingInfo = mainWindows.constFirst().scalingInfo;
|
|
if (firstScalingInfo.isValid()) {
|
|
for (auto &dw : allDockWidgets) {
|
|
// TODO: Determine the best main window. This only interesting for closed dock
|
|
// widget geometry which was previously floating. But they still have some other
|
|
// main window as parent.
|
|
dw->scaleSizes(firstScalingInfo);
|
|
}
|
|
}
|
|
}
|
|
|
|
LayoutSaver::MainWindow LayoutSaver::Layout::mainWindowForIndex(int index) const
|
|
{
|
|
if (index < 0 || index >= mainWindows.size())
|
|
return {};
|
|
|
|
return mainWindows.at(index);
|
|
}
|
|
|
|
LayoutSaver::FloatingWindow LayoutSaver::Layout::floatingWindowForIndex(int index) const
|
|
{
|
|
if (index < 0 || index >= floatingWindows.size())
|
|
return {};
|
|
|
|
return floatingWindows.at(index);
|
|
}
|
|
|
|
QStringList LayoutSaver::Layout::mainWindowNames() const
|
|
{
|
|
QStringList names;
|
|
names.reserve(mainWindows.size());
|
|
for (const auto &mw : mainWindows) {
|
|
names << mw.uniqueName;
|
|
}
|
|
|
|
return names;
|
|
}
|
|
|
|
QStringList LayoutSaver::Layout::dockWidgetNames() const
|
|
{
|
|
QStringList names;
|
|
names.reserve(allDockWidgets.size());
|
|
for (const auto &dw : allDockWidgets) {
|
|
names << dw->uniqueName;
|
|
}
|
|
|
|
return names;
|
|
}
|
|
|
|
QStringList LayoutSaver::Layout::dockWidgetsToClose() const
|
|
{
|
|
// Before restoring a layout we close all dock widgets, unless they're a floating window with the DontCloseBeforeRestore flag
|
|
|
|
QStringList names;
|
|
names.reserve(allDockWidgets.size());
|
|
auto registry = DockRegistry::self();
|
|
for (const auto &dw : allDockWidgets) {
|
|
if (Controllers::DockWidget *dockWidget = registry->dockByName(dw->uniqueName)) {
|
|
|
|
bool doClose = true;
|
|
|
|
if (dockWidget->skipsRestore()) {
|
|
if (auto fw = dockWidget->floatingWindow()) {
|
|
if (fw->allDockWidgetsHave(Controllers::DockWidget::LayoutSaverOption::Skip)) {
|
|
// All dock widgets in this floating window skips float, so we can honour it for all.
|
|
doClose = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (doClose)
|
|
names << dw->uniqueName;
|
|
}
|
|
}
|
|
|
|
return names;
|
|
}
|
|
|
|
bool LayoutSaver::Layout::containsDockWidget(const QString &uniqueName) const
|
|
{
|
|
return std::find_if(allDockWidgets.cbegin(), allDockWidgets.cend(), [uniqueName](const std::shared_ptr<LayoutSaver::DockWidget> &dock) {
|
|
return dock->uniqueName == uniqueName;
|
|
})
|
|
!= allDockWidgets.cend();
|
|
}
|
|
|
|
bool LayoutSaver::Frame::isValid() const
|
|
{
|
|
if (isNull)
|
|
return true;
|
|
|
|
if (!geometry.isValid()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid geometry";
|
|
return false;
|
|
}
|
|
|
|
if (id.isEmpty()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid id";
|
|
return false;
|
|
}
|
|
|
|
if (!dockWidgets.isEmpty()) {
|
|
if (currentTabIndex >= dockWidgets.size() || currentTabIndex < 0) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid tab index" << currentTabIndex << dockWidgets.size();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (auto &dw : dockWidgets) {
|
|
if (!dw->isValid())
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool LayoutSaver::Frame::hasSingleDockWidget() const
|
|
{
|
|
return dockWidgets.size() == 1;
|
|
}
|
|
|
|
bool LayoutSaver::Frame::skipsRestore() const
|
|
{
|
|
return std::all_of(dockWidgets.cbegin(), dockWidgets.cend(), [](LayoutSaver::DockWidget::Ptr dw) {
|
|
return dw->skipsRestore();
|
|
});
|
|
}
|
|
|
|
LayoutSaver::DockWidget::Ptr LayoutSaver::Frame::singleDockWidget() const
|
|
{
|
|
if (!hasSingleDockWidget())
|
|
return {};
|
|
|
|
return dockWidgets.first();
|
|
}
|
|
|
|
bool LayoutSaver::DockWidget::isValid() const
|
|
{
|
|
return !uniqueName.isEmpty();
|
|
}
|
|
|
|
void LayoutSaver::DockWidget::scaleSizes(const ScalingInfo &scalingInfo)
|
|
{
|
|
lastPosition.scaleSizes(scalingInfo);
|
|
}
|
|
|
|
bool LayoutSaver::DockWidget::skipsRestore() const
|
|
{
|
|
if (Controllers::DockWidget *dw = DockRegistry::self()->dockByName(uniqueName))
|
|
return dw->skipsRestore();
|
|
|
|
return false;
|
|
}
|
|
|
|
bool LayoutSaver::FloatingWindow::isValid() const
|
|
{
|
|
if (!multiSplitterLayout.isValid())
|
|
return false;
|
|
|
|
if (!geometry.isValid()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid geometry";
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool LayoutSaver::FloatingWindow::hasSingleDockWidget() const
|
|
{
|
|
return multiSplitterLayout.hasSingleDockWidget();
|
|
}
|
|
|
|
LayoutSaver::DockWidget::Ptr LayoutSaver::FloatingWindow::singleDockWidget() const
|
|
{
|
|
return multiSplitterLayout.singleDockWidget();
|
|
}
|
|
|
|
bool LayoutSaver::FloatingWindow::skipsRestore() const
|
|
{
|
|
return multiSplitterLayout.skipsRestore();
|
|
}
|
|
|
|
void LayoutSaver::FloatingWindow::scaleSizes(const ScalingInfo &scalingInfo)
|
|
{
|
|
scalingInfo.applyFactorsTo(/*by-ref*/ geometry);
|
|
}
|
|
|
|
bool LayoutSaver::MainWindow::isValid() const
|
|
{
|
|
if (!multiSplitterLayout.isValid())
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void LayoutSaver::MainWindow::scaleSizes()
|
|
{
|
|
if (scalingInfo.isValid()) {
|
|
// Doesn't happen, it's called only once
|
|
Q_ASSERT(false);
|
|
return;
|
|
}
|
|
|
|
scalingInfo = ScalingInfo(uniqueName, geometry, screenIndex);
|
|
}
|
|
|
|
bool LayoutSaver::MultiSplitter::isValid() const
|
|
{
|
|
if (layout.isEmpty())
|
|
return false;
|
|
|
|
/*if (!size.isValid()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid size";
|
|
return false;
|
|
}*/
|
|
|
|
return true;
|
|
}
|
|
|
|
bool LayoutSaver::MultiSplitter::hasSingleDockWidget() const
|
|
{
|
|
return frames.size() == 1 && frames.cbegin()->hasSingleDockWidget();
|
|
}
|
|
|
|
LayoutSaver::DockWidget::Ptr LayoutSaver::MultiSplitter::singleDockWidget() const
|
|
{
|
|
if (!hasSingleDockWidget())
|
|
return {};
|
|
|
|
return frames.cbegin()->singleDockWidget();
|
|
}
|
|
|
|
bool LayoutSaver::MultiSplitter::skipsRestore() const
|
|
{
|
|
return std::all_of(frames.cbegin(), frames.cend(), [](const LayoutSaver::Frame &frame) {
|
|
return frame.skipsRestore();
|
|
});
|
|
}
|
|
|
|
void LayoutSaver::Position::scaleSizes(const ScalingInfo &scalingInfo)
|
|
{
|
|
scalingInfo.applyFactorsTo(/*by-ref*/ lastFloatingGeometry);
|
|
}
|
|
|
|
QVariantMap LayoutSaver::Placeholder::toVariantMap() const
|
|
{
|
|
QVariantMap map;
|
|
map.insert(QStringLiteral("isFloatingWindow"), isFloatingWindow);
|
|
map.insert(QStringLiteral("itemIndex"), itemIndex);
|
|
|
|
if (isFloatingWindow)
|
|
map.insert(QStringLiteral("indexOfFloatingWindow"), indexOfFloatingWindow);
|
|
else
|
|
map.insert(QStringLiteral("mainWindowUniqueName"), mainWindowUniqueName);
|
|
|
|
return map;
|
|
}
|
|
|
|
void LayoutSaver::Placeholder::fromVariantMap(const QVariantMap &map)
|
|
{
|
|
isFloatingWindow = map.value(QStringLiteral("isFloatingWindow")).toBool();
|
|
indexOfFloatingWindow = map.value(QStringLiteral("indexOfFloatingWindow"), -1).toInt();
|
|
itemIndex = map.value(QStringLiteral("itemIndex")).toInt();
|
|
mainWindowUniqueName = map.value(QStringLiteral("mainWindowUniqueName")).toString();
|
|
}
|
|
|
|
static QScreen *screenForMainWindow(Controllers::MainWindow *mw)
|
|
{
|
|
return mw->view()->screen();
|
|
}
|
|
|
|
LayoutSaver::ScalingInfo::ScalingInfo(const QString &mainWindowId, QRect savedMainWindowGeo, int screenIndex)
|
|
{
|
|
auto mainWindow = DockRegistry::self()->mainWindowByName(mainWindowId);
|
|
if (!mainWindow) {
|
|
qWarning() << Q_FUNC_INFO << "Failed to find main window with name" << mainWindowName;
|
|
return;
|
|
}
|
|
|
|
if (!savedMainWindowGeo.isValid() || savedMainWindowGeo.isNull()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid saved main window geometry" << savedMainWindowGeo;
|
|
return;
|
|
}
|
|
|
|
if (!mainWindow->geometry().isValid() || mainWindow->geometry().isNull()) {
|
|
qWarning() << Q_FUNC_INFO << "Invalid main window geometry" << mainWindow->geometry();
|
|
return;
|
|
}
|
|
|
|
const int currentScreenIndex = qApp->screens().indexOf(screenForMainWindow(mainWindow));
|
|
|
|
this->mainWindowName = mainWindowId;
|
|
this->savedMainWindowGeometry = savedMainWindowGeo;
|
|
realMainWindowGeometry = mainWindow->window()->geometry(); // window() as our main window might be embedded
|
|
widthFactor = double(realMainWindowGeometry.width()) / savedMainWindowGeo.width();
|
|
heightFactor = double(realMainWindowGeometry.height()) / savedMainWindowGeo.height();
|
|
mainWindowChangedScreen = currentScreenIndex != screenIndex;
|
|
}
|
|
|
|
void LayoutSaver::ScalingInfo::translatePos(QPoint &pt) const
|
|
{
|
|
const int deltaX = pt.x() - savedMainWindowGeometry.x();
|
|
const int deltaY = pt.y() - savedMainWindowGeometry.y();
|
|
|
|
const double newDeltaX = deltaX * widthFactor;
|
|
const double newDeltaY = deltaY * heightFactor;
|
|
|
|
pt.setX(qCeil(savedMainWindowGeometry.x() + newDeltaX));
|
|
pt.setY(qCeil(savedMainWindowGeometry.y() + newDeltaY));
|
|
}
|
|
|
|
void LayoutSaver::ScalingInfo::applyFactorsTo(QPoint &pt) const
|
|
{
|
|
translatePos(pt);
|
|
}
|
|
|
|
void LayoutSaver::ScalingInfo::applyFactorsTo(QSize &sz) const
|
|
{
|
|
sz.setWidth(int(widthFactor * sz.width()));
|
|
sz.setHeight(int(heightFactor * sz.height()));
|
|
}
|
|
|
|
void LayoutSaver::ScalingInfo::applyFactorsTo(QRect &rect) const
|
|
{
|
|
if (rect.isEmpty())
|
|
return;
|
|
|
|
QPoint pos = rect.topLeft();
|
|
QSize size = rect.size();
|
|
|
|
applyFactorsTo(/*by-ref*/ size);
|
|
|
|
|
|
if (!mainWindowChangedScreen) {
|
|
// Don't play with floating window position if the main window changed screen.
|
|
// There's too many corner cases that push the floating windows out of bounds, and
|
|
// we're not even considering monitors with different HDPI. We can support only the simple case.
|
|
// For complex cases we'll try to guarantee the window is placed somewhere reasonable.
|
|
applyFactorsTo(/*by-ref*/ pos);
|
|
}
|
|
|
|
rect.moveTopLeft(pos);
|
|
rect.setSize(size);
|
|
}
|
|
|
|
LayoutSaver::Private::RAIIIsRestoring::RAIIIsRestoring()
|
|
{
|
|
LayoutSaver::Private::s_restoreInProgress = true;
|
|
}
|
|
|
|
LayoutSaver::Private::RAIIIsRestoring::~RAIIIsRestoring()
|
|
{
|
|
LayoutSaver::Private::s_restoreInProgress = false;
|
|
}
|