1
0
mirror of https://github.com/f4exb/sdrangel.git synced 2024-11-15 12:51:49 -05:00

Merge pull request #1216 from srcejon/v7_stack_layout

V7: Add stack window layout
This commit is contained in:
Edouard Griffiths 2022-04-26 20:32:23 +02:00 committed by GitHub
commit 7a5ebcb8cf
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 458 additions and 36 deletions

View File

@ -487,7 +487,7 @@
<widget class="QComboBox" name="udpFormat"> <widget class="QComboBox" name="udpFormat">
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>60</width> <width>66</width>
<height>0</height> <height>0</height>
</size> </size>
</property> </property>

View File

@ -6,25 +6,25 @@
<rect> <rect>
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>374</width> <width>360</width>
<height>584</height> <height>584</height>
</rect> </rect>
</property> </property>
<property name="sizePolicy"> <property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Fixed"> <sizepolicy hsizetype="Expanding" vsizetype="Fixed">
<horstretch>0</horstretch> <horstretch>0</horstretch>
<verstretch>0</verstretch> <verstretch>0</verstretch>
</sizepolicy> </sizepolicy>
</property> </property>
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>374</width> <width>360</width>
<height>584</height> <height>584</height>
</size> </size>
</property> </property>
<property name="maximumSize"> <property name="maximumSize">
<size> <size>
<width>374</width> <width>560</width>
<height>584</height> <height>584</height>
</size> </size>
</property> </property>
@ -59,7 +59,7 @@
<string>Calculators</string> <string>Calculators</string>
</property> </property>
<property name="currentIndex"> <property name="currentIndex">
<number>1</number> <number>0</number>
</property> </property>
<widget class="QWidget" name="dipoleTab"> <widget class="QWidget" name="dipoleTab">
<attribute name="title"> <attribute name="title">
@ -70,7 +70,7 @@
<widget class="QComboBox" name="dipoleFrequencySelect"> <widget class="QComboBox" name="dipoleFrequencySelect">
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>94</width> <width>90</width>
<height>0</height> <height>0</height>
</size> </size>
</property> </property>
@ -266,7 +266,7 @@
</property> </property>
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>94</width> <width>90</width>
<height>0</height> <height>0</height>
</size> </size>
</property> </property>

View File

@ -668,10 +668,6 @@ void APRSGUI::resizeEvent(QResizeEvent* size)
plotWeather(); plotWeather();
plotTelemetry(); plotTelemetry();
plotMotion(); plotMotion();
int maxWidth = getRollupContents()->maximumWidth();
int minHeight = getRollupContents()->minimumHeight() + getAdditionalHeight();
resize(width() < maxWidth ? width() : maxWidth, minHeight);
size->accept();
} }
void APRSGUI::onMenuDialogCalled(const QPoint &p) void APRSGUI::onMenuDialogCalled(const QPoint &p)

View File

