2020-03-03 21:34:50 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
#include "command_event.h"
|
|
|
|
#include "core/dive.h"
|
2020-10-25 08:14:16 +00:00
|
|
|
#include "core/event.h"
|
2020-03-05 22:10:44 +00:00
|
|
|
#include "core/selection.h"
|
2020-03-03 21:34:50 +00:00
|
|
|
#include "core/subsurface-qt/divelistnotifier.h"
|
|
|
|
#include "core/libdivecomputer.h"
|
2020-03-03 21:54:54 +00:00
|
|
|
#include "core/gettextfromc.h"
|
2020-03-04 20:10:05 +00:00
|
|
|
#include <QVector>
|
2020-03-03 21:34:50 +00:00
|
|
|
|
|
|
|
namespace Command {
|
|
|
|
|
2020-03-04 16:35:02 +00:00
|
|
|
EventBase::EventBase(struct dive *dIn, int dcNrIn) :
|
|
|
|
d(dIn), dcNr(dcNrIn)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void EventBase::redo()
|
|
|
|
{
|
|
|
|
redoit(); // Call actual function in base class
|
2020-03-05 22:10:44 +00:00
|
|
|
updateDive();
|
2020-03-04 16:35:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void EventBase::undo()
|
|
|
|
{
|
|
|
|
undoit(); // Call actual function in base class
|
2020-03-05 22:10:44 +00:00
|
|
|
updateDive();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EventBase::updateDive()
|
|
|
|
{
|
2020-03-04 16:35:02 +00:00
|
|
|
invalidate_dive_cache(d);
|
|
|
|
emit diveListNotifier.eventsChanged(d);
|
2022-04-04 16:57:28 +00:00
|
|
|
setSelection({ d }, d, dcNr);
|
2020-03-04 16:35:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AddEventBase::AddEventBase(struct dive *d, int dcNr, struct event *ev) : EventBase(d, dcNr),
|
|
|
|
eventToAdd(ev)
|
2020-03-03 21:34:50 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AddEventBase::workToBeDone()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-03-04 16:35:02 +00:00
|
|
|
void AddEventBase::redoit()
|
2020-03-03 21:34:50 +00:00
|
|
|
{
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
eventToRemove = eventToAdd.get();
|
|
|
|
add_event_to_dc(dc, eventToAdd.release()); // return ownership to backend
|
|
|
|
}
|
|
|
|
|
2020-03-04 16:35:02 +00:00
|
|
|
void AddEventBase::undoit()
|
2020-03-03 21:34:50 +00:00
|
|
|
{
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
remove_event_from_dc(dc, eventToRemove);
|
|
|
|
eventToAdd.reset(eventToRemove); // take ownership of event
|
|
|
|
eventToRemove = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
AddEventBookmark::AddEventBookmark(struct dive *d, int dcNr, int seconds) :
|
|
|
|
AddEventBase(d, dcNr, create_event(seconds, SAMPLE_EVENT_BOOKMARK, 0, 0, "bookmark"))
|
|
|
|
{
|
2020-11-08 22:38:59 +00:00
|
|
|
setText(Command::Base::tr("Add bookmark"));
|
2020-03-03 21:34:50 +00:00
|
|
|
}
|
|
|
|
|
2020-03-03 21:54:54 +00:00
|
|
|
AddEventDivemodeSwitch::AddEventDivemodeSwitch(struct dive *d, int dcNr, int seconds, int divemode) :
|
|
|
|
AddEventBase(d, dcNr, create_event(seconds, SAMPLE_EVENT_BOOKMARK, 0, divemode, QT_TRANSLATE_NOOP("gettextFromC", "modechange")))
|
|
|
|
{
|
2020-11-08 22:38:59 +00:00
|
|
|
setText(Command::Base::tr("Add dive mode switch to %1").arg(gettextFromC::tr(divemode_text_ui[divemode])));
|
2020-03-03 21:54:54 +00:00
|
|
|
}
|
|
|
|
|
2020-03-03 22:31:46 +00:00
|
|
|
AddEventSetpointChange::AddEventSetpointChange(struct dive *d, int dcNr, int seconds, pressure_t pO2) :
|
2021-01-11 15:52:02 +00:00
|
|
|
AddEventBase(d, dcNr, create_event(seconds, SAMPLE_EVENT_PO2, 0, pO2.mbar, QT_TRANSLATE_NOOP("gettextFromC", "SP change"))),
|
|
|
|
divemode(CCR)
|
2020-03-03 22:31:46 +00:00
|
|
|
{
|
2020-11-08 22:38:59 +00:00
|
|
|
setText(Command::Base::tr("Add set point change")); // TODO: format pO2 value in bar or psi.
|
2020-03-03 22:31:46 +00:00
|
|
|
}
|
|
|
|
|
2021-01-11 15:52:02 +00:00
|
|
|
void AddEventSetpointChange::undoit()
|
|
|
|
{
|
|
|
|
AddEventBase::undoit();
|
|
|
|
std::swap(get_dive_dc(d, dcNr)->divemode, divemode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddEventSetpointChange::redoit()
|
|
|
|
{
|
|
|
|
AddEventBase::redoit();
|
|
|
|
std::swap(get_dive_dc(d, dcNr)->divemode, divemode);
|
|
|
|
}
|
|
|
|
|
2020-03-04 17:13:02 +00:00
|
|
|
RenameEvent::RenameEvent(struct dive *d, int dcNr, struct event *ev, const char *name) : EventBase(d, dcNr),
|
|
|
|
eventToAdd(clone_event_rename(ev, name)),
|
|
|
|
eventToRemove(ev)
|
|
|
|
{
|
2020-11-08 22:38:59 +00:00
|
|
|
setText(Command::Base::tr("Rename bookmark to %1").arg(name));
|
2020-03-04 17:13:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool RenameEvent::workToBeDone()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenameEvent::redoit()
|
|
|
|
{
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
swap_event(dc, eventToRemove, eventToAdd.get());
|
|
|
|
event *tmp = eventToRemove;
|
|
|
|
eventToRemove = eventToAdd.release();
|
|
|
|
eventToAdd.reset(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenameEvent::undoit()
|
|
|
|
{
|
|
|
|
// Undo and redo do the same thing - they simply swap events
|
|
|
|
redoit();
|
|
|
|
}
|
|
|
|
|
2020-03-04 17:25:47 +00:00
|
|
|
RemoveEvent::RemoveEvent(struct dive *d, int dcNr, struct event *ev) : EventBase(d, dcNr),
|
2020-03-06 22:20:58 +00:00
|
|
|
eventToRemove(ev),
|
|
|
|
cylinder(ev->type == SAMPLE_EVENT_GASCHANGE2 || ev->type == SAMPLE_EVENT_GASCHANGE ?
|
|
|
|
ev->gas.index : -1)
|
2020-03-04 17:25:47 +00:00
|
|
|
{
|
2020-11-08 22:38:59 +00:00
|
|
|
setText(Command::Base::tr("Remove %1 event").arg(ev->name));
|
2020-03-04 17:25:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool RemoveEvent::workToBeDone()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveEvent::redoit()
|
|
|
|
{
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
remove_event_from_dc(dc, eventToRemove);
|
|
|
|
eventToAdd.reset(eventToRemove); // take ownership of event
|
|
|
|
eventToRemove = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveEvent::undoit()
|
|
|
|
{
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
eventToRemove = eventToAdd.get();
|
|
|
|
add_event_to_dc(dc, eventToAdd.release()); // return ownership to backend
|
|
|
|
}
|
|
|
|
|
2020-03-06 22:20:58 +00:00
|
|
|
void RemoveEvent::post() const
|
|
|
|
{
|
|
|
|
if (cylinder < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fixup_dive(d);
|
|
|
|
emit diveListNotifier.cylinderEdited(d, cylinder);
|
|
|
|
|
|
|
|
// TODO: This is silly we send a DURATION change event so that the statistics are recalculated.
|
|
|
|
// We should instead define a proper DiveField that expresses the change caused by a gas switch.
|
|
|
|
emit diveListNotifier.divesChanged(QVector<dive *>{ d }, DiveField::DURATION | DiveField::DEPTH);
|
|
|
|
}
|
|
|
|
|
2020-03-04 20:10:05 +00:00
|
|
|
AddGasSwitch::AddGasSwitch(struct dive *d, int dcNr, int seconds, int tank) : EventBase(d, dcNr)
|
|
|
|
{
|
|
|
|
// If there is a gas change at this time stamp, remove it before adding the new one.
|
|
|
|
// There shouldn't be more than one gas change per time stamp. Just in case we'll
|
|
|
|
// support that anyway.
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
struct event *gasChangeEvent = dc->events;
|
|
|
|
while ((gasChangeEvent = get_next_event_mutable(gasChangeEvent, "gaschange")) != NULL) {
|
|
|
|
if (gasChangeEvent->time.seconds == seconds) {
|
|
|
|
eventsToRemove.push_back(gasChangeEvent);
|
|
|
|
int idx = gasChangeEvent->gas.index;
|
|
|
|
if (std::find(cylinders.begin(), cylinders.end(), idx) == cylinders.end())
|
|
|
|
cylinders.push_back(idx); // cylinders might have changed their status
|
|
|
|
}
|
|
|
|
gasChangeEvent = gasChangeEvent->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
eventsToAdd.emplace_back(create_gas_switch_event(d, dc, seconds, tank));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AddGasSwitch::workToBeDone()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddGasSwitch::redoit()
|
|
|
|
{
|
2024-05-04 17:15:47 +00:00
|
|
|
std::vector<std::unique_ptr<event>> newEventsToAdd;
|
2020-03-04 20:10:05 +00:00
|
|
|
std::vector<event *> newEventsToRemove;
|
|
|
|
newEventsToAdd.reserve(eventsToRemove.size());
|
|
|
|
newEventsToRemove.reserve(eventsToAdd.size());
|
|
|
|
struct divecomputer *dc = get_dive_dc(d, dcNr);
|
|
|
|
|
|
|
|
for (event *ev: eventsToRemove) {
|
|
|
|
remove_event_from_dc(dc, ev);
|
|
|
|
newEventsToAdd.emplace_back(ev); // take ownership of event
|
|
|
|
}
|
2024-05-04 17:15:47 +00:00
|
|
|
for (auto &ev: eventsToAdd) {
|
2020-03-04 20:10:05 +00:00
|
|
|
newEventsToRemove.push_back(ev.get());
|
|
|
|
add_event_to_dc(dc, ev.release()); // return ownership to backend
|
|
|
|
}
|
|
|
|
eventsToAdd = std::move(newEventsToAdd);
|
|
|
|
eventsToRemove = std::move(newEventsToRemove);
|
|
|
|
|
|
|
|
// this means we potentially have a new tank that is being used and needs to be shown
|
|
|
|
fixup_dive(d);
|
|
|
|
|
|
|
|
for (int idx: cylinders)
|
|
|
|
emit diveListNotifier.cylinderEdited(d, idx);
|
|
|
|
|
|
|
|
// TODO: This is silly we send a DURATION change event so that the statistics are recalculated.
|
|
|
|
// We should instead define a proper DiveField that expresses the change caused by a gas switch.
|
|
|
|
emit diveListNotifier.divesChanged(QVector<dive *>{ d }, DiveField::DURATION | DiveField::DEPTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddGasSwitch::undoit()
|
|
|
|
{
|
|
|
|
// Undo and redo do the same thing, as the dives-to-be-added and dives-to-be-removed are exchanged.
|
|
|
|
redoit();
|
|
|
|
}
|
|
|
|
|
2020-03-03 21:34:50 +00:00
|
|
|
} // namespace Command
|