mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-21 23:35:27 +00:00
e38a473a4d
In a dive, when you choose a very low GFlow (like 5 or 9) and a trimix with quite some He (12/48 in the example) and descend fast, the ceiling seems to do strange things in the first minutes of the dive (very very deep for example or jumping around). To understand what is going on we have to recall what gradient factors do in detail: Plain Buehlmann gives you for each tissue a maximal inert gas pressure that is a straight line when plotted against the ambient pressure. So for each depth (=ambient pressure) there is a maximally allowed over-pressure. The idea of gradient factors is that one does not use all the possible over-pressure that Buehlmann gives us but only a depth dependent fraction. GFhigh is the fraction of the possible over-pressure at the surface while GFlow is the fraction at the first deco stop. In between, the fraction is linearly interpolated. As the Buehlmann over-pressure is increasing with depth and typically also the allowed overpressure after applications of gradient factors increases with depth or said differently: the tissue saturation has to be lower if the diver wants to ascent. The main problem is: What is the first stop (where to apply GFlow)? In a planned dive, we could take the first deco stop, but in a real dive from a dive computer download it is impossible to say what constitutes a stop and what is only a slow ascent? What I have used so far is not exactly the first stop but rather the first theoretical stop: During all of the dive, I have calculated the ceiling under the assumption that GFlow applies everywhere (and not just at a single depth). The deepest of these ceilings I have used as the “first stop depth”, the depth at which GFlow applies. Even more, I only wanted to use the information that a diver has during the dive, so I actually only considered the ceilings in the past (and not in the future of a given sample). But this brings with it the problem that early in the dive, in particular during the descent the lowest ceiling so far is very shallow (as not much gas has built up in the body so far). This problem now interferes with a second one: If at the start of the dive when the all compartments have 790mbar N2 the diver starts breathing a He-heavy mix (like 12/48) and descents fast the He builds up in the tissues before the N2 can diffuse out. So right at the start, we already encounter high tissue loadings. If now we have a large difference between GFhigh and GFlow but they apply at very similar depth (the surface and a very shallow depth of the deepest ceiling (which for a non-decompression dive would be theoretically at negative depth) so far) it can happen that the linear interpolation as opposite slope then in the typical case above: The allowed over-pressure is degreasing with depth, shallower depth do not require lower gas loading in the tissue (i.e. can be reached after further off-gasing) but but tolerate higher loadings. In that situation the ceiling disappears (or is rather a floor). So far, I got rid of that problem, by stating that the minimum depth for GFlow was 20m (after all, GFlow is about deep stops, so it should better not be too shallow). Now the dive reported in ticket #549 takes values to an extreme in such away that 20m (which is determined by buehlmann_config.gf_low_position_min in deco.c) was not enough to prevent this inversion problem (or in a milder form that the interpolation of gradient factors is in fact an extrapolation with quite extreme values). This patch that gets rid of the problem for the dive described above but still it is possible to find (more extreme) parameter choices that lead to non-realistic ceilings. Let me close by pointing out that all this is only about the descent, as it is about too shallow depth for GFlow. So no real deco (i.e. later part of the dive) is inflicted. This is only about a theoretical ceiling displayed possibly in the first minutes of a dive. So this is more an aesthetically than a practical problem. Fixes #549 Signed-off-by: Robert C. Helling <helling@atdotde.de> Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
742 lines
23 KiB
C++
742 lines
23 KiB
C++
#include "diveprofileitem.h"
|
|
#include "diveplotdatamodel.h"
|
|
#include "divecartesianaxis.h"
|
|
#include "graphicsview-common.h"
|
|
#include "divetextitem.h"
|
|
#include "profilewidget2.h"
|
|
#include "profile.h"
|
|
#include "dive.h"
|
|
#include "preferences.h"
|
|
#include "helpers.h"
|
|
#include "diveplanner.h"
|
|
|
|
#include <QPen>
|
|
#include <QPainter>
|
|
#include <QLinearGradient>
|
|
#include <QDebug>
|
|
#include <QApplication>
|
|
#include <QGraphicsItem>
|
|
#include <QSettings>
|
|
|
|
AbstractProfilePolygonItem::AbstractProfilePolygonItem() : QObject(), QGraphicsPolygonItem(), hAxis(NULL), vAxis(NULL), dataModel(NULL), hDataColumn(-1), vDataColumn(-1)
|
|
{
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), this, SLOT(settingsChanged()));
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::settingsChanged()
|
|
{
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::setHorizontalAxis(DiveCartesianAxis *horizontal)
|
|
{
|
|
hAxis = horizontal;
|
|
connect(hAxis, SIGNAL(sizeChanged()), this, SLOT(modelDataChanged()));
|
|
modelDataChanged();
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::setHorizontalDataColumn(int column)
|
|
{
|
|
hDataColumn = column;
|
|
modelDataChanged();
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::setModel(DivePlotDataModel *model)
|
|
{
|
|
dataModel = model;
|
|
connect(dataModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(modelDataChanged(QModelIndex, QModelIndex)));
|
|
connect(dataModel, SIGNAL(rowsAboutToBeRemoved(QModelIndex, int, int)), this, SLOT(modelDataRemoved(QModelIndex, int, int)));
|
|
modelDataChanged();
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::modelDataRemoved(const QModelIndex &parent, int from, int to)
|
|
{
|
|
setPolygon(QPolygonF());
|
|
qDeleteAll(texts);
|
|
texts.clear();
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::setVerticalAxis(DiveCartesianAxis *vertical)
|
|
{
|
|
vAxis = vertical;
|
|
connect(vAxis, SIGNAL(sizeChanged()), this, SLOT(modelDataChanged()));
|
|
connect(vAxis, SIGNAL(maxChanged()), this, SLOT(modelDataChanged()));
|
|
modelDataChanged();
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::setVerticalDataColumn(int column)
|
|
{
|
|
vDataColumn = column;
|
|
modelDataChanged();
|
|
}
|
|
|
|
bool AbstractProfilePolygonItem::shouldCalculateStuff(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
if (!hAxis || !vAxis)
|
|
return false;
|
|
if (!dataModel || dataModel->rowCount() == 0)
|
|
return false;
|
|
if (hDataColumn == -1 || vDataColumn == -1)
|
|
return false;
|
|
if (topLeft.isValid() && bottomRight.isValid()) {
|
|
if ((topLeft.column() >= vDataColumn || topLeft.column() >= hDataColumn) &&
|
|
(bottomRight.column() <= vDataColumn || topLeft.column() <= hDataColumn)) {
|
|
return true;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void AbstractProfilePolygonItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
// We don't have enougth data to calculate things, quit.
|
|
|
|
// Calculate the polygon. This is the polygon that will be painted on screen
|
|
// on the ::paint method. Here we calculate the correct position of the points
|
|
// regarting our cartesian plane ( made by the hAxis and vAxis ), the QPolygonF
|
|
// is an array of QPointF's, so we basically get the point from the model, convert
|
|
// to our coordinates, store. no painting is done here.
|
|
QPolygonF poly;
|
|
for (int i = 0, modelDataCount = dataModel->rowCount(); i < modelDataCount; i++) {
|
|
qreal horizontalValue = dataModel->index(i, hDataColumn).data().toReal();
|
|
qreal verticalValue = dataModel->index(i, vDataColumn).data().toReal();
|
|
QPointF point(hAxis->posAtValue(horizontalValue), vAxis->posAtValue(verticalValue));
|
|
poly.append(point);
|
|
}
|
|
setPolygon(poly);
|
|
|
|
qDeleteAll(texts);
|
|
texts.clear();
|
|
}
|
|
|
|
DiveProfileItem::DiveProfileItem() : show_reported_ceiling(0), reported_ceiling_in_red(0)
|
|
{
|
|
}
|
|
|
|
void DiveProfileItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
Q_UNUSED(widget);
|
|
if (polygon().isEmpty())
|
|
return;
|
|
|
|
// This paints the Polygon + Background. I'm setting the pen to QPen() so we don't get a black line here,
|
|
// after all we need to plot the correct velocities colors later.
|
|
setPen(Qt::NoPen);
|
|
QGraphicsPolygonItem::paint(painter, option, widget);
|
|
|
|
// Here we actually paint the boundaries of the Polygon using the colors that the model provides.
|
|
// Those are the speed colors of the dives.
|
|
QPen pen;
|
|
pen.setCosmetic(true);
|
|
pen.setWidth(2);
|
|
QPolygonF poly = polygon();
|
|
// This paints the colors of the velocities.
|
|
for (int i = 1, count = dataModel->rowCount(); i < count; i++) {
|
|
QModelIndex colorIndex = dataModel->index(i, DivePlotDataModel::COLOR);
|
|
pen.setBrush(QBrush(colorIndex.data(Qt::BackgroundRole).value<QColor>()));
|
|
painter->setPen(pen);
|
|
painter->drawLine(poly[i - 1], poly[i]);
|
|
}
|
|
}
|
|
|
|
int DiveProfileItem::maxCeiling(int row)
|
|
{
|
|
int max = -1;
|
|
plot_data *entry = dataModel->data().entry + row;
|
|
for (int tissue = 0; tissue < 16; tissue++) {
|
|
if (max < entry->ceilings[tissue])
|
|
max = entry->ceilings[tissue];
|
|
}
|
|
return max;
|
|
}
|
|
|
|
void DiveProfileItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
|
|
AbstractProfilePolygonItem::modelDataChanged(topLeft, bottomRight);
|
|
if (polygon().isEmpty())
|
|
return;
|
|
|
|
show_reported_ceiling = prefs.dcceiling;
|
|
reported_ceiling_in_red = prefs.redceiling;
|
|
profileColor = getColor(DEPTH_BOTTOM);
|
|
|
|
int currState = qobject_cast<ProfileWidget2 *>(scene()->views().first())->currentState;
|
|
if (currState == ProfileWidget2::PLAN) {
|
|
plot_data *entry = dataModel->data().entry + dataModel->rowCount() - 1;
|
|
for (int i = dataModel->rowCount() - 1; i >= 0; i--, entry--) {
|
|
int max = maxCeiling(i);
|
|
// Don't screem if we violate the ceiling by a few cm
|
|
if (entry->depth < max - 100)
|
|
profileColor = QColor(Qt::red);
|
|
}
|
|
}
|
|
|
|
/* Show any ceiling we may have encountered */
|
|
if (prefs.dcceiling && !prefs.redceiling) {
|
|
QPolygonF p = polygon();
|
|
plot_data *entry = dataModel->data().entry + dataModel->rowCount() - 1;
|
|
for (int i = dataModel->rowCount() - 1; i >= 0; i--, entry--) {
|
|
if (!entry->in_deco) {
|
|
/* not in deco implies this is a safety stop, no ceiling */
|
|
p.append(QPointF(hAxis->posAtValue(entry->sec), vAxis->posAtValue(0)));
|
|
} else {
|
|
p.append(QPointF(hAxis->posAtValue(entry->sec), vAxis->posAtValue(qMin(entry->stopdepth, entry->depth))));
|
|
}
|
|
}
|
|
setPolygon(p);
|
|
}
|
|
|
|
// This is the blueish gradient that the Depth Profile should have.
|
|
// It's a simple QLinearGradient with 2 stops, starting from top to bottom.
|
|
QLinearGradient pat(0, polygon().boundingRect().top(), 0, polygon().boundingRect().bottom());
|
|
pat.setColorAt(1, profileColor);
|
|
pat.setColorAt(0, getColor(DEPTH_TOP));
|
|
setBrush(QBrush(pat));
|
|
|
|
int last = -1;
|
|
for (int i = 0, count = dataModel->rowCount(); i < count; i++) {
|
|
|
|
struct plot_data *entry = dataModel->data().entry + i;
|
|
if (entry->depth < 2000)
|
|
continue;
|
|
|
|
if ((entry == entry->max[2]) && entry->depth / 100 != last) {
|
|
plot_depth_sample(entry, Qt::AlignHCenter | Qt::AlignBottom, getColor(SAMPLE_DEEP));
|
|
last = entry->depth / 100;
|
|
}
|
|
|
|
if ((entry == entry->min[2]) && entry->depth / 100 != last) {
|
|
plot_depth_sample(entry, Qt::AlignHCenter | Qt::AlignTop, getColor(SAMPLE_SHALLOW));
|
|
last = entry->depth / 100;
|
|
}
|
|
|
|
if (entry->depth != last)
|
|
last = -1;
|
|
}
|
|
}
|
|
|
|
void DiveProfileItem::settingsChanged()
|
|
{
|
|
//TODO: Only modelDataChanged() here if we need to rebuild the graph ( for instance,
|
|
// if the prefs.dcceiling are enabled, but prefs.redceiling is disabled
|
|
// and only if it changed something. let's not waste cpu cycles repoloting something we don't need to.
|
|
modelDataChanged();
|
|
}
|
|
|
|
void DiveProfileItem::plot_depth_sample(struct plot_data *entry, QFlags<Qt::AlignmentFlag> flags, const QColor &color)
|
|
{
|
|
int decimals;
|
|
double d = get_depth_units(entry->depth, &decimals, NULL);
|
|
DiveTextItem *item = new DiveTextItem(this);
|
|
item->setPos(hAxis->posAtValue(entry->sec), vAxis->posAtValue(entry->depth));
|
|
item->setText(QString("%1").arg(d, 0, 'f', 1));
|
|
item->setAlignment(flags);
|
|
item->setBrush(color);
|
|
texts.append(item);
|
|
}
|
|
|
|
DiveHeartrateItem::DiveHeartrateItem()
|
|
{
|
|
QPen pen;
|
|
pen.setBrush(QBrush(getColor(::HR_PLOT)));
|
|
pen.setCosmetic(true);
|
|
pen.setWidth(1);
|
|
setPen(pen);
|
|
visible = true;
|
|
}
|
|
|
|
void DiveHeartrateItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
int last = -300, last_printed_hr = 0, sec = 0;
|
|
struct {
|
|
int sec;
|
|
int hr;
|
|
} hist[3] = {};
|
|
|
|
// We don't have enougth data to calculate things, quit.
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
|
|
qDeleteAll(texts);
|
|
texts.clear();
|
|
// Ignore empty values. a heartrate of 0 would be a bad sign.
|
|
QPolygonF poly;
|
|
for (int i = 0, modelDataCount = dataModel->rowCount(); i < modelDataCount; i++) {
|
|
int hr = dataModel->index(i, vDataColumn).data().toInt();
|
|
if (!hr)
|
|
continue;
|
|
sec = dataModel->index(i, hDataColumn).data().toInt();
|
|
QPointF point(hAxis->posAtValue(sec), vAxis->posAtValue(hr));
|
|
poly.append(point);
|
|
if (hr == hist[2].hr)
|
|
// same as last one, no point in looking at printing
|
|
continue;
|
|
hist[0] = hist[1];
|
|
hist[1] = hist[2];
|
|
hist[2].sec = sec;
|
|
hist[2].hr = hr;
|
|
// don't print a HR
|
|
// if it's not a local min / max
|
|
// if it's been less than 5min and less than a 20 beats change OR
|
|
// if it's been less than 2min OR if the change from the
|
|
// last print is less than 10 beats
|
|
// to test min / max requires three points, so we now look at the
|
|
// previous one
|
|
sec = hist[1].sec;
|
|
hr = hist[1].hr;
|
|
if ((hist[0].hr < hr && hr < hist[2].hr) ||
|
|
(hist[0].hr > hr && hr > hist[2].hr) ||
|
|
((sec < last + 300) && (abs(hr - last_printed_hr) < 20)) ||
|
|
(sec < last + 120) ||
|
|
(abs(hr - last_printed_hr) < 10))
|
|
continue;
|
|
last = sec;
|
|
createTextItem(sec, hr);
|
|
last_printed_hr = hr;
|
|
}
|
|
setPolygon(poly);
|
|
|
|
if (texts.count())
|
|
texts.last()->setAlignment(Qt::AlignLeft | Qt::AlignBottom);
|
|
}
|
|
|
|
void DiveHeartrateItem::createTextItem(int sec, int hr)
|
|
{
|
|
DiveTextItem *text = new DiveTextItem(this);
|
|
text->setAlignment(Qt::AlignRight | Qt::AlignBottom);
|
|
text->setBrush(getColor(HR_TEXT));
|
|
text->setPos(QPointF(hAxis->posAtValue(sec), vAxis->posAtValue(hr)));
|
|
text->setScale(0.7); // need to call this BEFORE setText()
|
|
text->setText(QString("%1").arg(hr));
|
|
texts.append(text);
|
|
}
|
|
|
|
void DiveHeartrateItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if (polygon().isEmpty())
|
|
return;
|
|
painter->setPen(pen());
|
|
painter->drawPolyline(polygon());
|
|
}
|
|
|
|
void DiveHeartrateItem::settingsChanged()
|
|
{
|
|
QSettings s;
|
|
s.beginGroup("TecDetails");
|
|
visible = s.value(visibilityKey).toBool();
|
|
setVisible(visible);
|
|
}
|
|
|
|
void DiveHeartrateItem::setVisibilitySettingsKey(const QString &key)
|
|
{
|
|
visibilityKey = key;
|
|
}
|
|
|
|
bool DiveHeartrateItem::isVisible()
|
|
{
|
|
return visible == true;
|
|
}
|
|
|
|
DiveTemperatureItem::DiveTemperatureItem()
|
|
{
|
|
QPen pen;
|
|
pen.setBrush(QBrush(getColor(::TEMP_PLOT)));
|
|
pen.setCosmetic(true);
|
|
pen.setWidth(2);
|
|
setPen(pen);
|
|
}
|
|
|
|
void DiveTemperatureItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
int last = -300, last_printed_temp = 0, sec = 0, last_valid_temp = 0;
|
|
// We don't have enougth data to calculate things, quit.
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
|
|
qDeleteAll(texts);
|
|
texts.clear();
|
|
// Ignore empty values. things do not look good with '0' as temperature in kelvin...
|
|
QPolygonF poly;
|
|
for (int i = 0, modelDataCount = dataModel->rowCount(); i < modelDataCount; i++) {
|
|
int mkelvin = dataModel->index(i, vDataColumn).data().toInt();
|
|
if (!mkelvin)
|
|
continue;
|
|
last_valid_temp = mkelvin;
|
|
sec = dataModel->index(i, hDataColumn).data().toInt();
|
|
QPointF point(hAxis->posAtValue(sec), vAxis->posAtValue(mkelvin));
|
|
poly.append(point);
|
|
|
|
/* don't print a temperature
|
|
* if it's been less than 5min and less than a 2K change OR
|
|
* if it's been less than 2min OR if the change from the
|
|
* last print is less than .4K (and therefore less than 1F) */
|
|
if (((sec < last + 300) && (abs(mkelvin - last_printed_temp) < 2000)) ||
|
|
(sec < last + 120) ||
|
|
(abs(mkelvin - last_printed_temp) < 400))
|
|
continue;
|
|
last = sec;
|
|
if (mkelvin > 200000)
|
|
createTextItem(sec, mkelvin);
|
|
last_printed_temp = mkelvin;
|
|
}
|
|
setPolygon(poly);
|
|
|
|
/* it would be nice to print the end temperature, if it's
|
|
* different or if the last temperature print has been more
|
|
* than a quarter of the dive back */
|
|
if (last_valid_temp > 200000 &&
|
|
((abs(last_valid_temp - last_printed_temp) > 500) || ((double)last / (double)sec < 0.75))) {
|
|
createTextItem(sec, last_valid_temp);
|
|
}
|
|
if (texts.count())
|
|
texts.last()->setAlignment(Qt::AlignLeft | Qt::AlignBottom);
|
|
}
|
|
|
|
void DiveTemperatureItem::createTextItem(int sec, int mkelvin)
|
|
{
|
|
double deg;
|
|
const char *unit;
|
|
deg = get_temp_units(mkelvin, &unit);
|
|
|
|
DiveTextItem *text = new DiveTextItem(this);
|
|
text->setAlignment(Qt::AlignRight | Qt::AlignBottom);
|
|
text->setBrush(getColor(TEMP_TEXT));
|
|
text->setPos(QPointF(hAxis->posAtValue(sec), vAxis->posAtValue(mkelvin)));
|
|
text->setScale(0.8); // need to call this BEFORE setText()
|
|
text->setText(QString("%1%2").arg(deg, 0, 'f', 1).arg(unit));
|
|
texts.append(text);
|
|
}
|
|
|
|
void DiveTemperatureItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if (polygon().isEmpty())
|
|
return;
|
|
painter->setPen(pen());
|
|
painter->drawPolyline(polygon());
|
|
}
|
|
|
|
void DiveGasPressureItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
// We don't have enougth data to calculate things, quit.
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
int last_index = -1;
|
|
QPolygonF boundingPoly; // This is the "Whole Item", but a pressure can be divided in N Polygons.
|
|
polygons.clear();
|
|
|
|
for (int i = 0, count = dataModel->rowCount(); i < count; i++) {
|
|
plot_data *entry = dataModel->data().entry + i;
|
|
int mbar = GET_PRESSURE(entry);
|
|
|
|
if (entry->cylinderindex != last_index) {
|
|
polygons.append(QPolygonF()); // this is the polygon that will be actually drawned on screen.
|
|
last_index = entry->cylinderindex;
|
|
}
|
|
if (!mbar) {
|
|
continue;
|
|
}
|
|
|
|
QPointF point(hAxis->posAtValue(entry->sec), vAxis->posAtValue(mbar));
|
|
boundingPoly.push_back(point); // The BoundingRect
|
|
polygons.last().push_back(point); // The polygon thta will be plotted.
|
|
}
|
|
setPolygon(boundingPoly);
|
|
qDeleteAll(texts);
|
|
texts.clear();
|
|
int mbar, cyl;
|
|
int seen_cyl[MAX_CYLINDERS] = { false, };
|
|
int last_pressure[MAX_CYLINDERS] = { 0, };
|
|
int last_time[MAX_CYLINDERS] = { 0, };
|
|
struct plot_data *entry;
|
|
struct dive *dive = get_dive_by_uniq_id(dataModel->id());
|
|
|
|
cyl = -1;
|
|
for (int i = 0, count = dataModel->rowCount(); i < count; i++) {
|
|
entry = dataModel->data().entry + i;
|
|
mbar = GET_PRESSURE(entry);
|
|
|
|
if (!mbar)
|
|
continue;
|
|
if (cyl != entry->cylinderindex) {
|
|
cyl = entry->cylinderindex;
|
|
if (!seen_cyl[cyl]) {
|
|
plotPressureValue(mbar, entry->sec, Qt::AlignRight | Qt::AlignTop);
|
|
plotGasValue(mbar, entry->sec, Qt::AlignRight | Qt::AlignBottom,
|
|
dive->cylinder[cyl].gasmix);
|
|
seen_cyl[cyl] = true;
|
|
}
|
|
}
|
|
last_pressure[cyl] = mbar;
|
|
last_time[cyl] = entry->sec;
|
|
}
|
|
|
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) {
|
|
if (last_time[cyl]) {
|
|
plotPressureValue(last_pressure[cyl], last_time[cyl], Qt::AlignLeft | Qt::AlignTop);
|
|
}
|
|
}
|
|
}
|
|
|
|
void DiveGasPressureItem::plotPressureValue(int mbar, int sec, QFlags<Qt::AlignmentFlag> flags)
|
|
{
|
|
const char *unit;
|
|
int pressure = get_pressure_units(mbar, &unit);
|
|
DiveTextItem *text = new DiveTextItem(this);
|
|
text->setPos(hAxis->posAtValue(sec), vAxis->posAtValue(mbar) - 0.5);
|
|
text->setText(QString("%1 %2").arg(pressure).arg(unit));
|
|
text->setAlignment(flags);
|
|
text->setBrush(getColor(PRESSURE_TEXT));
|
|
texts.push_back(text);
|
|
}
|
|
|
|
void DiveGasPressureItem::plotGasValue(int mbar, int sec, QFlags<Qt::AlignmentFlag> flags, struct gasmix gasmix)
|
|
{
|
|
QString gas = gasToStr(gasmix);
|
|
DiveTextItem *text = new DiveTextItem(this);
|
|
text->setPos(hAxis->posAtValue(sec), vAxis->posAtValue(mbar));
|
|
text->setText(gas);
|
|
text->setAlignment(flags);
|
|
text->setBrush(getColor(PRESSURE_TEXT));
|
|
texts.push_back(text);
|
|
}
|
|
|
|
void DiveGasPressureItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if (polygon().isEmpty())
|
|
return;
|
|
QPen pen;
|
|
pen.setCosmetic(true);
|
|
pen.setWidth(2);
|
|
struct dive *d = get_dive_by_uniq_id(dataModel->id());
|
|
struct plot_data *entry = dataModel->data().entry;
|
|
Q_FOREACH (const QPolygonF &poly, polygons) {
|
|
for (int i = 1, count = poly.count(); i < count; i++, entry++) {
|
|
pen.setBrush(getSacColor(entry->sac, d->sac));
|
|
painter->setPen(pen);
|
|
painter->drawLine(poly[i - 1], poly[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
DiveCalculatedCeiling::DiveCalculatedCeiling() : is3mIncrement(false), gradientFactor(new DiveTextItem(this))
|
|
{
|
|
gradientFactor->setY(0);
|
|
gradientFactor->setBrush(getColor(PRESSURE_TEXT));
|
|
gradientFactor->setAlignment(Qt::AlignHCenter | Qt::AlignBottom);
|
|
settingsChanged();
|
|
}
|
|
|
|
void DiveCalculatedCeiling::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
// We don't have enougth data to calculate things, quit.
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
AbstractProfilePolygonItem::modelDataChanged(topLeft, bottomRight);
|
|
// Add 2 points to close the polygon.
|
|
QPolygonF poly = polygon();
|
|
if (poly.isEmpty())
|
|
return;
|
|
QPointF p1 = poly.first();
|
|
QPointF p2 = poly.last();
|
|
|
|
poly.prepend(QPointF(p1.x(), vAxis->posAtValue(0)));
|
|
poly.append(QPointF(p2.x(), vAxis->posAtValue(0)));
|
|
setPolygon(poly);
|
|
|
|
QLinearGradient pat(0, polygon().boundingRect().top(), 0, polygon().boundingRect().bottom());
|
|
pat.setColorAt(0, getColor(CALC_CEILING_SHALLOW));
|
|
pat.setColorAt(1, getColor(CALC_CEILING_DEEP));
|
|
setPen(QPen(QBrush(Qt::NoBrush), 0));
|
|
setBrush(pat);
|
|
|
|
gradientFactor->setX(poly.boundingRect().width() / 2 + poly.boundingRect().x());
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
if (plannerModel->isPlanner()) {
|
|
struct diveplan &diveplan = plannerModel->getDiveplan();
|
|
gradientFactor->setText(QString("GF %1/%2").arg(diveplan.gflow).arg(diveplan.gfhigh));
|
|
} else {
|
|
gradientFactor->setText(QString("GF %1/%2").arg(prefs.gflow).arg(prefs.gfhigh));
|
|
}
|
|
}
|
|
|
|
void DiveCalculatedCeiling::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if (polygon().isEmpty())
|
|
return;
|
|
QGraphicsPolygonItem::paint(painter, option, widget);
|
|
}
|
|
|
|
DiveCalculatedTissue::DiveCalculatedTissue()
|
|
{
|
|
settingsChanged();
|
|
}
|
|
|
|
void DiveCalculatedTissue::settingsChanged()
|
|
{
|
|
setVisible(prefs.calcalltissues && prefs.calcceiling);
|
|
}
|
|
|
|
void DiveReportedCeiling::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
|
|
QPolygonF p;
|
|
p.append(QPointF(hAxis->posAtValue(0), vAxis->posAtValue(0)));
|
|
plot_data *entry = dataModel->data().entry;
|
|
for (int i = 0, count = dataModel->rowCount(); i < count; i++, entry++) {
|
|
if (entry->in_deco && entry->stopdepth) {
|
|
p.append(QPointF(hAxis->posAtValue(entry->sec), vAxis->posAtValue(qMin(entry->stopdepth, entry->depth))));
|
|
} else {
|
|
p.append(QPointF(hAxis->posAtValue(entry->sec), vAxis->posAtValue(0)));
|
|
}
|
|
}
|
|
setPolygon(p);
|
|
QLinearGradient pat(0, p.boundingRect().top(), 0, p.boundingRect().bottom());
|
|
// does the user want the ceiling in "surface color" or in red?
|
|
if (prefs.redceiling) {
|
|
pat.setColorAt(0, getColor(CEILING_SHALLOW));
|
|
pat.setColorAt(1, getColor(CEILING_DEEP));
|
|
} else {
|
|
pat.setColorAt(0, getColor(BACKGROUND_TRANS));
|
|
pat.setColorAt(1, getColor(BACKGROUND_TRANS));
|
|
}
|
|
setPen(QPen(QBrush(Qt::NoBrush), 0));
|
|
setBrush(pat);
|
|
}
|
|
|
|
void DiveCalculatedCeiling::settingsChanged()
|
|
{
|
|
if (dataModel && is3mIncrement != prefs.calcceiling3m) {
|
|
// recalculate that part.
|
|
dataModel->calculateDecompression();
|
|
}
|
|
is3mIncrement = prefs.calcceiling3m;
|
|
setVisible(prefs.calcceiling);
|
|
}
|
|
|
|
void DiveReportedCeiling::settingsChanged()
|
|
{
|
|
setVisible(prefs.dcceiling);
|
|
}
|
|
|
|
void DiveReportedCeiling::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if (polygon().isEmpty())
|
|
return;
|
|
QGraphicsPolygonItem::paint(painter, option, widget);
|
|
}
|
|
|
|
MeanDepthLine::MeanDepthLine() : meanDepth(0), leftText(new DiveTextItem(this)), rightText(new DiveTextItem(this))
|
|
{
|
|
leftText->setAlignment(Qt::AlignRight | Qt::AlignBottom);
|
|
leftText->setBrush(getColor(MEAN_DEPTH));
|
|
rightText->setAlignment(Qt::AlignLeft | Qt::AlignBottom);
|
|
rightText->setBrush(getColor(MEAN_DEPTH));
|
|
leftText->setPos(0, 0);
|
|
rightText->setPos(line().length(), 0);
|
|
}
|
|
|
|
void MeanDepthLine::setLine(qreal x1, qreal y1, qreal x2, qreal y2)
|
|
{
|
|
QGraphicsLineItem::setLine(x1, y1, x2, y2);
|
|
leftText->setPos(x1, 0);
|
|
rightText->setPos(x2, 0);
|
|
}
|
|
|
|
void MeanDepthLine::setMeanDepth(int value)
|
|
{
|
|
leftText->setText(get_depth_string(value, true, true));
|
|
rightText->setText(get_depth_string(value, true, true));
|
|
meanDepth = value;
|
|
}
|
|
|
|
void MeanDepthLine::setAxis(DiveCartesianAxis *a)
|
|
{
|
|
connect(a, SIGNAL(sizeChanged()), this, SLOT(axisLineChanged()));
|
|
}
|
|
|
|
void MeanDepthLine::axisLineChanged()
|
|
{
|
|
DiveCartesianAxis *axis = qobject_cast<DiveCartesianAxis *>(sender());
|
|
animateMoveTo(x(), axis->posAtValue(meanDepth));
|
|
}
|
|
|
|
void PartialPressureGasItem::modelDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
|
|
{
|
|
//AbstractProfilePolygonItem::modelDataChanged();
|
|
if (!shouldCalculateStuff(topLeft, bottomRight))
|
|
return;
|
|
|
|
plot_data *entry = dataModel->data().entry;
|
|
QPolygonF poly;
|
|
QPolygonF alertpoly;
|
|
alertPolygons.clear();
|
|
QSettings s;
|
|
s.beginGroup("TecDetails");
|
|
double threshould = s.value(threshouldKey).toDouble();
|
|
bool inAlertFragment = false;
|
|
for (int i = 0; i < dataModel->rowCount(); i++, entry++) {
|
|
double value = dataModel->index(i, vDataColumn).data().toDouble();
|
|
int time = dataModel->index(i, hDataColumn).data().toInt();
|
|
QPointF point(hAxis->posAtValue(time), vAxis->posAtValue(value));
|
|
poly.push_back(point);
|
|
if (value >= threshould) {
|
|
if (inAlertFragment) {
|
|
alertPolygons.back().push_back(point);
|
|
} else {
|
|
alertpoly.clear();
|
|
alertpoly.push_back(point);
|
|
alertPolygons.append(alertpoly);
|
|
inAlertFragment = true;
|
|
}
|
|
} else {
|
|
inAlertFragment = false;
|
|
}
|
|
}
|
|
setPolygon(poly);
|
|
/*
|
|
createPPLegend(trUtf8("pN" UTF8_SUBSCRIPT_2),getColor(PN2), legendPos);
|
|
*/
|
|
}
|
|
|
|
void PartialPressureGasItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
const qreal pWidth = 0.0;
|
|
painter->setPen(QPen(normalColor, pWidth));
|
|
painter->drawPolyline(polygon());
|
|
|
|
QPolygonF poly;
|
|
painter->setPen(QPen(alertColor, pWidth));
|
|
Q_FOREACH (const QPolygonF &poly, alertPolygons)
|
|
painter->drawPolyline(poly);
|
|
}
|
|
|
|
void PartialPressureGasItem::setThreshouldSettingsKey(const QString &threshouldSettingsKey)
|
|
{
|
|
threshouldKey = threshouldSettingsKey;
|
|
}
|
|
|
|
PartialPressureGasItem::PartialPressureGasItem()
|
|
{
|
|
}
|
|
|
|
void PartialPressureGasItem::settingsChanged()
|
|
{
|
|
QSettings s;
|
|
s.beginGroup("TecDetails");
|
|
setVisible(s.value(visibilityKey).toBool());
|
|
}
|
|
|
|
void PartialPressureGasItem::setVisibilitySettingsKey(const QString &key)
|
|
{
|
|
visibilityKey = key;
|
|
}
|
|
|
|
void PartialPressureGasItem::setColors(const QColor &normal, const QColor &alert)
|
|
{
|
|
normalColor = normal;
|
|
alertColor = alert;
|
|
}
|