@ -11,7 +11,7 @@
</rect> </rect>
</property> </property>
<property name="sizePolicy"> <property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Fixed"> <sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch> <horstretch>0</horstretch>
<verstretch>0</verstretch> <verstretch>0</verstretch>
</sizepolicy> </sizepolicy>
@ -22,12 +22,6 @@
<height>0</height> <height>0</height>
</size> </size>
</property> </property>
<property name="maximumSize">
<size>
<width>700</width>
<height>16777215</height>
</size>
</property>
<property name="font"> <property name="font">
<font> <font>
<family>Liberation Sans</family> <family>Liberation Sans</family>
@ -154,7 +148,7 @@
</rect> </rect>
</property> </property>
<property name="sizePolicy"> <property name="sizePolicy">
<sizepolicy hsizetype="Preferred" vsizetype="Preferred"> <sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch> <horstretch>0</horstretch>
<verstretch>0</verstretch> <verstretch>0</verstretch>
</sizepolicy> </sizepolicy>
@ -187,6 +181,12 @@
<number>0</number> <number>0</number>
</property> </property>
<widget class="QWidget" name="stationTab"> <widget class="QWidget" name="stationTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Stations and Objects</string> <string>Stations and Objects</string>
</attribute> </attribute>
@ -568,6 +568,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="weatherTab"> <widget class="QWidget" name="weatherTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Weather</string> <string>Weather</string>
</attribute> </attribute>
@ -840,6 +846,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="motionTab"> <widget class="QWidget" name="motionTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Motion</string> <string>Motion</string>
</attribute> </attribute>
@ -1007,6 +1019,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="telemetryTab"> <widget class="QWidget" name="telemetryTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Telemetry</string> <string>Telemetry</string>
</attribute> </attribute>
@ -1271,6 +1289,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="statusTab"> <widget class="QWidget" name="statusTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Status</string> <string>Status</string>
</attribute> </attribute>
@ -1323,6 +1347,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="packetsTab"> <widget class="QWidget" name="packetsTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Packets</string> <string>Packets</string>
</attribute> </attribute>
@ -1374,6 +1404,12 @@
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="messagesTab"> <widget class="QWidget" name="messagesTab">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<attribute name="title"> <attribute name="title">
<string>Messages</string> <string>Messages</string>
</attribute> </attribute>
@ -1453,17 +1489,17 @@
</widget> </widget>
</widget> </widget>
<customwidgets> <customwidgets>
<customwidget>
<class>ButtonSwitch</class>
<extends>QToolButton</extends>
<header>gui/buttonswitch.h</header>
</customwidget>
<customwidget> <customwidget>
<class>RollupContents</class> <class>RollupContents</class>
<extends>QWidget</extends> <extends>QWidget</extends>
<header>gui/rollupcontents.h</header> <header>gui/rollupcontents.h</header>
<container>1</container> <container>1</container>
</customwidget> </customwidget>
<customwidget>
<class>ButtonSwitch</class>
<extends>QToolButton</extends>
<header>gui/buttonswitch.h</header>
</customwidget>
<customwidget> <customwidget>
<class>QChartView</class> <class>QChartView</class>
<extends>QGraphicsView</extends> <extends>QGraphicsView</extends>

View File

@ -24,7 +24,7 @@
</property> </property>
<property name="maximumSize"> <property name="maximumSize">
<size> <size>
<width>360</width> <width>560</width>
<height>155</height> <height>155</height>
</size> </size>
</property> </property>

View File

@ -6,7 +6,7 @@
<rect> <rect>
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>410</width> <width>360</width>
<height>234</height> <height>234</height>
</rect> </rect>
</property> </property>
@ -18,13 +18,13 @@
</property> </property>
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>410</width> <width>360</width>
<height>234</height> <height>234</height>
</size> </size>
</property> </property>
<property name="maximumSize"> <property name="maximumSize">
<size> <size>
<width>410</width> <width>360</width>
<height>234</height> <height>234</height>
</size> </size>
</property> </property>

View File

@ -6,7 +6,7 @@
<rect> <rect>
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>381</width> <width>360</width>
<height>331</height> <height>331</height>
</rect> </rect>
</property> </property>
@ -18,13 +18,13 @@
</property> </property>
<property name="minimumSize"> <property name="minimumSize">
<size> <size>
<width>381</width> <width>360</width>
<height>331</height> <height>331</height>
</size> </size>
</property> </property>
<property name="maximumSize"> <property name="maximumSize">
<size> <size>
<width>381</width> <width>360</width>
<height>331</height> <height>331</height>
</size> </size>
</property> </property>

View File

@ -95,7 +95,7 @@ protected:
void mouseMoveEvent(QMouseEvent* event) override; void mouseMoveEvent(QMouseEvent* event) override;
void resetContextMenuType() { m_contextMenuType = ContextMenuNone; } void resetContextMenuType() { m_contextMenuType = ContextMenuNone; }
void updateIndexLabel(); void updateIndexLabel();
int getAdditionalHeight() const { return 25 + 22; } int getAdditionalHeight() const { return 22 + 22; } // height of top and bottom bars
void setHighlighted(bool highlighted); void setHighlighted(bool highlighted);
DeviceType m_deviceType; DeviceType m_deviceType;

View File

@ -83,7 +83,7 @@ protected:
void mouseReleaseEvent(QMouseEvent* event) override; void mouseReleaseEvent(QMouseEvent* event) override;
void mouseMoveEvent(QMouseEvent* event) override; void mouseMoveEvent(QMouseEvent* event) override;
void resetContextMenuType() { m_contextMenuType = ContextMenuNone; } void resetContextMenuType() { m_contextMenuType = ContextMenuNone; }
int getAdditionalHeight() const { return 25 + 22; } int getAdditionalHeight() const { return 26 + 22; } // height of top and bottom bars
DeviceType m_deviceType; DeviceType m_deviceType;
int m_deviceSetIndex; int m_deviceSetIndex;

