From 5436f9b8590e6022ce2dabefb23ffb037020784a Mon Sep 17 00:00:00 2001 From: Berthold Stoeger Date: Thu, 14 Feb 2019 23:07:12 +0100 Subject: Undo: move dive-list logic into edit commands The edit-commands were called with a list of selected dives and the original value. Move the creation of the list and extraction of the original value into the edit-commmands. This removes the "current is last" rule and allows for more flexibility. Since the depth- and duration editing applies only to the current dive and not all selected dives, add a parameter to the edit-commands controlling whether only the current or all selected dives are edited. Signed-off-by: Berthold Stoeger --- desktop-widgets/command.cpp | 56 +++++++++++++------------- desktop-widgets/command.h | 28 ++++++------- desktop-widgets/command_edit.cpp | 71 +++++++++++++++++++++++---------- desktop-widgets/command_edit.h | 16 ++++---- desktop-widgets/tab-widgets/maintab.cpp | 39 +++++++----------- 5 files changed, 112 insertions(+), 98 deletions(-) diff --git a/desktop-widgets/command.cpp b/desktop-widgets/command.cpp index 2f73477fe..0486b2bce 100644 --- a/desktop-widgets/command.cpp +++ b/desktop-widgets/command.cpp @@ -130,74 +130,74 @@ void purgeUnusedDiveSites() } // Dive editing related commands -void editNotes(const QVector dives, const QString &newValue, const QString &oldValue) +void editNotes(const QString &newValue, bool currentDiveOnly) { - execute(new EditNotes(dives, newValue, oldValue)); + execute(new EditNotes(newValue, currentDiveOnly)); } -void editMode(const QVector dives, int index, int newValue, int oldValue) +void editMode(int index, int newValue, bool currentDiveOnly) { - execute(new EditMode(dives, index, newValue, oldValue)); + execute(new EditMode(index, newValue, currentDiveOnly)); } -void editSuit(const QVector dives, const QString &newValue, const QString &oldValue) +void editSuit(const QString &newValue, bool currentDiveOnly) { - execute(new EditSuit(dives, newValue, oldValue)); + execute(new EditSuit(newValue, currentDiveOnly)); } -void editRating(const QVector dives, int newValue, int oldValue) +void editRating(int newValue, bool currentDiveOnly) { - execute(new EditRating(dives, newValue, oldValue)); + execute(new EditRating(newValue, currentDiveOnly)); } -void editVisibility(const QVector dives, int newValue, int oldValue) +void editVisibility(int newValue, bool currentDiveOnly) { - execute(new EditVisibility(dives, newValue, oldValue)); + execute(new EditVisibility(newValue, currentDiveOnly)); } -void editAirTemp(const QVector dives, int newValue, int oldValue) +void editAirTemp(int newValue, bool currentDiveOnly) { - execute(new EditAirTemp(dives, newValue, oldValue)); + execute(new EditAirTemp(newValue, currentDiveOnly)); } -void editWaterTemp(const QVector dives, int newValue, int oldValue) +void editWaterTemp(int newValue, bool currentDiveOnly) { - execute(new EditWaterTemp(dives, newValue, oldValue)); + execute(new EditWaterTemp(newValue, currentDiveOnly)); } -void editDepth(const QVector dives, int newValue, int oldValue) +void editDepth(int newValue, bool currentDiveOnly) { - execute(new EditDepth(dives, newValue, oldValue)); + execute(new EditDepth(newValue, currentDiveOnly)); } -void editDuration(const QVector dives, int newValue, int oldValue) +void editDuration(int newValue, bool currentDiveOnly) { - execute(new EditDuration(dives, newValue, oldValue)); + execute(new EditDuration(newValue, currentDiveOnly)); } -void editDiveSite(const QVector dives, struct dive_site *newValue, struct dive_site *oldValue) +void editDiveSite(struct dive_site *newValue, bool currentDiveOnly) { - execute(new EditDiveSite(dives, newValue, oldValue)); + execute(new EditDiveSite(newValue, currentDiveOnly)); } -void editDiveSiteNew(const QVector dives, const QString &newName, struct dive_site *oldValue) +void editDiveSiteNew(const QString &newName, bool currentDiveOnly) { - execute(new EditDiveSiteNew(dives, newName, oldValue)); + execute(new EditDiveSiteNew(newName, currentDiveOnly)); } -void editTags(const QVector &dives, const QStringList &newList, struct dive *d) +void editTags(const QStringList &newList, bool currentDiveOnly) { - execute(new EditTags(dives, newList, d)); + execute(new EditTags(newList, currentDiveOnly)); } -void editBuddies(const QVector &dives, const QStringList &newList, struct dive *d) +void editBuddies(const QStringList &newList, bool currentDiveOnly) { - execute(new EditBuddies(dives, newList, d)); + execute(new EditBuddies(newList, currentDiveOnly)); } -void editDiveMaster(const QVector &dives, const QStringList &newList, struct dive *d) +void editDiveMaster(const QStringList &newList, bool currentDiveOnly) { - execute(new EditDiveMaster(dives, newList, d)); + execute(new EditDiveMaster(newList, currentDiveOnly)); } } // namespace Command diff --git a/desktop-widgets/command.h b/desktop-widgets/command.h index 4754a6937..7667f3482 100644 --- a/desktop-widgets/command.h +++ b/desktop-widgets/command.h @@ -52,20 +52,20 @@ void purgeUnusedDiveSites(); // 4) Dive editing related commands -void editNotes(const QVector dives, const QString &newValue, const QString &oldValue); -void editSuit(const QVector dives, const QString &newValue, const QString &oldValue); -void editMode(const QVector dives, int index, int newValue, int oldValue); -void editRating(const QVector dives, int newValue, int oldValue); -void editVisibility(const QVector dives, int newValue, int oldValue); -void editAirTemp(const QVector dives, int newValue, int oldValue); -void editWaterTemp(const QVector dives, int newValue, int oldValue); -void editDepth(const QVector dives, int newValue, int oldValue); -void editDuration(const QVector dives, int newValue, int oldValue); -void editDiveSite(const QVector dives, struct dive_site *newValue, struct dive_site *oldValue); -void editDiveSiteNew(const QVector dives, const QString &newName, struct dive_site *oldValue); -void editTags(const QVector &dives, const QStringList &newList, struct dive *d); -void editBuddies(const QVector &dives, const QStringList &newList, struct dive *d); -void editDiveMaster(const QVector &dives, const QStringList &newList, struct dive *d); +void editNotes(const QString &newValue, bool currentDiveOnly); +void editSuit(const QString &newValue, bool currentDiveOnly); +void editMode(int index, int newValue, bool currentDiveOnly); +void editRating(int newValue, bool currentDiveOnly); +void editVisibility(int newValue, bool currentDiveOnly); +void editAirTemp(int newValue, bool currentDiveOnly); +void editWaterTemp(int newValue, bool currentDiveOnly); +void editDepth(int newValue, bool currentDiveOnly); +void editDuration(int newValue, bool currentDiveOnly); +void editDiveSite(struct dive_site *newValue, bool currentDiveOnly); +void editDiveSiteNew(const QString &newName, bool currentDiveOnly); +void editTags(const QStringList &newList, bool currentDiveOnly); +void editBuddies(const QStringList &newList, bool currentDiveOnly); +void editDiveMaster(const QStringList &newList, bool currentDiveOnly); } // namespace Command diff --git a/desktop-widgets/command_edit.cpp b/desktop-widgets/command_edit.cpp index dd681e1d8..b8727bfb6 100644 --- a/desktop-widgets/command_edit.cpp +++ b/desktop-widgets/command_edit.cpp @@ -8,16 +8,28 @@ namespace Command { +static std::vector getSelectedDives(bool currentDiveOnly) +{ + if (currentDiveOnly) + return current_dive ? std::vector { current_dive } + : std::vector { }; + + std::vector res; + struct dive *d; + int i; + for_each_dive (i, d) { + if (d->selected) + res.push_back(d); + } + return res; +} + template -EditBase::EditBase(const QVector &divesIn, T newValue, T oldValue) : +EditBase::EditBase(T newValue, bool currentDiveOnly) : value(std::move(newValue)), - old(std::move(oldValue)), - dives(divesIn.toStdVector()) + dives(getSelectedDives(currentDiveOnly)), + current(current_dive) { - // If there is nothing to do, clear the dives vector. - // This signals that no action has to be taken. - if (old == value) - dives.clear(); } // This is quite hackish: we can't use virtual functions in the constructor and @@ -28,6 +40,16 @@ EditBase::EditBase(const QVector &divesIn, T newValue, T oldValue) : template bool EditBase::workToBeDone() { + // First, let's fetch the old value, i.e. the value of the current dive. + // If no current dive exists, bail. + if (!current) + return false; + old = data(current); + + // If there is no change - do nothing. + if (old == value) + return false; + std::vector divesNew; divesNew.reserve(dives.size()); for (dive *d: dives) { @@ -42,7 +64,7 @@ bool EditBase::workToBeDone() //: remove the part in parantheses for %n = 1 setText(tr("Edit %1 (%n dive(s))", "", num_dives).arg(fieldName())); - return num_dives; + return num_dives > 0; } template @@ -74,11 +96,11 @@ void EditBase::undo() // don't have their own constructor. They simply delegate to the base // class by virtue of a "using" declaration. template -EditBase::EditBase(const QVector &dives, QString oldValue, QString newValue); +EditBase::EditBase(QString newValue, bool currentDiveOnly); template -EditBase::EditBase(const QVector &dives, int oldValue, int newValue); +EditBase::EditBase(int newValue, bool currentDiveOnly); template -EditBase::EditBase(const QVector &dives, struct dive_site *oldValue, struct dive_site *newValue); +EditBase::EditBase(struct dive_site *newValue, bool currentDiveOnly); // Undo and redo do the same as just the stored value is exchanged template @@ -305,9 +327,10 @@ void EditDiveSite::redo() EditDiveSite::undo(); // Undo and redo do the same } -static struct dive_site *createDiveSite(const QString &name, struct dive_site *old) +static struct dive_site *createDiveSite(const QString &name) { struct dive_site *ds = alloc_dive_site(); + struct dive_site *old = current_dive ? current_dive->dive_site : nullptr; if (old) { copy_dive_site(old, ds); free(ds->name); // Free name, as we will overwrite it with our own version @@ -316,8 +339,8 @@ static struct dive_site *createDiveSite(const QString &name, struct dive_site *o return ds; } -EditDiveSiteNew::EditDiveSiteNew(const QVector &dives, const QString &newName, struct dive_site *oldValue) : - EditDiveSite(dives, createDiveSite(newName, oldValue), oldValue), +EditDiveSiteNew::EditDiveSiteNew(const QString &newName, bool currentDiveOnly) : + EditDiveSite(createDiveSite(newName), currentDiveOnly), diveSiteToAdd(value), diveSiteToRemove(nullptr) { @@ -353,8 +376,8 @@ void EditDiveSiteNew::redo() // - Not derive EditMode from EditBase. // - Change the semantics of the mode-editing. // The future will tell. -EditMode::EditMode(const QVector &dives, int indexIn, int newValue, int oldValue) - : EditBase(dives, newValue, oldValue), index(indexIn) +EditMode::EditMode(int indexIn, int newValue, bool currentDiveOnly) + : EditBase(newValue, currentDiveOnly), index(indexIn) { } @@ -380,10 +403,10 @@ DiveField EditMode::fieldId() const } // ***** Tag based commands ***** -EditTagsBase::EditTagsBase(const QVector &divesIn, const QStringList &newListIn, struct dive *d): - dives(divesIn.toStdVector()), - newList(newListIn), - oldDive(d) +EditTagsBase::EditTagsBase(const QStringList &newListIn, bool currentDiveOnly) : + dives(getSelectedDives(currentDiveOnly)), + current(current_dive), + newList(newListIn) { } @@ -412,8 +435,12 @@ bool EditTagsBase::workToBeDone() // here's what I think... add the tags that were added to the displayed dive and remove the tags // that were removed from it + // If there is no current dive, bail. + if (!current) + return false; + // Calculate tags to add and tags to remove - QStringList oldList = data(oldDive); + QStringList oldList = data(current); for (const QString &s: newList) { if (!oldList.contains(s)) tagsToAdd.push_back(s); @@ -441,7 +468,7 @@ bool EditTagsBase::workToBeDone() //: remove the part in parantheses for %n = 1 setText(tr("Edit %1 (%n dive(s))", "", num_dives).arg(fieldName())); - return num_dives; + return num_dives != 0; } void EditTagsBase::undo() diff --git a/desktop-widgets/command_edit.h b/desktop-widgets/command_edit.h index 081ea11bb..4363cb9bd 100644 --- a/desktop-widgets/command_edit.h +++ b/desktop-widgets/command_edit.h @@ -33,12 +33,10 @@ protected: void redo() override; bool workToBeDone() override; - // Dives to be edited. For historical reasons, the *last* entry was - // the active dive when the user initialized the action. This dive - // will be made the current dive on redo / undo. - std::vector dives; + std::vector dives; // Dives to be edited. + struct dive *current; // On undo, we set the current dive at the time of the operation. public: - EditBase(const QVector &dives, T newValue, T oldValue); + EditBase(T newValue, bool currentDiveOnly); protected: // Get and set functions to be overriden by sub-classes. @@ -139,7 +137,7 @@ class EditDiveSiteNew : public EditDiveSite { public: OwningDiveSitePtr diveSiteToAdd; struct dive_site *diveSiteToRemove; - EditDiveSiteNew(const QVector &dives, const QString &newName, struct dive_site *oldValue); + EditDiveSiteNew(const QString &newName, bool currentDiveOnly); void undo() override; void redo() override; }; @@ -147,7 +145,7 @@ public: class EditMode : public EditBase { int index; public: - EditMode(const QVector &dives, int indexIn, int newValue, int oldValue); + EditMode(int indexIn, int newValue, bool currentDiveOnly); void set(struct dive *d, int i) const override; int data(struct dive *d) const override; QString fieldName() const override; @@ -164,10 +162,10 @@ class EditTagsBase : public Base { // the active dive when the user initialized the action. This dive // will be made the current dive on redo / undo. std::vector dives; + struct dive *current; QStringList newList; // Temporary until initialized - struct dive *oldDive; // Temporary until initialized public: - EditTagsBase(const QVector &dives, const QStringList &newList, struct dive *d); + EditTagsBase(const QStringList &newList, bool currentDiveOnly); protected: QStringList tagsToAdd; diff --git a/desktop-widgets/tab-widgets/maintab.cpp b/desktop-widgets/tab-widgets/maintab.cpp index 31fe4d130..19b096f0f 100644 --- a/desktop-widgets/tab-widgets/maintab.cpp +++ b/desktop-widgets/tab-widgets/maintab.cpp @@ -737,13 +737,6 @@ static QVector getSelectedDivesCurrentLast() return res; } -// When editing depth and duration, we only edit a single dive. Therefore, return the current dive as a list -static QVector getCurrentAsList() -{ - return current_dive ? QVector { current_dive } - : QVector { }; -} - void MainTab::acceptChanges() { int i, addedId = -1; @@ -1016,7 +1009,7 @@ void MainTab::on_buddy_editingFinished() if (editMode == IGNORE || !current_dive) return; - Command::editBuddies(getSelectedDivesCurrentLast(), stringToList(ui.buddy->toPlainText()), current_dive); + Command::editBuddies(stringToList(ui.buddy->toPlainText()), false); } void MainTab::on_divemaster_editingFinished() @@ -1024,7 +1017,7 @@ void MainTab::on_divemaster_editingFinished() if (editMode == IGNORE || !current_dive) return; - Command::editDiveMaster(getSelectedDivesCurrentLast(), stringToList(ui.divemaster->toPlainText()), current_dive); + Command::editDiveMaster(stringToList(ui.divemaster->toPlainText()), false); } void MainTab::on_duration_editingFinished() @@ -1033,7 +1026,7 @@ void MainTab::on_duration_editingFinished() return; // Duration editing is special: we only edit the current dive. - Command::editDuration(getCurrentAsList(), parseDurationToSeconds(ui.duration->text()), displayed_dive.dc.duration.seconds); + Command::editDuration(parseDurationToSeconds(ui.duration->text()), true); } void MainTab::on_depth_editingFinished() @@ -1042,32 +1035,28 @@ void MainTab::on_depth_editingFinished() return; // Depth editing is special: we only edit the current dive. - Command::editDepth(getCurrentAsList(), parseLengthToMm(ui.depth->text()), current_dive->dc.maxdepth.mm); + Command::editDepth(parseLengthToMm(ui.depth->text()), true); } void MainTab::on_airtemp_editingFinished() { if (editMode == IGNORE || !current_dive) return; - Command::editAirTemp(getSelectedDivesCurrentLast(), - parseTemperatureToMkelvin(ui.airtemp->text()), current_dive->airtemp.mkelvin); + Command::editAirTemp(parseTemperatureToMkelvin(ui.airtemp->text()), false); } void MainTab::divetype_Changed(int index) { if (editMode == IGNORE || !current_dive) return; - Command::editMode(getSelectedDivesCurrentLast(), dc_number, (enum divemode_t)index, - get_dive_dc(current_dive, dc_number)->divemode); + Command::editMode(dc_number, (enum divemode_t)index, false); } void MainTab::on_watertemp_editingFinished() { if (editMode == IGNORE || !current_dive) return; - Command::editWaterTemp(getSelectedDivesCurrentLast(), - parseTemperatureToMkelvin(ui.watertemp->text()), - current_dive->watertemp.mkelvin); + Command::editWaterTemp(parseTemperatureToMkelvin(ui.watertemp->text()), false); } // Editing of the dive time is different. If multiple dives are edited, @@ -1182,7 +1171,7 @@ void MainTab::on_tagWidget_editingFinished() if (editMode == IGNORE || !current_dive) return; - Command::editTags(getSelectedDivesCurrentLast(), ui.tagWidget->getBlockStringList(), current_dive); + Command::editTags(ui.tagWidget->getBlockStringList(), false); } void MainTab::on_location_diveSiteSelected() @@ -1192,9 +1181,9 @@ void MainTab::on_location_diveSiteSelected() struct dive_site *newDs = ui.location->currDiveSite(); if (newDs == RECENTLY_ADDED_DIVESITE) - Command::editDiveSiteNew(getSelectedDivesCurrentLast(), ui.location->text(), current_dive->dive_site); + Command::editDiveSiteNew(ui.location->text(), false); else - Command::editDiveSite(getSelectedDivesCurrentLast(), newDs, current_dive->dive_site); + Command::editDiveSite(newDs, false); } void MainTab::on_diveTripLocation_textEdited(const QString& text) @@ -1211,7 +1200,7 @@ void MainTab::on_suit_editingFinished() if (editMode == IGNORE || !current_dive) return; - Command::editSuit(getSelectedDivesCurrentLast(), ui.suit->text(), QString(current_dive->suit)); + Command::editSuit(ui.suit->text(), false); } void MainTab::on_notes_textChanged() @@ -1235,7 +1224,7 @@ void MainTab::on_notes_editingFinished() QString notes = ui.notes->toHtml().indexOf("toHtml() : ui.notes->toPlainText(); - Command::editNotes(getSelectedDivesCurrentLast(), notes, QString(current_dive->notes)); + Command::editNotes(notes, false); } void MainTab::on_rating_valueChanged(int value) @@ -1243,7 +1232,7 @@ void MainTab::on_rating_valueChanged(int value) if (editMode == IGNORE || !current_dive) return; - Command::editRating(getSelectedDivesCurrentLast(), value, current_dive->rating); + Command::editRating(value, false); } void MainTab::on_visibility_valueChanged(int value) @@ -1251,7 +1240,7 @@ void MainTab::on_visibility_valueChanged(int value) if (editMode == IGNORE || !current_dive) return; - Command::editVisibility(getSelectedDivesCurrentLast(), value, current_dive->visibility); + Command::editVisibility(value, false); } #undef MODIFY_DIVES -- cgit v1.2.3-70-g09d2