aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2020-06-28 15:24:19 +0200
committerGravatar Dirk Hohndel <dirk@hohndel.org>2020-09-29 16:13:03 -0700
commitf9721fce4b12faee87398d23fd2f8b4d1b9a0309 (patch)
tree0ca37231d70028eb2fdaf615a812af8c9983b1cf
parent631be569fe5b277c4c1729019305e06b02cec611 (diff)
downloadsubsurface-f9721fce4b12faee87398d23fd2f8b4d1b9a0309.tar.gz
filter: implement importing of filter presets
When importing a divelog, import filter presets. If there are equal names, import only if the presets differ. In that case, disambiguate the name. This made things a bit more complicated, as comparison of filter presets had to be implemented. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
-rw-r--r--commands/command.cpp5
-rw-r--r--commands/command.h4
-rw-r--r--commands/command_divelist.cpp38
-rw-r--r--commands/command_divelist.h11
-rw-r--r--core/divefilter.cpp6
-rw-r--r--core/divefilter.h1
-rw-r--r--core/filterconstraint.cpp16
-rw-r--r--core/filterconstraint.h1
-rw-r--r--core/filterpreset.cpp3
-rw-r--r--core/fulltext.cpp1
-rw-r--r--desktop-widgets/divelogimportdialog.cpp2
-rw-r--r--desktop-widgets/mainwindow.cpp2
-rw-r--r--desktop-widgets/subsurfacewebservices.cpp2
-rw-r--r--qt-models/diveimportedmodel.cpp2
14 files changed, 81 insertions, 13 deletions
diff --git a/commands/command.cpp b/commands/command.cpp
index f04d5cc7a..eaf84bab3 100644
--- a/commands/command.cpp
+++ b/commands/command.cpp
@@ -17,9 +17,10 @@ void addDive(dive *d, bool autogroup, bool newNumber)
execute(new AddDive(d, autogroup, newNumber));
}
-void importDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites, int flags, const QString &source)
+void importDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites,
+ filter_preset_table_t *presets, int flags, const QString &source)
{
- execute(new ImportDives(dives, trips, sites, flags, source));
+ execute(new ImportDives(dives, trips, sites, presets, flags, source));
}
void deleteDive(const QVector<struct dive*> &divesToDelete)
diff --git a/commands/command.h b/commands/command.h
index 3cdec6d4e..8936355e5 100644
--- a/commands/command.h
+++ b/commands/command.h
@@ -3,6 +3,7 @@
#define COMMAND_H
#include "core/dive.h"
+#include "core/filterpreset.h"
#include "core/pictureobj.h"
#include <QVector>
#include <QAction>
@@ -32,7 +33,8 @@ QString changesMade(); // return a string with the texts from all commands on
// insertion position.
void addDive(dive *d, bool autogroup, bool newNumber);
void importDives(struct dive_table *dives, struct trip_table *trips,
- struct dive_site_table *sites, int flags, const QString &source); // The tables are consumed!
+ struct dive_site_table *sites, filter_preset_table_t *filter_presets,
+ int flags, const QString &source); // The tables are consumed!
void deleteDive(const QVector<struct dive*> &divesToDelete);
void shiftTime(const std::vector<dive *> &changedDives, int amount);
void renumberDives(const QVector<QPair<dive *, int>> &divesToRenumber);
diff --git a/commands/command_divelist.cpp b/commands/command_divelist.cpp
index 6d81482e0..39232b731 100644
--- a/commands/command_divelist.cpp
+++ b/commands/command_divelist.cpp
@@ -469,7 +469,8 @@ void AddDive::undoit()
setSelection(selection, currentDive);
}
-ImportDives::ImportDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites, int flags, const QString &source)
+ImportDives::ImportDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites,
+ filter_preset_table_t *filter_presets, int flags, const QString &source)
{
setText(Command::Base::tr("import %n dive(s) from %1", "", dives->nr).arg(source));
@@ -509,6 +510,22 @@ ImportDives::ImportDives(struct dive_table *dives, struct trip_table *trips, str
divesAndSitesToRemove.dives.reserve(dives_to_remove.nr);
for (int i = 0; i < dives_to_remove.nr; ++i)
divesAndSitesToRemove.dives.push_back(dives_to_remove.dives[i]);
+
+ // When encountering filter presets with equal names, check whether they are
+ // the same. If they are, ignore them.
+ if (filter_presets) {
+ for (const filter_preset &preset: *filter_presets) {
+ QString name = preset.name;
+ auto it = std::find_if(filter_preset_table.begin(), filter_preset_table.end(),
+ [&name](const filter_preset &preset) { return preset.name == name; });
+ if (it != filter_preset_table.end() && it->data == preset.data)
+ continue;
+ filterPresetsToAdd.emplace_back(preset.name, preset.data);
+ }
+
+ // Consume the table for analogy with the other tables.
+ filter_presets->clear();
+ }
}
bool ImportDives::workToBeDone()
@@ -532,6 +549,13 @@ void ImportDives::redoit()
// Remember dives and sites to remove
divesAndSitesToRemove = std::move(divesAndSitesToRemoveNew);
+
+ // Add new filter presets
+ for (auto &it: filterPresetsToAdd) {
+ filterPresetsToRemove.push_back(filter_preset_add(it.first, it.second));
+ emit diveListNotifier.filterPresetAdded(filterPresetsToRemove.back());
+ }
+ filterPresetsToAdd.clear();
}
void ImportDives::undoit()
@@ -547,6 +571,18 @@ void ImportDives::undoit()
// ...and restore the selection
setSelection(selection, currentDive);
+
+ // Remove filter presets. Do this in reverse order.
+ for (auto it = filterPresetsToRemove.rbegin(); it != filterPresetsToRemove.rend(); ++it) {
+ int index = *it;
+ QString oldName = filter_preset_name_qstring(index);
+ FilterData oldData = filter_preset_get(index);
+ filter_preset_delete(index);
+ emit diveListNotifier.filterPresetRemoved(index);
+ filterPresetsToAdd.emplace_back(oldName, oldData);
+ }
+ filterPresetsToRemove.clear();
+ std::reverse(filterPresetsToAdd.begin(), filterPresetsToAdd.end());
}
DeleteDive::DeleteDive(const QVector<struct dive*> &divesToDeleteIn)
diff --git a/commands/command_divelist.h b/commands/command_divelist.h
index f1bbc7a91..9569da031 100644
--- a/commands/command_divelist.h
+++ b/commands/command_divelist.h
@@ -5,6 +5,7 @@
#define COMMAND_DIVELIST_H
#include "command_base.h"
+#include "core/filterpreset.h"
#include <QVector>
@@ -97,7 +98,8 @@ private:
class ImportDives : public DiveListBase {
public:
// Note: dives and trips are consumed - after the call they will be empty.
- ImportDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites, int flags, const QString &source);
+ ImportDives(struct dive_table *dives, struct trip_table *trips, struct dive_site_table *sites,
+ filter_preset_table_t *filter_presets, int flags, const QString &source);
private:
void undoit() override;
void redoit() override;
@@ -109,11 +111,14 @@ private:
// For redo
std::vector<OwningDiveSitePtr> sitesToAdd;
+ std::vector<std::pair<QString,FilterData>>
+ filterPresetsToAdd;
// For undo
std::vector<dive_site *> sitesToRemove;
- std::vector<dive *> selection;
- dive * currentDive;
+ std::vector<dive *> selection;
+ dive *currentDive;
+ std::vector<int> filterPresetsToRemove;
};
class DeleteDive : public DiveListBase {
diff --git a/core/divefilter.cpp b/core/divefilter.cpp
index 9f54f5cca..23d023b9d 100644
--- a/core/divefilter.cpp
+++ b/core/divefilter.cpp
@@ -22,6 +22,12 @@ static void updateDiveStatus(dive *d, bool newStatus, ShownChange &change)
}
}
+bool FilterData::operator==(const FilterData &f2) const
+{
+ return fullText.originalQuery == f2.fullText.originalQuery &&
+ fulltextStringMode == f2.fulltextStringMode &&
+ constraints == f2.constraints;
+}
bool FilterData::validFilter() const
{
diff --git a/core/divefilter.h b/core/divefilter.h
index 3ebb049ef..d8d28ab04 100644
--- a/core/divefilter.h
+++ b/core/divefilter.h
@@ -32,6 +32,7 @@ struct FilterData {
StringFilterMode fulltextStringMode = StringFilterMode::STARTSWITH;
std::vector<filter_constraint> constraints;
bool validFilter() const;
+ bool operator==(const FilterData &) const;
};
class DiveFilter {
diff --git a/core/filterconstraint.cpp b/core/filterconstraint.cpp
index 4bddc7b53..fb8b49e8c 100644
--- a/core/filterconstraint.cpp
+++ b/core/filterconstraint.cpp
@@ -602,6 +602,22 @@ filter_constraint &filter_constraint::operator=(const filter_constraint &c)
return *this;
}
+bool filter_constraint::operator==(const filter_constraint &f2) const
+{
+ if (type != f2.type || string_mode != f2.string_mode || range_mode != f2.range_mode || negate != f2.negate)
+ return false;
+ if (filter_constraint_is_timestamp(type))
+ return data.timestamp_range.from == f2.data.timestamp_range.from &&
+ data.timestamp_range.to == f2.data.timestamp_range.to;
+ else if (filter_constraint_is_string(type))
+ return *data.string_list == *(f2.data.string_list);
+ else if (filter_constraint_is_multiple_choice(type))
+ return data.multiple_choice == f2.data.multiple_choice;
+ else
+ return data.numerical_range.from == f2.data.numerical_range.from &&
+ data.numerical_range.to == f2.data.numerical_range.to;
+}
+
filter_constraint::~filter_constraint()
{
if (filter_constraint_is_string(type))
diff --git a/core/filterconstraint.h b/core/filterconstraint.h
index d6596d09d..b3bd72354 100644
--- a/core/filterconstraint.h
+++ b/core/filterconstraint.h
@@ -86,6 +86,7 @@ struct filter_constraint {
filter_constraint(const filter_constraint &);
filter_constraint &operator=(const filter_constraint &);
~filter_constraint();
+ bool operator==(const filter_constraint &f2) const;
#endif
};
diff --git a/core/filterpreset.cpp b/core/filterpreset.cpp
index b21306502..fa48a9fd2 100644
--- a/core/filterpreset.cpp
+++ b/core/filterpreset.cpp
@@ -133,8 +133,9 @@ FilterData filter_preset_get(int preset)
return filter_preset_table[preset].data;
}
-int filter_preset_add(const QString &name, const FilterData &d)
+int filter_preset_add(const QString &nameIn, const FilterData &d)
{
+ QString name = get_unique_preset_name(nameIn, filter_preset_table);
return filter_preset_add_to_table(name, d, filter_preset_table);
}
diff --git a/core/fulltext.cpp b/core/fulltext.cpp
index 38afb0798..acebc36ba 100644
--- a/core/fulltext.cpp
+++ b/core/fulltext.cpp
@@ -18,7 +18,6 @@ struct full_text_cache {
class FullText {
std::map<QString, std::vector<dive *>> words; // Dives that belong to each word
public:
-
void populate(); // Rebuild from current dive_table
void registerDive(struct dive *d); // Note: can be called repeatedly
void unregisterDive(struct dive *d); // Note: can be called repeatedly
diff --git a/desktop-widgets/divelogimportdialog.cpp b/desktop-widgets/divelogimportdialog.cpp
index f3d512009..d613a810a 100644
--- a/desktop-widgets/divelogimportdialog.cpp
+++ b/desktop-widgets/divelogimportdialog.cpp
@@ -1023,7 +1023,7 @@ void DiveLogImportDialog::on_buttonBox_accepted()
}
QString source = fileNames.size() == 1 ? fileNames[0] : tr("multiple files");
- Command::importDives(&table, &trips, &sites, IMPORT_MERGE_ALL_TRIPS, source);
+ Command::importDives(&table, &trips, &sites, nullptr, IMPORT_MERGE_ALL_TRIPS, source);
}
TagDragDelegate::TagDragDelegate(QObject *parent) : QStyledItemDelegate(parent)
diff --git a/desktop-widgets/mainwindow.cpp b/desktop-widgets/mainwindow.cpp
index 1683a4122..91395d077 100644
--- a/desktop-widgets/mainwindow.cpp
+++ b/desktop-widgets/mainwindow.cpp
@@ -1556,7 +1556,7 @@ void MainWindow::importFiles(const QStringList fileNames)
parse_file(fileNamePtr.data(), &table, &trips, &sites, &filter_presets);
}
QString source = fileNames.size() == 1 ? fileNames[0] : tr("multiple files");
- Command::importDives(&table, &trips, &sites, IMPORT_MERGE_ALL_TRIPS, source);
+ Command::importDives(&table, &trips, &sites, &filter_presets, IMPORT_MERGE_ALL_TRIPS, source);
}
void MainWindow::loadFiles(const QStringList fileNames)
diff --git a/desktop-widgets/subsurfacewebservices.cpp b/desktop-widgets/subsurfacewebservices.cpp
index fa1b003da..27e80b31c 100644
--- a/desktop-widgets/subsurfacewebservices.cpp
+++ b/desktop-widgets/subsurfacewebservices.cpp
@@ -458,7 +458,7 @@ void DivelogsDeWebServices::buttonClicked(QAbstractButton *button)
struct dive_site_table sites = empty_dive_site_table;
filter_preset_table_t filter_presets;
parse_file(QFile::encodeName(zipFile.fileName()), &table, &trips, &sites, &filter_presets);
- Command::importDives(&table, &trips, &sites, IMPORT_MERGE_ALL_TRIPS, QStringLiteral("divelogs.de"));
+ Command::importDives(&table, &trips, &sites, nullptr, IMPORT_MERGE_ALL_TRIPS, QStringLiteral("divelogs.de"));
/* store last entered user/pass in config */
qPrefCloudStorage::set_divelogde_user(ui.userID->text());
diff --git a/qt-models/diveimportedmodel.cpp b/qt-models/diveimportedmodel.cpp
index 902d2099a..3413e03a5 100644
--- a/qt-models/diveimportedmodel.cpp
+++ b/qt-models/diveimportedmodel.cpp
@@ -194,7 +194,7 @@ void DiveImportedModel::recordDives(int flags)
std::pair<struct dive_table, struct dive_site_table> tables = consumeTables();
if (tables.first.nr > 0) {
auto data = thread.data();
- Command::importDives(&tables.first, nullptr, &tables.second, flags, data->devName());
+ Command::importDives(&tables.first, nullptr, &tables.second, nullptr, flags, data->devName());
} else {
clear_dive_site_table(&tables.second);
}