diff options
Diffstat (limited to 'desktop-widgets/undocommands.cpp')
-rw-r--r-- | desktop-widgets/undocommands.cpp | 163 |
1 files changed, 149 insertions, 14 deletions
diff --git a/desktop-widgets/undocommands.cpp b/desktop-widgets/undocommands.cpp index 45c796af8..9a56b4e9d 100644 --- a/desktop-widgets/undocommands.cpp +++ b/desktop-widgets/undocommands.cpp @@ -44,6 +44,50 @@ static dive *addDive(DiveToAdd &d) 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. +static std::vector<DiveToAdd> removeDives(std::vector<dive *> &divesToDelete) +{ + std::vector<DiveToAdd> res; + res.reserve(divesToDelete.size()); + + for (dive *d: divesToDelete) + res.push_back(removeDive(d)); + divesToDelete.clear(); + + 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. +static std::vector<dive *> addDives(std::vector<DiveToAdd> &divesToAdd) +{ + std::vector<dive *> res; + res.reserve(divesToAdd.size()); + + for (auto it = divesToAdd.rbegin(); it != divesToAdd.rend(); ++it) + res.push_back(addDive(*it)); + divesToAdd.clear(); + + 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 "UndoCommands". +static void renumberDives(QVector<QPair<int, int>> &divesToRenumber) +{ + for (auto &pair: divesToRenumber) { + dive *d = get_dive_by_uniq_id(pair.first); + if (!d) + continue; + std::swap(d->number, pair.second); + } +} + UndoAddDive::UndoAddDive(dive *d) { setText(gettextFromC::tr("add dive")); @@ -78,18 +122,16 @@ void UndoAddDive::undo() MainWindow::instance()->refreshDisplay(); } -UndoDeleteDive::UndoDeleteDive(const QVector<struct dive*> &divesToDeleteIn) : divesToDelete(divesToDeleteIn) +UndoDeleteDive::UndoDeleteDive(const QVector<struct dive*> &divesToDeleteIn) : divesToDelete(divesToDeleteIn.toStdVector()) { setText(tr("delete %n dive(s)", "", divesToDelete.size())); } void UndoDeleteDive::undo() { - for (auto it = divesToAdd.rbegin(); it != divesToAdd.rend(); ++it) - divesToDelete.append(addDive(*it)); + divesToDelete = addDives(divesToAdd); mark_divelist_changed(true); - divesToAdd.clear(); // Finally, do the UI stuff: MainWindow::instance()->refreshDisplay(); @@ -97,10 +139,7 @@ void UndoDeleteDive::undo() void UndoDeleteDive::redo() { - for (dive *d: divesToDelete) - divesToAdd.push_back(removeDive(d)); - - divesToDelete.clear(); + divesToAdd = removeDives(divesToDelete); mark_divelist_changed(true); // Finally, do the UI stuff: @@ -145,12 +184,7 @@ UndoRenumberDives::UndoRenumberDives(const QVector<QPair<int, int>> &divesToRenu void UndoRenumberDives::undo() { - for (auto &pair: divesToRenumber) { - dive *d = get_dive_by_uniq_id(pair.first); - if (!d) - continue; - std::swap(d->number, pair.second); - } + renumberDives(divesToRenumber); mark_divelist_changed(true); // Finally, do the UI stuff: @@ -260,3 +294,104 @@ void UndoSplitDives::undo() MainWindow::instance()->refreshDisplay(); MainWindow::instance()->refreshProfile(); } + +UndoMergeDives::UndoMergeDives(const QVector <dive *> &dives) +{ + setText(gettextFromC::tr("merge dive")); + + // We start in redo mode + diveToUnmerge = nullptr; + + // 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)); + + // 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<int,int>(dive_table.dives[i]->id, newnr)); + previousnr = newnr; + } + } + + mergedDive.dive = std::move(d); + mergedDive.idx = get_divenr(dives[0]); + mergedDive.trip = preferred_trip; + divesToMerge = dives.toStdVector(); +} + +void UndoMergeDives::redo() +{ + renumberDives(divesToRenumber); + diveToUnmerge = addDive(mergedDive); + unmergedDives = removeDives(divesToMerge); + + // Finally, do the UI stuff: + MainWindow::instance()->refreshDisplay(); + MainWindow::instance()->refreshProfile(); +} + +void UndoMergeDives::undo() +{ + divesToMerge = addDives(unmergedDives); + mergedDive = removeDive(diveToUnmerge); + renumberDives(divesToRenumber); + + // Finally, do the UI stuff: + MainWindow::instance()->refreshDisplay(); + MainWindow::instance()->refreshProfile(); +} |