summaryrefslogtreecommitdiffstats
path: root/qt-models
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2020-05-15 17:05:34 +0200
committerGravatar Dirk Hohndel <dirk@hohndel.org>2020-09-29 16:13:03 -0700
commitaf9d379a4147b01ea599c86e0ff001cace926c1c (patch)
tree09fc7cb69d0f186a8575eb4efabd9eb19240ec19 /qt-models
parent634152ae43f37a62a67de5a5d5fc4e6b02829ce0 (diff)
downloadsubsurface-af9d379a4147b01ea599c86e0ff001cace926c1c.tar.gz
filter: add filter constraint model
Add a model that keeps track of a list of filter constraint and makes them accessible from Qt. Sadly, this is mostly repetitive boiler-plate code, but this is due to Qt's model/view-API, which is a perfect example of how *not* to design a reasonable modern API. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
Diffstat (limited to 'qt-models')
-rw-r--r--qt-models/CMakeLists.txt2
-rw-r--r--qt-models/filterconstraintmodel.cpp186
-rw-r--r--qt-models/filterconstraintmodel.h58
3 files changed, 246 insertions, 0 deletions
diff --git a/qt-models/CMakeLists.txt b/qt-models/CMakeLists.txt
index 8d88e0712..280abbd41 100644
--- a/qt-models/CMakeLists.txt
+++ b/qt-models/CMakeLists.txt
@@ -19,6 +19,8 @@ set(SUBSURFACE_GENERIC_MODELS_LIB_SRCS
diveplotdatamodel.h
divetripmodel.cpp
divetripmodel.h
+ filterconstraintmodel.cpp
+ filterconstraintmodel.h
maplocationmodel.cpp
maplocationmodel.h
models.cpp
diff --git a/qt-models/filterconstraintmodel.cpp b/qt-models/filterconstraintmodel.cpp
new file mode 100644
index 000000000..8ccc23d81
--- /dev/null
+++ b/qt-models/filterconstraintmodel.cpp
@@ -0,0 +1,186 @@
+// SPDX-License-Identifier: GPL-2.0
+#include "filterconstraintmodel.h"
+#include "core/qthelper.h" // for timestamp conversions
+#include <QTime>
+
+FilterConstraintModel::~FilterConstraintModel()
+{
+}
+
+// QTime <-> seconds in integer conversion functions
+static QTime secondsToTime(int seconds)
+{
+ return QTime::fromMSecsSinceStartOfDay(seconds * 1000);
+}
+
+static int timeToSeconds(const QTime &t)
+{
+ return t.msecsSinceStartOfDay() / 1000;
+}
+
+QVariant FilterConstraintModel::data(const QModelIndex &index, int role) const
+{
+ if (!index.isValid() || index.row() >= (int)constraints.size())
+ return QVariant();
+
+ const filter_constraint &constraint = constraints[index.row()];
+
+ switch (role) {
+ case TYPE_ROLE:
+ return constraint.type;
+ case IS_STAR_WIDGET_ROLE:
+ return filter_constraint_is_star(constraint.type);
+ case HAS_DATE_WIDGET_ROLE:
+ return filter_constraint_has_date_widget(constraint.type);
+ case HAS_TIME_WIDGET_ROLE:
+ return filter_constraint_has_time_widget(constraint.type);
+ case NUM_DECIMALS_ROLE:
+ return filter_constraint_num_decimals(constraint.type);
+ case NEGATE_COMBO_ROLE:
+ return filter_constraint_negate_list_translated();
+ case STRING_MODE_COMBO_ROLE:
+ return filter_constraint_has_string_mode(constraint.type) ?
+ filter_constraint_string_mode_list_translated() : QStringList();
+ case RANGE_MODE_COMBO_ROLE:
+ return filter_constraint_has_range_mode(constraint.type) ?
+ filter_constraint_range_mode_list_translated() : QStringList();
+ case MULTIPLE_CHOICE_LIST_ROLE:
+ return filter_contraint_multiple_choice_translated(constraint.type);
+ case STRING_MODE_ROLE:
+ return static_cast<int>(constraint.string_mode);
+ case RANGE_MODE_ROLE:
+ return static_cast<int>(constraint.range_mode);
+ case TYPE_DISPLAY_ROLE:
+ return filter_constraint_type_to_string_translated(constraint.type);
+ case NEGATE_DISPLAY_ROLE:
+ return filter_constraint_negate_to_string_translated(constraint.negate);
+ case STRING_MODE_DISPLAY_ROLE:
+ return filter_constraint_string_mode_to_string_translated(constraint.string_mode);
+ case RANGE_MODE_DISPLAY_ROLE:
+ return filter_constraint_range_mode_to_string_translated(constraint.range_mode);
+ case TYPE_INDEX_ROLE:
+ return filter_constraint_type_to_index(constraint.type);
+ case NEGATE_INDEX_ROLE:
+ return static_cast<int>(constraint.negate);
+ case STRING_MODE_INDEX_ROLE:
+ return filter_constraint_string_mode_to_index(constraint.string_mode);
+ case RANGE_MODE_INDEX_ROLE:
+ return filter_constraint_range_mode_to_index(constraint.range_mode);
+ case UNIT_ROLE:
+ return filter_constraint_get_unit(constraint.type);
+ case STRING_ROLE:
+ return filter_constraint_get_string(constraint);
+ case INTEGER_FROM_ROLE:
+ return filter_constraint_get_integer_from(constraint);
+ case INTEGER_TO_ROLE:
+ return filter_constraint_get_integer_to(constraint);
+ case FLOAT_FROM_ROLE:
+ return filter_constraint_get_float_from(constraint); // Converts from integers to metric or imperial units
+ case FLOAT_TO_ROLE:
+ return filter_constraint_get_float_to(constraint); // Converts from integers to metric or imperial units
+ case TIMESTAMP_FROM_ROLE:
+ return timestampToDateTime(filter_constraint_get_timestamp_from(constraint));
+ case TIMESTAMP_TO_ROLE:
+ return timestampToDateTime(filter_constraint_get_timestamp_to(constraint));
+ case TIME_FROM_ROLE:
+ return secondsToTime(filter_constraint_get_integer_from(constraint));
+ case TIME_TO_ROLE:
+ return secondsToTime(filter_constraint_get_integer_from(constraint));
+ case MULTIPLE_CHOICE_ROLE:
+ return (qulonglong)filter_constraint_get_multiple_choice(constraint);
+ }
+ return QVariant();
+}
+
+bool FilterConstraintModel::setData(const QModelIndex &index, const QVariant &value, int role)
+{
+ if (!index.isValid() || index.row() > (int)constraints.size())
+ return false;
+ filter_constraint &constraint = constraints[index.row()];
+
+ switch (role) {
+ case NEGATE_INDEX_ROLE:
+ constraint.negate = value.value<bool>();
+ break;
+ case STRING_ROLE:
+ filter_constraint_set_stringlist(constraint, value.value<QString>());
+ break;
+ case STRING_MODE_INDEX_ROLE:
+ constraint.string_mode = filter_constraint_string_mode_from_index(value.value<int>());
+ break;
+ case RANGE_MODE_INDEX_ROLE:
+ constraint.range_mode = filter_constraint_range_mode_from_index(value.value<int>());
+ break;
+ case INTEGER_FROM_ROLE:
+ filter_constraint_set_integer_from(constraint, value.value<int>());
+ break;
+ case INTEGER_TO_ROLE:
+ filter_constraint_set_integer_to(constraint, value.value<int>());
+ break;
+ case FLOAT_FROM_ROLE:
+ filter_constraint_set_float_from(constraint, value.value<double>());
+ break;
+ case FLOAT_TO_ROLE:
+ filter_constraint_set_float_to(constraint, value.value<double>());
+ break;
+ case TIMESTAMP_FROM_ROLE:
+ filter_constraint_set_timestamp_from(constraint, dateTimeToTimestamp(value.value<QDateTime>()));
+ break;
+ case TIMESTAMP_TO_ROLE:
+ filter_constraint_set_timestamp_to(constraint, dateTimeToTimestamp(value.value<QDateTime>()));
+ break;
+ case TIME_FROM_ROLE:
+ filter_constraint_set_integer_from(constraint, timeToSeconds(value.value<QTime>()));
+ break;
+ case TIME_TO_ROLE:
+ filter_constraint_set_integer_to(constraint, timeToSeconds(value.value<QTime>()));
+ break;
+ case MULTIPLE_CHOICE_ROLE:
+ filter_constraint_set_multiple_choice(constraint, value.value<uint64_t>());
+ break;
+ default:
+ return false;
+ }
+ emit dataChanged(index, index, QVector<int> { role });
+ return true;
+}
+
+int FilterConstraintModel::rowCount(const QModelIndex&) const
+{
+ return constraints.size();
+}
+
+void FilterConstraintModel::reload(const std::vector<filter_constraint> &constraintsIn)
+{
+ beginResetModel();
+ constraints = constraintsIn;
+ endResetModel();
+}
+
+std::vector<filter_constraint> FilterConstraintModel::getConstraints() const
+{
+ std::vector<filter_constraint> res;
+ res.reserve(constraints.size());
+ for (const filter_constraint &c: constraints) {
+ if (filter_constraint_is_valid(&c))
+ res.push_back(c);
+ }
+ return res;
+}
+
+void FilterConstraintModel::addConstraint(filter_constraint_type type)
+{
+ int count = (int)constraints.size();
+ beginInsertRows(QModelIndex(), count, count);
+ constraints.emplace_back(type);
+ endInsertRows();
+}
+
+void FilterConstraintModel::deleteConstraint(int row)
+{
+ if (row < 0 || row >= (int)constraints.size())
+ return;
+ beginRemoveRows(QModelIndex(), row, row);
+ constraints.erase(constraints.begin() + row);
+ endRemoveRows();
+}
diff --git a/qt-models/filterconstraintmodel.h b/qt-models/filterconstraintmodel.h
new file mode 100644
index 000000000..e7e70960d
--- /dev/null
+++ b/qt-models/filterconstraintmodel.h
@@ -0,0 +1,58 @@
+// SPDX-License-Identifier: GPL-2.0
+#ifndef FILTERCONSTRAINTMODEL_H
+#define FILTERCONSTRAINTMODEL_H
+
+#include "core/filterconstraint.h"
+#include <QAbstractTableModel>
+#include <vector>
+
+class FilterConstraintModel : public QAbstractListModel {
+ Q_OBJECT
+public:
+ enum Roles {
+ TYPE_ROLE = Qt::UserRole + 1, // enum filter_constraint_type cast to int
+ IS_STAR_WIDGET_ROLE, // represent as a star widget
+ HAS_DATE_WIDGET_ROLE, // has a date widget
+ HAS_TIME_WIDGET_ROLE, // has a time widget
+ NUM_DECIMALS_ROLE, // number of decimal places for numeric data
+ NEGATE_COMBO_ROLE, // combo box entries for negate
+ STRING_MODE_COMBO_ROLE, // combo box entries for string mode or empty list if no string mode
+ RANGE_MODE_COMBO_ROLE, // combo box entries for range mode or empty list if no range mode
+ MULTIPLE_CHOICE_LIST_ROLE, // list of translated multiple-choice items
+ STRING_MODE_ROLE, // enum filter_constraint_string_mode_role cast to int
+ RANGE_MODE_ROLE, // enum filter_constraint_range_mode cast to int
+ TYPE_DISPLAY_ROLE, // type for display (i.e. translated)
+ NEGATE_DISPLAY_ROLE, // negate flag for display (i.e. translated)
+ STRING_MODE_DISPLAY_ROLE, // string mode for display (i.e. translated)
+ RANGE_MODE_DISPLAY_ROLE, // range mode for display (i.e. translated)
+ NEGATE_INDEX_ROLE, // negate index in combo box
+ TYPE_INDEX_ROLE, // type index in combo box
+ STRING_MODE_INDEX_ROLE, // string mode index in combo box
+ RANGE_MODE_INDEX_ROLE, // range mode index in combo box
+ UNIT_ROLE, // unit, if any
+ STRING_ROLE, // string data
+ INTEGER_FROM_ROLE,
+ INTEGER_TO_ROLE,
+ FLOAT_FROM_ROLE,
+ FLOAT_TO_ROLE,
+ TIMESTAMP_FROM_ROLE,
+ TIMESTAMP_TO_ROLE,
+ TIME_FROM_ROLE,
+ TIME_TO_ROLE,
+ MULTIPLE_CHOICE_ROLE
+ };
+private:
+ QVariant data(const QModelIndex &index, int role) const override;
+ std::vector<filter_constraint> constraints;
+public:
+ using QAbstractListModel::QAbstractListModel;
+ ~FilterConstraintModel();
+ void reload(const std::vector<filter_constraint> &);
+ std::vector<filter_constraint> getConstraints() const; // filters out constraints with no user input
+ void addConstraint(filter_constraint_type type);
+ void deleteConstraint(int row);
+ bool setData(const QModelIndex &index, const QVariant &value, int role) override;
+ int rowCount(const QModelIndex &parent) const override;
+};
+
+#endif