summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2019-01-19 09:59:33 +0100
committerGravatar Dirk Hohndel <dirk@hohndel.org>2019-01-19 13:59:24 -0800
commit8fa156eb0d2364f9dc17ac80805408f62343799d (patch)
tree3dfd58e7f768a025a9fbe0fca530678c01008c61
parentf67f07223f93d18353ff7e0621af6b05269c33be (diff)
downloadsubsurface-8fa156eb0d2364f9dc17ac80805408f62343799d.tar.gz
Filter: make tags, people and location filter logically-and
If the user provides multiple tags, they probably want to search for dive with *all* of these tags. Replace the convoluted loops by std::all_of(). This makes it trivial to change logically-and to logically-or: Replace std::all_of() by std::any_of(). Reported-by: Jan Mulder <jlmulder@xs4all.nl> Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
-rw-r--r--qt-models/filtermodels.cpp85
1 files changed, 38 insertions, 47 deletions
diff --git a/qt-models/filtermodels.cpp b/qt-models/filtermodels.cpp
index 6e983c835..2f3ce99a8 100644
--- a/qt-models/filtermodels.cpp
+++ b/qt-models/filtermodels.cpp
@@ -16,57 +16,48 @@
#include <algorithm>
namespace {
- bool hasTag(const QStringList tags, const struct dive *d)
+ // Check if a string-list contains at least one string containing the second argument.
+ // Comparison is non case sensitive and removes white space.
+ bool listContainsSuperstring(const QStringList &list, const QString &s)
{
- if (!tags.isEmpty()) {
- auto dive_tags = get_taglist_string(d->tag_list).split(",");
- bool found_tag = false;
- for (const auto& filter_tag : tags)
- for (const auto& dive_tag : dive_tags)
- if (dive_tag.trimmed().toUpper().contains(filter_tag.trimmed().toUpper()))
- found_tag = true;
-
- return found_tag;
- }
- return true;
+ return std::any_of(list.begin(), list.end(), [&s](const QString &s2)
+ { return s2.trimmed().contains(s.trimmed(), Qt::CaseInsensitive); });
}
- bool hasPerson(const QStringList people, const struct dive *d)
+ bool hasTags(const QStringList &tags, const struct dive *d)
{
- if (!people.isEmpty()) {
- QStringList dive_people = QString(d->buddy).split(",", QString::SkipEmptyParts)
- + QString(d->divemaster).split(",", QString::SkipEmptyParts);
-
- bool found_person = false;
- for(const auto& filter_person : people)
- for(const auto& dive_person : dive_people)
- if (dive_person.trimmed().toUpper().contains(filter_person.trimmed().toUpper()))
- found_person = true;
-
- return found_person;
- }
- return true;
+ if (tags.isEmpty())
+ return true;
+ QStringList dive_tags = get_taglist_string(d->tag_list).split(",");
+
+ return std::all_of(tags.begin(), tags.end(), [&dive_tags](const QString &tag)
+ { return listContainsSuperstring(dive_tags, tag); });
}
- bool hasLocation(const QStringList locations, const struct dive *d)
+ bool hasPersons(const QStringList &people, const struct dive *d)
{
- if (!locations.isEmpty()) {
- QStringList diveLocations;
- if (d->divetrip)
- diveLocations.push_back(QString(d->divetrip->location));
-
- if (d->dive_site)
- diveLocations.push_back(QString(d->dive_site->name));
-
- bool found_location = false;
- for (const auto& filter_location : locations)
- for (const auto& dive_location : diveLocations)
- if (dive_location.trimmed().toUpper().contains(filter_location.trimmed().toUpper()))
- found_location = true;
-
- return found_location;
- }
- return true;
+ if (people.isEmpty())
+ return true;
+ QStringList dive_people = QString(d->buddy).split(",", QString::SkipEmptyParts)
+ + QString(d->divemaster).split(",", QString::SkipEmptyParts);
+
+ return std::all_of(people.begin(), people.end(), [&dive_people](const QString &person)
+ { return listContainsSuperstring(dive_people, person); });
+ }
+
+ bool hasLocations(const QStringList &locations, const struct dive *d)
+ {
+ if (locations.isEmpty())
+ return true;
+ QStringList diveLocations;
+ if (d->divetrip)
+ diveLocations.push_back(QString(d->divetrip->location));
+
+ if (d->dive_site)
+ diveLocations.push_back(QString(d->dive_site->name));
+
+ return std::all_of(locations.begin(), locations.end(), [&diveLocations](const QString &location)
+ { return listContainsSuperstring(diveLocations, location); });
}
// TODO: Finish this iimplementation.
@@ -130,15 +121,15 @@ bool MultiFilterSortModel::showDive(const struct dive *d) const
return false;
// tags.
- if (!hasTag(filterData.tags, d))
+ if (!hasTags(filterData.tags, d))
return false;
// people
- if (!hasPerson(filterData.people, d))
+ if (!hasPersons(filterData.people, d))
return false;
// Location
- if (!hasLocation(filterData.location, d))
+ if (!hasLocations(filterData.location, d))
return false;
if (!hasEquipment(filterData.equipment, d))