diff options
author | Berthold Stoeger <bstoeger@mail.tuwien.ac.at> | 2018-09-08 19:58:11 +0200 |
---|---|---|
committer | Dirk Hohndel <dirk@hohndel.org> | 2018-10-11 16:22:27 -0700 |
commit | b19adecb9f679b3b4523d679e65dedc2849d50e6 (patch) | |
tree | 4002a578dcfab0470e1bff0c27b6fa25a2484aa3 | |
parent | e0fcf99d0ac3ec23e6009a55b6e59c47c668081b (diff) | |
download | subsurface-b19adecb9f679b3b4523d679e65dedc2849d50e6.tar.gz |
Undo: make adding of planned dive undo-able
Planned dives were still added by directly calling core code.
This could confuse the undo-machinery, leading to crashes.
Instead, use the proper undo-command. The problem is that as
opposed to the other AddDive-commands, planned dives may
belong to a trip. Thus, the interface to the AddDive command
was changed to respect the divetrip field. Make sure that
the other callers reset that field (actually, it should never
be set). Add a comment describing the perhaps surprising
interface (the passed-in dive, usually displayed dive, is
reset).
Moreover, a dive cloned in the planner is not assigned a
new number. Thus, add an argument to the AddDive-command,
which expresses whether a new number should be generated
for the to-be-added dive.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
-rw-r--r-- | desktop-widgets/command.cpp | 4 | ||||
-rw-r--r-- | desktop-widgets/command.h | 11 | ||||
-rw-r--r-- | desktop-widgets/command_divelist.cpp | 22 | ||||
-rw-r--r-- | desktop-widgets/command_divelist.h | 2 | ||||
-rw-r--r-- | desktop-widgets/mainwindow.cpp | 14 | ||||
-rw-r--r-- | desktop-widgets/tab-widgets/maintab.cpp | 4 | ||||
-rw-r--r-- | qt-models/diveplannermodel.cpp | 45 |
7 files changed, 52 insertions, 50 deletions
diff --git a/desktop-widgets/command.cpp b/desktop-widgets/command.cpp index 54ca2d39f..2bff1aad0 100644 --- a/desktop-widgets/command.cpp +++ b/desktop-widgets/command.cpp @@ -6,9 +6,9 @@ namespace Command { // Dive-list related commands -void addDive(dive *d, bool autogroup) +void addDive(dive *d, bool autogroup, bool newNumber) { - execute(new AddDive(d, autogroup)); + execute(new AddDive(d, autogroup, newNumber)); } void deleteDive(const QVector<struct dive*> &divesToDelete) diff --git a/desktop-widgets/command.h b/desktop-widgets/command.h index 9a8ccfd6d..71ef2f9da 100644 --- a/desktop-widgets/command.h +++ b/desktop-widgets/command.h @@ -9,13 +9,18 @@ // We put everything in a namespace, so that we can shorten names without polluting the global namespace namespace Command { -// General commands +// 1) General commands + void clear(); // Reset the undo stack. Delete all commands. QAction *undoAction(QObject *parent); // Create an undo action. QAction *redoAction(QObject *parent); // Create an redo action. -// Dive-list related commands -void addDive(dive *d, bool autogroup); +// 2) Dive-list related commands + +void addDive(dive *d, bool autogroup, bool newNumber); // If d->dive_trip is null and autogroup is true, dives within the auto-group + // distance are added to a trip. dive d is consumed (the structure is reset)! + // If newNumber is true, the dive is assigned a new number, depending on the + // insertion position. void deleteDive(const QVector<struct dive*> &divesToDelete); void shiftTime(const QVector<dive *> &changedDives, int amount); void renumberDives(const QVector<QPair<dive *, int>> &divesToRenumber); diff --git a/desktop-widgets/command_divelist.cpp b/desktop-widgets/command_divelist.cpp index 1bc225086..e129fefdd 100644 --- a/desktop-widgets/command_divelist.cpp +++ b/desktop-widgets/command_divelist.cpp @@ -483,21 +483,28 @@ void DiveListBase::redo() finishWork(); } -AddDive::AddDive(dive *d, bool autogroup) +AddDive::AddDive(dive *d, bool autogroup, bool newNumber) { setText(tr("add dive")); + // By convention, d is "displayed dive" and can be overwritten. d->maxdepth.mm = 0; + d->dc.maxdepth.mm = 0; fixup_dive(d); - d->divetrip = nullptr; - // Get an owning pointer to a copy of the dive - // Note: this destroys the old dive! + // Get an owning pointer to a copied or moved dive + // Note: if move is true, this destroys the old dive! OwningDivePtr divePtr(clone_dive(d)); + divePtr->selected = false; // If we clone a planned dive, it might have been selected. + // We have to clear the flag, as selections will be managed + // on dive-addition. // If we alloc a new-trip for autogrouping, get an owning pointer to it. OwningTripPtr allocTrip; - dive_trip *trip = nullptr; - if (autogroup) { + dive_trip *trip = divePtr->divetrip; + // We have to delete the pointer-to-trip, because this would prevent the core from adding to the trip + // and we would get the count-of-dives in the trip wrong. Yes, that's all horribly subtle! + divePtr->divetrip = nullptr; + if (!trip && autogroup) { bool alloc; trip = get_trip_for_new_dive(divePtr.get(), &alloc); if (alloc) @@ -505,7 +512,8 @@ AddDive::AddDive(dive *d, bool autogroup) } int idx = dive_get_insertion_index(divePtr.get()); - divePtr->number = get_dive_nr_at_idx(idx); + if (newNumber) + divePtr->number = get_dive_nr_at_idx(idx); divesToAdd.push_back({ std::move(divePtr), std::move(allocTrip), trip, idx }); } diff --git a/desktop-widgets/command_divelist.h b/desktop-widgets/command_divelist.h index 650a64ba4..29c9c6e7b 100644 --- a/desktop-widgets/command_divelist.h +++ b/desktop-widgets/command_divelist.h @@ -82,7 +82,7 @@ private: class AddDive : public DiveListBase { public: - AddDive(dive *dive, bool autogroup); + AddDive(dive *dive, bool autogroup, bool newNumber); private: void undoit() override; void redoit() override; diff --git a/desktop-widgets/mainwindow.cpp b/desktop-widgets/mainwindow.cpp index 625c094c1..0476dfae8 100644 --- a/desktop-widgets/mainwindow.cpp +++ b/desktop-widgets/mainwindow.cpp @@ -890,19 +890,11 @@ void MainWindow::planCanceled() void MainWindow::planCreated() { - // get the new dive selected and assign a number if reasonable - graphics()->setProfileState(); - if (displayed_dive.id == 0) { - // we might have added a new dive (so displayed_dive was cleared out by clone_dive() - dive_list()->unselectDives(); - select_dive(get_dive(dive_table.nr - 1)); - dive_list()->selectDive(get_divenr(current_dive)); // TODO: don't convert dive->idx and back - set_dive_nr_for_current_dive(); - } // make sure our UI is in a consistent state - information()->updateDiveInfo(); showProfile(); - refreshDisplay(); + setApplicationState("Default"); + dive_list()->setEnabled(true); + dive_list()->setFocus(); } void MainWindow::setPlanNotes() diff --git a/desktop-widgets/tab-widgets/maintab.cpp b/desktop-widgets/tab-widgets/maintab.cpp index 92e55784f..d45bc246d 100644 --- a/desktop-widgets/tab-widgets/maintab.cpp +++ b/desktop-widgets/tab-widgets/maintab.cpp @@ -799,7 +799,7 @@ void MainTab::acceptChanges() updateDiveSite(ui.location->currDiveSiteUuid(), &displayed_dive); copyTagsToDisplayedDive(); - Command::addDive(&displayed_dive, autogroup); + Command::addDive(&displayed_dive, autogroup, true); editMode = NONE; MainWindow::instance()->exitEditState(); @@ -810,8 +810,6 @@ void MainTab::acceptChanges() ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty()); emit addDiveFinished(); DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::NOTHING); - int scrolledBy = MainWindow::instance()->dive_list()->verticalScrollBar()->sliderPosition(); - MainWindow::instance()->dive_list()->verticalScrollBar()->setSliderPosition(scrolledBy); MainWindow::instance()->dive_list()->setFocus(); resetPallete(); displayed_dive.divetrip = nullptr; // Should not be necessary, just in case! diff --git a/qt-models/diveplannermodel.cpp b/qt-models/diveplannermodel.cpp index b2d71d1ef..bab57e0df 100644 --- a/qt-models/diveplannermodel.cpp +++ b/qt-models/diveplannermodel.cpp @@ -7,6 +7,7 @@ #include "core/device.h" #include "core/qthelper.h" #include "core/settings/qPrefDivePlanner.h" +#include "desktop-widgets/command.h" #include "core/gettextfromc.h" #include <QApplication> #include <QTextDocument> @@ -1110,15 +1111,9 @@ void DivePlannerPointsModel::createPlan(bool replanCopy) computeVariations(plan_copy, &ds_after_previous_dives); free(cache); - if (!current_dive || displayed_dive.id != current_dive->id) { - // we were planning a new dive, not re-planning an existing on - record_dive(clone_dive(&displayed_dive)); - } else if (current_dive && displayed_dive.id == current_dive->id) { - // we are replanning a dive - make sure changes are reflected - // correctly in the dive structure and copy it back into the dive table - displayed_dive.maxdepth.mm = 0; - displayed_dive.dc.maxdepth.mm = 0; - fixup_dive(&displayed_dive); + + // Fixup planner notes. + if (current_dive && displayed_dive.id == current_dive->id) { // Try to identify old planner output and remove only this part // Treat user provided text as plain text. QTextDocument notesDocument; @@ -1134,24 +1129,28 @@ void DivePlannerPointsModel::createPlan(bool replanCopy) oldnotes.append(displayed_dive.notes); displayed_dive.notes = copy_qstring(oldnotes); // If we save as new create a copy of the dive here - if (replanCopy) { - struct dive *copy = alloc_dive(); - copy_dive(current_dive, copy); - copy->id = 0; - copy->selected = false; - copy->divetrip = NULL; - if (current_dive->divetrip) - add_dive_to_trip(copy, current_dive->divetrip); - record_dive(copy); - } + } + + setPlanMode(NOTHING); + planCreated(); // This signal will exit the profile from planner state. This must be *before* adding the dive, + // so that the Undo-commands update the display accordingly (see condition in updateDiveInfo(). + + // Now, add or modify the dive. + if (!current_dive || displayed_dive.id != current_dive->id) { + // we were planning a new dive, not re-planning an existing one + displayed_dive.divetrip = nullptr; // Should not be necessary, just in case! + Command::addDive(&displayed_dive, autogroup, true); + } else if (replanCopy) { + // we were planning an old dive and save as a new dive + displayed_dive.id = dive_getUniqID(); // Things will break horribly if we create dives with the same id. + Command::addDive(&displayed_dive, false, false); + } else { + // we were planning an old dive and rewrite the plan + mark_divelist_changed(true); copy_dive(&displayed_dive, current_dive); } - mark_divelist_changed(true); - sort_table(&dive_table); // Remove and clean the diveplan, so we don't delete // the dive by mistake. free_dps(&diveplan); - setPlanMode(NOTHING); - planCreated(); } |