diff options
author | Berthold Stoeger <bstoeger@mail.tuwien.ac.at> | 2019-03-12 23:51:39 +0100 |
---|---|---|
committer | Dirk Hohndel <dirk@hohndel.org> | 2019-04-12 18:19:07 +0300 |
commit | 0e1b0cf1da697851b0db4f8b860da8ac3a509d17 (patch) | |
tree | eea365971c6f509121efcd0277f352c6c88a1f31 | |
parent | 8e1f736d2b608784e1ea942ec8579d2361691c43 (diff) | |
download | subsurface-0e1b0cf1da697851b0db4f8b860da8ac3a509d17.tar.gz |
Undo: Implement undo of dive site name editing
Implement an undo command that edits the name of a dive site.
Connect it to the dive site table, so that names can be edited
directly in the table.
Send signals on undo / redo so that the dive site table and
the dive site edit widget can be updated.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
-rw-r--r-- | core/subsurface-qt/DiveListNotifier.h | 1 | ||||
-rw-r--r-- | desktop-widgets/command.cpp | 5 | ||||
-rw-r--r-- | desktop-widgets/command.h | 1 | ||||
-rw-r--r-- | desktop-widgets/command_divesite.cpp | 27 | ||||
-rw-r--r-- | desktop-widgets/command_divesite.h | 12 | ||||
-rw-r--r-- | desktop-widgets/locationinformation.cpp | 15 | ||||
-rw-r--r-- | desktop-widgets/locationinformation.h | 1 | ||||
-rw-r--r-- | qt-models/divelocationmodel.cpp | 32 | ||||
-rw-r--r-- | qt-models/divelocationmodel.h | 5 |
9 files changed, 97 insertions, 2 deletions
diff --git a/core/subsurface-qt/DiveListNotifier.h b/core/subsurface-qt/DiveListNotifier.h index 41a964955..40343791f 100644 --- a/core/subsurface-qt/DiveListNotifier.h +++ b/core/subsurface-qt/DiveListNotifier.h @@ -48,6 +48,7 @@ signals: void diveSiteAdded(dive_site *ds, int idx); void diveSiteDeleted(dive_site *ds, int idx); void diveSiteDiveCountChanged(dive_site *ds); + void diveSiteChanged(dive_site *ds, int field); // field according to LocationInformationModel public: // Desktop uses the QTreeView class to present the list of dives. The layout // of this class gives us a very fundamental problem, as we can not easily diff --git a/desktop-widgets/command.cpp b/desktop-widgets/command.cpp index 80a233404..82e5ebe07 100644 --- a/desktop-widgets/command.cpp +++ b/desktop-widgets/command.cpp @@ -83,4 +83,9 @@ void deleteDiveSites(const QVector <dive_site *> &sites) execute(new DeleteDiveSites(sites)); } +void editDiveSiteName(dive_site *ds, const QString &value) +{ + execute(new EditDiveSiteName(ds, value)); +} + } // namespace Command diff --git a/desktop-widgets/command.h b/desktop-widgets/command.h index 77920a81c..09fb45526 100644 --- a/desktop-widgets/command.h +++ b/desktop-widgets/command.h @@ -41,6 +41,7 @@ void mergeDives(const QVector <dive *> &dives); // 3) Dive-site related commands void deleteDiveSites(const QVector <dive_site *> &sites); +void editDiveSiteName(dive_site *ds, const QString &value); } // namespace Command diff --git a/desktop-widgets/command_divesite.cpp b/desktop-widgets/command_divesite.cpp index d4c5d87b8..eafa18a76 100644 --- a/desktop-widgets/command_divesite.cpp +++ b/desktop-widgets/command_divesite.cpp @@ -3,6 +3,8 @@ #include "command_divesite.h" #include "core/divesite.h" #include "core/subsurface-qt/DiveListNotifier.h" +#include "core/qthelper.h" +#include "qt-models/divelocationmodel.h" namespace Command { @@ -80,4 +82,29 @@ void DeleteDiveSites::undo() sitesToRemove = std::move(addDiveSites(sitesToAdd)); } +EditDiveSiteName::EditDiveSiteName(dive_site *dsIn, const QString &name) : ds(dsIn), + value(name) +{ +} + +bool EditDiveSiteName::workToBeDone() +{ + return value != QString(ds->name); +} + +void EditDiveSiteName::redo() +{ + QString s = ds->name; + free(ds->name); + ds->name = copy_qstring(value); + value = s; + emit diveListNotifier.diveSiteChanged(ds, LocationInformationModel::NAME); // Inform frontend of changed dive site. +} + +void EditDiveSiteName::undo() +{ + // Undo and redo do the same + redo(); +} + } // namespace Command diff --git a/desktop-widgets/command_divesite.h b/desktop-widgets/command_divesite.h index 9c4f93058..692e5c109 100644 --- a/desktop-widgets/command_divesite.h +++ b/desktop-widgets/command_divesite.h @@ -16,14 +16,26 @@ public: DeleteDiveSites(const QVector<dive_site *> &sites); private: bool workToBeDone() override; + void undo() override; + void redo() override; // For redo std::vector<dive_site *> sitesToRemove; // For undo std::vector<OwningDiveSitePtr> sitesToAdd; +}; + +class EditDiveSiteName : public Base { +public: + EditDiveSiteName(dive_site *ds, const QString &name); +private: + bool workToBeDone() override; void undo() override; void redo() override; + + dive_site *ds; + QString value; // Value to be set }; } // namespace Command diff --git a/desktop-widgets/locationinformation.cpp b/desktop-widgets/locationinformation.cpp index ea52dd8c6..d827f5006 100644 --- a/desktop-widgets/locationinformation.cpp +++ b/desktop-widgets/locationinformation.cpp @@ -8,6 +8,7 @@ #include "qt-models/filtermodels.h" #include "core/divesitehelpers.h" #include "desktop-widgets/modeldelegates.h" +#include "core/subsurface-qt/DiveListNotifier.h" #include <QDebug> #include <QShowEvent> @@ -38,6 +39,8 @@ LocationInformationWidget::LocationInformationWidget(QWidget *parent) : QGroupBo connect(ui.diveSiteCoordinates, SIGNAL(returnPressed()), this, SLOT(updateLocationOnMap())); ui.diveSiteCoordinates->installEventFilter(this); + connect(&diveListNotifier, &DiveListNotifier::diveSiteChanged, this, &LocationInformationWidget::diveSiteChanged); + ui.diveSiteListView->setModel(&filter_model); ui.diveSiteListView->setModelColumn(LocationInformationModel::NAME); ui.diveSiteListView->installEventFilter(this); @@ -121,6 +124,18 @@ void LocationInformationWidget::updateLabels() ui.locationTags->setText(constructLocationTags(&taxonomy, false)); } +void LocationInformationWidget::diveSiteChanged(struct dive_site *ds, int field) +{ + if (diveSite != ds) + return; // A different dive site was changed -> do nothing. + switch (field) { + case LocationInformationModel::NAME: + ui.diveSiteName->setText(diveSite->name); + default: + return; + } +} + void LocationInformationWidget::clearLabels() { ui.diveSiteName->clear(); diff --git a/desktop-widgets/locationinformation.h b/desktop-widgets/locationinformation.h index c741f49f3..3b297c741 100644 --- a/desktop-widgets/locationinformation.h +++ b/desktop-widgets/locationinformation.h @@ -39,6 +39,7 @@ public slots: private slots: void updateLabels(); void updateLocationOnMap(); + void diveSiteChanged(struct dive_site *ds, int field); signals: void endEditDiveSite(); void nameChanged(const QString &oldName, const QString &newName); diff --git a/qt-models/divelocationmodel.cpp b/qt-models/divelocationmodel.cpp index ebdffc185..c64d1990a 100644 --- a/qt-models/divelocationmodel.cpp +++ b/qt-models/divelocationmodel.cpp @@ -26,6 +26,7 @@ LocationInformationModel::LocationInformationModel(QObject *obj) : QAbstractTabl connect(&diveListNotifier, &DiveListNotifier::diveSiteDiveCountChanged, this, &LocationInformationModel::diveSiteDiveCountChanged); connect(&diveListNotifier, &DiveListNotifier::diveSiteAdded, this, &LocationInformationModel::diveSiteAdded); connect(&diveListNotifier, &DiveListNotifier::diveSiteDeleted, this, &LocationInformationModel::diveSiteDeleted); + connect(&diveListNotifier, &DiveListNotifier::diveSiteChanged, this, &LocationInformationModel::diveSiteChanged); } int LocationInformationModel::columnCount(const QModelIndex &) const @@ -173,6 +174,14 @@ void LocationInformationModel::diveSiteDeleted(struct dive_site *, int idx) endRemoveRows(); } +void LocationInformationModel::diveSiteChanged(struct dive_site *ds, int field) +{ + int idx = get_divesite_idx(ds, &dive_site_table); + if (idx < 0) + return; + dataChanged(createIndex(idx, field), createIndex(idx, field)); +} + bool DiveSiteSortedModel::filterAcceptsRow(int sourceRow, const QModelIndex &source_parent) const { // TODO: filtering @@ -223,14 +232,33 @@ QStringList DiveSiteSortedModel::allSiteNames() const return locationNames; } +struct dive_site *DiveSiteSortedModel::getDiveSite(const QModelIndex &idx) +{ + return get_dive_site(mapToSource(idx).row(), &dive_site_table); +} + #ifndef SUBSURFACE_MOBILE +bool DiveSiteSortedModel::setData(const QModelIndex &index, const QVariant &value, int role) +{ + struct dive_site *ds = getDiveSite(index); + if (!ds || value.isNull()) + return false; + switch (index.column()) { + case LocationInformationModel::NAME: + Command::editDiveSiteName(ds, value.toString()); + return true; + default: + return false; + } +} + // TODO: Remove from model. It doesn't make sense to call the model here, which calls the undo command, // which in turn calls the model. void DiveSiteSortedModel::remove(const QModelIndex &index) { if (index.column() != LocationInformationModel::REMOVE) return; - struct dive_site *ds = get_dive_site(mapToSource(index).row(), &dive_site_table); + struct dive_site *ds = getDiveSite(index); if (!ds) return; if (ds->dives.nr > 0 && @@ -240,7 +268,7 @@ void DiveSiteSortedModel::remove(const QModelIndex &index) return; Command::deleteDiveSites(QVector<dive_site *>{ds}); } -#endif +#endif // SUBSURFACE_MOBILE GeoReferencingOptionsModel *GeoReferencingOptionsModel::instance() { diff --git a/qt-models/divelocationmodel.h b/qt-models/divelocationmodel.h index 34ac34c04..4fd7b3d36 100644 --- a/qt-models/divelocationmodel.h +++ b/qt-models/divelocationmodel.h @@ -35,15 +35,20 @@ public slots: void diveSiteDiveCountChanged(struct dive_site *ds); void diveSiteAdded(struct dive_site *ds, int idx); void diveSiteDeleted(struct dive_site *ds, int idx); + void diveSiteChanged(struct dive_site *ds, int field); }; class DiveSiteSortedModel : public QSortFilterProxyModel { Q_OBJECT private: + struct dive_site *getDiveSite(const QModelIndex &idx); bool filterAcceptsRow(int sourceRow, const QModelIndex &source_parent) const override; bool lessThan(const QModelIndex &i1, const QModelIndex &i2) const override; +#ifndef SUBSURFACE_MOBILE + bool setData(const QModelIndex &index, const QVariant &value, int role) override; public slots: void remove(const QModelIndex &index); +#endif // SUBSURFACE_MOBILE public: DiveSiteSortedModel(); QStringList allSiteNames() const; |