1
0
mirror of https://github.com/githubuser0xFFFF/Qt-Advanced-Docking-System.git synced 2025-04-01 02:42:39 +08:00

Implements storeGeometry() and restoreGeometry() for ContainerWidget. It

correctly aligns all widgets... wohooo.
This commit is contained in:
mfreiholz 2016-02-11 14:33:02 +01:00
parent 7123410bb1
commit 9050d1910c
7 changed files with 181 additions and 29 deletions

View File

@ -50,6 +50,10 @@ public:
QMenu* createContextMenu() const; QMenu* createContextMenu() const;
private:
void saveGeometryWalk(QDataStream& out, QWidget* widget) const;
bool restoreGeometryWalk(QDataStream& in, QSplitter* currentSplitter = NULL);
signals: signals:
void orientationChanged(); void orientationChanged();
@ -63,8 +67,8 @@ public:
// Layout stuff // Layout stuff
QGridLayout* _mainLayout; QGridLayout* _mainLayout;
Qt::Orientation _orientation; ///< Default orientation of new sections. Qt::Orientation _orientation;
QPointer<QSplitter> _splitter; ///< Default/main splitter. QPointer<QSplitter> _splitter;
}; };
ADS_NAMESPACE_END ADS_NAMESPACE_END

View File

@ -5,6 +5,7 @@
#include <QWidget> #include <QWidget>
#include <QHash> #include <QHash>
#include <QSharedPointer> #include <QSharedPointer>
#include <QWeakPointer>
#include "ads/API.h" #include "ads/API.h"
@ -12,28 +13,32 @@ ADS_NAMESPACE_BEGIN
class SectionContent class SectionContent
{ {
friend class ContainerWidget;
private:
SectionContent(QWidget* title, QWidget* content, const QString& uniqueName = QString()); ///< Do not use!
public: public:
typedef QSharedPointer<SectionContent> RefPtr; typedef QSharedPointer<SectionContent> RefPtr;
typedef QWeakPointer<SectionContent> WeakPtr;
SectionContent(QWidget* title, QWidget* content, const QString& uniqueName = QString()); ///< Do not use!
virtual ~SectionContent(); virtual ~SectionContent();
int uid() const; int uid() const;
QString uniqueName() const; QString uniqueName() const;
QWidget* titleWidget() const; QWidget* titleWidget() const;
QWidget* contentWidget() const; QWidget* contentWidget() const;
static RefPtr newSectionContent(QWidget* title, QWidget* content); static RefPtr newSectionContent(QWidget* title, QWidget* content, const QString& uniqueName = QString());
static RefPtr newSectionContent(const QString& title, QWidget* content);
public: private:
const int _uid; const int _uid;
const QString _uniqueName; const QString _uniqueName;
QPointer<QWidget> _title; QPointer<QWidget> _title;
QPointer<QWidget> _content; QPointer<QWidget> _content;
static int NextUid; static int NextUid;
static QHash<int, SectionContent*> LookupMap; static QHash<int, SectionContent::WeakPtr> LookupMap;
static QHash<QString, SectionContent::WeakPtr> LookupMapByName;
}; };
ADS_NAMESPACE_END ADS_NAMESPACE_END

View File

@ -35,7 +35,7 @@ public:
QRect titleAreaGeometry() const; QRect titleAreaGeometry() const;
QRect contentAreaGeometry() const; QRect contentAreaGeometry() const;
QList<SectionContent::RefPtr> contents() const { return _contents; } const QList<SectionContent::RefPtr>& contents() const { return _contents; }
void addContent(SectionContent::RefPtr c); void addContent(SectionContent::RefPtr c);
void addContent(const InternalContentData& data, bool autoActivate); void addContent(const InternalContentData& data, bool autoActivate);
InternalContentData take(int uid, bool del = true); InternalContentData take(int uid, bool del = true);

View File

