summaryrefslogtreecommitdiffstats
path: root/qt-models
diff options
context:
space:
mode:
Diffstat (limited to 'qt-models')
-rw-r--r--qt-models/filtermodels.cpp588
-rw-r--r--qt-models/filtermodels.h109
2 files changed, 3 insertions, 694 deletions
diff --git a/qt-models/filtermodels.cpp b/qt-models/filtermodels.cpp
index a6e80aa8f..31c6fe0af 100644
--- a/qt-models/filtermodels.cpp
+++ b/qt-models/filtermodels.cpp
@@ -15,547 +15,10 @@
#include <QDebug>
#include <algorithm>
-#define CREATE_INSTANCE_METHOD(CLASS) \
- CLASS *CLASS::instance() \
- { \
- static CLASS *self = new CLASS(); \
- return self; \
- }
-
-CREATE_INSTANCE_METHOD(TagFilterModel)
-CREATE_INSTANCE_METHOD(BuddyFilterModel)
-CREATE_INSTANCE_METHOD(LocationFilterModel)
-CREATE_INSTANCE_METHOD(SuitsFilterModel)
-CREATE_INSTANCE_METHOD(MultiFilterSortModel)
-
-FilterModelBase::FilterModelBase(QObject *parent) : QAbstractListModel(parent),
- anyChecked(false),
- negate(false)
-{
-}
-
-// Get index of item with given name, but ignore last item, as this
-// is the "Show Empty Tags" item. Return -1 for not found.
-int FilterModelBase::indexOf(const QString &name) const
-{
- for (int i = 0; i < rowCount() - 1; i++) {
- if (name == items[i].name)
- return i;
- }
- return -1;
-}
-
-int FilterModelBase::findInsertionIndex(const QString &name)
-{
- // Find insertion position. Note: we search only up to the last
- // item, because the last item is the "Show Empty Tags" item.
- // N.B: We might do a binary search using std::lower_bound()
- int i;
- for (i = 0; i < rowCount() - 1; i++) {
- if (name < items[i].name)
- return i;
- }
- return i;
-}
-
-void FilterModelBase::addItem(const QString &name, bool checked, int count)
-{
- int idx = findInsertionIndex(name);
- beginInsertRows(QModelIndex(), idx, idx);
- items.insert(items.begin() + idx, { name, checked, count });
- endInsertRows();
-}
-
-void FilterModelBase::changeName(const QString &oldName, const QString &newName)
-{
- if (oldName.isEmpty() || newName.isEmpty() || oldName == newName)
- return;
- int oldIndex = indexOf(oldName);
- if (oldIndex < 0)
- return;
- int newIndex = indexOf(newName);
-
- if (newIndex >= 0) {
- // If there was already an entry with the new name, we are merging entries.
- // Thus, if the old entry was selected, also select the new entry.
- if (items[oldIndex].checked && !items[newIndex].checked) {
- items[newIndex].checked = true;
- dataChanged(createIndex(newIndex, 0), createIndex(newIndex, 0));
- }
- // Now, delete the old item
- beginRemoveRows(QModelIndex(), oldIndex, oldIndex);
- items.erase(items.begin() + oldIndex);
- endRemoveRows();
- } else {
- // There was no entry of the same name. We might have to move the item.
- newIndex = findInsertionIndex(newName);
- if (oldIndex != newIndex && oldIndex + 1 != newIndex) {
- beginMoveRows(QModelIndex(), oldIndex, oldIndex, QModelIndex(), newIndex);
- moveInVector(items, oldIndex, oldIndex + 1, newIndex);
- endMoveRows();
- }
-
- // The item was moved, but the name still has to be modified
- items[newIndex].name = newName;
- dataChanged(createIndex(newIndex, 0), createIndex(newIndex, 0));
- }
-}
-
-// Update the the items array.
-// The last item is supposed to be the "Show Empty Tags" entry.
-// All other items will be sorted alphabetically. Attention: the passed-in list is modified!
-void FilterModelBase::updateList(QStringList &newList)
-{
- // Sort list, but leave out last element by using std::prev()
- if (!newList.empty())
- std::sort(newList.begin(), std::prev(newList.end()));
-
- beginResetModel();
-
- // Keep copy of the old items array to reimport the checked state later.
- // Note that by using std::move(), this is an essentially free operation:
- // The data is moved from the old array to the new one and the old array
- // is reset to zero size.
- std::vector<Item> oldItems = std::move(items);
-
- // Resize the cleared array to the new size. This leaves the checked
- // flag in an undefined state (since we didn't define a default constructor).
- items.resize(newList.count());
-
- // First, reset all checked states to false and set the names
- anyChecked = false;
- for (int i = 0; i < rowCount(); ++i) {
- items[i].name = newList[i];
- items[i].checked = false;
- }
-
- // Then, restore the checked state. Ignore the last item, since
- // this is the "Show Empty Tags" entry.
- for (int i = 0; i < (int)oldItems.size() - 1; i++) {
- if (oldItems[i].checked) {
- int ind = newList.indexOf(oldItems[i].name);
- if (ind >= 0 && ind < newList.count() - 1) {
- items[ind].checked = true;
- anyChecked = true;
- }
- }
- }
-
- // Reset the state of the "Show Empty Tags" entry. But be careful:
- // on program startup, the old list is empty.
- if (!oldItems.empty() && !items.empty() && oldItems.back().checked) {
- items.back().checked = true;
- anyChecked = true;
- }
-
- // Finally, calculate and cache the counts. Ignore the last item, since
- // this is the "Show Empty Tags" entry.
- for (int i = 0; i < (int)newList.size() - 1; i++)
- items[i].count = countDives(qPrintable(newList[i]));
-
- // Calculate count of "Empty Tags".
- if (!items.empty())
- items.back().count = countDives("");
-
- endResetModel();
-}
-
-// Decrease count of entry with given name. Remove if count reaches zero.
-// Exception: Don't remove the "Show Empty Tags" entry.
-void FilterModelBase::decreaseCount(const QString &name)
-{
- if (name.isEmpty()) {
- // Decrease the "Show Empty Tags" entry. Keep it even if count reaches 0.
- if (items.empty() || items.back().count <= 0)
- return; // Shouldn't happen!
- --items.back().count;
- int idx = items.size() - 1;
- dataChanged(createIndex(idx, 0), createIndex(idx, 0));
- return;
- }
-
- int idx = indexOf(name);
- if (idx < 0 || items[idx].count <= 0)
- return; // Shouldn't happen
-
- if(--items[idx].count == 0) {
- beginRemoveRows(QModelIndex(), idx, idx);
- items.erase(items.begin() + idx);
- endRemoveRows();
- } else {
- dataChanged(createIndex(idx, 0), createIndex(idx, 0));
- }
-}
-
-// Increase count of entry with given name. If entry doesn't yet exist, add it.
-void FilterModelBase::increaseCount(const QString &name)
-{
- if (name.isEmpty()) {
- // Increase the "Show Empty Tags" entry. Keep it even if count reaches 0.
- if (items.empty())
- return; // Shouldn't happen!
- ++items.back().count;
- int idx = items.size() - 1;
- dataChanged(createIndex(idx, 0), createIndex(idx, 0));
- return;
- }
-
- int idx = indexOf(name);
- if (idx < 0) {
- idx = findInsertionIndex(name);
- beginInsertRows(QModelIndex(), idx, idx);
- items.insert(items.begin() + idx, { name, anyChecked, 1 });
- endInsertRows();
- } else {
- ++items[idx].count;
- dataChanged(createIndex(idx, 0), createIndex(idx, 0));
- }
-}
-
-Qt::ItemFlags FilterModelBase::flags(const QModelIndex &index) const
-{
- return QAbstractListModel::flags(index) | Qt::ItemIsUserCheckable;
-}
-
-bool FilterModelBase::setData(const QModelIndex &index, const QVariant &value, int role)
-{
- if (role == Qt::CheckStateRole) {
- items[index.row()].checked = value.toBool();
- anyChecked = false;
- for (const Item &item: items) {
- if (item.checked) {
- anyChecked = true;
- break;
- }
- }
- dataChanged(index, index, { role });
- return true;
- }
- return false;
-}
-
-int FilterModelBase::rowCount(const QModelIndex &) const
-{
- return items.size();
-}
-
-QVariant FilterModelBase::data(const QModelIndex &index, int role) const
-{
- if (role == Qt::CheckStateRole) {
- return items[index.row()].checked ? Qt::Checked : Qt::Unchecked;
- } else if (role == Qt::DisplayRole) {
- const Item &item = items[index.row()];
- return QStringLiteral("%1 (%2)").arg(item.name, QString::number(item.count));
- }
- return QVariant();
-}
-
-void FilterModelBase::clearFilter()
-{
- for (Item &item: items)
- item.checked = false;
- anyChecked = false;
- emit dataChanged(createIndex(0, 0), createIndex(rowCount() - 1, 0));
-}
-
-void FilterModelBase::selectAll()
-{
- for (Item &item: items)
- item.checked = true;
- anyChecked = true;
- emit dataChanged(createIndex(0, 0), createIndex(rowCount() - 1, 0));
-}
-
-void FilterModelBase::invertSelection()
-{
- for (Item &item: items)
- item.checked = !item.checked;
- anyChecked = std::any_of(items.begin(), items.end(), [](Item &item) { return !!item.checked; });
- emit dataChanged(createIndex(0, 0), createIndex(rowCount() - 1, 0));
-}
-
-void FilterModelBase::setNegate(bool negateParam)
-{
- negate = negateParam;
- emit dataChanged(createIndex(0, 0), createIndex(rowCount() - 1, 0));
-}
-
-SuitsFilterModel::SuitsFilterModel(QObject *parent) : FilterModelBase(parent)
-{
-}
-
-int SuitsFilterModel::countDives(const char *s) const
-{
- return count_dives_with_suit(s);
-}
-
-bool SuitsFilterModel::doFilter(const dive *d) const
-{
- // rowCount() == 0 should never happen, because we have the "no suits" row
- // let's handle it gracefully anyway.
- if (!anyChecked || rowCount() == 0)
- return true;
-
- // Checked means 'Show', Unchecked means 'Hide'.
- QString suit(d->suit);
- // only show empty suit dives if the user checked that.
- if (suit.isEmpty())
- return items[rowCount() - 1].checked != negate;
-
- // there is a suit selected
- // Ignore last item, since this is the "Show Empty Tags" entry
- for (int i = 0; i < rowCount() - 1; i++) {
- if (items[i].checked && suit == items[i].name)
- return !negate;
- }
- return negate;
-}
-
-void SuitsFilterModel::diveAdded(const dive *d)
-{
- increaseCount(QString(d->suit));
-}
-
-void SuitsFilterModel::diveDeleted(const dive *d)
-{
- decreaseCount(QString(d->suit));
-}
-
-void SuitsFilterModel::repopulate()
-{
- QStringList list;
- struct dive *dive;
- int i = 0;
- for_each_dive (i, dive) {
- QString suit(dive->suit);
- if (!suit.isEmpty() && !list.contains(suit)) {
- list.append(suit);
- }
- }
- list << tr("No suit set");
- updateList(list);
-}
-
-TagFilterModel::TagFilterModel(QObject *parent) : FilterModelBase(parent)
-{
-}
-
-int TagFilterModel::countDives(const char *s) const
-{
- return count_dives_with_tag(s);
-}
-
-void TagFilterModel::repopulate()
-{
- if (g_tag_list == NULL)
- return;
- QStringList list;
- struct tag_entry *current_tag_entry = g_tag_list;
- while (current_tag_entry != NULL) {
- if (count_dives_with_tag(current_tag_entry->tag->name) > 0)
- list.append(QString(current_tag_entry->tag->name));
- current_tag_entry = current_tag_entry->next;
- }
- list << tr("Empty tags");
- updateList(list);
-}
-
-bool TagFilterModel::doFilter(const dive *d) const
-{
- // If there's nothing checked, this should show everything
- // rowCount() == 0 should never happen, because we have the "no tags" row
- // let's handle it gracefully anyway.
- if (!anyChecked || rowCount() == 0)
- return true;
-
- // Checked means 'Show', Unchecked means 'Hide'.
- struct tag_entry *head = d->tag_list;
-
- if (!head) // last tag means "Show empty tags";
- return items[rowCount() - 1].checked != negate;
-
- // have at least one tag.
- while (head) {
- QString tagName(head->tag->name);
- int index = indexOf(tagName);
- if (index >= 0 && items[index].checked)
- return !negate;
- head = head->next;
- }
- return negate;
-}
-
-void TagFilterModel::diveAdded(const dive *d)
-{
- struct tag_entry *head = d->tag_list;
- if (!head) {
- increaseCount(QString());
- return;
- }
- while (head) {
- increaseCount(QString());
- increaseCount(QString(head->tag->name));
- head = head->next;
- }
-}
-
-void TagFilterModel::diveDeleted(const dive *d)
-{
- struct tag_entry *head = d->tag_list;
- if (!head) {
- decreaseCount(QString());
- return;
- }
- while (head) {
- decreaseCount(QString(head->tag->name));
- head = head->next;
- }
-}
-
-BuddyFilterModel::BuddyFilterModel(QObject *parent) : FilterModelBase(parent)
-{
-}
-
-int BuddyFilterModel::countDives(const char *s) const
-{
- return count_dives_with_person(s);
-}
-
-static QStringList getDiveBuddies(const dive *d)
-{
- QString persons = QString(d->buddy) + "," + QString(d->divemaster);
- QStringList personsList = persons.split(',', QString::SkipEmptyParts);
- for (QString &s: personsList)
- s = s.trimmed();
- return personsList;
-}
-
-bool BuddyFilterModel::doFilter(const dive *d) const
-{
- // If there's nothing checked, this should show everything
- // rowCount() == 0 should never happen, because we have the "no tags" row
- // let's handle it gracefully anyway.
- if (!anyChecked || rowCount() == 0)
- return true;
-
- QStringList personsList = getDiveBuddies(d);
- // only show empty buddie dives if the user checked that.
- if (personsList.isEmpty())
- return items[rowCount() - 1].checked != negate;
-
- // have at least one buddy
- // Ignore last item, since this is the "Show Empty Tags" entry
- for (int i = 0; i < rowCount() - 1; i++) {
- if (items[i].checked && personsList.contains(items[i].name, Qt::CaseInsensitive))
- return !negate;
- }
- return negate;
-}
-
-void BuddyFilterModel::diveAdded(const dive *d)
-{
- QStringList buddies = getDiveBuddies(d);
- if (buddies.empty()) {
- increaseCount(QString());
- return;
- }
- for(const QString &buddy: buddies)
- increaseCount(buddy);
-}
-
-void BuddyFilterModel::diveDeleted(const dive *d)
+MultiFilterSortModel *MultiFilterSortModel::instance()
{
- QStringList buddies = getDiveBuddies(d);
- if (buddies.empty()) {
- decreaseCount(QString());
- return;
- }
- for(const QString &buddy: buddies)
- decreaseCount(buddy);
-}
-
-void BuddyFilterModel::repopulate()
-{
- QStringList list;
- struct dive *dive;
- int i = 0;
- for_each_dive (i, dive) {
- QString persons = QString(dive->buddy) + "," + QString(dive->divemaster);
- Q_FOREACH (const QString &person, persons.split(',', QString::SkipEmptyParts)) {
- // Remove any leading spaces
- if (!list.contains(person.trimmed())) {
- list.append(person.trimmed());
- }
- }
- }
- list << tr("No buddies");
- updateList(list);
-}
-
-LocationFilterModel::LocationFilterModel(QObject *parent) : FilterModelBase(parent)
-{
-}
-
-int LocationFilterModel::countDives(const char *s) const
-{
- return count_dives_with_location(s);
-}
-
-bool LocationFilterModel::doFilter(const dive *d) const
-{
- // rowCount() == 0 should never happen, because we have the "no location" row
- // let's handle it gracefully anyway.
- if (!anyChecked || rowCount() == 0)
- return true;
-
- // Checked means 'Show', Unchecked means 'Hide'.
- QString location(get_dive_location(d));
- // only show empty location dives if the user checked that.
- if (location.isEmpty())
- return items[rowCount() - 1].checked != negate;
-
- // There is a location selected
- // Ignore last item, since this is the "Show Empty Tags" entry
- for (int i = 0; i < rowCount() - 1; i++) {
- if (items[i].checked && location == items[i].name)
- return !negate;
- }
- return negate;
-}
-
-void LocationFilterModel::diveAdded(const dive *d)
-{
- increaseCount(get_dive_location(d));
-}
-
-void LocationFilterModel::diveDeleted(const dive *d)
-{
- decreaseCount(get_dive_location(d));
-}
-
-void LocationFilterModel::repopulate()
-{
- QStringList list;
- struct dive *dive;
- int i = 0;
- for_each_dive (i, dive) {
- QString location(get_dive_location(dive));
- if (!location.isEmpty() && !list.contains(location)) {
- list.append(location);
- }
- }
- list << tr("No location set");
- updateList(list);
-}
-
-void LocationFilterModel::addName(const QString &newName)
-{
- if (newName.isEmpty() || indexOf(newName) >= 0)
- return;
- int count = countDives(qPrintable(newName));
- // If any old item was checked, also check the new one so that
- // dives with the added dive site are shown.
- addItem(newName, anyChecked, count);
+ static MultiFilterSortModel self;
+ return &self;
}
MultiFilterSortModel::MultiFilterSortModel(QObject *parent) : QSortFilterProxyModel(parent),
@@ -574,28 +37,6 @@ void MultiFilterSortModel::setLayout(DiveTripModel::Layout layout)
tripModel->setLayout(layout); // Note: setLayout() resets the whole model
}
-void MultiFilterSortModel::divesAdded(const QVector<dive *> &dives)
-{
- // TODO: We call diveAdded for every dive and model.
- // If multiple dives are added (e.g. import dive) this will lead to a large
- // number of model changes and might be a pessimization compared to a full
- // model reload. Instead, the models should take the vector, calculate the
- // new fields and add them at once.
- for (FilterModelBase *model: models) {
- for (const dive *d: dives)
- model->diveAdded(d);
- }
-}
-
-void MultiFilterSortModel::divesDeleted(const QVector<dive *> &dives)
-{
- // TODO: See comment for divesDeleted
- for (FilterModelBase *model: models) {
- for (const dive *d: dives)
- model->diveDeleted(d);
- }
-}
-
bool MultiFilterSortModel::showDive(const struct dive *d) const
{
if (curr_dive_site) {
@@ -605,14 +46,6 @@ bool MultiFilterSortModel::showDive(const struct dive *d) const
return ds == curr_dive_site || same_string(ds->name, curr_dive_site->name);
}
- if (models.isEmpty())
- return true;
-
- for (const FilterModelBase *model: models) {
- if (!model->doFilter(d))
- return false;
- }
-
return true;
}
@@ -674,23 +107,8 @@ void MultiFilterSortModel::myInvalidate()
#endif
}
-void MultiFilterSortModel::addFilterModel(FilterModelBase *model)
-{
- models.append(model);
- connect(model, &FilterModelBase::dataChanged, this, &MultiFilterSortModel::filterChanged);
-}
-
-void MultiFilterSortModel::removeFilterModel(FilterModelBase *model)
-{
- models.removeAll(model);
- disconnect(model, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(myInvalidate()));
-}
-
void MultiFilterSortModel::clearFilter()
{
- Q_FOREACH (FilterModelBase *iface, models) {
- iface->clearFilter();
- }
myInvalidate();
}
diff --git a/qt-models/filtermodels.h b/qt-models/filtermodels.h
index 6956146e1..61b82be33 100644
--- a/qt-models/filtermodels.h
+++ b/qt-models/filtermodels.h
@@ -34,119 +34,11 @@ struct FilterData {
bool invertFilter;
};
-class FilterModelBase : public QAbstractListModel {
- Q_OBJECT
-private:
- int findInsertionIndex(const QString &name);
-protected:
- struct Item {
- QString name;
- bool checked;
- int count;
- };
- std::vector<Item> items;
- int indexOf(const QString &name) const;
- void addItem(const QString &name, bool checked, int count);
- int rowCount(const QModelIndex &parent = QModelIndex()) const override;
- void decreaseCount(const QString &d);
- void increaseCount(const QString &d);
-public:
- virtual bool doFilter(const dive *d) const = 0;
- virtual void diveAdded(const dive *d) = 0;
- virtual void diveDeleted(const dive *d) = 0;
- void clearFilter();
- void selectAll();
- void invertSelection();
- bool anyChecked;
- bool negate;
-public
-slots:
- void setNegate(bool negate);
- void changeName(const QString &oldName, const QString &newName);
-protected:
- explicit FilterModelBase(QObject *parent = 0);
- void updateList(QStringList &new_list);
- virtual int countDives(const char *) const = 0;
-private:
- Qt::ItemFlags flags(const QModelIndex &index) const;
- QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const;
- bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole);
-};
-
-class TagFilterModel : public FilterModelBase {
- Q_OBJECT
-public:
- static TagFilterModel *instance();
- bool doFilter(const dive *d) const;
-public
-slots:
- void repopulate();
-
-private:
- explicit TagFilterModel(QObject *parent = 0);
- int countDives(const char *) const;
- void diveAdded(const dive *d);
- void diveDeleted(const dive *d);
-};
-
-class BuddyFilterModel : public FilterModelBase {
- Q_OBJECT
-public:
- static BuddyFilterModel *instance();
- bool doFilter(const dive *d) const;
-public
-slots:
- void repopulate();
-
-private:
- explicit BuddyFilterModel(QObject *parent = 0);
- int countDives(const char *) const;
- void diveAdded(const dive *d);
- void diveDeleted(const dive *d);
-};
-
-class LocationFilterModel : public FilterModelBase {
- Q_OBJECT
-public:
- static LocationFilterModel *instance();
- bool doFilter(const dive *d) const;
-public
-slots:
- void repopulate();
- void addName(const QString &newName);
-
-private:
- explicit LocationFilterModel(QObject *parent = 0);
- int countDives(const char *) const;
- void diveAdded(const dive *d);
- void diveDeleted(const dive *d);
-};
-
-class SuitsFilterModel : public FilterModelBase {
- Q_OBJECT
-public:
- static SuitsFilterModel *instance();
- bool doFilter(const dive *d) const;
-public
-slots:
- void repopulate();
-
-private:
- explicit SuitsFilterModel(QObject *parent = 0);
- int countDives(const char *) const;
- void diveAdded(const dive *d);
- void diveDeleted(const dive *d);
-};
-
class MultiFilterSortModel : public QSortFilterProxyModel {
Q_OBJECT
public:
static MultiFilterSortModel *instance();
bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const override;
- void addFilterModel(FilterModelBase *model);
- void removeFilterModel(FilterModelBase *model);
- void divesAdded(const QVector<dive *> &dives);
- void divesDeleted(const QVector<dive *> &dives);
bool showDive(const struct dive *d) const;
int divesDisplayed;
bool lessThan(const QModelIndex &, const QModelIndex &) const override;
@@ -165,7 +57,6 @@ signals:
private:
MultiFilterSortModel(QObject *parent = 0);
- QList<FilterModelBase *> models;
struct dive_site *curr_dive_site;
DiveTripModel *model;
FilterData filterData;