summaryrefslogtreecommitdiffstats
path: root/commands/command_edit.cpp
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2020-03-27 09:22:11 +0100
committerGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2020-04-07 00:13:35 +0200
commit4e8a838f746d7319b97b56a209651a65655aca7f (patch)
tree47fad5a73fd53063d3cb7f384ef7fbf15fb26915 /commands/command_edit.cpp
parent5b65776e4313bb088e3eb3dcb1e8d5732be3b1d0 (diff)
downloadsubsurface-4e8a838f746d7319b97b56a209651a65655aca7f.tar.gz
undo: store all cylinders in EditCylinderBase
We stored only one cylinder in EditCylinderBase, which is the base class of RemoveCylinder and EditCylinder. This turns out to be too crude: when removing the "same" cylinder from multiple dives, there are some "hidden variables" such as bestmix_o2 or manually_added, which might actually be different. We don't want to overwrite those on undo of delete. Moreover, the cylinder edit is way too crude at the moment, as it overwrites the whole cylinder even when the user edited only a single field. To enable a more refined edit, we have to store each changed cylinder. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
Diffstat (limited to 'commands/command_edit.cpp')
-rw-r--r--commands/command_edit.cpp62
1 files changed, 25 insertions, 37 deletions
diff --git a/commands/command_edit.cpp b/commands/command_edit.cpp
index 8914485c6..d5e2d12d0 100644
--- a/commands/command_edit.cpp
+++ b/commands/command_edit.cpp
@@ -1086,40 +1086,35 @@ static int find_cylinder_index(const struct dive *d, const cylinder_t &cyl, int
}
EditCylinderBase::EditCylinderBase(int index, bool currentDiveOnly, bool nonProtectedOnly, int sameCylinderFlags) :
- EditDivesBase(currentDiveOnly),
- cyl(empty_cylinder)
+ EditDivesBase(currentDiveOnly)
{
// Get the old cylinder, bail if index is invalid
if (!current || index < 0 || index >= current->cylinders.nr) {
dives.clear();
return;
}
- cyl = clone_cylinder(current->cylinders.cylinders[index]);
+ const cylinder_t &orig = current->cylinders.cylinders[index];
std::vector<dive *> divesNew;
divesNew.reserve(dives.size());
indexes.reserve(dives.size());
+ cyl.reserve(dives.size());
for (dive *d: dives) {
- if (d == current) {
- if (nonProtectedOnly && is_cylinder_prot(d, index))
- continue;
- divesNew.push_back(d);
- indexes.push_back(index);
- continue;
- }
- int idx = find_cylinder_index(d, cyl, sameCylinderFlags);
+ int idx = d == current ? index : find_cylinder_index(d, orig, sameCylinderFlags);
if (idx < 0 || (nonProtectedOnly && is_cylinder_prot(d, idx)))
continue;
divesNew.push_back(d);
indexes.push_back(idx);
+ cyl.push_back(clone_cylinder(d->cylinders.cylinders[idx]));
}
dives = std::move(divesNew);
}
EditCylinderBase::~EditCylinderBase()
{
- free_cylinder(cyl);
+ for (cylinder_t c: cyl)
+ free_cylinder(c);
}
bool EditCylinderBase::workToBeDone()
@@ -1141,7 +1136,7 @@ void RemoveCylinder::undo()
{
for (size_t i = 0; i < dives.size(); ++i) {
std::vector<int> mapping = get_cylinder_map_for_add(dives[i]->cylinders.nr, indexes[i]);
- add_to_cylinder_table(&dives[i]->cylinders, indexes[i], clone_cylinder(cyl));
+ add_to_cylinder_table(&dives[i]->cylinders, indexes[i], clone_cylinder(cyl[i]));
emit diveListNotifier.cylinderAdded(dives[i], indexes[i]);
}
}
@@ -1158,8 +1153,7 @@ void RemoveCylinder::redo()
// ***** Edit Cylinder *****
EditCylinder::EditCylinder(int index, cylinder_t cylIn, bool currentDiveOnly) :
- EditCylinderBase(index, currentDiveOnly, false, SAME_TYPE | SAME_PRESS | SAME_GAS),
- new_cyl(empty_cylinder)
+ EditCylinderBase(index, currentDiveOnly, false, SAME_TYPE | SAME_PRESS | SAME_GAS)
{
if (dives.empty())
return;
@@ -1170,44 +1164,38 @@ EditCylinder::EditCylinder(int index, cylinder_t cylIn, bool currentDiveOnly) :
setText(tr("Edit cylinder (%n dive(s))", "", dives.size()));
// Try to untranslate the cylinder type
- new_cyl = clone_cylinder(cylIn);
- QString vString(new_cyl.type.description);
+ QString description = cylIn.type.description;
for (int i = 0; i < MAX_TANK_INFO && tank_info[i].name; ++i) {
- if (gettextFromC::tr(tank_info[i].name) == vString) {
- free_cylinder(new_cyl);
- new_cyl.type.description = copy_string(tank_info[i].name);
+ if (gettextFromC::tr(tank_info[i].name) == description) {
+ description = tank_info[i].name;
break;
}
}
- // If that doesn't change anything, do nothing
- if (same_cylinder_with_flags(cyl, new_cyl, SAME_TYPE | SAME_PRESS | SAME_GAS)) {
- dives.clear();
- return;
- }
-
+ // Update the tank info model
TankInfoModel *tim = TankInfoModel::instance();
- QModelIndexList matches = tim->match(tim->index(0, 0), Qt::DisplayRole, gettextFromC::tr(new_cyl.type.description));
+ QModelIndexList matches = tim->match(tim->index(0, 0), Qt::DisplayRole, gettextFromC::tr(cylIn.type.description));
if (!matches.isEmpty()) {
- if (new_cyl.type.size.mliter != cyl.type.size.mliter)
- tim->setData(tim->index(matches.first().row(), TankInfoModel::ML), new_cyl.type.size.mliter);
- if (new_cyl.type.workingpressure.mbar != cyl.type.workingpressure.mbar)
- tim->setData(tim->index(matches.first().row(), TankInfoModel::BAR), new_cyl.type.workingpressure.mbar / 1000.0);
+ if (cylIn.type.size.mliter != cyl[0].type.size.mliter)
+ tim->setData(tim->index(matches.first().row(), TankInfoModel::ML), cylIn.type.size.mliter);
+ if (cylIn.type.workingpressure.mbar != cyl[0].type.workingpressure.mbar)
+ tim->setData(tim->index(matches.first().row(), TankInfoModel::BAR), cylIn.type.workingpressure.mbar / 1000.0);
}
-}
-EditCylinder::~EditCylinder()
-{
- free_cylinder(new_cyl);
+ // The base class copied the cylinders for us, let's edit them
+ for (int i = 0; i < (int)indexes.size(); ++i) {
+ free_cylinder(cyl[i]);
+ cyl[i] = cylIn;
+ cyl[i].type.description = copy_qstring(description);
+ }
}
void EditCylinder::redo()
{
for (size_t i = 0; i < dives.size(); ++i) {
- set_cylinder(dives[i], indexes[i], new_cyl);
+ std::swap(dives[i]->cylinders.cylinders[indexes[i]], cyl[i]);
emit diveListNotifier.cylinderEdited(dives[i], indexes[i]);
}
- std::swap(cyl, new_cyl);
}
// Undo and redo do the same as just the stored value is exchanged