@ -5,6 +5,8 @@
#include <QContextMenuEvent> #include <QContextMenuEvent>
#include <QMenu> #include <QMenu>
#include <QSplitter> #include <QSplitter>
#include <QDataStream>
#include <QtGlobal>
ADS_NAMESPACE_BEGIN ADS_NAMESPACE_BEGIN
@ -276,8 +278,18 @@ QByteArray ContainerWidget::saveGeometry() const
QByteArray ba; QByteArray ba;
QDataStream out(&ba, QIODevice::WriteOnly); QDataStream out(&ba, QIODevice::WriteOnly);
out.setVersion(QDataStream::Qt_4_5); out.setVersion(QDataStream::Qt_4_5);
out << (int) 0x00001337; out << (quint32) 0x00001337; // Magic
out << _sections.size(); out << (quint32) 1; // Version
// Walk through layout for splitters
// Well.. there actually shouldn't be more than one
for (int i = 0; i < _mainLayout->count(); ++i)
{
QLayoutItem* li = _mainLayout->itemAt(i);
if (!li->widget())
continue;
saveGeometryWalk(out, li->widget());
}
return ba; return ba;
} }
@ -286,10 +298,28 @@ bool ContainerWidget::restoreGeometry(const QByteArray& data)
QDataStream in(data); QDataStream in(data);
in.setVersion(QDataStream::Qt_4_5); in.setVersion(QDataStream::Qt_4_5);
int magic = 0; quint32 magic = 0;
in >> magic; in >> magic;
if (magic != 0x00001337)
return false; return false;
quint32 version = 0;
in >> version;
if (version != 1)
return false;
QList<SectionWidget*> currentSections = _sections;
_sections.clear();
const bool success = restoreGeometryWalk(in, NULL);
if (success)
{
QLayoutItem* old = _mainLayout->takeAt(0);
_mainLayout->addWidget(_splitter);
delete old;
qDeleteAll(currentSections);
}
return success;
} }
QMenu* ContainerWidget::createContextMenu() const QMenu* ContainerWidget::createContextMenu() const
@ -326,4 +356,91 @@ QMenu* ContainerWidget::createContextMenu() const
return m; return m;
} }
void ContainerWidget::saveGeometryWalk(QDataStream& out, QWidget* widget) const
{
QSplitter* sp = NULL;
SectionWidget* sw = NULL;
if (!widget)
{
out << 0;
}
else if ((sp = dynamic_cast<QSplitter*>(widget)) != NULL)
{
out << 1; // Type = QSplitter
out << ((sp->orientation() == Qt::Horizontal) ? (int) 1 : (int) 2);
out << sp->count();
for (int i = 0; i < sp->count(); ++i)
{
saveGeometryWalk(out, sp->widget(i));
}
}
else if ((sw = dynamic_cast<SectionWidget*>(widget)) != NULL)
{
out << 2; // Type = SectionWidget
out << sw->geometry();
out << sw->contents().count();
const QList<SectionContent::RefPtr>& contents = sw->contents();
for (int i = 0; i < contents.count(); ++i)
{
out << contents[i]->uniqueName();
}
}
}
bool ContainerWidget::restoreGeometryWalk(QDataStream& in, QSplitter* currentSplitter)
{
int type;
in >> type;
// Splitter
if (type == 1)
{
int orientation, count;
in >> orientation >> count;
QSplitter* sp = newSplitter((Qt::Orientation) orientation);
for (int i = 0; i < count; ++i)
{
if (!restoreGeometryWalk(in, sp))
return false;
}
if (!currentSplitter)
_splitter = sp;
else
currentSplitter->addWidget(sp);
}
// Section
else if (type == 2)
{
if (!currentSplitter)
{
qWarning() << "Missing splitter object for section";
return false;
}
QRect geom;
int count;
in >> geom >> count;
SectionWidget* sw = new SectionWidget(this);
sw->setGeometry(geom);
for (int i = 0; i < count; ++i)
{
QString name;
in >> name;
SectionContent::RefPtr sc = SectionContent::LookupMapByName.value(name).toStrongRef();
if (sc)
sw->addContent(sc);
}
currentSplitter->addWidget(sw);
}
// Unknown
else
{
qDebug() << QString("");
}
return true;
}
ADS_NAMESPACE_END ADS_NAMESPACE_END

View File

@ -5,20 +5,21 @@
ADS_NAMESPACE_BEGIN ADS_NAMESPACE_BEGIN
int SectionContent::NextUid = 1; int SectionContent::NextUid = 1;
QHash<int, SectionContent*> SectionContent::LookupMap; QHash<int, SectionContent::WeakPtr> SectionContent::LookupMap;
QHash<QString, SectionContent::WeakPtr> SectionContent::LookupMapByName;
SectionContent::SectionContent(QWidget* title, QWidget* content, const QString& uniqueName) : SectionContent::SectionContent(QWidget* title, QWidget* content, const QString& uniqueName) :
_uid(NextUid++), _uid(NextUid++),
_uniqueName(uniqueName) _uniqueName(uniqueName),
_title(title),
_content(content)
{ {
_title = title;
_content = content;
LookupMap.insert(_uid, this);
} }
SectionContent::~SectionContent() SectionContent::~SectionContent()
{ {
LookupMap.remove(_uid); LookupMap.remove(_uid);
LookupMapByName.remove(_uniqueName);
delete _title; delete _title;
delete _content; delete _content;
} }
@ -43,14 +44,13 @@ QWidget* SectionContent::contentWidget() const
return _content; return _content;
} }
SectionContent::RefPtr SectionContent::newSectionContent(QWidget* title, QWidget* content) SectionContent::RefPtr SectionContent::newSectionContent(QWidget* title, QWidget* content, const QString& uniqueName)
{ {
return QSharedPointer<SectionContent>(new SectionContent(title, content)); QSharedPointer<SectionContent> sc(new SectionContent(title, content, uniqueName));
} LookupMap.insert(sc->uid(), sc);
if (!sc->uniqueName().isEmpty())
SectionContent::RefPtr SectionContent::newSectionContent(const QString& title, QWidget* content) LookupMapByName.insert(sc->uniqueName(), sc);
{ return sc;
return QSharedPointer<SectionContent>(new SectionContent(new QLabel(title), content));
} }
ADS_NAMESPACE_END ADS_NAMESPACE_END

