diff options
author | Berthold Stoeger <bstoeger@mail.tuwien.ac.at> | 2018-07-19 22:35:25 +0200 |
---|---|---|
committer | Dirk Hohndel <dirk@hohndel.org> | 2018-10-11 16:22:27 -0700 |
commit | 12df9faaa2037b5155ebb84a7f6f6102491a0091 (patch) | |
tree | 33dcf0e400f669d088f1b20652e9c0c6f69aae2b /desktop-widgets | |
parent | 61467ea0d59b04f141a68452ee16c70760421d72 (diff) | |
download | subsurface-12df9faaa2037b5155ebb84a7f6f6102491a0091.tar.gz |
Undo: implement undo of manual dive-creation
Play manual addition of dives via an UndoCommand. Since this does in
large parts the same thing as undo/redo of dive deletion (just the
other way round and only a single instead of multiple dive), factor
out the functions that add/delete dives and take care of trips.
The UI-interaction is just mindless copy&paste and will have to
be adapted.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
Diffstat (limited to 'desktop-widgets')
-rw-r--r-- | desktop-widgets/command_divelist.cpp | 904 | ||||
-rw-r--r-- | desktop-widgets/tab-widgets/maintab.cpp | 43 | ||||
-rw-r--r-- | desktop-widgets/undocommands.cpp | 106 | ||||
-rw-r--r-- | desktop-widgets/undocommands.h | 29 |
4 files changed, 1029 insertions, 53 deletions
diff --git a/desktop-widgets/command_divelist.cpp b/desktop-widgets/command_divelist.cpp new file mode 100644 index 000000000..6f638b2a6 --- /dev/null +++ b/desktop-widgets/command_divelist.cpp @@ -0,0 +1,904 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include "command_divelist.h" +#include "desktop-widgets/mainwindow.h" +#include "desktop-widgets/divelistview.h" +#include "core/divelist.h" +#include "core/display.h" // for amount_selected +#include "core/subsurface-qt/DiveListNotifier.h" +#include "qt-models/filtermodels.h" + +namespace Command { + +// Generally, signals are sent in batches per trip. To avoid writing the same loop +// again and again, this template takes a vector of trip / dive pairs, sorts it +// by trip and then calls a function-object with trip and a QVector of dives in that trip. +// Input parameters: +// - dives: a vector of trip,dive pairs, which will be sorted and processed in batches by trip. +// - action: a function object, taking a trip-pointer and a QVector of dives, which will be called for each batch. +template<typename Function> +void processByTrip(std::vector<std::pair<dive_trip *, dive *>> &dives, Function action) +{ + // Use std::tie for lexicographical sorting of trip, then start-time + std::sort(dives.begin(), dives.end(), + [](const std::pair<dive_trip *, dive *> &e1, const std::pair<dive_trip *, dive *> &e2) + { return std::tie(e1.first, e1.second->when) < std::tie(e2.first, e2.second->when); }); + + // Then, process the dives in batches by trip + size_t i, j; // Begin and end of batch + for (i = 0; i < dives.size(); i = j) { + dive_trip *trip = dives[i].first; + for (j = i + 1; j < dives.size() && dives[j].first == trip; ++j) + ; // pass + // Copy dives into a QVector. Some sort of "range_view" would be ideal, but Qt doesn't work this way. + QVector<dive *> divesInTrip(j - i); + for (size_t k = i; k < j; ++k) + divesInTrip[k - i] = dives[k].second; + + // Finally, emit the signal + action(trip, divesInTrip); + } +} + + +// This helper function removes a dive, takes ownership of the dive and adds it to a DiveToAdd structure. +// It is crucial that dives are added in reverse order of deletion, so the the indices are correctly +// set and that the trips are added before they are used! +DiveToAdd DiveListBase::removeDive(struct dive *d) +{ + // If the dive to be removed is selected, we will inform the frontend + // later via a signal that the dive changed. + if (d->selected) + selectionChanged = true; + + // If the dive was the current dive, reset the current dive. The calling + // command is responsible of finding a new dive. + if (d == current_dive) { + selectionChanged = true; // Should have been set above, as current dive is always selected. + current_dive = nullptr; + } + + DiveToAdd res; + res.idx = get_divenr(d); + if (res.idx < 0) + qWarning() << "Deletion of unknown dive!"; + + // remove dive from trip - if this is the last dive in the trip + // remove the whole trip. + res.trip = unregister_dive_from_trip(d, false); + if (res.trip && res.trip->nrdives == 0) { + unregister_trip(res.trip); // Remove trip from backend + res.tripToAdd.reset(res.trip); // Take ownership of trip + } + + res.dive.reset(unregister_dive(res.idx)); // Remove dive from backend + + return res; +} + +// This helper function adds a dive and returns ownership to the backend. It may also add a dive trip. +// It is crucial that dives are added in reverse order of deletion (see comment above)! +// Returns pointer to added dive (which is owned by the backend!) +dive *DiveListBase::addDive(DiveToAdd &d) +{ + if (d.tripToAdd) + insert_trip_dont_merge(d.tripToAdd.release()); // Return ownership to backend + if (d.trip) + add_dive_to_trip(d.dive.get(), d.trip); + dive *res = d.dive.release(); // Give up ownership of dive + + // Set the filter flag according to current filter settings + bool show = MultiFilterSortModel::instance()->showDive(res); + res->hidden_by_filter = !show; + + add_single_dive(d.idx, res); // Return ownership to backend + + // If the dive to be removed is selected, we will inform the frontend + // later via a signal that the dive changed. + if (res->selected) + selectionChanged = true; + + return res; +} + +// This helper function calls removeDive() on a list of dives to be removed and +// returns a vector of corresponding DiveToAdd objects, which can later be readded. +// The passed in vector is cleared. +std::vector<DiveToAdd> DiveListBase::removeDives(std::vector<dive *> &divesToDelete) +{ + std::vector<DiveToAdd> res; + res.reserve(divesToDelete.size()); + + // First, tell the filters that dives are removed. This could + // be done later using the emitted signals, but we do this here + // for symmetry with addDives() + MultiFilterSortModel::instance()->divesDeleted(QVector<dive *>::fromStdVector(divesToDelete)); + + for (dive *d: divesToDelete) + res.push_back(removeDive(d)); + divesToDelete.clear(); + + // We send one dives-deleted signal per trip (see comments in DiveListNotifier.h). + // Therefore, collect all dives in an array and sort by trip. + std::vector<std::pair<dive_trip *, dive *>> dives; + dives.reserve(res.size()); + for (const DiveToAdd &entry: res) + dives.push_back({ entry.trip, entry.dive.get() }); + + // Send signals. + processByTrip(dives, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + // Now, let's check if this trip is supposed to be deleted, by checking if it was marked + // as "add it". We could be smarter here, but let's just check the whole array for brevity. + bool deleteTrip = trip && + std::find_if(res.begin(), res.end(), [trip](const DiveToAdd &entry) + { return entry.tripToAdd.get() == trip; }) != res.end(); + emit diveListNotifier.divesDeleted(trip, deleteTrip, divesInTrip); + }); + return res; +} + +// This helper function is the counterpart fo removeDives(): it calls addDive() on a list +// of dives to be (re)added and returns a vector of the added dives. It does this in reverse +// order, so that trips are created appropriately and indexing is correct. +// The passed in vector is cleared. +std::vector<dive *> DiveListBase::addDives(std::vector<DiveToAdd> &divesToAdd) +{ + std::vector<dive *> res; + res.resize(divesToAdd.size()); + + // First, tell the filters that new dives are added. We do this here + // instead of later by signals, so that the filter can set the + // checkboxes of the new rows to its liking. The added dives will + // then appear in the correct shown/hidden state. + QVector<dive *> divesForFilter; + for (const DiveToAdd &entry: divesToAdd) + divesForFilter.push_back(entry.dive.get()); + MultiFilterSortModel::instance()->divesAdded(divesForFilter); + + // At the end of the function, to send the proper dives-added signals, + // we the the list of added trips. Create this list now. + std::vector<dive_trip *> addedTrips; + for (const DiveToAdd &entry: divesToAdd) { + if (entry.tripToAdd) + addedTrips.push_back(entry.tripToAdd.get()); + } + + // Now, add the dives + // Note: the idiomatic STL-way would be std::transform, but let's use a loop since + // that is closer to classical C-style. + auto it2 = res.rbegin(); + for (auto it = divesToAdd.rbegin(); it != divesToAdd.rend(); ++it, ++it2) + *it2 = addDive(*it); + divesToAdd.clear(); + + // We send one dives-deleted signal per trip (see comments in DiveListNotifier.h). + // Therefore, collect all dives in a array and sort by trip. + std::vector<std::pair<dive_trip *, dive *>> dives; + dives.reserve(res.size()); + for (dive *d: res) + dives.push_back({ d->divetrip, d }); + + // Send signals. + processByTrip(dives, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + // Now, let's check if this trip is supposed to be created, by checking if it was marked + // as "add it". We could be smarter here, but let's just check the whole array for brevity. + bool createTrip = trip && std::find(addedTrips.begin(), addedTrips.end(), trip) != addedTrips.end(); + // Finally, emit the signal + emit diveListNotifier.divesAdded(trip, createTrip, divesInTrip); + }); + return res; +} + +// This helper function renumbers dives according to an array of id/number pairs. +// The old numbers are stored in the array, thus calling this function twice has no effect. +// TODO: switch from uniq-id to indexes once all divelist-actions are controlled by undo-able commands +static void renumberDives(QVector<QPair<dive *, int>> &divesToRenumber) +{ + for (auto &pair: divesToRenumber) { + dive *d = pair.first; + if (!d) + continue; + std::swap(d->number, pair.second); + } + + // Emit changed signals per trip. + // First, collect all dives and sort by trip + std::vector<std::pair<dive_trip *, dive *>> dives; + dives.reserve(divesToRenumber.size()); + for (const auto &pair: divesToRenumber) { + dive *d = pair.first; + dives.push_back({ d->divetrip, d }); + } + + // Send signals. + processByTrip(dives, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + emit diveListNotifier.divesChanged(trip, divesInTrip); + }); +} + +// This helper function moves a dive to a trip. The old trip is recorded in the +// passed-in structure. This means that calling the function twice on the same +// object is a no-op concerning the dive. If the old trip was deleted from the +// core, an owning pointer to the removed trip is returned, otherwise a null pointer. +static OwningTripPtr moveDiveToTrip(DiveToTrip &diveToTrip) +{ + // Firstly, check if we move to the same trip and bail if this is a no-op. + if (diveToTrip.trip == diveToTrip.dive->divetrip) + return {}; + + // Remove from old trip + OwningTripPtr res; + + // Remove dive from trip - if this is the last dive in the trip, remove the whole trip. + dive_trip *trip = unregister_dive_from_trip(diveToTrip.dive, false); + if (trip && trip->nrdives == 0) { + unregister_trip(trip); // Remove trip from backend + res.reset(trip); + } + + // Store old trip and get new trip we should associate this dive with + std::swap(trip, diveToTrip.trip); + add_dive_to_trip(diveToTrip.dive, trip); + return res; +} + +// This helper function moves a set of dives between trips using the +// moveDiveToTrip function. Before doing so, it adds the necessary trips to +// the core. Trips that are removed from the core because they are empty +// are recorded in the passed in struct. The vectors of trips and dives +// are reversed. Thus, calling the function twice on the same object is +// a no-op. +static void moveDivesBetweenTrips(DivesToTrip &dives) +{ + // We collect an array of created trips so that we can instruct + // the model to create a new entry + std::vector<dive_trip *> createdTrips; + createdTrips.reserve(dives.tripsToAdd.size()); + + // First, bring back the trip(s) + for (OwningTripPtr &trip: dives.tripsToAdd) { + dive_trip *t = trip.release(); // Give up ownership + createdTrips.push_back(t); + insert_trip_dont_merge(t); // Return ownership to backend + } + dives.tripsToAdd.clear(); + + for (DiveToTrip &dive: dives.divesToMove) { + OwningTripPtr tripToAdd = moveDiveToTrip(dive); + // register trips that we'll have to readd + if (tripToAdd) + dives.tripsToAdd.push_back(std::move(tripToAdd)); + } + + // We send one signal per from-trip/to-trip pair. + // First, collect all dives in a struct and sort by from-trip/to-trip. + struct DiveMoved { + dive_trip *from; + dive_trip *to; + dive *d; + }; + std::vector<DiveMoved> divesMoved; + divesMoved.reserve(dives.divesToMove.size()); + for (const DiveToTrip &entry: dives.divesToMove) + divesMoved.push_back({ entry.trip, entry.dive->divetrip, entry.dive }); + + // Sort lexicographically by from-trip, to-trip and by start-time. + // Use std::tie() for lexicographical sorting. + std::sort(divesMoved.begin(), divesMoved.end(), [] ( const DiveMoved &d1, const DiveMoved &d2) + { return std::tie(d1.from, d1.to, d1.d->when) < std::tie(d2.from, d2.to, d2.d->when); }); + + // Now, process the dives in batches by trip + // TODO: this is a bit different from the cases above, so we don't use the processByTrip template, + // but repeat the loop here. We might think about generalizing the template, if more of such + // "special cases" appear. + size_t i, j; // Begin and end of batch + for (i = 0; i < divesMoved.size(); i = j) { + dive_trip *from = divesMoved[i].from; + dive_trip *to = divesMoved[i].to; + for (j = i + 1; j < divesMoved.size() && divesMoved[j].from == from && divesMoved[j].to == to; ++j) + ; // pass + // Copy dives into a QVector. Some sort of "range_view" would be ideal, but Qt doesn't work this way. + QVector<dive *> divesInTrip(j - i); + for (size_t k = i; k < j; ++k) + divesInTrip[k - i] = divesMoved[k].d; + + // Check if the from-trip was deleted: If yes, it was recorded in the tripsToAdd structure + bool deleteFrom = from && + std::find_if(dives.tripsToAdd.begin(), dives.tripsToAdd.end(), + [from](const OwningTripPtr &trip) { return trip.get() == from; }) != dives.tripsToAdd.end(); + // Check if the to-trip has to be created. For this purpose, we saved an array of trips to be created. + bool createTo = false; + if (to) { + // Check if the element is there... + auto it = std::find(createdTrips.begin(), createdTrips.end(), to); + + // ...if it is - remove it as we don't want the model to create the trip twice! + if (it != createdTrips.end()) { + createTo = true; + // erase/remove would be more performant, but this is irrelevant in the big scheme of things. + createdTrips.erase(it); + } + } + + // Finally, emit the signal + emit diveListNotifier.divesMovedBetweenTrips(from, to, deleteFrom, createTo, divesInTrip); + } + + // Reverse the tripsToAdd and the divesToAdd, so that on undo/redo the operations + // will be performed in reverse order. + std::reverse(dives.tripsToAdd.begin(), dives.tripsToAdd.end()); + std::reverse(dives.divesToMove.begin(), dives.divesToMove.end()); +} + +// When we initialize the command we don't have to roll-back any selection change +DiveListBase::DiveListBase() : firstExecution(true) +{ +} + +// Turn current selection into a vector. +// TODO: This could be made much more efficient if we kept a sorted list of selected dives! +static std::vector<dive *> getDiveSelection() +{ + std::vector<dive *> res; + res.reserve(amount_selected); + + int i; + dive *d; + for_each_dive(i, d) { + if (d->selected) + res.push_back(d); + } + return res; +} + +void DiveListBase::initWork() +{ + selectionChanged = false; +} + +void DiveListBase::finishWork() +{ + if (selectionChanged) // If the selection changed -> tell the frontend + emit diveListNotifier.selectionChanged(); +} + +// Set the current dive either from a list of selected dives, +// or a newly selected dive. In both cases, try to select the +// dive that is newer that is newer than the given date. +// This mimics the old behavior when the current dive changed. +static void setClosestCurrentDive(timestamp_t when, const std::vector<dive *> &selection) +{ + // Start from back until we get the first dive that is before + // the supposed-to-be selected dive. (Note: this mimics the + // old behavior when the current dive changed). + for (auto it = selection.rbegin(); it < selection.rend(); ++it) { + if ((*it)->when > when && !(*it)->hidden_by_filter) { + current_dive = *it; + return; + } + } + + // We didn't find a more recent selected dive -> try to + // find *any* visible selected dive. + for (dive *d: selection) { + if (!d->hidden_by_filter) { + current_dive = d; + return; + } + } + + // No selected dive is visible! Take the closest dive. Note, this might + // return null, but that just means unsetting the current dive (as no + // dive is visible anyway). + current_dive = find_next_visible_dive(when); +} + +// Rese the selection to the dives of the "selection" vector and send the appropriate signals. +// Set the current dive to "currentDive". "currentDive" must be an element of "selection" (or +// null if "seletion" is empty). +void DiveListBase::restoreSelection(const std::vector<dive *> &selection, dive *currentDive) +{ + // To do so, generate vectors of dives to be selected and deselected. + // We send signals batched by trip, so keep track of trip/dive pairs. + std::vector<std::pair<dive_trip *, dive *>> divesToSelect; + std::vector<std::pair<dive_trip *, dive *>> divesToDeselect; + + // TODO: We might want to keep track of selected dives in a more efficient way! + int i; + dive *d; + amount_selected = 0; // We recalculate amount_selected + for_each_dive(i, d) { + // We only modify dives that are currently visible. + if (d->hidden_by_filter) { + d->selected = false; // Note, not necessary, just to be sure + // that we get amount_selected right + continue; + } + + // Search the dive in the list of selected dives. + // TODO: By sorting the list in the same way as the backend, this could be made more efficient. + bool newState = std::find(selection.begin(), selection.end(), d) != selection.end(); + + // TODO: Instead of using select_dive() and deselect_dive(), we set selected directly. + // The reason is that deselect() automatically sets a new current dive, which we + // don't want, as we set it later anyway. + // There is other parts of the C++ code that touches the innards directly, but + // ultimately this should be pushed down to C. + if (newState && !d->selected) { + d->selected = true; + ++amount_selected; + divesToSelect.push_back({ d->divetrip, d }); + } else if (!newState && d->selected) { + d->selected = false; + divesToDeselect.push_back({ d->divetrip, d }); + } + } + + // Send the select and deselect signals + processByTrip(divesToSelect, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + emit diveListNotifier.divesSelected(trip, divesInTrip); + }); + processByTrip(divesToDeselect, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + emit diveListNotifier.divesDeselected(trip, divesInTrip); + }); + + bool currentDiveChanged = false; + if (current_dive != currentDive) { + currentDiveChanged = true; + + // We cannot simply change the currentd dive to the given dive. + // It might be hidden by a filter and thus not be selected. + if (currentDive->selected) + // Current dive is visible and selected. Excellent. + current_dive = currentDive; + else + // Current not visible -> find a different dive. + setClosestCurrentDive(currentDive->when, selection); + emit diveListNotifier.currentDiveChanged(); + } + + // If anything changed (selection or current dive), send a final signal. + if (!divesToSelect.empty() || !divesToDeselect.empty() || currentDiveChanged) + selectionChanged = true; +} + +void DiveListBase::undo() +{ + auto marker = diveListNotifier.enterCommand(); + initWork(); + undoit(); + finishWork(); +} + +void DiveListBase::redo() +{ + auto marker = diveListNotifier.enterCommand(); + initWork(); + redoit(); + finishWork(); +} + +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); + + // 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 = 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) + allocTrip.reset(trip); + } + + int idx = dive_get_insertion_index(divePtr.get()); + if (newNumber) + divePtr->number = get_dive_nr_at_idx(idx); + + divesToAdd.push_back({ std::move(divePtr), std::move(allocTrip), trip, idx }); +} + +bool AddDive::workToBeDone() +{ + return true; +} + +void AddDive::redoit() +{ + // Remember selection so that we can undo it + selection = getDiveSelection(); + currentDive = current_dive; + + divesToRemove = addDives(divesToAdd); + mark_divelist_changed(true); + + // Select the newly added dive + restoreSelection(divesToRemove, divesToRemove[0]); + + // Exit from edit mode, but don't recalculate dive list + // TODO: Remove edit mode + MainWindow::instance()->refreshDisplay(false); +} + +void AddDive::undoit() +{ + // Simply remove the dive that was previously added... + divesToAdd = removeDives(divesToRemove); + + // ...and restore the selection + restoreSelection(selection, currentDive); + + // Exit from edit mode, but don't recalculate dive list + // TODO: Remove edit mode + MainWindow::instance()->refreshDisplay(false); +} + +DeleteDive::DeleteDive(const QVector<struct dive*> &divesToDeleteIn) : divesToDelete(divesToDeleteIn.toStdVector()) +{ + setText(tr("delete %n dive(s)", "", divesToDelete.size())); +} + +bool DeleteDive::workToBeDone() +{ + return !divesToDelete.empty(); +} + +void DeleteDive::undoit() +{ + divesToDelete = addDives(divesToAdd); + mark_divelist_changed(true); + + // Select all re-added dives and make the first one current + dive *currentDive = !divesToDelete.empty() ? divesToDelete[0] : nullptr; + restoreSelection(divesToDelete, currentDive); +} + +void DeleteDive::redoit() +{ + divesToAdd = removeDives(divesToDelete); + mark_divelist_changed(true); + + // Deselect all dives and select dive that was close to the first deleted dive + dive *newCurrent = nullptr; + if (!divesToAdd.empty()) { + timestamp_t when = divesToAdd[0].dive->when; + newCurrent = find_next_visible_dive(when); + } + if (newCurrent) + restoreSelection(std::vector<dive *>{ newCurrent }, newCurrent); + else + restoreSelection(std::vector<dive *>(), nullptr); +} + + +ShiftTime::ShiftTime(const QVector<dive *> &changedDives, int amount) + : diveList(changedDives), timeChanged(amount) +{ + setText(tr("shift time of %n dives", "", changedDives.count())); +} + +void ShiftTime::redoit() +{ + for (dive *d: diveList) + d->when -= timeChanged; + + // Changing times may have unsorted the dive table + sort_table(&dive_table); + + // We send one dives-deleted signal per trip (see comments in DiveListNotifier.h). + // Therefore, collect all dives in a array and sort by trip. + std::vector<std::pair<dive_trip *, dive *>> dives; + dives.reserve(diveList.size()); + for (dive *d: diveList) + dives.push_back({ d->divetrip, d }); + + // Send signals. + processByTrip(dives, [&](dive_trip *trip, const QVector<dive *> &divesInTrip) { + emit diveListNotifier.divesTimeChanged(trip, timeChanged, divesInTrip); + }); + + // Negate the time-shift so that the next call does the reverse + timeChanged = -timeChanged; + + mark_divelist_changed(true); +} + +bool ShiftTime::workToBeDone() +{ + return !diveList.isEmpty(); +} + +void ShiftTime::undoit() +{ + // Same as redoit(), since after redoit() we reversed the timeOffset + redoit(); +} + + +RenumberDives::RenumberDives(const QVector<QPair<dive *, int>> &divesToRenumberIn) : divesToRenumber(divesToRenumberIn) +{ + setText(tr("renumber %n dive(s)", "", divesToRenumber.count())); +} + +void RenumberDives::undoit() +{ + renumberDives(divesToRenumber); + mark_divelist_changed(true); +} + +bool RenumberDives::workToBeDone() +{ + return !divesToRenumber.isEmpty(); +} + +void RenumberDives::redoit() +{ + // Redo and undo do the same thing! + undoit(); +} + +bool TripBase::workToBeDone() +{ + return !divesToMove.divesToMove.empty(); +} + +void TripBase::redoit() +{ + moveDivesBetweenTrips(divesToMove); + + mark_divelist_changed(true); +} + +void TripBase::undoit() +{ + // Redo and undo do the same thing! + redoit(); +} + +RemoveDivesFromTrip::RemoveDivesFromTrip(const QVector<dive *> &divesToRemove) +{ + setText(tr("remove %n dive(s) from trip", "", divesToRemove.size())); + divesToMove.divesToMove.reserve(divesToRemove.size()); + for (dive *d: divesToRemove) + divesToMove.divesToMove.push_back( {d, nullptr} ); +} + +RemoveAutogenTrips::RemoveAutogenTrips() +{ + setText(tr("remove autogenerated trips")); + // TODO: don't touch core-innards directly + int i; + struct dive *dive; + for_each_dive(i, dive) { + if (dive->divetrip && dive->divetrip->autogen) + divesToMove.divesToMove.push_back( {dive, nullptr} ); + } +} + +AddDivesToTrip::AddDivesToTrip(const QVector<dive *> &divesToAddIn, dive_trip *trip) +{ + setText(tr("add %n dives to trip", "", divesToAddIn.size())); + for (dive *d: divesToAddIn) + divesToMove.divesToMove.push_back( {d, trip} ); +} + +CreateTrip::CreateTrip(const QVector<dive *> &divesToAddIn) +{ + setText(tr("create trip")); + + if (divesToAddIn.isEmpty()) + return; + + dive_trip *trip = create_trip_from_dive(divesToAddIn[0]); + divesToMove.tripsToAdd.emplace_back(trip); + for (dive *d: divesToAddIn) + divesToMove.divesToMove.push_back( {d, trip} ); +} + +AutogroupDives::AutogroupDives() +{ + setText(tr("autogroup dives")); + + dive_trip *trip; + bool alloc; + int from, to; + for(int i = 0; (trip = get_dives_to_autogroup(i, &from, &to, &alloc)) != NULL; i = to) { + // If this is an allocated trip, take ownership + if (alloc) + divesToMove.tripsToAdd.emplace_back(trip); + for (int j = from; j < to; ++j) + divesToMove.divesToMove.push_back( { get_dive(j), trip } ); + } +} + +MergeTrips::MergeTrips(dive_trip *trip1, dive_trip *trip2) +{ + if (trip1 == trip2) + return; + dive_trip *newTrip = combine_trips_create(trip1, trip2); + divesToMove.tripsToAdd.emplace_back(newTrip); + for (dive *d = trip1->dives; d; d = d->next) + divesToMove.divesToMove.push_back( { d, newTrip } ); + for (dive *d = trip2->dives; d; d = d->next) + divesToMove.divesToMove.push_back( { d, newTrip } ); +} + +SplitDives::SplitDives(dive *d, duration_t time) +{ + setText(tr("split dive")); + + // Split the dive + dive *new1, *new2; + int idx = time.seconds < 0 ? + split_dive_dont_insert(d, &new1, &new2) : + split_dive_at_time_dont_insert(d, time, &new1, &new2); + + // If this didn't work, simply return. Empty arrays indicate that nothing is to be done. + if (idx < 0) + return; + + // Currently, the core code selects the dive -> this is not what we want, as + // we manually manage the selection post-command. + // TODO: Reset selection in core. + new1->selected = false; + new2->selected = false; + + diveToSplit.push_back(d); + splitDives.resize(2); + splitDives[0].dive.reset(new1); + splitDives[0].trip = d->divetrip; + splitDives[0].idx = idx; + splitDives[1].dive.reset(new2); + splitDives[1].trip = d->divetrip; + splitDives[1].idx = idx + 1; +} + +bool SplitDives::workToBeDone() +{ + return !diveToSplit.empty(); +} + +void SplitDives::redoit() +{ + divesToUnsplit = addDives(splitDives); + unsplitDive = removeDives(diveToSplit); + mark_divelist_changed(true); + + // Select split dives and make first dive current + restoreSelection(divesToUnsplit, divesToUnsplit[0]); +} + +void SplitDives::undoit() +{ + // Note: reverse order with respect to redoit() + diveToSplit = addDives(unsplitDive); + splitDives = removeDives(divesToUnsplit); + mark_divelist_changed(true); + + // Select unsplit dive and make it current + restoreSelection(diveToSplit, diveToSplit[0] ); +} + +MergeDives::MergeDives(const QVector <dive *> &dives) +{ + setText(tr("merge dive")); + + // Just a safety check - if there's not two or more dives - do nothing + // The caller should have made sure that this doesn't happen. + if (dives.count() < 2) { + qWarning() << "Merging less than two dives"; + return; + } + + dive_trip *preferred_trip; + OwningDivePtr d(merge_dives(dives[0], dives[1], dives[1]->when - dives[0]->when, false, &preferred_trip)); + + // Currently, the core code selects the dive -> this is not what we want, as + // we manually manage the selection post-command. + // TODO: Remove selection code from core. + d->selected = false; + + // Set the preferred dive trip, so that for subsequent merges the better trip can be selected + d->divetrip = preferred_trip; + for (int i = 2; i < dives.count(); ++i) { + d.reset(merge_dives(d.get(), dives[i], dives[i]->when - d->when, false, &preferred_trip)); + // Set the preferred dive trip, so that for subsequent merges the better trip can be selected + d->divetrip = preferred_trip; + } + + // We got our preferred trip, so now the reference can be deleted from the newly generated dive + d->divetrip = nullptr; + + // The merged dive gets the number of the first dive + d->number = dives[0]->number; + + // We will only renumber the remaining dives if the joined dives are consecutive. + // Otherwise all bets are off concerning what the user wanted and doing nothing seems + // like the best option. + int idx = get_divenr(dives[0]); + int num = dives.count(); + if (idx < 0 || idx + num > dive_table.nr) { + // It was the callers responsibility to pass only known dives. + // Something is seriously wrong - give up. + qWarning() << "Merging unknown dives"; + return; + } + // std::equal compares two ranges. The parameters are (begin_range1, end_range1, begin_range2). + // Here, we can compare C-arrays, because QVector guarantees contiguous storage. + if (std::equal(&dives[0], &dives[0] + num, &dive_table.dives[idx]) && + dives[0]->number && dives.last()->number && dives[0]->number < dives.last()->number) { + // We have a consecutive set of dives. Rename all following dives according to the + // number of erased dives. This considers that there might be missing numbers. + // Comment copied from core/divelist.c: + // So if you had a dive list 1 3 6 7 8, and you + // merge 1 and 3, the resulting numbered list will + // be 1 4 5 6, because we assume that there were + // some missing dives (originally dives 4 and 5), + // that now will still be missing (dives 2 and 3 + // in the renumbered world). + // + // Obviously the normal case is that everything is + // consecutive, and the difference will be 1, so the + // above example is not supposed to be normal. + int diff = dives.last()->number - dives[0]->number; + divesToRenumber.reserve(dive_table.nr - idx - num); + int previousnr = dives[0]->number; + for (int i = idx + num; i < dive_table.nr; ++i) { + int newnr = dive_table.dives[i]->number - diff; + + // Stop renumbering if stuff isn't in order (see also core/divelist.c) + if (newnr <= previousnr) + break; + divesToRenumber.append(QPair<dive *,int>(dive_table.dives[i], newnr)); + previousnr = newnr; + } + } + + mergedDive.resize(1); + mergedDive[0].dive = std::move(d); + mergedDive[0].idx = get_divenr(dives[0]); + mergedDive[0].trip = preferred_trip; + divesToMerge = dives.toStdVector(); +} + +bool MergeDives::workToBeDone() +{ + return !mergedDive.empty(); +} + +void MergeDives::redoit() +{ + renumberDives(divesToRenumber); + diveToUnmerge = addDives(mergedDive); + unmergedDives = removeDives(divesToMerge); + + // Select merged dive and make it current + restoreSelection(diveToUnmerge, diveToUnmerge[0]); +} + +void MergeDives::undoit() +{ + divesToMerge = addDives(unmergedDives); + mergedDive = removeDives(diveToUnmerge); + renumberDives(divesToRenumber); + + // Select unmerged dives and make first one current + restoreSelection(divesToMerge, divesToMerge[0]); +} + +} // namespace Command diff --git a/desktop-widgets/tab-widgets/maintab.cpp b/desktop-widgets/tab-widgets/maintab.cpp index 766d32414..e5974a834 100644 --- a/desktop-widgets/tab-widgets/maintab.cpp +++ b/desktop-widgets/tab-widgets/maintab.cpp @@ -26,6 +26,7 @@ #include "core/subsurface-string.h" #include "core/gettextfromc.h" #include "desktop-widgets/locationinformation.h" +#include "desktop-widgets/undocommands.h" #include "TabDiveExtraInfo.h" #include "TabDiveInformation.h" @@ -788,23 +789,30 @@ void MainTab::acceptChanges() hideMessage(); ui.equipmentTab->setEnabled(true); if (editMode == ADD) { - // We need to add the dive we just created to the dive list and select it. - // Easy, right? - struct dive *added_dive = clone_dive(&displayed_dive); - record_dive(added_dive); - addedId = added_dive->id; // make sure that the dive site is handled as well - updateDiveSite(ui.location->currDiveSiteUuid(), added_dive); + updateDiveSite(ui.location->currDiveSiteUuid(), &displayed_dive); - // unselect everything as far as the UI is concerned and select the new - // dive - we'll have to undo/redo this later after we resort the dive_table - // but we need the dive selected for the middle part of this function - this - // way we can reuse the code used for editing dives - MainWindow::instance()->dive_list()->unselectDives(); - selected_dive = get_divenr(added_dive); - amount_selected = 1; - // finally, make sure we get the tags - saveTags(); + UndoAddDive *undoCommand = new UndoAddDive(&displayed_dive); + MainWindow::instance()->undoStack->push(undoCommand); + + editMode = NONE; + MainWindow::instance()->exitEditState(); + cylindersModel->changed = false; + weightModel->changed = false; + MainWindow::instance()->setEnabledToolbar(true); + acceptingEdit = false; + ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty()); + emit addDiveFinished(); + MainWindow::instance()->dive_list()->reload(DiveTripModel::CURRENT, true); + 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(); + saveTags(QVector<dive *>{ &displayed_dive }); + displayed_dive.divetrip = nullptr; // Should not be necessary, just in case! + Command::addDive(&displayed_dive, autogroup, true); + return; } else if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) { /* now figure out if things have changed */ if (displayedTrip.notes && !same_string(displayedTrip.notes, currentTrip->notes)) { @@ -957,7 +965,7 @@ void MainTab::acceptChanges() current_dive->divetrip->when = current_dive->when; find_new_trip_start_time(current_dive->divetrip); } - if (editMode == ADD || editMode == MANUALLY_ADDED_DIVE) { + if (editMode == MANUALLY_ADDED_DIVE) { // we just added or edited the dive, let fixup_dive() make // sure we get the max. depth right current_dive->maxdepth.mm = current_dc->maxdepth.mm = 0; @@ -969,7 +977,7 @@ void MainTab::acceptChanges() } int scrolledBy = MainWindow::instance()->dive_list()->verticalScrollBar()->sliderPosition(); resetPallete(); - if (editMode == ADD || editMode == MANUALLY_ADDED_DIVE) { + if (editMode == MANUALLY_ADDED_DIVE) { // since a newly added dive could be in the middle of the dive_table we need // to resort the dive list and make sure the newly added dive gets selected again sort_table(&dive_table); @@ -980,7 +988,6 @@ void MainTab::acceptChanges() editMode = NONE; MainWindow::instance()->refreshDisplay(); MainWindow::instance()->graphics()->replot(); - emit addDiveFinished(); } else { editMode = NONE; if (do_replot) diff --git a/desktop-widgets/undocommands.cpp b/desktop-widgets/undocommands.cpp index d58ac7a4b..668a84f9e 100644 --- a/desktop-widgets/undocommands.cpp +++ b/desktop-widgets/undocommands.cpp @@ -1,10 +1,83 @@ // SPDX-License-Identifier: GPL-2.0 #include "desktop-widgets/undocommands.h" #include "desktop-widgets/mainwindow.h" +#include "desktop-widgets/divelistview.h" #include "core/divelist.h" #include "core/subsurface-string.h" #include "core/gettextfromc.h" +// This helper function removes a dive, takes ownership of the dive and adds it to a DiveToAdd structure. +// It is crucial that dives are added in reverse order of deletion, so the the indices are correctly +// set and that the trips are added before they are used! +static DiveToAdd removeDive(struct dive *d) +{ + DiveToAdd res; + res.idx = get_divenr(d); + if (res.idx < 0) + qWarning() << "Deletion of unknown dive!"; + + // remove dive from trip - if this is the last dive in the trip + // remove the whole trip. + res.trip = unregister_dive_from_trip(d, false); + if (res.trip && res.trip->nrdives == 0) { + unregister_trip(res.trip); // Remove trip from backend + res.tripToAdd.reset(res.trip); // Take ownership of trip + } + + res.dive.reset(unregister_dive(res.idx)); // Remove dive from backend + return res; +} + +// This helper function adds a dive and returns ownership to the backend. It may also add a dive trip. +// It is crucial that dives are added in reverse order of deletion (see comment above)! +// Returns pointer to added dive (which is owned by the backend!) +static dive *addDive(DiveToAdd &d) +{ + if (d.tripToAdd) { + dive_trip *t = d.tripToAdd.release(); // Give up ownership of trip + insert_trip(&t); // Return ownership to backend + } + if (d.trip) + add_dive_to_trip(d.dive.get(), d.trip); + dive *res = d.dive.release(); // Give up ownership of dive + add_single_dive(d.idx, res); // Return ownership to backend + return res; +} + +UndoAddDive::UndoAddDive(dive *d) +{ + setText(gettextFromC::tr("add dive")); + // TODO: handle tags + //saveTags(); + d->maxdepth.mm = 0; + fixup_dive(d); + diveToAdd.trip = d->divetrip; + d->divetrip = nullptr; + diveToAdd.idx = dive_get_insertion_index(d); + d->number = get_dive_nr_at_idx(diveToAdd.idx); + diveToAdd.dive.reset(clone_dive(d)); +} + +void UndoAddDive::redo() +{ + diveToRemove = addDive(diveToAdd); + mark_divelist_changed(true); + + // Finally, do the UI stuff: + MainWindow::instance()->dive_list()->unselectDives(); + MainWindow::instance()->dive_list()->selectDive(diveToAdd.idx, true); + MainWindow::instance()->refreshDisplay(); +} + +void UndoAddDive::undo() +{ + // Simply remove the dive that was previously added + diveToAdd = removeDive(diveToRemove); + + // Finally, do the UI stuff: + MainWindow::instance()->refreshDisplay(); +} + UndoDeleteDive::UndoDeleteDive(const QVector<struct dive*> &divesToDeleteIn) : divesToDelete(divesToDeleteIn) { setText(tr("delete %n dive(s)", "", divesToDelete.size())); @@ -12,19 +85,9 @@ UndoDeleteDive::UndoDeleteDive(const QVector<struct dive*> &divesToDeleteIn) : d void UndoDeleteDive::undo() { - // first bring back the trip(s) - for (auto &trip: tripsToAdd) { - dive_trip *t = trip.release(); // Give up ownership - insert_trip(&t); // Return ownership to backend - } - tripsToAdd.clear(); + for (auto it = divesToAdd.rbegin(); it != divesToAdd.rend(); ++it) + divesToDelete.append(addDive(*it)); - for (DiveToAdd &d: divesToAdd) { - if (d.trip) - add_dive_to_trip(d.dive.get(), d.trip); - divesToDelete.append(d.dive.get()); // Delete dive on redo - add_single_dive(d.idx, d.dive.release()); // Return ownership to backend - } mark_divelist_changed(true); divesToAdd.clear(); @@ -34,24 +97,9 @@ void UndoDeleteDive::undo() void UndoDeleteDive::redo() { - for (dive *d: divesToDelete) { - int idx = get_divenr(d); - if (idx < 0) { - qWarning() << "Deletion of unknown dive!"; - continue; - } - // remove dive from trip - if this is the last dive in the trip - // remove the whole trip. - dive_trip *trip = unregister_dive_from_trip(d, false); - if (trip && trip->nrdives == 0) { - unregister_trip(trip); // Remove trip from backend - tripsToAdd.emplace_back(trip); // Take ownership of trip - } + for (dive *d: divesToDelete) + divesToAdd.push_back(removeDive(d)); - unregister_dive(idx); // Remove dive from backend - divesToAdd.push_back({ OwningDivePtr(d), trip, idx }); - // Take ownership for dive - } divesToDelete.clear(); mark_divelist_changed(true); diff --git a/desktop-widgets/undocommands.h b/desktop-widgets/undocommands.h index bbc1f05bd..4c4073454 100644 --- a/desktop-widgets/undocommands.h +++ b/desktop-widgets/undocommands.h @@ -148,6 +148,29 @@ struct TripDeleter { typedef std::unique_ptr<dive, DiveDeleter> OwningDivePtr; typedef std::unique_ptr<dive_trip, TripDeleter> OwningTripPtr; +// This helper structure describes a dive that we want to add. +// Potentially it also adds a trip (if deletion of the dive resulted in deletion of the trip) +struct DiveToAdd { + OwningDivePtr dive; // Dive to add + OwningTripPtr tripToAdd; // Not-null if we also have to add a dive + dive_trip *trip; // Trip the dive belongs to, may be null + int idx; // Position in divelist +}; + +class UndoAddDive : public QUndoCommand { +public: + UndoAddDive(dive *dive); // Warning: old dive will be erased (moved in C++-speak)! +private: + void undo() override; + void redo() override; + + // For redo + DiveToAdd diveToAdd; + + // For undo + dive *diveToRemove; +}; + class UndoDeleteDive : public QUndoCommand { Q_DECLARE_TR_FUNCTIONS(Command) public: @@ -159,12 +182,6 @@ private: // For redo QVector<struct dive*> divesToDelete; - // For undo - struct DiveToAdd { - OwningDivePtr dive; // Dive to add - dive_trip *trip; // Trip, may be null - int idx; // Position in divelist - }; std::vector<OwningTripPtr> tripsToAdd; std::vector<DiveToAdd> divesToAdd; }; |