View File

@ -72,7 +72,7 @@ protected:
void mouseReleaseEvent(QMouseEvent* event) override; void mouseReleaseEvent(QMouseEvent* event) override;
void mouseMoveEvent(QMouseEvent* event) override; void mouseMoveEvent(QMouseEvent* event) override;
void resetContextMenuType() { m_contextMenuType = ContextMenuNone; } void resetContextMenuType() { m_contextMenuType = ContextMenuNone; }
int getAdditionalHeight() const { return 25 + 22; } int getAdditionalHeight() const { return 22 + 22; } // height of top and bottom bars
int m_featureIndex; int m_featureIndex;
QString m_helpURL; QString m_helpURL;

View File

@ -16,6 +16,8 @@
// along with this program. If not, see <http://www.gnu.org/licenses/>. // // along with this program. If not, see <http://www.gnu.org/licenses/>. //
/////////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////////
#include <algorithm>
#include <QHBoxLayout> #include <QHBoxLayout>
#include <QLabel> #include <QLabel>
#include <QPushButton> #include <QPushButton>
@ -23,14 +25,23 @@
#include <QMdiArea> #include <QMdiArea>
#include <QMdiSubWindow> #include <QMdiSubWindow>
#include <QFrame> #include <QFrame>
#include <QDebug>
#include <QApplication>
#include "gui/samplingdevicedialog.h" #include "gui/samplingdevicedialog.h"
#include "gui/rollupcontents.h"
#include "channel/channelgui.h"
#include "feature/featuregui.h"
#include "device/devicegui.h"
#include "mainspectrum/mainspectrumgui.h"
#include "workspace.h" #include "workspace.h"
Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) : Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) :
QDockWidget(parent, flags), QDockWidget(parent, flags),
m_index(index), m_index(index),
m_featureAddDialog(this) m_featureAddDialog(this),
m_stacking(false),
m_userChannelMinWidth(0)
{ {
m_mdi = new QMdiArea(this); m_mdi = new QMdiArea(this);
m_mdi->setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded); m_mdi->setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded);
@ -100,6 +111,19 @@ Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) :
m_tileSubWindows->setToolTip("Tile sub windows"); m_tileSubWindows->setToolTip("Tile sub windows");
m_tileSubWindows->setFixedSize(20, 20); m_tileSubWindows->setFixedSize(20, 20);
m_stackSubWindows = new QPushButton("S");
//QIcon stackSubWindowsIcon(":/stack.png"); // FIXME
//m_stackSubWindows->setIcon(stackSubWindowsIcon);
m_stackSubWindows->setToolTip("Stack sub windows");
m_stackSubWindows->setFixedSize(20, 20);
m_autoStackSubWindows = new QPushButton("AS");
m_autoStackSubWindows->setCheckable(true);
//QIcon autoStackSubWindowsIcon(":/autostack.png"); // FIXME
//m_autoStackSubWindows->setIcon(autoStackSubWindowsIcon);
m_autoStackSubWindows->setToolTip("Automatically stack sub windows");
m_autoStackSubWindows->setFixedSize(20, 20);
m_normalButton = new QPushButton(); m_normalButton = new QPushButton();
QIcon normalIcon(":/dock.png"); QIcon normalIcon(":/dock.png");
m_normalButton->setIcon(normalIcon); m_normalButton->setIcon(normalIcon);
@ -122,6 +146,8 @@ Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) :
m_titleBarLayout->addWidget(m_vline2); m_titleBarLayout->addWidget(m_vline2);
m_titleBarLayout->addWidget(m_cascadeSubWindows); m_titleBarLayout->addWidget(m_cascadeSubWindows);
m_titleBarLayout->addWidget(m_tileSubWindows); m_titleBarLayout->addWidget(m_tileSubWindows);
m_titleBarLayout->addWidget(m_stackSubWindows);
m_titleBarLayout->addWidget(m_autoStackSubWindows);
m_titleBarLayout->addStretch(1); m_titleBarLayout->addStretch(1);
m_titleBarLayout->addWidget(m_normalButton); m_titleBarLayout->addWidget(m_normalButton);
m_titleBarLayout->addWidget(m_closeButton); m_titleBarLayout->addWidget(m_closeButton);
@ -176,6 +202,20 @@ Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) :
&Workspace::tileSubWindows &Workspace::tileSubWindows
); );
QObject::connect(
m_stackSubWindows,
&QPushButton::clicked,
this,
&Workspace::stackSubWindows
);
QObject::connect(
m_autoStackSubWindows,
&QPushButton::clicked,
this,
&Workspace::autoStackSubWindows
);
QObject::connect( QObject::connect(
m_normalButton, m_normalButton,
&QPushButton::clicked, &QPushButton::clicked,
@ -191,6 +231,7 @@ Workspace::Workspace(int index, QWidget *parent, Qt::WindowFlags flags) :
this, this,
&Workspace::addFeatureEmitted &Workspace::addFeatureEmitted
); );
} }
Workspace::~Workspace() Workspace::~Workspace()
@ -198,6 +239,8 @@ Workspace::~Workspace()
qDebug("Workspace::~Workspace"); qDebug("Workspace::~Workspace");
delete m_closeButton; delete m_closeButton;
delete m_normalButton; delete m_normalButton;
delete m_autoStackSubWindows;
delete m_stackSubWindows;
delete m_tileSubWindows; delete m_tileSubWindows;
delete m_cascadeSubWindows; delete m_cascadeSubWindows;
delete m_vline2; delete m_vline2;
@ -288,15 +331,344 @@ void Workspace::tileSubWindows()
m_mdi->tileSubWindows(); m_mdi->tileSubWindows();
} }
void Workspace::orderByIndex(QList<ChannelGUI *> &list)
{
std::sort(list.begin(), list.end(),
[](const ChannelGUI *a, const ChannelGUI *b) -> bool
{
if (a->getDeviceSetIndex() == b->getDeviceSetIndex()) {
return a->getIndex() < b->getIndex();
} else {
return a->getDeviceSetIndex() < b->getDeviceSetIndex();
}
});
}
void Workspace::orderByIndex(QList<FeatureGUI *> &list)
{
std::sort(list.begin(), list.end(),
[](const FeatureGUI *a, const FeatureGUI *b) -> bool
{
return a->getIndex() < b->getIndex();
});
}
void Workspace::orderByIndex(QList<DeviceGUI *> &list)
{
std::sort(list.begin(), list.end(),
[](const DeviceGUI *a, const DeviceGUI *b) -> bool
{
return a->getIndex() < b->getIndex();
});
}
void Workspace::orderByIndex(QList<MainSpectrumGUI *> &list)
{
std::sort(list.begin(), list.end(),
[](const MainSpectrumGUI *a, const MainSpectrumGUI *b) -> bool
{
return a->getIndex() < b->getIndex();
});
}
// Try to arrange windows somewhat like in earlier versions of SDRangel
// Devices and fixed size features stacked on left
// Spectrum and expandable features stacked in centre
// Channels stacked on right
void Workspace::stackSubWindows()
{
// Set a flag so event handler knows if it's this code or the user that
// resizes a window
m_stacking = true;
// Categorise windows according to type
QList<QMdiSubWindow *> windows = m_mdi->subWindowList(QMdiArea::CreationOrder);
QList<DeviceGUI *> devices;
QList<MainSpectrumGUI *> spectrums;
QList<ChannelGUI *> channels;
QList<FeatureGUI *> fixedFeatures;
QList<FeatureGUI *> features;
for (auto window : windows)
{
if (window->isVisible())
{
if (window->inherits("DeviceGUI")) {
devices.append(qobject_cast<DeviceGUI *>(window));
} else if (window->inherits("MainSpectrumGUI")) {
spectrums.append(qobject_cast<MainSpectrumGUI *>(window));
} else if (window->inherits("ChannelGUI")) {
channels.append(qobject_cast<ChannelGUI *>(window));
} else if (window->inherits("FeatureGUI")) {
if (window->sizePolicy().verticalPolicy() == QSizePolicy::Fixed) { // Test vertical, as horizontal can be adjusted a little bit
fixedFeatures.append(qobject_cast<FeatureGUI *>(window));
} else {
features.append(qobject_cast<FeatureGUI *>(window));
}
}
}
}
// Order windows by device/feature/channel index
orderByIndex(devices);
orderByIndex(spectrums);
orderByIndex(channels);
orderByIndex(fixedFeatures);
orderByIndex(features);
// Spacing between windows
const int spacing = 2;
// Calculate width and height needed for devices
int deviceMinWidth = 0;
int deviceTotalMinHeight = 0;
for (auto window : devices)
{
int winMinWidth = std::max(window->minimumSizeHint().width(), window->minimumWidth());
deviceMinWidth = std::max(deviceMinWidth, winMinWidth);
deviceTotalMinHeight += window->minimumSizeHint().height() + spacing;
}
// Calculate width & height needed for spectrums
int spectrumMinWidth = 0;
int spectrumTotalMinHeight = 0;
int expandingSpectrums = 0;
for (auto window : spectrums)
{
int winMinWidth = std::max(window->minimumSizeHint().width(), window->minimumWidth());
spectrumMinWidth = std::max(spectrumMinWidth, winMinWidth);
spectrumTotalMinHeight += window->minimumSizeHint().height() + spacing;
expandingSpectrums++;
}
// Calculate width & height needed for channels
int channelMinWidth = m_userChannelMinWidth;
int channelTotalMinHeight = 0;
int expandingChannels = 0;
for (auto window : channels)
{
int winMinWidth = std::max(window->minimumSizeHint().width(), window->minimumWidth());
channelMinWidth = std::max(channelMinWidth, winMinWidth);
channelTotalMinHeight += window->minimumSizeHint().height() + spacing;
if (window->sizePolicy().verticalPolicy() == QSizePolicy::Expanding) {
expandingChannels++;
}
}
// Calculate width & height needed for features
// These are spilt in to two groups - fixed size and expandable
int fixedFeaturesWidth = 0;
int fixedFeaturesTotalMinHeight = 0;
int featuresMinWidth = 0;
int featuresTotalMinHeight = 0;
int expandingFeatures = 0;
for (auto window : fixedFeatures)
{
int winMinWidth = std::max(window->minimumSizeHint().width(), window->minimumWidth());
fixedFeaturesWidth = std::max(fixedFeaturesWidth, winMinWidth);
fixedFeaturesTotalMinHeight += window->minimumSizeHint().height() + spacing;
}
for (auto window : features)
{
int winMinWidth = std::max(window->minimumSizeHint().width(), window->minimumWidth());
featuresMinWidth = std::max(featuresMinWidth, winMinWidth);
featuresTotalMinHeight += window->minimumSizeHint().height() + spacing;
expandingFeatures++;
}
// Calculate width for left hand column
int devicesFeaturesWidth = std::max(deviceMinWidth, fixedFeaturesWidth);
// Calculate min width for centre column
int spectrumFeaturesMinWidth = std::max(spectrumMinWidth, featuresMinWidth);
// Calculate spacing between columns
int spacing1 = devicesFeaturesWidth > 0 ? spacing : 0;
int spacing2 = spectrumFeaturesMinWidth > 0 ? spacing : 0;
// Will we need scroll bars?
QSize mdiSize = m_mdi->size();
int minWidth = devicesFeaturesWidth + spacing1 + spectrumFeaturesMinWidth + spacing2 + channelMinWidth;
int minHeight = std::max(std::max(deviceTotalMinHeight + fixedFeaturesTotalMinHeight, channelTotalMinHeight), channelTotalMinHeight + featuresTotalMinHeight);
bool requiresHScrollBar = minWidth > mdiSize.width();
bool requiresVScrollBar = minHeight > mdiSize.height();
// Reduce available size if scroll bars needed
int sbWidth = qApp->style()->pixelMetric(QStyle::PM_ScrollBarExtent);
if (requiresVScrollBar) {
mdiSize.setWidth(mdiSize.width() - sbWidth);
}
if (requiresHScrollBar) {
mdiSize.setHeight(mdiSize.height() - sbWidth);
}
// Now position the windows
int x = 0;
int y = 0;
// Put devices down left hand side
for (auto window : devices)
{
window->move(x, y);
y += window->size().height() + spacing;
}
// Put fixed height features underneath devices
// Resize them to be same width
for (auto window : fixedFeatures)
{
window->move(x, y);
window->resize(devicesFeaturesWidth, window->size().height());
y += window->size().height() + spacing;
}
// Calculate width needed for spectrum and features in the centre - use all available space
int spectrumFeaturesWidth = std::max(mdiSize.width() - channelMinWidth - devicesFeaturesWidth - spacing1 - spacing2, spectrumFeaturesMinWidth);
// Put channels on right hand side
// Try to resize them horizontally so they are the same width
// Share any available vertical space between expanding channels
x = devicesFeaturesWidth + spacing1 + spectrumFeaturesWidth + spacing2;
y = 0;
int extraSpacePerWindow;
int extraSpaceFirstWindow;
if ((channelTotalMinHeight < mdiSize.height()) && (expandingChannels > 0))
{
extraSpacePerWindow = (mdiSize.height() - channelTotalMinHeight) / expandingChannels;
extraSpaceFirstWindow = (mdiSize.height() - channelTotalMinHeight) % expandingChannels;
}
else
{
extraSpacePerWindow = 0;
extraSpaceFirstWindow = 0;
}
for (auto window : channels)
{
window->move(x, y);
int channelHeight = window->minimumSizeHint().height();
if (window->sizePolicy().verticalPolicy() == QSizePolicy::Expanding)
{
channelHeight += extraSpacePerWindow + extraSpaceFirstWindow;
extraSpaceFirstWindow = 0;
}
window->resize(channelMinWidth, channelHeight);
y += window->size().height() + spacing;
}
// Split remaining space in the middle between spectrums and expandable features, with spectrums stacked on top
x = devicesFeaturesWidth + spacing1;
y = 0;
if ((spectrumTotalMinHeight + featuresTotalMinHeight < mdiSize.height()) && (expandingSpectrums + expandingFeatures > 0))
{
int h = mdiSize.height() - spectrumTotalMinHeight - featuresTotalMinHeight;
int f = expandingSpectrums + expandingFeatures;
extraSpacePerWindow = h / f;
extraSpaceFirstWindow = h % f;
}
else
{
extraSpacePerWindow = 0;
extraSpaceFirstWindow = 0;
}
for (auto window : spectrums)
{
window->move(x, y);
int w = spectrumFeaturesWidth;
int h = window->minimumSizeHint().height() + extraSpacePerWindow + extraSpaceFirstWindow;
window->resize(w, h);
extraSpaceFirstWindow = 0;
y += window->size().height() + spacing;
}
for (auto window : features)
{
window->move(x, y);
int w = spectrumFeaturesWidth;
int h = window->minimumSizeHint().height() + extraSpacePerWindow + extraSpaceFirstWindow;
window->resize(w, h);
extraSpaceFirstWindow = 0;
y += window->size().height() + spacing;
}
m_stacking = false;
}
void Workspace::autoStackSubWindows()
{
// FIXME: Need to save whether this is checked as a preference
if (m_autoStackSubWindows->isChecked()) {
stackSubWindows();
}
}
void Workspace::resizeEvent(QResizeEvent *event)
{
QDockWidget::resizeEvent(event);
autoStackSubWindows();
}
void Workspace::addToMdiArea(QMdiSubWindow *sub) void Workspace::addToMdiArea(QMdiSubWindow *sub)
{ {
// Add event handler to auto-stack when sub window shown or hidden
sub->installEventFilter(this);
// Can't use Close event, as it's before window is closed, so
// catch sub-window destroyed signal instead
connect(sub, &QObject::destroyed, this, &Workspace::autoStackSubWindows);
m_mdi->addSubWindow(sub); m_mdi->addSubWindow(sub);
sub->show(); sub->show();
// Auto-stack when sub-window's widgets are rolled up
ChannelGUI *channel = qobject_cast<ChannelGUI *>(sub);
if (channel) {
connect(channel->getRollupContents(), &RollupContents::widgetRolled, this, &Workspace::autoStackSubWindows);
}
FeatureGUI *feature = qobject_cast<FeatureGUI *>(sub);
if (feature) {
connect(feature->getRollupContents(), &RollupContents::widgetRolled, this, &Workspace::autoStackSubWindows);
}
} }
void Workspace::removeFromMdiArea(QMdiSubWindow *sub) void Workspace::removeFromMdiArea(QMdiSubWindow *sub)
{ {
m_mdi->removeSubWindow(sub); m_mdi->removeSubWindow(sub);
sub->removeEventFilter(this);
disconnect(sub, &QObject::destroyed, this, &Workspace::autoStackSubWindows);
ChannelGUI *channel = qobject_cast<ChannelGUI *>(sub);
if (channel) {
disconnect(channel->getRollupContents(), &RollupContents::widgetRolled, this, &Workspace::autoStackSubWindows);
}
FeatureGUI *feature = qobject_cast<FeatureGUI *>(sub);
if (feature) {
disconnect(feature->getRollupContents(), &RollupContents::widgetRolled, this, &Workspace::autoStackSubWindows);
}
}
bool Workspace::eventFilter(QObject *obj, QEvent *event)
{
if (event->type() == QEvent::Show)
{
autoStackSubWindows();
}
else if (event->type() == QEvent::Hide)
{
autoStackSubWindows();
}
else if (event->type() == QEvent::Resize)
{
if (!m_stacking && m_autoStackSubWindows->isChecked())
{
ChannelGUI *channel = qobject_cast<ChannelGUI *>(obj);
if (channel)
{
// Allow width of channels column to be set by user when they
// resize a channel window
QResizeEvent *resizeEvent = static_cast<QResizeEvent *>(event);
m_userChannelMinWidth = resizeEvent->size().width();
stackSubWindows();
}
}
}
return QDockWidget::eventFilter(obj, event);
} }
int Workspace::getNumberOfSubWindows() const int Workspace::getNumberOfSubWindows() const