View File

@ -31,13 +31,16 @@ static ads::SectionContent::RefPtr createLongTextLabelSC()
l->setText(QString("Lorem Ipsum ist ein einfacher Demo-Text für die Print- und Schriftindustrie. Lorem Ipsum ist in der Industrie bereits der Standard Demo-Text seit 1500, als ein unbekannter Schriftsteller eine Hand voll Wörter nahm und diese durcheinander warf um ein Musterbuch zu erstellen. Es hat nicht nur 5 Jahrhunderte überlebt, sondern auch in Spruch in die elektronische Schriftbearbeitung geschafft (bemerke, nahezu unverändert). Bekannt wurde es 1960, mit dem erscheinen von Letrase, welches Passagen von Lorem Ipsum enhielt, so wie Desktop Software wie Aldus PageMaker - ebenfalls mit Lorem Ipsum.")); l->setText(QString("Lorem Ipsum ist ein einfacher Demo-Text für die Print- und Schriftindustrie. Lorem Ipsum ist in der Industrie bereits der Standard Demo-Text seit 1500, als ein unbekannter Schriftsteller eine Hand voll Wörter nahm und diese durcheinander warf um ein Musterbuch zu erstellen. Es hat nicht nur 5 Jahrhunderte überlebt, sondern auch in Spruch in die elektronische Schriftbearbeitung geschafft (bemerke, nahezu unverändert). Bekannt wurde es 1960, mit dem erscheinen von Letrase, welches Passagen von Lorem Ipsum enhielt, so wie Desktop Software wie Aldus PageMaker - ebenfalls mit Lorem Ipsum."));
bl->addWidget(l); bl->addWidget(l);
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Label %1").arg(++CONTENT_COUNT)), w); const int index = ++CONTENT_COUNT;
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Label %1").arg(index)), w, QString("uname-%1").arg(index));
} }
static ads::SectionContent::RefPtr createCalendarSC() static ads::SectionContent::RefPtr createCalendarSC()
{ {
QCalendarWidget* w = new QCalendarWidget(); QCalendarWidget* w = new QCalendarWidget();
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Calendar %1").arg(++CONTENT_COUNT)), w);
const int index = ++CONTENT_COUNT;
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Calendar %1").arg(index)), w, QString("uname-%1").arg(index));
} }
static ads::SectionContent::RefPtr createFileSystemTreeSC() static ads::SectionContent::RefPtr createFileSystemTreeSC()
@ -46,7 +49,9 @@ static ads::SectionContent::RefPtr createFileSystemTreeSC()
// QFileSystemModel* m = new QFileSystemModel(w); // QFileSystemModel* m = new QFileSystemModel(w);
// m->setRootPath(QDir::currentPath()); // m->setRootPath(QDir::currentPath());
// w->setModel(m); // w->setModel(m);
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Filesystem %1").arg(++CONTENT_COUNT)), w);
const int index = ++CONTENT_COUNT;
return ads::SectionContent::newSectionContent(new IconTitleWidget(QIcon(), QString("Filesystem %1").arg(index)), w, QString("uname-%1").arg(index));
} }
/////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////
@ -82,6 +87,15 @@ MainWindow::MainWindow(QWidget *parent) :
section = new ads::SectionWidget(_container); section = new ads::SectionWidget(_container);
section->addContent(createCalendarSC()); section->addContent(createCalendarSC());
_container->addSection(section); _container->addSection(section);
QByteArray ba;
QFile f("test.dat");
if (f.open(QFile::ReadOnly))
{
ba = f.readAll();
f.close();
}
_container->restoreGeometry(ba);
} }
MainWindow::~MainWindow() MainWindow::~MainWindow()
@ -108,3 +122,14 @@ void MainWindow::contextMenuEvent(QContextMenuEvent* e)
m->exec(QCursor::pos()); m->exec(QCursor::pos());
delete m; delete m;
} }
void MainWindow::closeEvent(QCloseEvent* e)
{
QByteArray ba = _container->saveGeometry();
QFile f("test.dat");
if (f.open(QFile::WriteOnly))
{
f.write(ba);
f.close();
}
}

View File

@ -25,6 +25,7 @@ private slots:
protected: protected:
virtual void contextMenuEvent(QContextMenuEvent* e); virtual void contextMenuEvent(QContextMenuEvent* e);
virtual void closeEvent(QCloseEvent* e);
private: private:
Ui::MainWindow *ui; Ui::MainWindow *ui;