2017-04-27 20:26:05 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2016-04-04 22:02:03 -07:00
|
|
|
#include "desktop-widgets/tableview.h"
|
|
|
|
#include "desktop-widgets/modeldelegates.h"
|
2013-09-03 13:57:38 -03:00
|
|
|
|
|
|
|
#include <QPushButton>
|
|
|
|
#include <QSettings>
|
|
|
|
|
2014-10-17 21:22:49 +02:00
|
|
|
TableView::TableView(QWidget *parent) : QGroupBox(parent)
|
2013-10-03 11:54:25 -07:00
|
|
|
{
|
|
|
|
ui.setupUi(this);
|
2014-01-15 18:52:42 +01:00
|
|
|
ui.tableView->setItemDelegate(new DiveListDelegate(this));
|
2014-10-15 15:30:48 +02:00
|
|
|
|
2019-11-02 22:52:27 +01:00
|
|
|
connect(ui.tableView, &QTableView::clicked, this, &TableView::itemClicked);
|
|
|
|
|
2014-10-15 15:30:48 +02:00
|
|
|
QFontMetrics fm(defaultModelFont());
|
|
|
|
int text_ht = fm.height();
|
2014-10-15 15:30:52 +02:00
|
|
|
|
|
|
|
metrics.icon = &defaultIconMetrics();
|
2014-10-15 15:30:48 +02:00
|
|
|
|
2014-10-19 16:15:20 +02:00
|
|
|
metrics.rm_col_width = metrics.icon->sz_small + 2*metrics.icon->spacing;
|
2014-10-15 15:30:48 +02:00
|
|
|
metrics.header_ht = text_ht + 10; // TODO DPI
|
|
|
|
|
2014-10-19 16:15:21 +02:00
|
|
|
/* We want to get rid of the margin around the table, but
|
|
|
|
* we must be careful with some styles (e.g. GTK+) where the top
|
|
|
|
* margin is actually used to hold the label. We thus check the
|
|
|
|
* rectangles for the label and contents to make sure they do not
|
|
|
|
* overlap, and adjust the top contentsMargin accordingly
|
|
|
|
*/
|
|
|
|
|
|
|
|
// start by setting all the margins at zero
|
|
|
|
QMargins margins;
|
|
|
|
|
|
|
|
// grab the label and contents dimensions and positions
|
|
|
|
QStyleOptionGroupBox option;
|
|
|
|
initStyleOption(&option);
|
|
|
|
QRect labelRect = style()->subControlRect(QStyle::CC_GroupBox, &option, QStyle::SC_GroupBoxLabel, this);
|
|
|
|
QRect contentsRect = style()->subControlRect(QStyle::CC_GroupBox, &option, QStyle::SC_GroupBoxContents, this);
|
|
|
|
|
|
|
|
/* we need to ensure that the bottom of the label is higher
|
|
|
|
* than the top of the contents */
|
|
|
|
int delta = contentsRect.top() - labelRect.bottom();
|
|
|
|
const int min_gap = metrics.icon->spacing;
|
|
|
|
if (delta <= min_gap) {
|
|
|
|
margins.setTop(min_gap - delta);
|
|
|
|
}
|
|
|
|
layout()->setContentsMargins(margins);
|
|
|
|
|
2017-11-29 10:57:08 +01:00
|
|
|
QIcon plusIcon(":list-add-icon");
|
2014-10-17 21:22:49 +02:00
|
|
|
plusBtn = new QPushButton(plusIcon, QString(), this);
|
2013-09-03 13:57:38 -03:00
|
|
|
plusBtn->setFlat(true);
|
2014-10-19 16:15:22 +02:00
|
|
|
|
|
|
|
/* now determine the icon and button size. Since the button will be
|
|
|
|
* placed in the label, make sure that we do not overflow, as it might
|
|
|
|
* get clipped
|
|
|
|
*/
|
|
|
|
int iconSize = metrics.icon->sz_small;
|
|
|
|
int btnSize = iconSize + 2*min_gap;
|
|
|
|
if (btnSize > labelRect.height()) {
|
|
|
|
btnSize = labelRect.height();
|
|
|
|
iconSize = btnSize - 2*min_gap;
|
|
|
|
}
|
|
|
|
plusBtn->setIconSize(QSize(iconSize, iconSize));
|
2020-09-12 17:10:19 -07:00
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(5, 15, 0)
|
|
|
|
// with Qt 5.15, this leads to an inoperable button
|
2014-10-19 16:15:22 +02:00
|
|
|
plusBtn->resize(btnSize, btnSize);
|
2020-09-12 17:10:19 -07:00
|
|
|
#endif
|
2013-09-03 13:57:38 -03:00
|
|
|
connect(plusBtn, SIGNAL(clicked(bool)), this, SIGNAL(addButtonClicked()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TableView::~TableView()
|
|
|
|
{
|
|
|
|
QSettings s;
|
|
|
|
s.beginGroup(objectName());
|
2014-08-26 17:52:05 -07:00
|
|
|
// remove the old default
|
|
|
|
bool oldDefault = (ui.tableView->columnWidth(0) == 30);
|
2017-04-17 13:16:56 +03:00
|
|
|
for (int i = 1; oldDefault && ui.tableView->model() && i < ui.tableView->model()->columnCount(); i++) {
|
2014-08-26 17:52:05 -07:00
|
|
|
if (ui.tableView->columnWidth(i) != 80)
|
|
|
|
oldDefault = false;
|
|
|
|
}
|
|
|
|
if (oldDefault) {
|
|
|
|
s.remove("");
|
2017-04-04 19:50:46 +02:00
|
|
|
} else if (ui.tableView->model()) {
|
2014-08-26 17:52:05 -07:00
|
|
|
for (int i = 0; i < ui.tableView->model()->columnCount(); i++) {
|
desktop: fix saving of column-widths of device and site tables
Qt's memory management scheme is completely broken and messes
with common expectations.
QObjects are organized as a tree. The children are destroyed
in the destructor of QObject. This means that they are destructed
after the destructor of the parent object has run and its
sub-object were destructed. Obviously, this makes no sense as
the child objects should be able to access their parent at
any time.
To restore the commonly expected deterministic order of
construction and destruction, one might simply do away with
Qt's silly object tree and organise things using classical
subobjects. However, that breaks with the Qt-generated UI
classes: The objects generated by these classes are *not*
destructed with the UI class. Instead, they are attached
to the widget's QObject tree. Thus these are again destructed
*after* the widget! Who comes up with such a scheme?
In our case this means that we cannot have models used for
TableViews as subobjects, because the TableView needs the
model to save the column widths in the destructor. Which,
as detailed above is called *after* the desctructor of the
widget! Thus, turn these models into heap-allocated objects
and add them to the QObject tree.
Funilly, this exposes another insanity of Qt's QObject tree:
Children are destructed in order of construction! One would
expect that if objects are constructed in the sequence
A, B, C one can expect that C can, at any time, access B and A.
Not so in Qt: The destruction order is likewise A, B, C!
Thus, take care to init the widgets before the model. Jeez.
Finally, print a warning in the column-saving code of
TableWidget, so that these kind of subtleties are caught
in the future.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2020-11-07 13:20:46 +01:00
|
|
|
if (ui.tableView->columnWidth(i) == defaultColumnWidth(i)) {
|
2014-08-26 17:52:05 -07:00
|
|
|
s.remove(QString("colwidth%1").arg(i));
|
desktop: fix saving of column-widths of device and site tables
Qt's memory management scheme is completely broken and messes
with common expectations.
QObjects are organized as a tree. The children are destroyed
in the destructor of QObject. This means that they are destructed
after the destructor of the parent object has run and its
sub-object were destructed. Obviously, this makes no sense as
the child objects should be able to access their parent at
any time.
To restore the commonly expected deterministic order of
construction and destruction, one might simply do away with
Qt's silly object tree and organise things using classical
subobjects. However, that breaks with the Qt-generated UI
classes: The objects generated by these classes are *not*
destructed with the UI class. Instead, they are attached
to the widget's QObject tree. Thus these are again destructed
*after* the widget! Who comes up with such a scheme?
In our case this means that we cannot have models used for
TableViews as subobjects, because the TableView needs the
model to save the column widths in the destructor. Which,
as detailed above is called *after* the desctructor of the
widget! Thus, turn these models into heap-allocated objects
and add them to the QObject tree.
Funilly, this exposes another insanity of Qt's QObject tree:
Children are destructed in order of construction! One would
expect that if objects are constructed in the sequence
A, B, C one can expect that C can, at any time, access B and A.
Not so in Qt: The destruction order is likewise A, B, C!
Thus, take care to init the widgets before the model. Jeez.
Finally, print a warning in the column-saving code of
TableWidget, so that these kind of subtleties are caught
in the future.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2020-11-07 13:20:46 +01:00
|
|
|
} else {
|
2014-08-26 17:52:05 -07:00
|
|
|
s.setValue(QString("colwidth%1").arg(i), ui.tableView->columnWidth(i));
|
desktop: fix saving of column-widths of device and site tables
Qt's memory management scheme is completely broken and messes
with common expectations.
QObjects are organized as a tree. The children are destroyed
in the destructor of QObject. This means that they are destructed
after the destructor of the parent object has run and its
sub-object were destructed. Obviously, this makes no sense as
the child objects should be able to access their parent at
any time.
To restore the commonly expected deterministic order of
construction and destruction, one might simply do away with
Qt's silly object tree and organise things using classical
subobjects. However, that breaks with the Qt-generated UI
classes: The objects generated by these classes are *not*
destructed with the UI class. Instead, they are attached
to the widget's QObject tree. Thus these are again destructed
*after* the widget! Who comes up with such a scheme?
In our case this means that we cannot have models used for
TableViews as subobjects, because the TableView needs the
model to save the column widths in the destructor. Which,
as detailed above is called *after* the desctructor of the
widget! Thus, turn these models into heap-allocated objects
and add them to the QObject tree.
Funilly, this exposes another insanity of Qt's QObject tree:
Children are destructed in order of construction! One would
expect that if objects are constructed in the sequence
A, B, C one can expect that C can, at any time, access B and A.
Not so in Qt: The destruction order is likewise A, B, C!
Thus, take care to init the widgets before the model. Jeez.
Finally, print a warning in the column-saving code of
TableWidget, so that these kind of subtleties are caught
in the future.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2020-11-07 13:20:46 +01:00
|
|
|
}
|
2014-08-26 17:52:05 -07:00
|
|
|
}
|
desktop: fix saving of column-widths of device and site tables
Qt's memory management scheme is completely broken and messes
with common expectations.
QObjects are organized as a tree. The children are destroyed
in the destructor of QObject. This means that they are destructed
after the destructor of the parent object has run and its
sub-object were destructed. Obviously, this makes no sense as
the child objects should be able to access their parent at
any time.
To restore the commonly expected deterministic order of
construction and destruction, one might simply do away with
Qt's silly object tree and organise things using classical
subobjects. However, that breaks with the Qt-generated UI
classes: The objects generated by these classes are *not*
destructed with the UI class. Instead, they are attached
to the widget's QObject tree. Thus these are again destructed
*after* the widget! Who comes up with such a scheme?
In our case this means that we cannot have models used for
TableViews as subobjects, because the TableView needs the
model to save the column widths in the destructor. Which,
as detailed above is called *after* the desctructor of the
widget! Thus, turn these models into heap-allocated objects
and add them to the QObject tree.
Funilly, this exposes another insanity of Qt's QObject tree:
Children are destructed in order of construction! One would
expect that if objects are constructed in the sequence
A, B, C one can expect that C can, at any time, access B and A.
Not so in Qt: The destruction order is likewise A, B, C!
Thus, take care to init the widgets before the model. Jeez.
Finally, print a warning in the column-saving code of
TableWidget, so that these kind of subtleties are caught
in the future.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2020-11-07 13:20:46 +01:00
|
|
|
} else {
|
|
|
|
qWarning("TableView %s without model", qPrintable(objectName()));
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|
|
|
|
s.endGroup();
|
|
|
|
}
|
|
|
|
|
2014-02-27 20:09:57 -08:00
|
|
|
void TableView::setBtnToolTip(const QString &tooltip)
|
2013-09-03 13:57:38 -03:00
|
|
|
{
|
|
|
|
plusBtn->setToolTip(tooltip);
|
|
|
|
}
|
|
|
|
|
2014-01-16 11:50:56 +07:00
|
|
|
void TableView::setModel(QAbstractItemModel *model)
|
|
|
|
{
|
2013-10-03 11:54:25 -07:00
|
|
|
ui.tableView->setModel(model);
|
2013-09-03 13:57:38 -03:00
|
|
|
|
|
|
|
QSettings s;
|
|
|
|
s.beginGroup(objectName());
|
2013-11-21 23:16:19 -02:00
|
|
|
const int columnCount = ui.tableView->model()->columnCount();
|
|
|
|
for (int i = 0; i < columnCount; i++) {
|
2014-10-15 15:30:48 +02:00
|
|
|
QVariant width = s.value(QString("colwidth%1").arg(i), defaultColumnWidth(i));
|
2013-11-21 23:16:19 -02:00
|
|
|
ui.tableView->setColumnWidth(i, width.toInt());
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|
|
|
|
s.endGroup();
|
|
|
|
|
2014-10-15 15:30:48 +02:00
|
|
|
ui.tableView->horizontalHeader()->setMinimumHeight(metrics.header_ht);
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void TableView::fixPlusPosition()
|
|
|
|
{
|
2014-10-17 21:22:49 +02:00
|
|
|
QStyleOptionGroupBox option;
|
|
|
|
initStyleOption(&option);
|
|
|
|
QRect labelRect = style()->subControlRect(QStyle::CC_GroupBox, &option, QStyle::SC_GroupBoxLabel, this);
|
|
|
|
QRect contentsRect = style()->subControlRect(QStyle::CC_GroupBox, &option, QStyle::QStyle::SC_GroupBoxFrame, this);
|
|
|
|
plusBtn->setGeometry( contentsRect.width() - plusBtn->width(), labelRect.y(), plusBtn->width(), labelRect.height());
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// We need to manually position the 'plus' on cylinder and weight.
|
2014-02-27 20:09:57 -08:00
|
|
|
void TableView::resizeEvent(QResizeEvent *event)
|
2013-09-03 13:57:38 -03:00
|
|
|
{
|
|
|
|
fixPlusPosition();
|
|
|
|
QWidget::resizeEvent(event);
|
|
|
|
}
|
|
|
|
|
2014-02-27 20:09:57 -08:00
|
|
|
void TableView::showEvent(QShowEvent *event)
|
2013-09-03 13:57:38 -03:00
|
|
|
{
|
|
|
|
QWidget::showEvent(event);
|
|
|
|
fixPlusPosition();
|
|
|
|
}
|
|
|
|
|
2014-02-27 20:09:57 -08:00
|
|
|
void TableView::edit(const QModelIndex &index)
|
2014-01-16 11:50:56 +07:00
|
|
|
{
|
2013-10-03 11:54:25 -07:00
|
|
|
ui.tableView->edit(index);
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|
|
|
|
|
2014-10-15 15:30:48 +02:00
|
|
|
int TableView::defaultColumnWidth(int col)
|
|
|
|
{
|
2016-12-31 21:17:20 -08:00
|
|
|
int width;
|
2015-03-17 17:56:52 -03:00
|
|
|
QString text = ui.tableView->model()->headerData(col, Qt::Horizontal).toString();
|
2020-01-01 17:14:15 -08:00
|
|
|
#if QT_VERSION < QT_VERSION_CHECK(5, 11, 0)
|
|
|
|
int textSpace = defaultModelFontMetrics().width(text) + 4;
|
|
|
|
#else // QT 5.11 or newer
|
|
|
|
int textSpace = defaultModelFontMetrics().horizontalAdvance(text) + 4;
|
|
|
|
#endif
|
|
|
|
width = text.isEmpty() ? metrics.rm_col_width : textSpace + 4; // add small margin
|
2016-12-31 21:17:20 -08:00
|
|
|
#if defined(Q_OS_MAC)
|
|
|
|
width += 10; // Mac needs more margin
|
|
|
|
#endif
|
|
|
|
return width;
|
2014-10-15 15:30:48 +02:00
|
|
|
}
|
|
|
|
|
2014-01-16 11:50:56 +07:00
|
|
|
QTableView *TableView::view()
|
|
|
|
{
|
2013-10-03 11:54:25 -07:00
|
|
|
return ui.tableView;
|
2013-09-03 13:57:38 -03:00
|
|
|
}
|