View File

@ -31,6 +31,10 @@ class QStringList;
class QMdiArea; class QMdiArea;
class QMdiSubWindow; class QMdiSubWindow;
class QFrame; class QFrame;
class ChannelGUI;
class FeatureGUI;
class DeviceGUI;
class MainSpectrumGUI;
class SDRGUI_API Workspace : public QDockWidget class SDRGUI_API Workspace : public QDockWidget
{ {
@ -49,6 +53,10 @@ public:
QByteArray saveMdiGeometry(); QByteArray saveMdiGeometry();
void restoreMdiGeometry(const QByteArray& blob); void restoreMdiGeometry(const QByteArray& blob);
QList<QMdiSubWindow *> getSubWindowList() const; QList<QMdiSubWindow *> getSubWindowList() const;
void orderByIndex(QList<ChannelGUI *> &list);
void orderByIndex(QList<FeatureGUI *> &list);
void orderByIndex(QList<DeviceGUI *> &list);
void orderByIndex(QList<MainSpectrumGUI *> &list);
private: private:
int m_index; int m_index;
@ -61,6 +69,8 @@ private:
QFrame *m_vline2; QFrame *m_vline2;
QPushButton *m_cascadeSubWindows; QPushButton *m_cascadeSubWindows;
QPushButton *m_tileSubWindows; QPushButton *m_tileSubWindows;
QPushButton *m_stackSubWindows;
QPushButton *m_autoStackSubWindows;
QWidget *m_titleBar; QWidget *m_titleBar;
QHBoxLayout *m_titleBarLayout; QHBoxLayout *m_titleBarLayout;
QLabel *m_titleLabel; QLabel *m_titleLabel;
@ -68,6 +78,12 @@ private:
QPushButton *m_closeButton; QPushButton *m_closeButton;
FeatureAddDialog m_featureAddDialog; FeatureAddDialog m_featureAddDialog;
QMdiArea *m_mdi; QMdiArea *m_mdi;
bool m_stacking; // Set when stackSubWindows() is running
int m_userChannelMinWidth; // Minimum width of channels column for stackSubWindows(), set by user resizing a channel window
protected:
void resizeEvent(QResizeEvent *event) override;
bool eventFilter(QObject *obj, QEvent *event) override;
private slots: private slots:
void addRxDeviceClicked(); void addRxDeviceClicked();
@ -77,6 +93,8 @@ private slots:
void featurePresetsDialog(); void featurePresetsDialog();
void cascadeSubWindows(); void cascadeSubWindows();
void tileSubWindows(); void tileSubWindows();
void stackSubWindows();
void autoStackSubWindows();
void addFeatureEmitted(int featureIndex); void addFeatureEmitted(int featureIndex);
void toggleFloating(); void toggleFloating();