mirror of
https://github.com/subsurface/subsurface.git
synced 2025-02-19 22:16:15 +00:00
This has UI changes: - The unhiding options are accessed by a field that appears when there are hidden events. - Only event-types of this particular dive are unhidden. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
212 lines
7.9 KiB
C++
212 lines
7.9 KiB
C++
// SPDX-License-Identifier: GPL-2.0
|
|
#include "profile-widget/diveeventitem.h"
|
|
#include "profile-widget/divecartesianaxis.h"
|
|
#include "profile-widget/divepixmapcache.h"
|
|
#include "profile-widget/animationfunctions.h"
|
|
#include "core/dive.h"
|
|
#include "core/event.h"
|
|
#include "core/eventtype.h"
|
|
#include "core/format.h"
|
|
#include "core/profile.h"
|
|
#include "core/gettextfromc.h"
|
|
#include "core/sample.h"
|
|
#include "core/subsurface-string.h"
|
|
|
|
#define DEPTH_NOT_FOUND (-2342)
|
|
|
|
static int depthAtTime(const plot_info &pi, duration_t time);
|
|
|
|
DiveEventItem::DiveEventItem(const struct dive *d, int idx, const struct event &ev, struct gasmix lastgasmix,
|
|
const plot_info &pi, DiveCartesianAxis *hAxis, DiveCartesianAxis *vAxis,
|
|
int speed, const DivePixmaps &pixmaps, QGraphicsItem *parent) : QGraphicsPixmapItem(parent),
|
|
text(setupToolTipString(d, ev, lastgasmix)),
|
|
pixmap(setupPixmap(d, ev, lastgasmix, pixmaps)),
|
|
vAxis(vAxis),
|
|
hAxis(hAxis),
|
|
idx(idx),
|
|
ev(ev),
|
|
dive(d),
|
|
depth(depthAtTime(pi, ev.time))
|
|
{
|
|
setFlag(ItemIgnoresTransformations);
|
|
setPixmap(pixmap);
|
|
recalculatePos();
|
|
|
|
if (ev.type == SAMPLE_EVENT_BOOKMARK)
|
|
setOffset(QPointF(0.0, -pixmap.height()));
|
|
}
|
|
|
|
DiveEventItem::~DiveEventItem()
|
|
{
|
|
}
|
|
|
|
QPixmap DiveEventItem::setupPixmap(const struct dive *dive, const struct event &ev, struct gasmix lastgasmix, const DivePixmaps &pixmaps)
|
|
{
|
|
event_severity severity = ev.get_severity();
|
|
if (ev.name.empty())
|
|
return pixmaps.warning;
|
|
|
|
if (same_string_caseinsensitive(ev.name.c_str(), "modechange"))
|
|
return ev.value == 0 ? pixmaps.bailout : pixmaps.onCCRLoop;
|
|
|
|
if (ev.type == SAMPLE_EVENT_BOOKMARK)
|
|
return pixmaps.bookmark;
|
|
|
|
if (ev.is_gaschange()) {
|
|
struct gasmix mix = dive->get_gasmix_from_event(ev);
|
|
struct icd_data icd_data;
|
|
bool icd = isobaric_counterdiffusion(lastgasmix, mix, &icd_data);
|
|
if (mix.he.permille)
|
|
return icd ? pixmaps.gaschangeTrimixICD : pixmaps.gaschangeTrimix;
|
|
if (gasmix_is_air(mix))
|
|
return icd ? pixmaps.gaschangeAirICD : pixmaps.gaschangeAir;
|
|
if (mix.o2.permille == 1000)
|
|
return icd ? pixmaps.gaschangeOxygenICD : pixmaps.gaschangeOxygen;
|
|
return icd ? pixmaps.gaschangeEANICD : pixmaps.gaschangeEAN;
|
|
}
|
|
if ((((ev.flags & SAMPLE_FLAGS_SEVERITY_MASK) >> SAMPLE_FLAGS_SEVERITY_SHIFT) == 1) ||
|
|
// those are useless internals of the dive computer
|
|
same_string_caseinsensitive(ev.name.c_str(), "heading") ||
|
|
(same_string_caseinsensitive(ev.name.c_str(), "SP change") && ev.time.seconds == 0)) {
|
|
// 2 cases:
|
|
// a) some dive computers have heading in every sample
|
|
// b) at t=0 we might have an "SP change" to indicate dive type
|
|
// in both cases we want to get the right data into the tooltip but don't want the visual clutter
|
|
// so set an "almost invisible" pixmap (a narrow but somewhat tall, basically transparent pixmap)
|
|
// that allows tooltips to work when we don't want to show a specific
|
|
// pixmap for an event, but want to show the event value in the tooltip
|
|
return pixmaps.transparent;
|
|
}
|
|
if (severity == EVENT_SEVERITY_INFO)
|
|
return pixmaps.info;
|
|
if (severity == EVENT_SEVERITY_WARN)
|
|
return pixmaps.warning;
|
|
if (severity == EVENT_SEVERITY_ALARM)
|
|
return pixmaps.violation;
|
|
if (same_string_caseinsensitive(ev.name.c_str(), "violation") || // generic libdivecomputer
|
|
same_string_caseinsensitive(ev.name.c_str(), "Safety stop violation") || // the rest are from the Uemis downloader
|
|
same_string_caseinsensitive(ev.name.c_str(), "pO₂ ascend alarm") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "RGT alert") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "Dive time alert") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "Low battery alert") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "Speed alarm"))
|
|
return pixmaps.violation;
|
|
if (same_string_caseinsensitive(ev.name.c_str(), "non stop time") || // generic libdivecomputer
|
|
same_string_caseinsensitive(ev.name.c_str(), "safety stop") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "safety stop (voluntary)") ||
|
|
same_string_caseinsensitive(ev.name.c_str(), "Tank change suggested") || // Uemis downloader
|
|
same_string_caseinsensitive(ev.name.c_str(), "Marker"))
|
|
return pixmaps.info;
|
|
|
|
// we should do some guessing based on the type / name of the event;
|
|
// for now they all get the warning icon
|
|
return pixmaps.warning;
|
|
}
|
|
|
|
QString DiveEventItem::setupToolTipString(const struct dive *dive, const struct event &ev, struct gasmix lastgasmix)
|
|
{
|
|
// we display the event on screen - so translate
|
|
QString name = gettextFromC::tr(ev.name.c_str());
|
|
int value = ev.value;
|
|
int type = ev.type;
|
|
|
|
if (ev.is_gaschange()) {
|
|
struct icd_data icd_data;
|
|
struct gasmix mix = dive->get_gasmix_from_event(ev);
|
|
name += ": ";
|
|
name += QString::fromStdString(mix.name());
|
|
|
|
/* Do we have an explicit cylinder index? Show it. */
|
|
if (ev.gas.index >= 0)
|
|
name += tr(" (cyl. %1)").arg(ev.gas.index + 1);
|
|
bool icd = isobaric_counterdiffusion(lastgasmix, mix, &icd_data);
|
|
if (icd_data.dHe < 0) {
|
|
name += qasprintf_loc("\n%s %s:%+.3g%% %s:%+.3g%%%s%+.3g%%",
|
|
qPrintable(tr("ICD")),
|
|
qPrintable(tr("ΔHe")), icd_data.dHe / 10.0,
|
|
qPrintable(tr("ΔN₂")), icd_data.dN2 / 10.0,
|
|
icd ? ">" : "<", lrint(-icd_data.dHe / 5.0) / 10.0);
|
|
}
|
|
} else if (ev.name == "modechange") {
|
|
name += QString(": %1").arg(gettextFromC::tr(divemode_text_ui[ev.value]));
|
|
} else if (value) {
|
|
if (type == SAMPLE_EVENT_PO2 && ev.name == "SP change") {
|
|
name += QString(": %1bar").arg((double)value / 1000, 0, 'f', 1);
|
|
} else if (type == SAMPLE_EVENT_CEILING && ev.name == "planned waypoint above ceiling") {
|
|
const char *depth_unit;
|
|
double depth_value = get_depth_units(value*1000, NULL, &depth_unit);
|
|
name += QString(": %1%2").arg((int) round(depth_value)).arg(depth_unit);
|
|
} else {
|
|
name += QString(": %1").arg(value);
|
|
}
|
|
} else if (type == SAMPLE_EVENT_PO2 && ev.name == "SP change") {
|
|
// this is a bad idea - we are abusing an existing event type that is supposed to
|
|
// warn of high or low pO₂ and are turning it into a setpoint change event
|
|
name += ":\n" + tr("Manual switch to OC");
|
|
} else {
|
|
name += ev.flags & SAMPLE_FLAGS_BEGIN ? tr(" begin", "Starts with space!") :
|
|
ev.flags & SAMPLE_FLAGS_END ? tr(" end", "Starts with space!") : "";
|
|
}
|
|
return name;
|
|
}
|
|
|
|
void DiveEventItem::eventVisibilityChanged(const QString&, bool)
|
|
{
|
|
//WARN: lookslike we should implement this.
|
|
}
|
|
|
|
static int depthAtTime(const plot_info &pi, duration_t time)
|
|
{
|
|
// Do a binary search for the timestamp
|
|
auto it = std::lower_bound(pi.entry.begin(), pi.entry.end(), time,
|
|
[](const plot_data &d1, duration_t t) { return d1.sec < t.seconds; });
|
|
if (it == pi.entry.end() || it->sec != time.seconds) {
|
|
qWarning("can't find a spot in the dataModel");
|
|
return DEPTH_NOT_FOUND;
|
|
}
|
|
return it->depth;
|
|
}
|
|
|
|
bool DiveEventItem::isInteresting(const struct dive *d, const struct divecomputer *dc,
|
|
const struct event &ev, const plot_info &pi,
|
|
int firstSecond, int lastSecond)
|
|
{
|
|
/*
|
|
* Ignore items outside of plot range
|
|
*/
|
|
if (ev.time.seconds < firstSecond || ev.time.seconds >= lastSecond)
|
|
return false;
|
|
|
|
/*
|
|
* Some gas change events are special. Some dive computers just tell us the initial gas this way.
|
|
* Don't bother showing those
|
|
*/
|
|
if (ev.name == "gaschange" &&
|
|
(ev.time.seconds == 0 ||
|
|
(!dc->samples.empty() && ev.time.seconds == dc->samples[0].time.seconds) ||
|
|
depthAtTime(pi, ev.time) < SURFACE_THRESHOLD))
|
|
return false;
|
|
|
|
/*
|
|
* Some divecomputers give "surface" events that just aren't interesting.
|
|
* Like at the beginning or very end of a dive. Well, duh.
|
|
*/
|
|
if (ev.name == "surface") {
|
|
int time = ev.time.seconds;
|
|
if (time <= 30 || time + 30 >= (int)dc->duration.seconds)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void DiveEventItem::recalculatePos()
|
|
{
|
|
if (depth == DEPTH_NOT_FOUND) {
|
|
hide();
|
|
return;
|
|
}
|
|
setVisible(!ev.hidden && !is_event_type_hidden(ev));
|
|
double x = hAxis->posAtValue(ev.time.seconds);
|
|
double y = vAxis->posAtValue(depth);
|
|
setPos(x, y);
|
|
}
|