2017-04-27 18:26:36 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "profile-widget/profilewidget2.h"
|
2016-03-06 14:52:55 +00:00
|
|
|
#include "qt-models/diveplotdatamodel.h"
|
2018-05-11 15:25:41 +00:00
|
|
|
#include "core/subsurface-string.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "core/helpers.h"
|
|
|
|
#include "core/profile.h"
|
|
|
|
#include "profile-widget/diveeventitem.h"
|
|
|
|
#include "profile-widget/divetextitem.h"
|
|
|
|
#include "profile-widget/divetooltipitem.h"
|
|
|
|
#include "core/planner.h"
|
|
|
|
#include "core/device.h"
|
|
|
|
#include "profile-widget/ruleritem.h"
|
|
|
|
#include "profile-widget/tankitem.h"
|
|
|
|
#include "core/pref.h"
|
2016-03-06 14:52:55 +00:00
|
|
|
#include "qt-models/diveplannermodel.h"
|
|
|
|
#include "qt-models/models.h"
|
|
|
|
#include "qt-models/divepicturemodel.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "core/divelist.h"
|
|
|
|
#include "core/subsurface-qt/SettingsObjectWrapper.h"
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/diveplanner.h"
|
|
|
|
#include "desktop-widgets/simplewidgets.h"
|
|
|
|
#include "desktop-widgets/divepicturewidget.h"
|
2017-01-07 02:01:14 +00:00
|
|
|
#include "core/qthelper.h"
|
2015-11-06 00:05:44 +00:00
|
|
|
#endif
|
2015-02-09 20:27:59 +00:00
|
|
|
|
2014-03-16 04:12:34 +00:00
|
|
|
#include <libdivecomputer/parser.h>
|
2014-02-04 23:47:50 +00:00
|
|
|
#include <QScrollBar>
|
2014-03-10 15:59:02 +00:00
|
|
|
#include <QtCore/qmath.h>
|
2014-03-16 04:12:34 +00:00
|
|
|
#include <QMessageBox>
|
2014-04-03 19:16:15 +00:00
|
|
|
#include <QInputDialog>
|
2015-01-17 15:02:03 +00:00
|
|
|
#include <QDebug>
|
2015-02-09 20:58:40 +00:00
|
|
|
#include <QWheelEvent>
|
2015-11-05 18:51:12 +00:00
|
|
|
#include <QSettings>
|
2015-11-05 20:50:51 +00:00
|
|
|
#include <QMenu>
|
2014-01-16 17:02:32 +00:00
|
|
|
|
2014-01-15 15:20:05 +00:00
|
|
|
#ifndef QT_NO_DEBUG
|
|
|
|
#include <QTableView>
|
|
|
|
#endif
|
2016-03-06 14:52:55 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/preferences/preferencesdialog.h"
|
2016-03-06 14:52:55 +00:00
|
|
|
#endif
|
2015-11-03 20:17:50 +00:00
|
|
|
#include <QtWidgets>
|
2014-01-15 15:20:05 +00:00
|
|
|
|
2015-11-05 18:05:22 +00:00
|
|
|
// a couple of helpers we need
|
|
|
|
extern bool haveFilesOnCommandLine();
|
|
|
|
|
2014-02-07 17:32:39 +00:00
|
|
|
/* This is the global 'Item position' variable.
|
|
|
|
* it should tell you where to position things up
|
|
|
|
* on the canvas.
|
|
|
|
*
|
|
|
|
* please, please, please, use this instead of
|
|
|
|
* hard coding the item on the scene with a random
|
|
|
|
* value.
|
|
|
|
*/
|
2014-02-28 04:09:57 +00:00
|
|
|
static struct _ItemPos {
|
|
|
|
struct _Pos {
|
2014-02-07 17:32:39 +00:00
|
|
|
QPointF on;
|
|
|
|
QPointF off;
|
|
|
|
};
|
2014-02-28 04:09:57 +00:00
|
|
|
struct _Axis {
|
|
|
|
_Pos pos;
|
|
|
|
QLineF shrinked;
|
|
|
|
QLineF expanded;
|
2014-09-28 08:11:38 +00:00
|
|
|
QLineF intermediate;
|
2014-02-07 17:32:39 +00:00
|
|
|
};
|
|
|
|
_Pos background;
|
2014-02-07 23:17:14 +00:00
|
|
|
_Pos dcLabel;
|
2014-08-15 13:30:31 +00:00
|
|
|
_Pos tankBar;
|
2014-02-07 17:32:39 +00:00
|
|
|
_Axis depth;
|
2014-02-11 18:55:07 +00:00
|
|
|
_Axis partialPressure;
|
2014-09-28 08:11:38 +00:00
|
|
|
_Axis partialPressureTissue;
|
2014-08-15 14:11:14 +00:00
|
|
|
_Axis partialPressureWithTankBar;
|
2014-09-15 12:09:00 +00:00
|
|
|
_Axis percentage;
|
2014-09-28 08:11:38 +00:00
|
|
|
_Axis percentageWithTankBar;
|
2014-02-07 17:32:39 +00:00
|
|
|
_Axis time;
|
|
|
|
_Axis cylinder;
|
|
|
|
_Axis temperature;
|
2014-11-02 18:41:41 +00:00
|
|
|
_Axis temperatureAll;
|
2014-02-20 01:18:26 +00:00
|
|
|
_Axis heartBeat;
|
2015-01-17 18:54:10 +00:00
|
|
|
_Axis heartBeatWithTankBar;
|
2014-02-07 17:32:39 +00:00
|
|
|
} itemPos;
|
2014-02-07 16:59:58 +00:00
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
ProfileWidget2::ProfileWidget2(QWidget *parent) : QGraphicsView(parent),
|
2014-01-15 14:00:23 +00:00
|
|
|
currentState(INVALID),
|
2014-06-09 21:06:00 +00:00
|
|
|
dataModel(new DivePlotDataModel(this)),
|
2014-02-04 23:47:50 +00:00
|
|
|
zoomLevel(0),
|
2014-03-10 15:59:02 +00:00
|
|
|
zoomFactor(1.15),
|
2014-02-28 04:09:57 +00:00
|
|
|
background(new DivePixmapItem()),
|
2017-11-29 09:57:08 +00:00
|
|
|
backgroundFile(":poster-icon"),
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-02-05 16:34:45 +00:00
|
|
|
toolTipItem(new ToolTipItem()),
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2015-10-06 05:21:44 +00:00
|
|
|
isPlotZoomed(prefs.zoomed_plot),// no! bad use of prefs. 'PreferencesDialog::loadSettings' NOT CALLED yet.
|
2015-11-06 11:54:35 +00:00
|
|
|
profileYAxis(new DepthAxis(this)),
|
|
|
|
gasYAxis(new PartialGasPressureAxis(this)),
|
|
|
|
temperatureAxis(new TemperatureAxis(this)),
|
|
|
|
timeAxis(new TimeAxis(this)),
|
2014-02-07 16:14:36 +00:00
|
|
|
diveProfileItem(new DiveProfileItem()),
|
|
|
|
temperatureItem(new DiveTemperatureItem()),
|
2015-01-04 21:36:01 +00:00
|
|
|
meanDepthItem(new DiveMeanDepthItem()),
|
2015-11-06 11:54:35 +00:00
|
|
|
cylinderPressureAxis(new DiveCartesianAxis(this)),
|
2014-02-07 16:14:36 +00:00
|
|
|
gasPressureItem(new DiveGasPressureItem()),
|
2014-01-21 16:59:19 +00:00
|
|
|
diveComputerText(new DiveTextItem()),
|
2016-02-06 21:25:58 +00:00
|
|
|
reportedCeiling(new DiveReportedCeiling()),
|
2014-02-28 04:09:57 +00:00
|
|
|
pn2GasItem(new PartialPressureGasItem()),
|
|
|
|
pheGasItem(new PartialPressureGasItem()),
|
|
|
|
po2GasItem(new PartialPressureGasItem()),
|
2015-01-05 07:20:26 +00:00
|
|
|
o2SetpointGasItem(new PartialPressureGasItem()),
|
2015-01-20 18:13:53 +00:00
|
|
|
ccrsensor1GasItem(new PartialPressureGasItem()),
|
|
|
|
ccrsensor2GasItem(new PartialPressureGasItem()),
|
|
|
|
ccrsensor3GasItem(new PartialPressureGasItem()),
|
2018-03-14 15:13:37 +00:00
|
|
|
ocpo2GasItem(new PartialPressureGasItem()),
|
2017-03-25 12:03:37 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
diveCeiling(new DiveCalculatedCeiling(this)),
|
|
|
|
decoModelParameters(new DiveTextItem()),
|
2015-11-06 11:54:35 +00:00
|
|
|
heartBeatAxis(new DiveCartesianAxis(this)),
|
2014-02-27 15:39:53 +00:00
|
|
|
heartBeatItem(new DiveHeartrateItem()),
|
2015-11-06 11:54:35 +00:00
|
|
|
percentageAxis(new DiveCartesianAxis(this)),
|
2014-09-15 12:09:00 +00:00
|
|
|
ambPressureItem(new DiveAmbPressureItem()),
|
|
|
|
gflineItem(new DiveGFLineItem()),
|
2014-08-05 21:27:00 +00:00
|
|
|
mouseFollowerVertical(new DiveLineItem()),
|
2014-08-06 17:54:37 +00:00
|
|
|
mouseFollowerHorizontal(new DiveLineItem()),
|
2014-03-25 21:34:09 +00:00
|
|
|
rulerItem(new RulerItem2()),
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-08-15 00:22:27 +00:00
|
|
|
tankItem(new TankItem()),
|
2014-03-25 21:34:09 +00:00
|
|
|
isGrayscale(false),
|
2014-05-26 20:51:46 +00:00
|
|
|
printMode(false),
|
2014-06-30 22:08:16 +00:00
|
|
|
shouldCalculateMaxTime(true),
|
2014-07-09 20:09:52 +00:00
|
|
|
shouldCalculateMaxDepth(true),
|
|
|
|
fontPrintScale(1.0)
|
2014-01-14 16:30:13 +00:00
|
|
|
{
|
2015-10-06 05:21:44 +00:00
|
|
|
// would like to be able to ASSERT here that PreferencesDialog::loadSettings has been called.
|
|
|
|
isPlotZoomed = prefs.zoomed_plot; // now it seems that 'prefs' has loaded our preferences
|
|
|
|
|
2014-02-09 18:11:17 +00:00
|
|
|
memset(&plotInfo, 0, sizeof(plotInfo));
|
|
|
|
|
2014-02-07 16:59:58 +00:00
|
|
|
setupSceneAndFlags();
|
|
|
|
setupItemSizes();
|
|
|
|
setupItemOnScene();
|
|
|
|
addItemsToScene();
|
|
|
|
scene()->installEventFilter(this);
|
2015-11-12 00:37:18 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2015-11-03 20:17:50 +00:00
|
|
|
setAcceptDrops(true);
|
|
|
|
|
2018-01-10 15:50:57 +00:00
|
|
|
addActionShortcut(Qt::Key_Escape, &ProfileWidget2::keyEscAction);
|
|
|
|
addActionShortcut(Qt::Key_Delete, &ProfileWidget2::keyDeleteAction);
|
|
|
|
addActionShortcut(Qt::Key_Up, &ProfileWidget2::keyUpAction);
|
|
|
|
addActionShortcut(Qt::Key_Down, &ProfileWidget2::keyDownAction);
|
|
|
|
addActionShortcut(Qt::Key_Left, &ProfileWidget2::keyLeftAction);
|
|
|
|
addActionShortcut(Qt::Key_Right, &ProfileWidget2::keyRightAction);
|
2015-11-12 00:37:18 +00:00
|
|
|
#endif // SUBSURFACE_MOBILE
|
2014-05-24 15:39:40 +00:00
|
|
|
|
2015-02-02 21:12:11 +00:00
|
|
|
#if !defined(QT_NO_DEBUG) && defined(SHOW_PLOT_INFO_TABLE)
|
2014-02-07 16:59:58 +00:00
|
|
|
QTableView *diveDepthTableView = new QTableView();
|
|
|
|
diveDepthTableView->setModel(dataModel);
|
2014-03-07 15:42:13 +00:00
|
|
|
diveDepthTableView->show();
|
2014-02-07 16:59:58 +00:00
|
|
|
#endif
|
|
|
|
}
|
2014-01-15 14:23:40 +00:00
|
|
|
|
2018-01-10 15:50:57 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
void ProfileWidget2::addActionShortcut(const Qt::Key shortcut, void (ProfileWidget2::*slot)())
|
|
|
|
{
|
|
|
|
QAction *action = new QAction(this);
|
|
|
|
action->setShortcut(shortcut);
|
|
|
|
action->setShortcutContext(Qt::WindowShortcut);
|
|
|
|
addAction(action);
|
|
|
|
connect(action, &QAction::triggered, this, slot);
|
|
|
|
actionsForKeys[shortcut] = action;
|
|
|
|
}
|
|
|
|
#endif // SUBSURFACE_MOBILE
|
|
|
|
|
2014-05-19 05:39:34 +00:00
|
|
|
#define SUBSURFACE_OBJ_DATA 1
|
|
|
|
#define SUBSURFACE_OBJ_DC_TEXT 0x42
|
|
|
|
|
2014-02-07 16:59:58 +00:00
|
|
|
void ProfileWidget2::addItemsToScene()
|
|
|
|
{
|
|
|
|
scene()->addItem(background);
|
|
|
|
scene()->addItem(profileYAxis);
|
|
|
|
scene()->addItem(gasYAxis);
|
|
|
|
scene()->addItem(temperatureAxis);
|
|
|
|
scene()->addItem(timeAxis);
|
|
|
|
scene()->addItem(diveProfileItem);
|
|
|
|
scene()->addItem(cylinderPressureAxis);
|
|
|
|
scene()->addItem(temperatureItem);
|
2015-01-02 00:28:37 +00:00
|
|
|
scene()->addItem(meanDepthItem);
|
2014-02-07 16:59:58 +00:00
|
|
|
scene()->addItem(gasPressureItem);
|
2014-05-19 05:39:34 +00:00
|
|
|
// I cannot seem to figure out if an object that I find with itemAt() on the scene
|
|
|
|
// is the object I am looking for - my guess is there's a simple way in Qt to do that
|
|
|
|
// but nothing I tried worked.
|
|
|
|
// so instead this adds a special magic key/value pair to the object to mark it
|
|
|
|
diveComputerText->setData(SUBSURFACE_OBJ_DATA, SUBSURFACE_OBJ_DC_TEXT);
|
2014-02-07 16:59:58 +00:00
|
|
|
scene()->addItem(diveComputerText);
|
2016-02-06 21:25:58 +00:00
|
|
|
scene()->addItem(reportedCeiling);
|
2017-02-04 09:46:02 +00:00
|
|
|
scene()->addItem(tankItem);
|
2014-02-07 16:59:58 +00:00
|
|
|
scene()->addItem(pn2GasItem);
|
|
|
|
scene()->addItem(pheGasItem);
|
|
|
|
scene()->addItem(po2GasItem);
|
2015-01-05 07:20:26 +00:00
|
|
|
scene()->addItem(o2SetpointGasItem);
|
2015-01-20 18:13:53 +00:00
|
|
|
scene()->addItem(ccrsensor1GasItem);
|
|
|
|
scene()->addItem(ccrsensor2GasItem);
|
|
|
|
scene()->addItem(ccrsensor3GasItem);
|
2018-03-14 15:13:37 +00:00
|
|
|
scene()->addItem(ocpo2GasItem);
|
2017-03-25 12:03:37 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
scene()->addItem(toolTipItem);
|
|
|
|
scene()->addItem(diveCeiling);
|
|
|
|
scene()->addItem(decoModelParameters);
|
2014-09-15 12:09:00 +00:00
|
|
|
scene()->addItem(percentageAxis);
|
2014-02-20 01:18:26 +00:00
|
|
|
scene()->addItem(heartBeatAxis);
|
|
|
|
scene()->addItem(heartBeatItem);
|
2014-02-27 18:20:03 +00:00
|
|
|
scene()->addItem(rulerItem);
|
|
|
|
scene()->addItem(rulerItem->sourceNode());
|
|
|
|
scene()->addItem(rulerItem->destNode());
|
2014-08-05 21:27:00 +00:00
|
|
|
scene()->addItem(mouseFollowerHorizontal);
|
|
|
|
scene()->addItem(mouseFollowerVertical);
|
2014-08-05 22:47:02 +00:00
|
|
|
QPen pen(QColor(Qt::red).lighter());
|
|
|
|
pen.setWidth(0);
|
|
|
|
mouseFollowerHorizontal->setPen(pen);
|
|
|
|
mouseFollowerVertical->setPen(pen);
|
2016-02-06 21:25:58 +00:00
|
|
|
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (DiveCalculatedTissue *tissue, allTissues) {
|
2014-02-07 16:59:58 +00:00
|
|
|
scene()->addItem(tissue);
|
|
|
|
}
|
2014-09-15 12:09:00 +00:00
|
|
|
Q_FOREACH (DivePercentageItem *percentage, allPercentages) {
|
|
|
|
scene()->addItem(percentage);
|
|
|
|
}
|
|
|
|
scene()->addItem(ambPressureItem);
|
|
|
|
scene()->addItem(gflineItem);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-02-07 16:59:58 +00:00
|
|
|
}
|
2014-01-15 14:23:40 +00:00
|
|
|
|
2014-02-07 16:59:58 +00:00
|
|
|
void ProfileWidget2::setupItemOnScene()
|
|
|
|
{
|
2014-02-07 18:34:42 +00:00
|
|
|
background->setZValue(9999);
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-03-11 22:09:54 +00:00
|
|
|
toolTipItem->setZValue(9998);
|
2014-02-07 16:59:58 +00:00
|
|
|
toolTipItem->setTimeAxis(timeAxis);
|
2014-03-11 22:09:54 +00:00
|
|
|
rulerItem->setZValue(9997);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-08-15 00:22:27 +00:00
|
|
|
tankItem->setZValue(100);
|
2014-01-23 19:54:34 +00:00
|
|
|
|
2014-02-07 19:59:21 +00:00
|
|
|
profileYAxis->setOrientation(DiveCartesianAxis::TopToBottom);
|
|
|
|
profileYAxis->setMinimum(0);
|
2014-02-28 04:09:57 +00:00
|
|
|
profileYAxis->setTickInterval(M_OR_FT(10, 30));
|
2014-03-19 20:06:18 +00:00
|
|
|
profileYAxis->setTickSize(0.5);
|
2014-02-16 00:43:27 +00:00
|
|
|
profileYAxis->setLineSize(96);
|
|
|
|
|
2014-02-16 22:16:39 +00:00
|
|
|
timeAxis->setLineSize(92);
|
2014-03-19 20:06:18 +00:00
|
|
|
timeAxis->setTickSize(-0.5);
|
2014-02-07 19:59:21 +00:00
|
|
|
|
2014-01-23 19:54:34 +00:00
|
|
|
gasYAxis->setOrientation(DiveCartesianAxis::BottomToTop);
|
|
|
|
gasYAxis->setTickInterval(1);
|
2014-02-07 21:42:47 +00:00
|
|
|
gasYAxis->setTickSize(1);
|
2014-01-27 17:14:42 +00:00
|
|
|
gasYAxis->setMinimum(0);
|
|
|
|
gasYAxis->setModel(dataModel);
|
2014-02-15 19:15:57 +00:00
|
|
|
gasYAxis->setFontLabelScale(0.7);
|
2014-02-16 00:43:27 +00:00
|
|
|
gasYAxis->setLineSize(96);
|
2014-01-16 20:39:13 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-02-20 01:18:26 +00:00
|
|
|
heartBeatAxis->setOrientation(DiveCartesianAxis::BottomToTop);
|
2014-04-07 20:15:21 +00:00
|
|
|
heartBeatAxis->setTickSize(0.2);
|
2014-02-20 01:18:26 +00:00
|
|
|
heartBeatAxis->setTickInterval(10);
|
|
|
|
heartBeatAxis->setFontLabelScale(0.7);
|
|
|
|
heartBeatAxis->setLineSize(96);
|
|
|
|
|
2014-09-15 12:09:00 +00:00
|
|
|
percentageAxis->setOrientation(DiveCartesianAxis::BottomToTop);
|
|
|
|
percentageAxis->setTickSize(0.2);
|
|
|
|
percentageAxis->setTickInterval(10);
|
|
|
|
percentageAxis->setFontLabelScale(0.7);
|
|
|
|
percentageAxis->setLineSize(96);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-09-15 12:09:00 +00:00
|
|
|
|
2014-01-16 21:28:33 +00:00
|
|
|
temperatureAxis->setOrientation(DiveCartesianAxis::BottomToTop);
|
2014-01-16 20:39:13 +00:00
|
|
|
temperatureAxis->setTickSize(2);
|
|
|
|
temperatureAxis->setTickInterval(300);
|
|
|
|
|
2014-01-17 17:34:15 +00:00
|
|
|
cylinderPressureAxis->setOrientation(DiveCartesianAxis::BottomToTop);
|
2014-01-17 16:28:59 +00:00
|
|
|
cylinderPressureAxis->setTickSize(2);
|
|
|
|
cylinderPressureAxis->setTickInterval(30000);
|
|
|
|
|
2014-01-15 14:23:40 +00:00
|
|
|
|
2014-02-16 22:16:39 +00:00
|
|
|
diveComputerText->setAlignment(Qt::AlignRight | Qt::AlignTop);
|
2014-03-25 21:34:10 +00:00
|
|
|
diveComputerText->setBrush(getColor(TIME_TEXT, isGrayscale));
|
2014-01-21 12:48:26 +00:00
|
|
|
|
2014-08-15 00:22:27 +00:00
|
|
|
tankItem->setHorizontalAxis(timeAxis);
|
2014-02-27 19:42:00 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
rulerItem->setAxis(timeAxis, profileYAxis);
|
|
|
|
|
2015-10-24 09:03:46 +00:00
|
|
|
// show the deco model parameters at the top in the center
|
|
|
|
decoModelParameters->setY(0);
|
|
|
|
decoModelParameters->setX(50);
|
|
|
|
decoModelParameters->setBrush(getColor(PRESSURE_TEXT));
|
|
|
|
decoModelParameters->setAlignment(Qt::AlignHCenter | Qt::AlignBottom);
|
2018-01-10 16:08:56 +00:00
|
|
|
setupItem(diveCeiling, profileYAxis, DivePlotDataModel::CEILING, DivePlotDataModel::TIME, 1);
|
2014-02-28 04:09:57 +00:00
|
|
|
for (int i = 0; i < 16; i++) {
|
2015-11-06 12:40:20 +00:00
|
|
|
DiveCalculatedTissue *tissueItem = new DiveCalculatedTissue(this);
|
2018-01-10 16:08:56 +00:00
|
|
|
setupItem(tissueItem, profileYAxis, DivePlotDataModel::TISSUE_1 + i, DivePlotDataModel::TIME, 1 + i);
|
2014-01-23 17:02:12 +00:00
|
|
|
allTissues.append(tissueItem);
|
2014-09-15 12:09:00 +00:00
|
|
|
DivePercentageItem *percentageItem = new DivePercentageItem(i);
|
2018-01-10 16:08:56 +00:00
|
|
|
setupItem(percentageItem, percentageAxis, DivePlotDataModel::PERCENTAGE_1 + i, DivePlotDataModel::TIME, 1 + i);
|
2014-09-15 12:09:00 +00:00
|
|
|
allPercentages.append(percentageItem);
|
2014-01-23 17:02:12 +00:00
|
|
|
}
|
2018-01-10 16:08:56 +00:00
|
|
|
setupItem(heartBeatItem, heartBeatAxis, DivePlotDataModel::HEARTBEAT, DivePlotDataModel::TIME, 1);
|
|
|
|
setupItem(ambPressureItem, percentageAxis, DivePlotDataModel::AMBPRESSURE, DivePlotDataModel::TIME, 1);
|
|
|
|
setupItem(gflineItem, percentageAxis, DivePlotDataModel::GFLINE, DivePlotDataModel::TIME, 1);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2018-01-10 16:08:56 +00:00
|
|
|
setupItem(reportedCeiling, profileYAxis, DivePlotDataModel::CEILING, DivePlotDataModel::TIME, 1);
|
|
|
|
setupItem(gasPressureItem, cylinderPressureAxis, DivePlotDataModel::TEMPERATURE, DivePlotDataModel::TIME, 1);
|
|
|
|
setupItem(temperatureItem, temperatureAxis, DivePlotDataModel::TEMPERATURE, DivePlotDataModel::TIME, 1);
|
|
|
|
setupItem(diveProfileItem, profileYAxis, DivePlotDataModel::DEPTH, DivePlotDataModel::TIME, 0);
|
|
|
|
setupItem(meanDepthItem, profileYAxis, DivePlotDataModel::INSTANT_MEANDEPTH, DivePlotDataModel::TIME, 1);
|
2015-01-01 23:28:38 +00:00
|
|
|
|
2018-01-10 16:28:24 +00:00
|
|
|
createPPGas(pn2GasItem, DivePlotDataModel::PN2, PN2, PN2_ALERT, NULL, &prefs.pp_graphs.pn2_threshold);
|
|
|
|
createPPGas(pheGasItem, DivePlotDataModel::PHE, PHE, PHE_ALERT, NULL, &prefs.pp_graphs.phe_threshold);
|
|
|
|
createPPGas(po2GasItem, DivePlotDataModel::PO2, PO2, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
2018-04-29 05:00:17 +00:00
|
|
|
createPPGas(o2SetpointGasItem, DivePlotDataModel::O2SETPOINT, O2SETPOINT, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
2018-01-10 16:28:24 +00:00
|
|
|
createPPGas(ccrsensor1GasItem, DivePlotDataModel::CCRSENSOR1, CCRSENSOR1, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
|
|
|
createPPGas(ccrsensor2GasItem, DivePlotDataModel::CCRSENSOR2, CCRSENSOR2, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
|
|
|
createPPGas(ccrsensor3GasItem, DivePlotDataModel::CCRSENSOR3, CCRSENSOR3, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
2018-03-14 15:13:37 +00:00
|
|
|
createPPGas(ocpo2GasItem, DivePlotDataModel::SCR_OC_PO2, SCR_OCPO2, PO2_ALERT, &prefs.pp_graphs.po2_threshold_min, &prefs.pp_graphs.po2_threshold_max);
|
2017-03-24 18:11:31 +00:00
|
|
|
|
2017-03-25 12:03:37 +00:00
|
|
|
#undef CREATE_PP_GAS
|
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2017-03-24 18:11:31 +00:00
|
|
|
|
2016-01-25 19:20:18 +00:00
|
|
|
// Visibility Connections
|
|
|
|
connect(SettingsObjectWrapper::instance()->pp_gas, &PartialPressureGasSettings::showPheChanged, pheGasItem, &PartialPressureGasItem::setVisible);
|
|
|
|
connect(SettingsObjectWrapper::instance()->pp_gas, &PartialPressureGasSettings::showPo2Changed, po2GasItem, &PartialPressureGasItem::setVisible);
|
|
|
|
connect(SettingsObjectWrapper::instance()->pp_gas, &PartialPressureGasSettings::showPn2Changed, pn2GasItem, &PartialPressureGasItem::setVisible);
|
|
|
|
|
|
|
|
//WARNING: The old code was broken, I'm not sure what should trigger the visibility of those graphs, since the old code didn't triggered them
|
|
|
|
// because it was using a wrong settings.
|
2018-04-29 05:00:17 +00:00
|
|
|
connect(SettingsObjectWrapper::instance()->techDetails, &TechnicalDetailsSettings::showCCRSetpointChanged, o2SetpointGasItem, &PartialPressureGasItem::setVisible);
|
2018-03-14 15:13:37 +00:00
|
|
|
connect(SettingsObjectWrapper::instance()->techDetails, &TechnicalDetailsSettings::showSCROCpO2Changed, ocpo2GasItem, &PartialPressureGasItem::setVisible);
|
2016-01-25 19:20:18 +00:00
|
|
|
connect(SettingsObjectWrapper::instance()->techDetails, &TechnicalDetailsSettings::showCCRSensorsChanged, ccrsensor1GasItem, &PartialPressureGasItem::setVisible);
|
|
|
|
connect(SettingsObjectWrapper::instance()->techDetails, &TechnicalDetailsSettings::showCCRSensorsChanged, ccrsensor2GasItem, &PartialPressureGasItem::setVisible);
|
|
|
|
connect(SettingsObjectWrapper::instance()->techDetails, &TechnicalDetailsSettings::showCCRSensorsChanged, ccrsensor3GasItem, &PartialPressureGasItem::setVisible);
|
2014-01-23 20:03:28 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
heartBeatAxis->setTextVisible(true);
|
|
|
|
heartBeatAxis->setLinesVisible(true);
|
|
|
|
percentageAxis->setTextVisible(true);
|
|
|
|
percentageAxis->setLinesVisible(true);
|
|
|
|
#endif
|
2014-02-12 16:24:19 +00:00
|
|
|
temperatureAxis->setTextVisible(false);
|
2014-02-16 00:54:41 +00:00
|
|
|
temperatureAxis->setLinesVisible(false);
|
2014-02-12 16:24:19 +00:00
|
|
|
cylinderPressureAxis->setTextVisible(false);
|
2014-02-16 00:54:41 +00:00
|
|
|
cylinderPressureAxis->setLinesVisible(false);
|
|
|
|
timeAxis->setLinesVisible(true);
|
|
|
|
profileYAxis->setLinesVisible(true);
|
2014-02-28 04:09:57 +00:00
|
|
|
gasYAxis->setZValue(timeAxis->zValue() + 1);
|
2014-12-13 21:06:52 +00:00
|
|
|
|
|
|
|
replotEnabled = true;
|
2014-02-07 16:59:58 +00:00
|
|
|
}
|
2014-01-15 15:01:29 +00:00
|
|
|
|
2015-09-23 02:55:48 +00:00
|
|
|
void ProfileWidget2::replot(struct dive *d)
|
2014-03-09 13:59:31 +00:00
|
|
|
{
|
2014-12-13 21:06:52 +00:00
|
|
|
if (!replotEnabled)
|
|
|
|
return;
|
2014-04-03 19:52:05 +00:00
|
|
|
dataModel->clear();
|
2018-05-16 13:25:57 +00:00
|
|
|
plotDive(d, true, false);
|
2014-03-09 13:59:31 +00:00
|
|
|
}
|
|
|
|
|
2018-01-10 16:33:20 +00:00
|
|
|
void ProfileWidget2::createPPGas(PartialPressureGasItem *item, int verticalColumn, color_index_t color, color_index_t colorAlert,
|
2018-01-10 16:28:24 +00:00
|
|
|
double *thresholdSettingsMin, double *thresholdSettingsMax)
|
|
|
|
{
|
|
|
|
setupItem(item, gasYAxis, verticalColumn, DivePlotDataModel::TIME, 0);
|
2018-01-10 16:39:25 +00:00
|
|
|
item->setThresholdSettingsKey(thresholdSettingsMin, thresholdSettingsMax);
|
2018-01-10 16:28:24 +00:00
|
|
|
item->setColors(getColor(color, isGrayscale), getColor(colorAlert, isGrayscale));
|
|
|
|
item->settingsChanged();
|
|
|
|
item->setZValue(99);
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:59:58 +00:00
|
|
|
void ProfileWidget2::setupItemSizes()
|
|
|
|
{
|
2014-02-07 19:38:00 +00:00
|
|
|
// Scene is *always* (double) 100 / 100.
|
|
|
|
// Background Config
|
|
|
|
/* Much probably a better math is needed here.
|
|
|
|
* good thing is that we only need to change the
|
|
|
|
* Axis and everything else is auto-adjusted.*
|
|
|
|
*/
|
|
|
|
|
2014-02-07 17:32:39 +00:00
|
|
|
itemPos.background.on.setX(0);
|
|
|
|
itemPos.background.on.setY(0);
|
|
|
|
itemPos.background.off.setX(0);
|
|
|
|
itemPos.background.off.setY(110);
|
2014-02-07 19:38:00 +00:00
|
|
|
|
|
|
|
//Depth Axis Config
|
|
|
|
itemPos.depth.pos.on.setX(3);
|
|
|
|
itemPos.depth.pos.on.setY(3);
|
|
|
|
itemPos.depth.pos.off.setX(-2);
|
|
|
|
itemPos.depth.pos.off.setY(3);
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.depth.expanded.setP1(QPointF(0, 0));
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.depth.expanded.setP2(QPointF(0, 85));
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.depth.shrinked.setP1(QPointF(0, 0));
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.depth.shrinked.setP2(QPointF(0, 55));
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.depth.intermediate.setP1(QPointF(0, 0));
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.depth.intermediate.setP2(QPointF(0, 65));
|
2017-02-04 09:46:02 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
itemPos.depth.expanded.setP2(QPointF(0, 65));
|
|
|
|
#endif
|
2014-02-07 19:38:00 +00:00
|
|
|
|
|
|
|
// Time Axis Config
|
|
|
|
itemPos.time.pos.on.setX(3);
|
2014-02-16 22:16:39 +00:00
|
|
|
itemPos.time.pos.on.setY(95);
|
2016-01-05 14:00:06 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
2017-02-04 09:46:02 +00:00
|
|
|
itemPos.time.pos.on.setY(89.5);
|
2016-01-05 14:00:06 +00:00
|
|
|
#endif
|
2014-02-07 19:38:00 +00:00
|
|
|
itemPos.time.pos.off.setX(3);
|
|
|
|
itemPos.time.pos.off.setY(110);
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.time.expanded.setP1(QPointF(0, 0));
|
|
|
|
itemPos.time.expanded.setP2(QPointF(94, 0));
|
2014-02-07 21:42:47 +00:00
|
|
|
|
|
|
|
// Partial Gas Axis Config
|
2014-02-11 18:55:07 +00:00
|
|
|
itemPos.partialPressure.pos.on.setX(97);
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.partialPressure.pos.on.setY(75);
|
2017-03-25 12:03:37 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
itemPos.partialPressure.pos.on.setY(70);
|
|
|
|
#endif
|
2014-02-11 18:55:07 +00:00
|
|
|
itemPos.partialPressure.pos.off.setX(110);
|
2014-02-16 01:01:20 +00:00
|
|
|
itemPos.partialPressure.pos.off.setY(63);
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.partialPressure.expanded.setP1(QPointF(0, 0));
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.partialPressure.expanded.setP2(QPointF(0, 19));
|
2017-03-25 12:03:37 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
itemPos.partialPressure.expanded.setP2(QPointF(0, 20));
|
|
|
|
#endif
|
2014-08-15 14:11:14 +00:00
|
|
|
itemPos.partialPressureWithTankBar = itemPos.partialPressure;
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.partialPressureWithTankBar.expanded.setP2(QPointF(0, 17));
|
|
|
|
itemPos.partialPressureTissue = itemPos.partialPressure;
|
|
|
|
itemPos.partialPressureTissue.pos.on.setX(97);
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.partialPressureTissue.pos.on.setY(65);
|
|
|
|
itemPos.partialPressureTissue.expanded.setP2(QPointF(0, 16));
|
2014-02-07 21:42:47 +00:00
|
|
|
|
|
|
|
// cylinder axis config
|
|
|
|
itemPos.cylinder.pos.on.setX(3);
|
|
|
|
itemPos.cylinder.pos.on.setY(20);
|
|
|
|
itemPos.cylinder.pos.off.setX(-10);
|
|
|
|
itemPos.cylinder.pos.off.setY(20);
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.cylinder.expanded.setP1(QPointF(0, 15));
|
|
|
|
itemPos.cylinder.expanded.setP2(QPointF(0, 50));
|
|
|
|
itemPos.cylinder.shrinked.setP1(QPointF(0, 0));
|
|
|
|
itemPos.cylinder.shrinked.setP2(QPointF(0, 20));
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.cylinder.intermediate.setP1(QPointF(0, 0));
|
|
|
|
itemPos.cylinder.intermediate.setP2(QPointF(0, 20));
|
2014-02-07 23:17:14 +00:00
|
|
|
|
2014-02-16 22:16:39 +00:00
|
|
|
// Temperature axis config
|
2014-02-07 23:17:14 +00:00
|
|
|
itemPos.temperature.pos.on.setX(3);
|
2014-12-12 19:44:14 +00:00
|
|
|
itemPos.temperature.pos.on.setY(60);
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.temperatureAll.pos.on.setY(51);
|
2014-02-07 23:17:14 +00:00
|
|
|
itemPos.temperature.pos.off.setX(-10);
|
|
|
|
itemPos.temperature.pos.off.setY(40);
|
2014-11-02 18:41:41 +00:00
|
|
|
itemPos.temperature.expanded.setP1(QPointF(0, 20));
|
|
|
|
itemPos.temperature.expanded.setP2(QPointF(0, 33));
|
|
|
|
itemPos.temperature.shrinked.setP1(QPointF(0, 2));
|
|
|
|
itemPos.temperature.shrinked.setP2(QPointF(0, 12));
|
|
|
|
itemPos.temperature.intermediate.setP1(QPointF(0, 2));
|
|
|
|
itemPos.temperature.intermediate.setP2(QPointF(0, 12));
|
2017-02-04 09:46:02 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
itemPos.temperature.pos.on.setY(51);
|
|
|
|
itemPos.temperatureAll.pos.on.setY(47);
|
|
|
|
itemPos.temperature.expanded.setP1(QPointF(0, 20));
|
|
|
|
itemPos.temperature.expanded.setP2(QPointF(0, 33));
|
|
|
|
itemPos.temperature.intermediate.setP1(QPointF(0, 2));
|
|
|
|
itemPos.temperature.intermediate.setP2(QPointF(0, 12));
|
|
|
|
#endif
|
2014-02-07 23:17:14 +00:00
|
|
|
|
2017-02-20 08:29:20 +00:00
|
|
|
// Heart rate axis config
|
2014-02-20 01:18:26 +00:00
|
|
|
itemPos.heartBeat.pos.on.setX(3);
|
2015-01-17 18:54:10 +00:00
|
|
|
itemPos.heartBeat.pos.on.setY(82);
|
2014-02-28 04:09:57 +00:00
|
|
|
itemPos.heartBeat.expanded.setP1(QPointF(0, 0));
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.heartBeat.expanded.setP2(QPointF(0, 10));
|
2015-01-17 18:54:10 +00:00
|
|
|
itemPos.heartBeatWithTankBar = itemPos.heartBeat;
|
|
|
|
itemPos.heartBeatWithTankBar.expanded.setP2(QPointF(0, 7));
|
2014-09-15 12:09:00 +00:00
|
|
|
|
2014-09-28 08:11:38 +00:00
|
|
|
// Percentage axis config
|
2014-09-15 12:09:00 +00:00
|
|
|
itemPos.percentage.pos.on.setX(3);
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.percentage.pos.on.setY(80);
|
2014-09-15 12:09:00 +00:00
|
|
|
itemPos.percentage.expanded.setP1(QPointF(0, 0));
|
2014-09-19 06:48:19 +00:00
|
|
|
itemPos.percentage.expanded.setP2(QPointF(0, 15));
|
2014-09-28 08:11:38 +00:00
|
|
|
itemPos.percentageWithTankBar = itemPos.percentage;
|
2017-01-16 17:52:18 +00:00
|
|
|
itemPos.percentageWithTankBar.expanded.setP2(QPointF(0, 11.9));
|
2014-02-20 01:18:26 +00:00
|
|
|
|
2014-02-07 23:17:14 +00:00
|
|
|
itemPos.dcLabel.on.setX(3);
|
2014-02-16 22:16:39 +00:00
|
|
|
itemPos.dcLabel.on.setY(100);
|
2014-02-07 23:17:14 +00:00
|
|
|
itemPos.dcLabel.off.setX(-10);
|
2014-02-16 22:16:39 +00:00
|
|
|
itemPos.dcLabel.off.setY(100);
|
2014-08-15 13:30:31 +00:00
|
|
|
|
|
|
|
itemPos.tankBar.on.setX(0);
|
2017-01-16 17:52:18 +00:00
|
|
|
itemPos.tankBar.on.setY(91.95);
|
2017-02-04 09:46:02 +00:00
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
itemPos.tankBar.on.setY(86.4);
|
|
|
|
#endif
|
2014-01-14 16:30:13 +00:00
|
|
|
}
|
2014-02-07 16:59:58 +00:00
|
|
|
|
2018-01-10 16:08:56 +00:00
|
|
|
void ProfileWidget2::setupItem(AbstractProfilePolygonItem *item, DiveCartesianAxis *vAxis,
|
2015-01-02 00:28:37 +00:00
|
|
|
int vData, int hData, int zValue)
|
2014-02-07 16:14:36 +00:00
|
|
|
{
|
2018-01-10 16:08:56 +00:00
|
|
|
item->setHorizontalAxis(timeAxis);
|
2014-02-07 16:59:58 +00:00
|
|
|
item->setVerticalAxis(vAxis);
|
2018-01-10 16:08:56 +00:00
|
|
|
item->setModel(dataModel);
|
2014-02-07 16:59:58 +00:00
|
|
|
item->setVerticalDataColumn(vData);
|
|
|
|
item->setHorizontalDataColumn(hData);
|
|
|
|
item->setZValue(zValue);
|
2014-02-07 16:14:36 +00:00
|
|
|
}
|
|
|
|
|
2014-02-07 16:59:58 +00:00
|
|
|
void ProfileWidget2::setupSceneAndFlags()
|
|
|
|
{
|
2014-12-18 18:47:00 +00:00
|
|
|
setScene(new QGraphicsScene(this));
|
2014-02-07 16:59:58 +00:00
|
|
|
scene()->setSceneRect(0, 0, 100, 100);
|
|
|
|
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
scene()->setItemIndexMethod(QGraphicsScene::NoIndex);
|
|
|
|
setOptimizationFlags(QGraphicsView::DontSavePainterState);
|
|
|
|
setViewportUpdateMode(QGraphicsView::BoundingRectViewportUpdate);
|
|
|
|
setRenderHints(QPainter::Antialiasing | QPainter::TextAntialiasing | QPainter::SmoothPixmapTransform);
|
|
|
|
setMouseTracking(true);
|
|
|
|
background->setFlag(QGraphicsItem::ItemIgnoresTransformations);
|
|
|
|
}
|
2014-01-14 16:30:13 +00:00
|
|
|
|
2015-10-11 11:21:13 +00:00
|
|
|
void ProfileWidget2::resetZoom()
|
|
|
|
{
|
|
|
|
if (!zoomLevel)
|
|
|
|
return;
|
|
|
|
const qreal defScale = 1.0 / qPow(zoomFactor, (qreal)zoomLevel);
|
|
|
|
scale(defScale, defScale);
|
|
|
|
zoomLevel = 0;
|
|
|
|
}
|
|
|
|
|
2014-01-14 16:30:13 +00:00
|
|
|
// Currently just one dive, but the plan is to enable All of the selected dives.
|
2018-05-16 13:25:57 +00:00
|
|
|
void ProfileWidget2::plotDive(struct dive *d, bool force, bool doClearPictures)
|
2014-01-14 16:30:13 +00:00
|
|
|
{
|
2014-03-11 21:54:28 +00:00
|
|
|
static bool firstCall = true;
|
2015-12-01 23:36:20 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-06-04 20:41:50 +00:00
|
|
|
QTime measureDuration; // let's measure how long this takes us (maybe we'll turn of TTL calculation later
|
|
|
|
measureDuration.start();
|
2018-05-21 07:41:35 +00:00
|
|
|
#else
|
|
|
|
Q_UNUSED(doClearPictures);
|
2015-12-01 23:36:20 +00:00
|
|
|
#endif
|
2014-07-03 21:34:24 +00:00
|
|
|
if (currentState != ADD && currentState != PLAN) {
|
|
|
|
if (!d) {
|
|
|
|
if (selected_dive == -1)
|
|
|
|
return;
|
|
|
|
d = current_dive; // display the current dive
|
|
|
|
}
|
2014-01-15 15:34:55 +00:00
|
|
|
|
2014-07-03 21:34:24 +00:00
|
|
|
// No need to do this again if we are already showing the same dive
|
|
|
|
// computer of the same dive, so we check the unique id of the dive
|
|
|
|
// and the selected dive computer number against the ones we are
|
|
|
|
// showing (can't compare the dive pointers as those might change).
|
2014-07-09 18:19:08 +00:00
|
|
|
if (d->id == displayed_dive.id && dc_number == dataModel->dcShown() && !force)
|
2014-07-03 21:34:24 +00:00
|
|
|
return;
|
2014-07-03 03:00:57 +00:00
|
|
|
|
2014-07-03 21:34:24 +00:00
|
|
|
// this copies the dive and makes copies of all the relevant additional data
|
|
|
|
copy_dive(d, &displayed_dive);
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2017-01-07 02:01:14 +00:00
|
|
|
if (decoMode() == VPMB)
|
2016-09-24 08:02:07 +00:00
|
|
|
decoModelParameters->setText(QString("VPM-B +%1").arg(prefs.vpmb_conservatism));
|
2015-10-24 09:03:46 +00:00
|
|
|
else
|
|
|
|
decoModelParameters->setText(QString("GF %1/%2").arg(prefs.gflow).arg(prefs.gfhigh));
|
2014-07-03 21:34:24 +00:00
|
|
|
} else {
|
2014-05-22 02:31:26 +00:00
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
plannerModel->createTemporaryPlan();
|
2015-07-29 17:31:34 +00:00
|
|
|
struct diveplan &diveplan = plannerModel->getDiveplan();
|
|
|
|
if (!diveplan.dp) {
|
2014-05-22 02:31:26 +00:00
|
|
|
plannerModel->deleteTemporaryPlan();
|
|
|
|
return;
|
|
|
|
}
|
2017-01-07 02:01:14 +00:00
|
|
|
if (decoMode() == VPMB)
|
2016-09-24 08:02:08 +00:00
|
|
|
decoModelParameters->setText(QString("VPM-B +%1").arg(diveplan.vpmb_conservatism));
|
2015-10-24 09:03:46 +00:00
|
|
|
else
|
2016-03-19 02:45:04 +00:00
|
|
|
decoModelParameters->setText(QString("GF %1/%2").arg(diveplan.gflow).arg(diveplan.gfhigh));
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-05-22 02:31:26 +00:00
|
|
|
}
|
|
|
|
|
2014-07-03 03:00:57 +00:00
|
|
|
// special handling for the first time we display things
|
2014-07-21 22:10:31 +00:00
|
|
|
int animSpeedBackup = 0;
|
2015-11-05 18:05:22 +00:00
|
|
|
if (firstCall && haveFilesOnCommandLine()) {
|
2014-07-21 22:10:31 +00:00
|
|
|
animSpeedBackup = prefs.animation_speed;
|
|
|
|
prefs.animation_speed = 0;
|
2014-03-11 21:54:28 +00:00
|
|
|
firstCall = false;
|
|
|
|
}
|
|
|
|
|
2014-03-13 17:23:20 +00:00
|
|
|
// restore default zoom level
|
2015-10-11 11:21:13 +00:00
|
|
|
resetZoom();
|
2014-03-10 15:59:02 +00:00
|
|
|
|
2015-12-01 23:36:20 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-03-25 21:34:12 +00:00
|
|
|
// reset some item visibility on printMode changes
|
2014-06-17 18:01:15 +00:00
|
|
|
toolTipItem->setVisible(!printMode);
|
2015-01-28 18:55:55 +00:00
|
|
|
rulerItem->setVisible(prefs.rulergraph && !printMode && currentState != PLAN && currentState != ADD);
|
2015-12-01 23:36:20 +00:00
|
|
|
#endif
|
2014-05-13 00:26:12 +00:00
|
|
|
if (currentState == EMPTY)
|
|
|
|
setProfileState();
|
2014-02-26 19:42:55 +00:00
|
|
|
|
|
|
|
// next get the dive computer structure - if there are no samples
|
|
|
|
// let's create a fake profile that's somewhat reasonable for the
|
|
|
|
// data that we have
|
2014-07-03 03:00:57 +00:00
|
|
|
struct divecomputer *currentdc = select_dc(&displayed_dive);
|
2014-01-15 15:34:55 +00:00
|
|
|
Q_ASSERT(currentdc);
|
2018-05-05 17:26:48 +00:00
|
|
|
if (!currentdc || !currentdc->samples)
|
|
|
|
fake_dc(currentdc);
|
2014-01-15 15:34:55 +00:00
|
|
|
|
2015-01-23 07:47:33 +00:00
|
|
|
bool setpointflag = (currentdc->divemode == CCR) && prefs.pp_graphs.po2 && current_dive;
|
|
|
|
bool sensorflag = setpointflag && prefs.show_ccr_sensors;
|
|
|
|
o2SetpointGasItem->setVisible(setpointflag && prefs.show_ccr_setpoint);
|
|
|
|
ccrsensor1GasItem->setVisible(sensorflag);
|
|
|
|
ccrsensor2GasItem->setVisible(sensorflag && (currentdc->no_o2sensors > 1));
|
|
|
|
ccrsensor3GasItem->setVisible(sensorflag && (currentdc->no_o2sensors > 2));
|
2018-03-14 15:13:37 +00:00
|
|
|
ocpo2GasItem->setVisible((currentdc->divemode == PSCR) && prefs.show_scr_ocpo2);
|
2015-01-05 07:20:26 +00:00
|
|
|
|
2014-01-15 15:34:55 +00:00
|
|
|
/* This struct holds all the data that's about to be plotted.
|
|
|
|
* I'm not sure this is the best approach ( but since we are
|
|
|
|
* interpolating some points of the Dive, maybe it is... )
|
|
|
|
* The Calculation of the points should be done per graph,
|
|
|
|
* so I'll *not* calculate everything if something is not being
|
|
|
|
* shown.
|
|
|
|
*/
|
2018-01-09 18:32:41 +00:00
|
|
|
|
2015-01-23 19:05:32 +00:00
|
|
|
plotInfo = calculate_max_limits_new(&displayed_dive, currentdc);
|
2018-01-09 18:32:41 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2017-11-24 13:17:01 +00:00
|
|
|
create_plot_info_new(&displayed_dive, currentdc, &plotInfo, !shouldCalculateMaxDepth, &DivePlannerPointsModel::instance()->final_deco_state);
|
2018-01-09 18:32:41 +00:00
|
|
|
#else
|
|
|
|
create_plot_info_new(&displayed_dive, currentdc, &plotInfo, !shouldCalculateMaxDepth, nullptr);
|
|
|
|
#endif
|
2017-01-29 05:00:47 +00:00
|
|
|
int newMaxtime = get_maxtime(&plotInfo);
|
|
|
|
if (shouldCalculateMaxTime || newMaxtime > maxtime)
|
|
|
|
maxtime = newMaxtime;
|
2014-06-30 22:08:16 +00:00
|
|
|
|
2017-03-06 12:27:39 +00:00
|
|
|
/* Only update the max. depth if it's bigger than the current ones
|
2014-06-30 22:08:16 +00:00
|
|
|
* when we are dragging the handler to plan / add dive.
|
|
|
|
* otherwhise, update normally.
|
|
|
|
*/
|
2014-11-19 22:28:16 +00:00
|
|
|
int newMaxDepth = get_maxdepth(&plotInfo);
|
2015-01-02 00:28:37 +00:00
|
|
|
if (!shouldCalculateMaxDepth) {
|
2014-06-30 22:08:16 +00:00
|
|
|
if (maxdepth < newMaxDepth) {
|
|
|
|
maxdepth = newMaxDepth;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
maxdepth = newMaxDepth;
|
|
|
|
}
|
2014-01-15 15:34:55 +00:00
|
|
|
|
2014-11-19 22:28:16 +00:00
|
|
|
dataModel->setDive(&displayed_dive, plotInfo);
|
2015-12-01 23:36:20 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-11-19 22:28:16 +00:00
|
|
|
toolTipItem->setPlotInfo(plotInfo);
|
2015-12-01 23:36:20 +00:00
|
|
|
#endif
|
2014-01-16 17:02:32 +00:00
|
|
|
// It seems that I'll have a lot of boilerplate setting the model / axis for
|
|
|
|
// each item, I'll mostly like to fix this in the future, but I'll keep at this for now.
|
2014-01-18 22:33:12 +00:00
|
|
|
profileYAxis->setMaximum(maxdepth);
|
2014-01-15 15:34:55 +00:00
|
|
|
profileYAxis->updateTicks();
|
2014-02-07 19:59:21 +00:00
|
|
|
|
2014-11-19 22:28:16 +00:00
|
|
|
temperatureAxis->setMinimum(plotInfo.mintemp);
|
2014-12-10 22:48:19 +00:00
|
|
|
temperatureAxis->setMaximum(plotInfo.maxtemp - plotInfo.mintemp > 2000 ? plotInfo.maxtemp : plotInfo.mintemp + 2000);
|
2014-02-20 01:18:26 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-11-19 22:28:16 +00:00
|
|
|
if (plotInfo.maxhr) {
|
2018-03-15 22:12:45 +00:00
|
|
|
int heartBeatAxisMin = lrint(plotInfo.minhr / 5.0 - 0.5) * 5;
|
|
|
|
int heartBeatAxisMax, heartBeatAxisTick;
|
|
|
|
if (plotInfo.maxhr - plotInfo.minhr < 40)
|
|
|
|
heartBeatAxisTick = 10;
|
|
|
|
else if (plotInfo.maxhr - plotInfo.minhr < 80)
|
|
|
|
heartBeatAxisTick = 20;
|
|
|
|
else if (plotInfo.maxhr - plotInfo.minhr < 100)
|
|
|
|
heartBeatAxisTick = 25;
|
|
|
|
else
|
|
|
|
heartBeatAxisTick = 50;
|
|
|
|
for (heartBeatAxisMax = heartBeatAxisMin; heartBeatAxisMax < plotInfo.maxhr; heartBeatAxisMax += heartBeatAxisTick);
|
|
|
|
heartBeatAxis->setMinimum(heartBeatAxisMin);
|
|
|
|
heartBeatAxis->setMaximum(heartBeatAxisMax + 1);
|
|
|
|
heartBeatAxis->setTickInterval(heartBeatAxisTick);
|
2014-07-21 22:32:46 +00:00
|
|
|
heartBeatAxis->updateTicks(HR_AXIS); // this shows the ticks
|
2014-02-23 22:32:25 +00:00
|
|
|
}
|
2014-11-19 22:28:16 +00:00
|
|
|
heartBeatAxis->setVisible(prefs.hrgraph && plotInfo.maxhr);
|
2014-07-21 22:32:46 +00:00
|
|
|
|
2014-09-15 12:09:00 +00:00
|
|
|
percentageAxis->setMinimum(0);
|
|
|
|
percentageAxis->setMaximum(100);
|
|
|
|
percentageAxis->setVisible(false);
|
|
|
|
percentageAxis->updateTicks(HR_AXIS);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2017-12-01 11:46:51 +00:00
|
|
|
if (shouldCalculateMaxTime)
|
|
|
|
timeAxis->setMaximum(maxtime);
|
2014-01-27 19:09:08 +00:00
|
|
|
int i, incr;
|
2014-02-28 04:09:57 +00:00
|
|
|
static int increments[8] = { 10, 20, 30, 60, 5 * 60, 10 * 60, 15 * 60, 30 * 60 };
|
2014-01-27 19:09:08 +00:00
|
|
|
/* Time markers: at most every 10 seconds, but no more than 12 markers.
|
|
|
|
* We start out with 10 seconds and increment up to 30 minutes,
|
|
|
|
* depending on the dive time.
|
|
|
|
* This allows for 6h dives - enough (I hope) for even the craziest
|
|
|
|
* divers - but just in case, for those 8h depth-record-breaking dives,
|
|
|
|
* we double the interval if this still doesn't get us to 12 or fewer
|
|
|
|
* time markers */
|
|
|
|
i = 0;
|
|
|
|
while (i < 7 && maxtime / increments[i] > 12)
|
|
|
|
i++;
|
|
|
|
incr = increments[i];
|
|
|
|
while (maxtime / incr > 12)
|
|
|
|
incr *= 2;
|
|
|
|
timeAxis->setTickInterval(incr);
|
2014-01-15 15:34:55 +00:00
|
|
|
timeAxis->updateTicks();
|
2014-11-19 22:28:16 +00:00
|
|
|
cylinderPressureAxis->setMinimum(plotInfo.minpressure);
|
|
|
|
cylinderPressureAxis->setMaximum(plotInfo.maxpressure);
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-11-19 22:28:16 +00:00
|
|
|
rulerItem->setPlotInfo(plotInfo);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2017-02-05 09:56:43 +00:00
|
|
|
|
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
if (currentdc->divemode == CCR) {
|
2017-10-26 13:55:49 +00:00
|
|
|
gasYAxis->setPos(itemPos.partialPressure.pos.on);
|
2017-03-25 12:03:37 +00:00
|
|
|
gasYAxis->setLine(itemPos.partialPressure.expanded);
|
|
|
|
|
2017-02-05 09:56:43 +00:00
|
|
|
tankItem->setVisible(false);
|
2017-03-25 12:03:37 +00:00
|
|
|
pn2GasItem->setVisible(false);
|
|
|
|
po2GasItem->setVisible(prefs.pp_graphs.po2);
|
|
|
|
pheGasItem->setVisible(false);
|
|
|
|
o2SetpointGasItem->setVisible(prefs.show_ccr_setpoint);
|
|
|
|
ccrsensor1GasItem->setVisible(prefs.show_ccr_sensors);
|
|
|
|
ccrsensor2GasItem->setVisible(prefs.show_ccr_sensors && (currentdc->no_o2sensors > 1));
|
|
|
|
ccrsensor3GasItem->setVisible(prefs.show_ccr_sensors && (currentdc->no_o2sensors > 1));
|
2018-03-14 15:13:37 +00:00
|
|
|
ocpo2GasItem->setVisible((currentdc->divemode == PSCR) && prefs.show_scr_ocpo2);
|
2017-03-25 12:03:37 +00:00
|
|
|
temperatureItem->setVisible(false);
|
2017-02-05 09:56:43 +00:00
|
|
|
} else {
|
|
|
|
tankItem->setVisible(prefs.tankbar);
|
2017-03-25 12:03:37 +00:00
|
|
|
gasYAxis->setPos(itemPos.partialPressure.pos.off);
|
|
|
|
pn2GasItem->setVisible(false);
|
|
|
|
po2GasItem->setVisible(false);
|
|
|
|
pheGasItem->setVisible(false);
|
|
|
|
o2SetpointGasItem->setVisible(false);
|
|
|
|
ccrsensor1GasItem->setVisible(false);
|
|
|
|
ccrsensor2GasItem->setVisible(false);
|
|
|
|
ccrsensor3GasItem->setVisible(false);
|
2018-03-14 15:13:37 +00:00
|
|
|
ocpo2GasItem->setVisible(false);
|
2017-02-05 09:56:43 +00:00
|
|
|
}
|
|
|
|
#endif
|
2014-11-19 22:28:16 +00:00
|
|
|
tankItem->setData(dataModel, &plotInfo, &displayed_dive);
|
2014-01-23 19:54:34 +00:00
|
|
|
|
2014-01-27 17:14:42 +00:00
|
|
|
dataModel->emitDataChanged();
|
2014-01-23 18:12:18 +00:00
|
|
|
// The event items are a bit special since we don't know how many events are going to
|
|
|
|
// exist on a dive, so I cant create cache items for that. that's why they are here
|
|
|
|
// while all other items are up there on the constructor.
|
2014-01-16 17:02:32 +00:00
|
|
|
qDeleteAll(eventItems);
|
|
|
|
eventItems.clear();
|
|
|
|
struct event *event = currentdc->events;
|
2018-01-20 16:58:52 +00:00
|
|
|
struct event *ev;
|
|
|
|
struct gasmix lastgasmix = *get_gasmix(&displayed_dive, current_dc, 1, &ev, NULL);
|
2018-05-22 08:49:29 +00:00
|
|
|
|
2014-01-16 17:02:32 +00:00
|
|
|
while (event) {
|
2018-05-22 08:49:29 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2015-08-03 20:55:46 +00:00
|
|
|
// if print mode is selected only draw headings, SP change, gas events or bookmark event
|
|
|
|
if (printMode) {
|
2018-01-07 10:12:48 +00:00
|
|
|
if (empty_string(event->name) ||
|
2015-08-03 20:55:46 +00:00
|
|
|
!(strcmp(event->name, "heading") == 0 ||
|
|
|
|
(same_string(event->name, "SP change") && event->time.seconds == 0) ||
|
|
|
|
event_is_gaschange(event) ||
|
|
|
|
event->type == SAMPLE_EVENT_BOOKMARK)) {
|
|
|
|
event = event->next;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2018-05-22 08:49:29 +00:00
|
|
|
#else
|
|
|
|
// printMode is always selected for SUBSURFACE_MOBILE due to font problems
|
|
|
|
// BUT events are wanted.
|
|
|
|
#endif
|
2014-01-16 17:02:32 +00:00
|
|
|
DiveEventItem *item = new DiveEventItem();
|
|
|
|
item->setHorizontalAxis(timeAxis);
|
|
|
|
item->setVerticalAxis(profileYAxis);
|
|
|
|
item->setModel(dataModel);
|
2018-01-20 16:58:52 +00:00
|
|
|
item->setEvent(event, &lastgasmix);
|
2014-01-19 22:16:08 +00:00
|
|
|
item->setZValue(2);
|
2014-01-16 17:02:32 +00:00
|
|
|
scene()->addItem(item);
|
|
|
|
eventItems.push_back(item);
|
|
|
|
event = event->next;
|
|
|
|
}
|
2018-05-22 08:49:29 +00:00
|
|
|
|
2014-03-17 15:14:58 +00:00
|
|
|
// Only set visible the events that should be visible
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (DiveEventItem *event, eventItems) {
|
2014-03-16 22:10:03 +00:00
|
|
|
event->setVisible(!event->shouldBeHidden());
|
2014-02-23 16:33:58 +00:00
|
|
|
}
|
2014-06-29 15:31:37 +00:00
|
|
|
QString dcText = get_dc_nickname(currentdc->model, currentdc->deviceid);
|
2017-06-21 18:53:47 +00:00
|
|
|
if (dcText == "planned dive")
|
|
|
|
dcText = tr("Planned dive");
|
|
|
|
else if (dcText == "manually added dive")
|
|
|
|
dcText = tr("Manually added dive");
|
|
|
|
else if (dcText.isEmpty())
|
2015-01-23 18:54:16 +00:00
|
|
|
dcText = tr("Unknown dive computer");
|
2017-10-20 06:51:20 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
int nr;
|
|
|
|
if ((nr = number_of_computers(&displayed_dive)) > 1)
|
|
|
|
dcText += tr(" (#%1 of %2)").arg(dc_number + 1).arg(nr);
|
|
|
|
#endif
|
2014-05-19 05:29:43 +00:00
|
|
|
diveComputerText->setText(dcText);
|
2015-11-05 18:05:22 +00:00
|
|
|
if (haveFilesOnCommandLine() && animSpeedBackup != 0) {
|
2014-07-21 22:10:31 +00:00
|
|
|
prefs.animation_speed = animSpeedBackup;
|
2014-03-11 21:54:28 +00:00
|
|
|
}
|
2014-05-22 02:31:26 +00:00
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-05-25 20:49:26 +00:00
|
|
|
if (currentState == ADD || currentState == PLAN) { // TODO: figure a way to move this from here.
|
2014-05-22 02:31:26 +00:00
|
|
|
repositionDiveHandlers();
|
|
|
|
DivePlannerPointsModel *model = DivePlannerPointsModel::instance();
|
|
|
|
model->deleteTemporaryPlan();
|
|
|
|
}
|
2018-05-16 13:25:57 +00:00
|
|
|
if (doClearPictures)
|
2018-04-06 15:58:16 +00:00
|
|
|
clearPictures();
|
|
|
|
else
|
|
|
|
plotPictures();
|
2018-01-10 12:55:29 +00:00
|
|
|
#endif
|
2014-07-21 21:54:28 +00:00
|
|
|
|
2014-06-04 20:41:50 +00:00
|
|
|
// OK, how long did this take us? Anything above the second is way too long,
|
|
|
|
// so if we are calculation TTS / NDL then let's force that off.
|
2015-12-01 23:36:20 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-06-04 20:41:50 +00:00
|
|
|
if (measureDuration.elapsed() > 1000 && prefs.calcndltts) {
|
2016-11-01 14:18:12 +00:00
|
|
|
SettingsObjectWrapper::instance()->techDetails->setCalcndltts(false);
|
2015-11-05 18:51:12 +00:00
|
|
|
report_error(qPrintable(tr("Show NDL / TTS was disabled because of excessive processing time")));
|
2014-06-04 20:41:50 +00:00
|
|
|
}
|
2015-12-01 23:36:20 +00:00
|
|
|
#endif
|
2014-01-14 16:30:13 +00:00
|
|
|
}
|
|
|
|
|
2015-01-30 10:16:55 +00:00
|
|
|
void ProfileWidget2::recalcCeiling()
|
|
|
|
{
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2015-01-30 10:16:55 +00:00
|
|
|
diveCeiling->recalc();
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2015-01-30 10:16:55 +00:00
|
|
|
}
|
|
|
|
|
2015-11-06 12:40:20 +00:00
|
|
|
void ProfileWidget2::dateTimeChanged()
|
|
|
|
{
|
|
|
|
emit dateTimeChangedItems();
|
|
|
|
}
|
|
|
|
|
2018-05-21 16:20:29 +00:00
|
|
|
void ProfileWidget2::actionRequestedReplot(bool)
|
2016-01-25 17:54:23 +00:00
|
|
|
{
|
|
|
|
settingsChanged();
|
|
|
|
}
|
|
|
|
|
2014-01-14 16:30:13 +00:00
|
|
|
void ProfileWidget2::settingsChanged()
|
|
|
|
{
|
2014-05-05 22:58:25 +00:00
|
|
|
// if we are showing calculated ceilings then we have to replot()
|
|
|
|
// because the GF could have changed; otherwise we try to avoid replot()
|
2018-01-29 12:13:29 +00:00
|
|
|
// but always replot in PLAN/ADD/EDIT mode to avoid a bug of DiveHandlers not
|
|
|
|
// being redrawn on setting changes, causing them to become unattached
|
|
|
|
// to the profile
|
|
|
|
bool needReplot;
|
|
|
|
if (currentState == ADD || currentState == PLAN || currentState == EDIT)
|
|
|
|
needReplot = true;
|
|
|
|
else
|
|
|
|
needReplot = prefs.calcceiling;
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2017-11-25 09:38:15 +00:00
|
|
|
gasYAxis->settingsChanged(); // Initialize ticks of partial pressure graph
|
2015-01-17 18:54:10 +00:00
|
|
|
if ((prefs.percentagegraph||prefs.hrgraph) && PP_GRAPHS_ENABLED) {
|
2014-02-12 15:50:35 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.shrinked);
|
2014-11-02 18:41:41 +00:00
|
|
|
temperatureAxis->setPos(itemPos.temperatureAll.pos.on);
|
2014-02-12 16:08:01 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.shrinked);
|
2014-02-12 16:12:56 +00:00
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.shrinked);
|
2014-09-28 08:11:38 +00:00
|
|
|
|
|
|
|
if (prefs.tankbar) {
|
|
|
|
percentageAxis->setPos(itemPos.percentageWithTankBar.pos.on);
|
|
|
|
percentageAxis->animateChangeLine(itemPos.percentageWithTankBar.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeatWithTankBar.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeatWithTankBar.expanded);
|
2017-11-23 21:39:17 +00:00
|
|
|
} else {
|
2014-09-28 08:11:38 +00:00
|
|
|
percentageAxis->setPos(itemPos.percentage.pos.on);
|
2015-01-17 18:54:10 +00:00
|
|
|
percentageAxis->animateChangeLine(itemPos.percentage.expanded);
|
|
|
|
heartBeatAxis->setPos(itemPos.heartBeat.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeat.expanded);
|
2014-09-28 08:11:38 +00:00
|
|
|
}
|
|
|
|
gasYAxis->setPos(itemPos.partialPressureTissue.pos.on);
|
|
|
|
gasYAxis->animateChangeLine(itemPos.partialPressureTissue.expanded);
|
2014-12-10 22:49:12 +00:00
|
|
|
} else if (PP_GRAPHS_ENABLED || prefs.hrgraph || prefs.percentagegraph) {
|
2014-09-28 08:11:38 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.intermediate);
|
2014-11-02 18:41:41 +00:00
|
|
|
temperatureAxis->setPos(itemPos.temperature.pos.on);
|
2014-09-28 08:11:38 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.intermediate);
|
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.intermediate);
|
|
|
|
if (prefs.tankbar) {
|
|
|
|
percentageAxis->setPos(itemPos.percentageWithTankBar.pos.on);
|
|
|
|
percentageAxis->animateChangeLine(itemPos.percentageWithTankBar.expanded);
|
|
|
|
gasYAxis->setPos(itemPos.partialPressureWithTankBar.pos.on);
|
2017-11-23 21:39:17 +00:00
|
|
|
gasYAxis->animateChangeLine(itemPos.partialPressureWithTankBar.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeatWithTankBar.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeatWithTankBar.expanded);
|
2014-12-10 22:49:12 +00:00
|
|
|
} else {
|
2014-09-28 08:11:38 +00:00
|
|
|
gasYAxis->setPos(itemPos.partialPressure.pos.on);
|
|
|
|
gasYAxis->animateChangeLine(itemPos.partialPressure.expanded);
|
|
|
|
percentageAxis->setPos(itemPos.percentage.pos.on);
|
2017-11-23 21:39:17 +00:00
|
|
|
percentageAxis->animateChangeLine(itemPos.percentage.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeat.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeat.expanded);
|
2014-09-28 08:11:38 +00:00
|
|
|
}
|
2014-02-28 04:09:57 +00:00
|
|
|
} else {
|
2016-02-06 21:25:58 +00:00
|
|
|
#else
|
|
|
|
{
|
|
|
|
#endif
|
2014-02-12 15:50:35 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.expanded);
|
2014-12-12 19:44:14 +00:00
|
|
|
if (prefs.tankbar) {
|
|
|
|
temperatureAxis->setPos(itemPos.temperatureAll.pos.on);
|
|
|
|
} else {
|
|
|
|
temperatureAxis->setPos(itemPos.temperature.pos.on);
|
|
|
|
}
|
2014-02-12 16:08:01 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.expanded);
|
2014-02-12 16:12:56 +00:00
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.expanded);
|
2014-02-12 15:50:35 +00:00
|
|
|
}
|
2014-09-28 08:11:38 +00:00
|
|
|
|
2014-08-21 12:52:14 +00:00
|
|
|
tankItem->setVisible(prefs.tankbar);
|
2014-04-16 20:03:44 +00:00
|
|
|
if (prefs.zoomed_plot != isPlotZoomed) {
|
|
|
|
isPlotZoomed = prefs.zoomed_plot;
|
2014-05-05 22:58:25 +00:00
|
|
|
needReplot = true;
|
2014-02-27 15:39:53 +00:00
|
|
|
}
|
2014-05-05 22:58:25 +00:00
|
|
|
if (needReplot)
|
|
|
|
replot();
|
2014-01-14 16:30:13 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void ProfileWidget2::resizeEvent(QResizeEvent *event)
|
2014-01-14 16:30:13 +00:00
|
|
|
{
|
2014-02-11 04:45:08 +00:00
|
|
|
QGraphicsView::resizeEvent(event);
|
2014-01-15 14:55:33 +00:00
|
|
|
fitInView(sceneRect(), Qt::IgnoreAspectRatio);
|
2014-02-07 18:34:42 +00:00
|
|
|
fixBackgroundPos();
|
2014-01-15 14:55:33 +00:00
|
|
|
}
|
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-05-26 20:51:46 +00:00
|
|
|
void ProfileWidget2::mousePressEvent(QMouseEvent *event)
|
|
|
|
{
|
2015-03-11 01:23:45 +00:00
|
|
|
if (zoomLevel)
|
|
|
|
return;
|
2014-05-26 20:51:46 +00:00
|
|
|
QGraphicsView::mousePressEvent(event);
|
2017-01-29 05:00:47 +00:00
|
|
|
if (currentState == PLAN || currentState == ADD || currentState == EDIT)
|
2018-01-12 15:57:11 +00:00
|
|
|
shouldCalculateMaxDepth = shouldCalculateMaxTime = false;
|
2014-05-26 20:51:46 +00:00
|
|
|
}
|
|
|
|
|
2014-06-30 22:08:16 +00:00
|
|
|
void ProfileWidget2::divePlannerHandlerClicked()
|
|
|
|
{
|
2015-03-11 01:23:45 +00:00
|
|
|
if (zoomLevel)
|
|
|
|
return;
|
2018-01-12 15:57:11 +00:00
|
|
|
shouldCalculateMaxDepth = shouldCalculateMaxTime = false;
|
2014-06-30 22:08:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::divePlannerHandlerReleased()
|
|
|
|
{
|
2015-03-11 01:23:45 +00:00
|
|
|
if (zoomLevel)
|
|
|
|
return;
|
2018-01-12 15:57:11 +00:00
|
|
|
shouldCalculateMaxDepth = shouldCalculateMaxTime = true;
|
2014-06-30 22:08:16 +00:00
|
|
|
replot();
|
|
|
|
}
|
|
|
|
|
2014-05-26 20:51:46 +00:00
|
|
|
void ProfileWidget2::mouseReleaseEvent(QMouseEvent *event)
|
|
|
|
{
|
2015-03-11 01:23:45 +00:00
|
|
|
if (zoomLevel)
|
|
|
|
return;
|
2014-05-26 20:51:46 +00:00
|
|
|
QGraphicsView::mouseReleaseEvent(event);
|
2017-01-29 05:00:47 +00:00
|
|
|
if (currentState == PLAN || currentState == ADD || currentState == EDIT) {
|
2018-01-24 10:42:42 +00:00
|
|
|
shouldCalculateMaxTime = shouldCalculateMaxDepth = true;
|
2014-05-26 20:51:46 +00:00
|
|
|
replot();
|
|
|
|
}
|
|
|
|
}
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-05-26 20:51:46 +00:00
|
|
|
|
2014-01-15 14:55:33 +00:00
|
|
|
void ProfileWidget2::fixBackgroundPos()
|
|
|
|
{
|
2014-05-28 18:43:32 +00:00
|
|
|
static QPixmap toBeScaled(backgroundFile);
|
2014-02-28 04:09:57 +00:00
|
|
|
if (currentState != EMPTY)
|
2014-02-07 21:42:47 +00:00
|
|
|
return;
|
2014-03-10 15:59:01 +00:00
|
|
|
QPixmap p = toBeScaled.scaledToHeight(viewport()->height() - 40, Qt::SmoothTransformation);
|
2014-01-16 04:50:56 +00:00
|
|
|
int x = viewport()->width() / 2 - p.width() / 2;
|
2014-03-10 15:59:01 +00:00
|
|
|
int y = viewport()->height() / 2 - p.height() / 2;
|
2014-02-07 17:32:39 +00:00
|
|
|
background->setPixmap(p);
|
|
|
|
background->setX(mapToScene(x, 0).x());
|
2014-03-10 15:59:01 +00:00
|
|
|
background->setY(mapToScene(y, 20).y());
|
2014-01-14 16:30:13 +00:00
|
|
|
}
|
2014-02-04 23:47:50 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-02-28 04:09:57 +00:00
|
|
|
void ProfileWidget2::wheelEvent(QWheelEvent *event)
|
2014-02-04 23:47:50 +00:00
|
|
|
{
|
2014-03-10 15:59:01 +00:00
|
|
|
if (currentState == EMPTY)
|
|
|
|
return;
|
2014-02-05 16:34:45 +00:00
|
|
|
QPoint toolTipPos = mapFromScene(toolTipItem->pos());
|
2015-01-02 00:28:37 +00:00
|
|
|
if (event->buttons() == Qt::LeftButton)
|
2014-08-06 11:51:54 +00:00
|
|
|
return;
|
2014-02-04 23:47:50 +00:00
|
|
|
if (event->delta() > 0 && zoomLevel < 20) {
|
2014-03-10 15:59:02 +00:00
|
|
|
scale(zoomFactor, zoomFactor);
|
2014-02-04 23:47:50 +00:00
|
|
|
zoomLevel++;
|
|
|
|
} else if (event->delta() < 0 && zoomLevel > 0) {
|
|
|
|
// Zooming out
|
2014-03-10 15:59:02 +00:00
|
|
|
scale(1.0 / zoomFactor, 1.0 / zoomFactor);
|
2014-02-04 23:47:50 +00:00
|
|
|
zoomLevel--;
|
|
|
|
}
|
|
|
|
scrollViewTo(event->pos());
|
2014-02-05 16:34:45 +00:00
|
|
|
toolTipItem->setPos(mapToScene(toolTipPos));
|
2014-02-04 23:47:50 +00:00
|
|
|
}
|
|
|
|
|
2014-05-21 18:52:24 +00:00
|
|
|
void ProfileWidget2::mouseDoubleClickEvent(QMouseEvent *event)
|
|
|
|
{
|
|
|
|
if (currentState == PLAN || currentState == ADD) {
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
QPointF mappedPos = mapToScene(event->pos());
|
|
|
|
if (isPointOutOfBoundaries(mappedPos))
|
|
|
|
return;
|
|
|
|
|
2017-03-08 06:41:41 +00:00
|
|
|
int minutes = lrint(timeAxis->valueAt(mappedPos) / 60);
|
|
|
|
int milimeters = lrint(profileYAxis->valueAt(mappedPos) / M_OR_FT(1, 1)) * M_OR_FT(1, 1);
|
2017-02-04 13:12:43 +00:00
|
|
|
plannerModel->addStop(milimeters, minutes * 60, -1, 0, true);
|
2014-05-21 18:52:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ProfileWidget2::isPointOutOfBoundaries(const QPointF &point) const
|
|
|
|
{
|
|
|
|
double xpos = timeAxis->valueAt(point);
|
|
|
|
double ypos = profileYAxis->valueAt(point);
|
2018-02-17 20:21:16 +00:00
|
|
|
return xpos > timeAxis->maximum() ||
|
|
|
|
xpos < timeAxis->minimum() ||
|
|
|
|
ypos > profileYAxis->maximum() ||
|
|
|
|
ypos < profileYAxis->minimum();
|
2014-05-21 18:52:24 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void ProfileWidget2::scrollViewTo(const QPoint &pos)
|
2014-02-04 23:47:50 +00:00
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
/* since we cannot use translate() directly on the scene we hack on
|
2014-02-04 23:47:50 +00:00
|
|
|
* the scroll bars (hidden) functionality */
|
2014-03-10 15:59:01 +00:00
|
|
|
if (!zoomLevel || currentState == EMPTY)
|
2014-02-04 23:47:50 +00:00
|
|
|
return;
|
|
|
|
QScrollBar *vs = verticalScrollBar();
|
|
|
|
QScrollBar *hs = horizontalScrollBar();
|
2014-02-05 16:57:02 +00:00
|
|
|
const qreal yRat = (qreal)pos.y() / viewport()->height();
|
|
|
|
const qreal xRat = (qreal)pos.x() / viewport()->width();
|
2017-03-23 01:13:49 +00:00
|
|
|
vs->setValue(lrint(yRat * vs->maximum()));
|
|
|
|
hs->setValue(lrint(xRat * hs->maximum()));
|
2014-02-04 23:47:50 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void ProfileWidget2::mouseMoveEvent(QMouseEvent *event)
|
2014-02-04 23:47:50 +00:00
|
|
|
{
|
2015-01-14 16:41:56 +00:00
|
|
|
QPointF pos = mapToScene(event->pos());
|
|
|
|
toolTipItem->refresh(pos);
|
2014-02-04 23:47:50 +00:00
|
|
|
if (zoomLevel == 0) {
|
|
|
|
QGraphicsView::mouseMoveEvent(event);
|
2014-02-05 16:34:45 +00:00
|
|
|
} else {
|
2015-01-14 16:41:56 +00:00
|
|
|
QPoint toolTipPos = mapFromScene(toolTipItem->pos());
|
2014-02-04 23:47:50 +00:00
|
|
|
scrollViewTo(event->pos());
|
2014-02-07 23:38:06 +00:00
|
|
|
toolTipItem->setPos(mapToScene(toolTipPos));
|
2014-02-04 23:47:50 +00:00
|
|
|
}
|
2014-08-05 21:27:00 +00:00
|
|
|
|
|
|
|
qreal vValue = profileYAxis->valueAt(pos);
|
|
|
|
qreal hValue = timeAxis->valueAt(pos);
|
2016-02-06 21:25:58 +00:00
|
|
|
|
2015-01-02 00:28:37 +00:00
|
|
|
if (profileYAxis->maximum() >= vValue && profileYAxis->minimum() <= vValue) {
|
2014-08-05 21:27:00 +00:00
|
|
|
mouseFollowerHorizontal->setPos(timeAxis->pos().x(), pos.y());
|
|
|
|
}
|
2015-01-02 00:28:37 +00:00
|
|
|
if (timeAxis->maximum() >= hValue && timeAxis->minimum() <= hValue) {
|
2014-08-05 21:27:00 +00:00
|
|
|
mouseFollowerVertical->setPos(pos.x(), profileYAxis->line().y1());
|
|
|
|
}
|
2014-02-04 23:47:50 +00:00
|
|
|
}
|
|
|
|
|
2014-02-05 18:15:59 +00:00
|
|
|
bool ProfileWidget2::eventFilter(QObject *object, QEvent *event)
|
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
QGraphicsScene *s = qobject_cast<QGraphicsScene *>(object);
|
|
|
|
if (s && event->type() == QEvent::GraphicsSceneHelp) {
|
2014-02-05 18:15:59 +00:00
|
|
|
event->ignore();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return QGraphicsView::eventFilter(object, event);
|
|
|
|
}
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-02-07 18:34:42 +00:00
|
|
|
|
2018-01-10 15:36:45 +00:00
|
|
|
template <typename T>
|
|
|
|
static void hideAll(T &container)
|
|
|
|
{
|
|
|
|
Q_FOREACH (auto *item, container)
|
|
|
|
item->setVisible(false);
|
|
|
|
}
|
|
|
|
|
2014-02-07 18:34:42 +00:00
|
|
|
void ProfileWidget2::setEmptyState()
|
|
|
|
{
|
|
|
|
// Then starting Empty State, move the background up.
|
|
|
|
if (currentState == EMPTY)
|
|
|
|
return;
|
|
|
|
|
2014-05-22 00:23:19 +00:00
|
|
|
disconnectTemporaryConnections();
|
2014-05-21 23:34:06 +00:00
|
|
|
setBackgroundBrush(getColor(::BACKGROUND, isGrayscale));
|
2014-02-10 16:41:59 +00:00
|
|
|
dataModel->clear();
|
2014-02-07 18:34:42 +00:00
|
|
|
currentState = EMPTY;
|
2015-11-05 17:32:19 +00:00
|
|
|
emit enableToolbar(false);
|
2014-03-10 15:59:01 +00:00
|
|
|
|
2014-02-07 18:34:42 +00:00
|
|
|
fixBackgroundPos();
|
2014-02-10 17:01:04 +00:00
|
|
|
background->setVisible(true);
|
2014-03-10 15:59:01 +00:00
|
|
|
|
|
|
|
profileYAxis->setVisible(false);
|
|
|
|
gasYAxis->setVisible(false);
|
|
|
|
timeAxis->setVisible(false);
|
|
|
|
temperatureAxis->setVisible(false);
|
|
|
|
cylinderPressureAxis->setVisible(false);
|
2014-02-07 18:34:42 +00:00
|
|
|
diveComputerText->setVisible(false);
|
2016-02-06 21:25:58 +00:00
|
|
|
reportedCeiling->setVisible(false);
|
|
|
|
tankItem->setVisible(false);
|
2014-03-18 18:08:23 +00:00
|
|
|
pn2GasItem->setVisible(false);
|
|
|
|
po2GasItem->setVisible(false);
|
2017-03-25 12:03:37 +00:00
|
|
|
pheGasItem->setVisible(false);
|
2015-01-05 07:20:26 +00:00
|
|
|
o2SetpointGasItem->setVisible(false);
|
2015-01-20 18:13:53 +00:00
|
|
|
ccrsensor1GasItem->setVisible(false);
|
|
|
|
ccrsensor2GasItem->setVisible(false);
|
|
|
|
ccrsensor3GasItem->setVisible(false);
|
2018-03-14 15:13:37 +00:00
|
|
|
ocpo2GasItem->setVisible(false);
|
2017-03-25 12:03:37 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
toolTipItem->setVisible(false);
|
|
|
|
diveCeiling->setVisible(false);
|
|
|
|
decoModelParameters->setVisible(false);
|
|
|
|
rulerItem->setVisible(false);
|
2014-09-15 12:09:00 +00:00
|
|
|
ambPressureItem->setVisible(false);
|
|
|
|
gflineItem->setVisible(false);
|
2014-08-05 21:27:00 +00:00
|
|
|
mouseFollowerHorizontal->setVisible(false);
|
|
|
|
mouseFollowerVertical->setVisible(false);
|
2015-12-06 19:36:21 +00:00
|
|
|
heartBeatAxis->setVisible(false);
|
|
|
|
heartBeatItem->setVisible(false);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-05-28 18:43:32 +00:00
|
|
|
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(allTissues);
|
|
|
|
hideAll(allPercentages);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(eventItems);
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(handles);
|
2015-11-06 00:05:44 +00:00
|
|
|
#endif
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(gases);
|
2014-02-07 18:34:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::setProfileState()
|
|
|
|
{
|
2014-02-07 18:54:12 +00:00
|
|
|
// Then starting Empty State, move the background up.
|
|
|
|
if (currentState == PROFILE)
|
|
|
|
return;
|
2014-02-07 18:34:42 +00:00
|
|
|
|
2014-05-22 00:23:19 +00:00
|
|
|
disconnectTemporaryConnections();
|
2018-01-10 12:55:29 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2018-04-06 15:58:16 +00:00
|
|
|
connect(DivePictureModel::instance(), &DivePictureModel::dataChanged, this, &ProfileWidget2::updatePictures);
|
2015-01-02 00:28:37 +00:00
|
|
|
connect(DivePictureModel::instance(), SIGNAL(rowsInserted(const QModelIndex &, int, int)), this, SLOT(plotPictures()));
|
2018-05-19 19:18:39 +00:00
|
|
|
connect(DivePictureModel::instance(), &DivePictureModel::rowsRemoved, this, &ProfileWidget2::removePictures);
|
2018-05-18 18:33:01 +00:00
|
|
|
connect(DivePictureModel::instance(), &DivePictureModel::modelReset, this, &ProfileWidget2::plotPictures);
|
2018-01-10 12:55:29 +00:00
|
|
|
#endif
|
2014-06-03 22:09:12 +00:00
|
|
|
/* show the same stuff that the profile shows. */
|
|
|
|
|
2015-11-05 19:12:22 +00:00
|
|
|
emit enableShortcuts();
|
2014-05-24 15:39:40 +00:00
|
|
|
|
2014-02-07 18:54:12 +00:00
|
|
|
currentState = PROFILE;
|
2015-11-05 17:32:19 +00:00
|
|
|
emit enableToolbar(true);
|
2014-03-25 21:34:10 +00:00
|
|
|
setBackgroundBrush(getColor(::BACKGROUND, isGrayscale));
|
2014-02-07 19:38:00 +00:00
|
|
|
|
2014-02-07 21:42:47 +00:00
|
|
|
background->setVisible(false);
|
2014-03-10 15:59:01 +00:00
|
|
|
profileYAxis->setVisible(true);
|
|
|
|
gasYAxis->setVisible(true);
|
|
|
|
timeAxis->setVisible(true);
|
|
|
|
temperatureAxis->setVisible(true);
|
|
|
|
cylinderPressureAxis->setVisible(true);
|
2014-02-07 19:38:00 +00:00
|
|
|
|
2014-02-07 21:42:47 +00:00
|
|
|
profileYAxis->setPos(itemPos.depth.pos.on);
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
toolTipItem->readPos();
|
|
|
|
toolTipItem->setVisible(true);
|
2015-01-17 18:54:10 +00:00
|
|
|
if ((prefs.percentagegraph||prefs.hrgraph) && PP_GRAPHS_ENABLED) {
|
2014-09-28 08:11:38 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.shrinked);
|
2014-11-02 18:41:41 +00:00
|
|
|
temperatureAxis->setPos(itemPos.temperatureAll.pos.on);
|
2014-09-28 08:11:38 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.shrinked);
|
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.shrinked);
|
|
|
|
|
|
|
|
if (prefs.tankbar) {
|
|
|
|
percentageAxis->setPos(itemPos.percentageWithTankBar.pos.on);
|
|
|
|
percentageAxis->animateChangeLine(itemPos.percentageWithTankBar.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeatWithTankBar.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeatWithTankBar.expanded);
|
|
|
|
}else {
|
2014-09-28 08:11:38 +00:00
|
|
|
percentageAxis->setPos(itemPos.percentage.pos.on);
|
2015-01-17 18:54:10 +00:00
|
|
|
percentageAxis->animateChangeLine(itemPos.percentage.expanded);
|
|
|
|
heartBeatAxis->setPos(itemPos.heartBeat.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeat.expanded);
|
2014-09-28 08:11:38 +00:00
|
|
|
}
|
|
|
|
gasYAxis->setPos(itemPos.partialPressureTissue.pos.on);
|
|
|
|
gasYAxis->animateChangeLine(itemPos.partialPressureTissue.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
|
2014-12-10 22:49:12 +00:00
|
|
|
} else if (PP_GRAPHS_ENABLED || prefs.hrgraph || prefs.percentagegraph) {
|
2014-09-28 08:11:38 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.intermediate);
|
2014-11-02 18:41:41 +00:00
|
|
|
temperatureAxis->setPos(itemPos.temperature.pos.on);
|
2014-09-28 08:11:38 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.intermediate);
|
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.intermediate);
|
|
|
|
if (prefs.tankbar) {
|
|
|
|
percentageAxis->setPos(itemPos.percentageWithTankBar.pos.on);
|
|
|
|
percentageAxis->animateChangeLine(itemPos.percentageWithTankBar.expanded);
|
|
|
|
gasYAxis->setPos(itemPos.partialPressureWithTankBar.pos.on);
|
|
|
|
gasYAxis->setLine(itemPos.partialPressureWithTankBar.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeatWithTankBar.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeatWithTankBar.expanded);
|
2014-12-10 22:49:12 +00:00
|
|
|
} else {
|
2014-09-28 08:11:38 +00:00
|
|
|
gasYAxis->setPos(itemPos.partialPressure.pos.on);
|
|
|
|
gasYAxis->animateChangeLine(itemPos.partialPressure.expanded);
|
|
|
|
percentageAxis->setPos(itemPos.percentage.pos.on);
|
|
|
|
percentageAxis->setLine(itemPos.percentage.expanded);
|
2015-01-17 18:54:10 +00:00
|
|
|
heartBeatAxis->setPos(itemPos.heartBeat.pos.on);
|
|
|
|
heartBeatAxis->animateChangeLine(itemPos.heartBeat.expanded);
|
2014-09-28 08:11:38 +00:00
|
|
|
}
|
2014-02-28 04:09:57 +00:00
|
|
|
} else {
|
2016-02-06 21:25:58 +00:00
|
|
|
#else
|
|
|
|
{
|
|
|
|
#endif
|
2014-09-28 08:11:38 +00:00
|
|
|
profileYAxis->animateChangeLine(itemPos.depth.expanded);
|
2014-12-12 19:44:14 +00:00
|
|
|
if (prefs.tankbar) {
|
|
|
|
temperatureAxis->setPos(itemPos.temperatureAll.pos.on);
|
|
|
|
} else {
|
|
|
|
temperatureAxis->setPos(itemPos.temperature.pos.on);
|
|
|
|
}
|
2014-09-28 08:11:38 +00:00
|
|
|
temperatureAxis->animateChangeLine(itemPos.temperature.expanded);
|
|
|
|
cylinderPressureAxis->animateChangeLine(itemPos.cylinder.expanded);
|
2014-02-07 21:42:47 +00:00
|
|
|
}
|
2016-02-06 21:25:58 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-04-16 20:03:44 +00:00
|
|
|
pn2GasItem->setVisible(prefs.pp_graphs.pn2);
|
|
|
|
po2GasItem->setVisible(prefs.pp_graphs.po2);
|
|
|
|
pheGasItem->setVisible(prefs.pp_graphs.phe);
|
2015-01-23 07:47:33 +00:00
|
|
|
|
2015-01-24 22:42:25 +00:00
|
|
|
bool setpointflag = current_dive && (current_dc->divemode == CCR) && prefs.pp_graphs.po2;
|
2015-01-23 07:47:33 +00:00
|
|
|
bool sensorflag = setpointflag && prefs.show_ccr_sensors;
|
|
|
|
o2SetpointGasItem->setVisible(setpointflag && prefs.show_ccr_setpoint);
|
|
|
|
ccrsensor1GasItem->setVisible(sensorflag);
|
|
|
|
ccrsensor2GasItem->setVisible(sensorflag && (current_dc->no_o2sensors > 1));
|
|
|
|
ccrsensor3GasItem->setVisible(sensorflag && (current_dc->no_o2sensors > 2));
|
2018-05-02 11:21:00 +00:00
|
|
|
ocpo2GasItem->setVisible(current_dive && (current_dc->divemode == PSCR) && prefs.show_scr_ocpo2);
|
2014-02-07 21:42:47 +00:00
|
|
|
|
2014-04-16 20:03:44 +00:00
|
|
|
heartBeatItem->setVisible(prefs.hrgraph);
|
|
|
|
diveCeiling->setVisible(prefs.calcceiling);
|
2015-10-24 09:03:46 +00:00
|
|
|
decoModelParameters->setVisible(prefs.calcceiling);
|
2014-02-07 23:28:59 +00:00
|
|
|
|
2014-04-16 20:03:44 +00:00
|
|
|
if (prefs.calcalltissues) {
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (DiveCalculatedTissue *tissue, allTissues) {
|
2014-02-07 23:28:59 +00:00
|
|
|
tissue->setVisible(true);
|
|
|
|
}
|
|
|
|
}
|
2014-09-15 12:09:00 +00:00
|
|
|
if (prefs.percentagegraph) {
|
|
|
|
Q_FOREACH (DivePercentageItem *percentage, allPercentages) {
|
|
|
|
percentage->setVisible(true);
|
|
|
|
}
|
2014-09-28 08:11:38 +00:00
|
|
|
}
|
2014-09-15 12:09:00 +00:00
|
|
|
|
2014-05-21 15:57:48 +00:00
|
|
|
rulerItem->setVisible(prefs.rulergraph);
|
2016-02-06 21:25:58 +00:00
|
|
|
|
|
|
|
#endif
|
|
|
|
timeAxis->setPos(itemPos.time.pos.on);
|
|
|
|
timeAxis->setLine(itemPos.time.expanded);
|
|
|
|
|
|
|
|
cylinderPressureAxis->setPos(itemPos.cylinder.pos.on);
|
|
|
|
meanDepthItem->setVisible(prefs.show_average_depth);
|
|
|
|
|
|
|
|
diveComputerText->setVisible(true);
|
|
|
|
diveComputerText->setPos(itemPos.dcLabel.on);
|
|
|
|
|
|
|
|
reportedCeiling->setVisible(prefs.dcceiling);
|
|
|
|
|
2014-08-15 14:11:14 +00:00
|
|
|
tankItem->setVisible(prefs.tankbar);
|
2014-08-15 13:30:31 +00:00
|
|
|
tankItem->setPos(itemPos.tankBar.on);
|
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(handles);
|
2016-02-06 21:25:58 +00:00
|
|
|
mouseFollowerHorizontal->setVisible(false);
|
|
|
|
mouseFollowerVertical->setVisible(false);
|
2015-11-06 00:05:44 +00:00
|
|
|
#endif
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(gases);
|
2014-02-07 18:34:42 +00:00
|
|
|
}
|
2014-02-17 22:15:40 +00:00
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-08-20 01:59:34 +00:00
|
|
|
void ProfileWidget2::clearHandlers()
|
|
|
|
{
|
|
|
|
if (handles.count()) {
|
|
|
|
foreach (DiveHandler *handle, handles) {
|
|
|
|
scene()->removeItem(handle);
|
2015-03-13 00:32:49 +00:00
|
|
|
delete handle;
|
2014-08-20 01:59:34 +00:00
|
|
|
}
|
|
|
|
handles.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-31 18:32:02 +00:00
|
|
|
void ProfileWidget2::setToolTipVisibile(bool visible)
|
|
|
|
{
|
|
|
|
toolTipItem->setVisible(visible);
|
|
|
|
}
|
|
|
|
|
2014-05-21 18:52:24 +00:00
|
|
|
void ProfileWidget2::setAddState()
|
|
|
|
{
|
|
|
|
if (currentState == ADD)
|
|
|
|
return;
|
|
|
|
|
2015-03-13 00:32:50 +00:00
|
|
|
clearHandlers();
|
2014-05-22 02:31:26 +00:00
|
|
|
setProfileState();
|
2014-08-06 19:33:36 +00:00
|
|
|
mouseFollowerHorizontal->setVisible(true);
|
|
|
|
mouseFollowerVertical->setVisible(true);
|
|
|
|
mouseFollowerHorizontal->setLine(timeAxis->line());
|
|
|
|
mouseFollowerVertical->setLine(QLineF(0, profileYAxis->pos().y(), 0, timeAxis->pos().y()));
|
2014-05-22 00:23:19 +00:00
|
|
|
disconnectTemporaryConnections();
|
2015-11-05 19:12:22 +00:00
|
|
|
emit disableShortcuts(false);
|
2014-05-24 15:39:40 +00:00
|
|
|
actionsForKeys[Qt::Key_Left]->setShortcut(Qt::Key_Left);
|
|
|
|
actionsForKeys[Qt::Key_Right]->setShortcut(Qt::Key_Right);
|
|
|
|
actionsForKeys[Qt::Key_Up]->setShortcut(Qt::Key_Up);
|
|
|
|
actionsForKeys[Qt::Key_Down]->setShortcut(Qt::Key_Down);
|
|
|
|
actionsForKeys[Qt::Key_Escape]->setShortcut(Qt::Key_Escape);
|
|
|
|
actionsForKeys[Qt::Key_Delete]->setShortcut(Qt::Key_Delete);
|
|
|
|
|
2014-05-22 02:31:26 +00:00
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
connect(plannerModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(replot()));
|
|
|
|
connect(plannerModel, SIGNAL(cylinderModelEdited()), this, SLOT(replot()));
|
2015-11-12 00:37:18 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-05-22 02:31:26 +00:00
|
|
|
connect(plannerModel, SIGNAL(rowsInserted(const QModelIndex &, int, int)),
|
|
|
|
this, SLOT(pointInserted(const QModelIndex &, int, int)));
|
|
|
|
connect(plannerModel, SIGNAL(rowsRemoved(const QModelIndex &, int, int)),
|
|
|
|
this, SLOT(pointsRemoved(const QModelIndex &, int, int)));
|
2015-11-12 00:37:18 +00:00
|
|
|
#endif
|
2014-05-21 18:52:24 +00:00
|
|
|
/* show the same stuff that the profile shows. */
|
|
|
|
currentState = ADD; /* enable the add state. */
|
2014-05-27 19:55:29 +00:00
|
|
|
diveCeiling->setVisible(true);
|
2015-10-24 09:03:46 +00:00
|
|
|
decoModelParameters->setVisible(true);
|
2014-06-30 21:46:51 +00:00
|
|
|
setBackgroundBrush(QColor("#A7DCFF"));
|
2014-05-21 18:52:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::setPlanState()
|
|
|
|
{
|
|
|
|
if (currentState == PLAN)
|
|
|
|
return;
|
2014-05-25 18:12:00 +00:00
|
|
|
|
2014-05-22 02:31:26 +00:00
|
|
|
setProfileState();
|
2014-08-06 19:33:36 +00:00
|
|
|
mouseFollowerHorizontal->setVisible(true);
|
|
|
|
mouseFollowerVertical->setVisible(true);
|
|
|
|
mouseFollowerHorizontal->setLine(timeAxis->line());
|
|
|
|
mouseFollowerVertical->setLine(QLineF(0, profileYAxis->pos().y(), 0, timeAxis->pos().y()));
|
2014-05-22 00:23:19 +00:00
|
|
|
disconnectTemporaryConnections();
|
2015-11-05 19:12:22 +00:00
|
|
|
emit disableShortcuts(true);
|
2014-05-25 18:12:00 +00:00
|
|
|
actionsForKeys[Qt::Key_Left]->setShortcut(Qt::Key_Left);
|
|
|
|
actionsForKeys[Qt::Key_Right]->setShortcut(Qt::Key_Right);
|
|
|
|
actionsForKeys[Qt::Key_Up]->setShortcut(Qt::Key_Up);
|
|
|
|
actionsForKeys[Qt::Key_Down]->setShortcut(Qt::Key_Down);
|
|
|
|
actionsForKeys[Qt::Key_Escape]->setShortcut(Qt::Key_Escape);
|
|
|
|
actionsForKeys[Qt::Key_Delete]->setShortcut(Qt::Key_Delete);
|
2014-05-25 17:16:43 +00:00
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
connect(plannerModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(replot()));
|
|
|
|
connect(plannerModel, SIGNAL(cylinderModelEdited()), this, SLOT(replot()));
|
|
|
|
connect(plannerModel, SIGNAL(rowsInserted(const QModelIndex &, int, int)),
|
|
|
|
this, SLOT(pointInserted(const QModelIndex &, int, int)));
|
|
|
|
connect(plannerModel, SIGNAL(rowsRemoved(const QModelIndex &, int, int)),
|
|
|
|
this, SLOT(pointsRemoved(const QModelIndex &, int, int)));
|
2014-05-25 18:12:00 +00:00
|
|
|
/* show the same stuff that the profile shows. */
|
2014-05-21 18:52:24 +00:00
|
|
|
currentState = PLAN; /* enable the add state. */
|
2014-05-27 19:55:29 +00:00
|
|
|
diveCeiling->setVisible(true);
|
2015-10-24 09:03:46 +00:00
|
|
|
decoModelParameters->setVisible(true);
|
2014-06-30 21:46:51 +00:00
|
|
|
setBackgroundBrush(QColor("#D7E3EF"));
|
2014-05-21 18:52:24 +00:00
|
|
|
}
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-05-21 18:52:24 +00:00
|
|
|
|
2014-02-17 22:15:40 +00:00
|
|
|
extern struct ev_select *ev_namelist;
|
|
|
|
extern int evn_allocated;
|
|
|
|
extern int evn_used;
|
|
|
|
|
2014-05-26 22:17:34 +00:00
|
|
|
bool ProfileWidget2::isPlanner()
|
|
|
|
{
|
|
|
|
return currentState == PLAN;
|
|
|
|
}
|
|
|
|
|
2014-11-19 22:31:28 +00:00
|
|
|
struct plot_data *ProfileWidget2::getEntryFromPos(QPointF pos)
|
|
|
|
{
|
|
|
|
// find the time stamp corresponding to the mouse position
|
2017-03-23 01:13:49 +00:00
|
|
|
int seconds = lrint(timeAxis->valueAt(pos));
|
2015-06-22 04:35:04 +00:00
|
|
|
struct plot_data *entry = NULL;
|
2014-11-19 22:31:28 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < plotInfo.nr; i++) {
|
|
|
|
entry = plotInfo.entry + i;
|
2016-03-10 04:08:03 +00:00
|
|
|
if ((int)entry->sec >= seconds)
|
2014-11-19 22:31:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2014-12-13 21:06:52 +00:00
|
|
|
void ProfileWidget2::setReplot(bool state)
|
|
|
|
{
|
|
|
|
replotEnabled = state;
|
|
|
|
}
|
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-02-28 04:09:57 +00:00
|
|
|
void ProfileWidget2::contextMenuEvent(QContextMenuEvent *event)
|
2014-02-17 22:15:40 +00:00
|
|
|
{
|
2014-05-25 03:33:39 +00:00
|
|
|
if (currentState == ADD || currentState == PLAN) {
|
|
|
|
QGraphicsView::contextMenuEvent(event);
|
|
|
|
return;
|
|
|
|
}
|
2014-05-19 05:39:34 +00:00
|
|
|
QMenu m;
|
|
|
|
bool isDCName = false;
|
2014-02-17 22:15:40 +00:00
|
|
|
if (selected_dive == -1)
|
|
|
|
return;
|
2014-05-19 05:39:34 +00:00
|
|
|
// figure out if we are ontop of the dive computer name in the profile
|
|
|
|
QGraphicsItem *sceneItem = itemAt(mapFromGlobal(event->globalPos()));
|
|
|
|
if (sceneItem) {
|
|
|
|
QGraphicsItem *parentItem = sceneItem;
|
|
|
|
while (parentItem) {
|
|
|
|
if (parentItem->data(SUBSURFACE_OBJ_DATA) == SUBSURFACE_OBJ_DC_TEXT) {
|
|
|
|
isDCName = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
parentItem = parentItem->parentItem();
|
|
|
|
}
|
|
|
|
if (isDCName) {
|
2014-06-11 20:56:33 +00:00
|
|
|
if (dc_number == 0 && count_divecomputers() == 1)
|
|
|
|
// nothing to do, can't delete or reorder
|
2014-05-19 05:39:34 +00:00
|
|
|
return;
|
|
|
|
// create menu to show when right clicking on dive computer name
|
2014-06-11 20:56:33 +00:00
|
|
|
if (dc_number > 0)
|
2017-02-20 07:15:18 +00:00
|
|
|
m.addAction(tr("Make first dive computer"), this, SLOT(makeFirstDC()));
|
2014-06-11 20:56:33 +00:00
|
|
|
if (count_divecomputers() > 1)
|
2017-02-20 07:15:18 +00:00
|
|
|
m.addAction(tr("Delete this dive computer"), this, SLOT(deleteCurrentDC()));
|
2014-05-19 05:39:34 +00:00
|
|
|
m.exec(event->globalPos());
|
|
|
|
// don't show the regular profile context menu
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// create the profile context menu
|
2014-02-17 22:15:40 +00:00
|
|
|
GasSelectionModel *model = GasSelectionModel::instance();
|
|
|
|
model->repopulate();
|
|
|
|
int rowCount = model->rowCount();
|
2014-11-19 21:03:58 +00:00
|
|
|
if (rowCount > 1) {
|
|
|
|
// if we have more than one gas, offer to switch to another one
|
|
|
|
QMenu *gasChange = m.addMenu(tr("Add gas change"));
|
2016-03-23 17:11:39 +00:00
|
|
|
for (int i = 0; i < rowCount; i++) {
|
2014-11-19 21:03:58 +00:00
|
|
|
QAction *action = new QAction(&m);
|
2017-11-18 08:35:06 +00:00
|
|
|
action->setText(model->data(model->index(i, 0), Qt::DisplayRole).toString() + QString(tr(" (cyl. %1)")).arg(i + 1));
|
2014-11-19 21:03:58 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(changeGas()));
|
|
|
|
action->setData(event->globalPos());
|
|
|
|
gasChange->addAction(action);
|
|
|
|
}
|
2014-02-17 22:15:40 +00:00
|
|
|
}
|
2017-03-01 09:18:06 +00:00
|
|
|
QAction *setpointAction = m.addAction(tr("Add setpoint change"), this, SLOT(addSetpointChange()));
|
2014-11-26 13:22:41 +00:00
|
|
|
setpointAction->setData(event->globalPos());
|
2014-07-11 17:39:05 +00:00
|
|
|
QAction *action = m.addAction(tr("Add bookmark"), this, SLOT(addBookmark()));
|
2014-02-17 22:15:40 +00:00
|
|
|
action->setData(event->globalPos());
|
2018-04-08 20:09:26 +00:00
|
|
|
struct event *ev = NULL;
|
2018-05-08 14:24:51 +00:00
|
|
|
enum divemode_t divemode = UNDEF_COMP_TYPE;
|
2018-04-08 20:09:26 +00:00
|
|
|
QPointF scenePos = mapToScene(mapFromGlobal(event->globalPos()));
|
|
|
|
QString gas = action->text();
|
|
|
|
qreal sec_val = timeAxis->valueAt(scenePos);
|
|
|
|
int seconds = (sec_val < 0.0) ? 0 : (int)sec_val;
|
|
|
|
|
|
|
|
get_current_divemode(current_dc, seconds, &ev, &divemode);
|
|
|
|
QMenu *changeMode = m.addMenu(tr("Change divemode"));
|
|
|
|
if (divemode != OC) {
|
|
|
|
QAction *action = new QAction(&m);
|
2018-05-17 08:04:41 +00:00
|
|
|
action->setText(divemode_text_ui[OC]);
|
2018-04-08 20:09:26 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(addDivemodeSwith()));
|
|
|
|
action->setData(event->globalPos());
|
|
|
|
changeMode->addAction(action);
|
|
|
|
}
|
|
|
|
if (divemode != CCR) {
|
|
|
|
QAction *action = new QAction(&m);
|
2018-05-17 08:04:41 +00:00
|
|
|
action->setText(divemode_text_ui[CCR]);
|
2018-04-08 20:09:26 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(addDivemodeSwith()));
|
|
|
|
action->setData(event->globalPos());
|
|
|
|
changeMode->addAction(action);
|
|
|
|
}
|
|
|
|
if (divemode != PSCR) {
|
|
|
|
QAction *action = new QAction(&m);
|
2018-05-17 08:04:41 +00:00
|
|
|
action->setText(divemode_text_ui[PSCR]);
|
2018-04-08 20:09:26 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(addDivemodeSwith()));
|
|
|
|
action->setData(event->globalPos());
|
|
|
|
changeMode->addAction(action);
|
2018-04-08 06:07:57 +00:00
|
|
|
}
|
2015-04-23 20:56:24 +00:00
|
|
|
|
|
|
|
if (same_string(current_dc->model, "manually added dive"))
|
2015-11-07 21:02:58 +00:00
|
|
|
m.addAction(tr("Edit the profile"), this, SIGNAL(editCurrentDive()));
|
2015-04-23 20:56:24 +00:00
|
|
|
|
2014-03-16 04:12:34 +00:00
|
|
|
if (DiveEventItem *item = dynamic_cast<DiveEventItem *>(sceneItem)) {
|
2014-02-17 22:15:40 +00:00
|
|
|
action = new QAction(&m);
|
2014-07-11 17:39:05 +00:00
|
|
|
action->setText(tr("Remove event"));
|
2014-02-28 04:09:57 +00:00
|
|
|
action->setData(QVariant::fromValue<void *>(item)); // so we know what to remove.
|
2014-02-17 22:15:40 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(removeEvent()));
|
|
|
|
m.addAction(action);
|
|
|
|
action = new QAction(&m);
|
|
|
|
action->setText(tr("Hide similar events"));
|
2014-02-28 04:09:57 +00:00
|
|
|
action->setData(QVariant::fromValue<void *>(item));
|
2014-02-17 22:15:40 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(hideEvents()));
|
|
|
|
m.addAction(action);
|
2014-11-20 00:15:10 +00:00
|
|
|
struct event *dcEvent = item->getEvent();
|
|
|
|
if (dcEvent->type == SAMPLE_EVENT_BOOKMARK) {
|
2014-04-03 19:16:15 +00:00
|
|
|
action = new QAction(&m);
|
|
|
|
action->setText(tr("Edit name"));
|
|
|
|
action->setData(QVariant::fromValue<void *>(item));
|
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(editName()));
|
|
|
|
m.addAction(action);
|
|
|
|
}
|
2014-11-20 00:15:10 +00:00
|
|
|
#if 0 // FIXME::: FINISH OR DISABLE
|
2017-08-26 19:11:44 +00:00
|
|
|
QPointF scenePos = mapToScene(event->pos());
|
|
|
|
struct plot_data *entry = getEntryFromPos(scenePos);
|
2014-11-20 00:15:10 +00:00
|
|
|
// this shows how to figure out if we should ask the user if they want adjust interpolated pressures
|
|
|
|
// at either side of a gas change
|
|
|
|
if (dcEvent->type == SAMPLE_EVENT_GASCHANGE || dcEvent->type == SAMPLE_EVENT_GASCHANGE2) {
|
|
|
|
qDebug() << "figure out if there are interpolated pressures";
|
|
|
|
struct plot_data *gasChangeEntry = entry;
|
|
|
|
struct plot_data *newGasEntry;
|
|
|
|
while (gasChangeEntry > plotInfo.entry) {
|
|
|
|
--gasChangeEntry;
|
|
|
|
if (gasChangeEntry->sec <= dcEvent->time.seconds)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qDebug() << "at gas change at" << gasChangeEntry->sec << ": sensor pressure" << gasChangeEntry->pressure[0] << "interpolated" << gasChangeEntry->pressure[1];
|
|
|
|
// now gasChangeEntry points at the gas change, that entry has the final pressure of
|
|
|
|
// the old tank, the next entry has the starting pressure of the next tank
|
|
|
|
if (gasChangeEntry + 1 <= plotInfo.entry + plotInfo.nr) {
|
|
|
|
newGasEntry = gasChangeEntry + 1;
|
|
|
|
qDebug() << "after gas change at " << newGasEntry->sec << ": sensor pressure" << newGasEntry->pressure[0] << "interpolated" << newGasEntry->pressure[1];
|
Start cleaning up sensor indexing for multiple sensors
This is a very timid start at making us actually use multiple sensors
without the magical special case for just CCR oxygen tracking.
It mainly does:
- turn the "sample->sensor" index into an array of two indexes, to
match the pressures themselves.
- get rid of dive->{oxygen_cylinder_index,diluent_cylinder_index},
since a CCR dive should now simply set the sample->sensor[] indices
correctly instead.
- in a couple of places, start actually looping over the sensors rather
than special-case the O2 case (although often the small "loops" are
just unrolled, since it's just two cases.
but in many cases we still end up only covering the zero sensor case,
because the CCR O2 sensor code coverage was fairly limited.
It's entirely possible (even likely) that this migth break some existing
case: it tries to be a fairly direct ("stupid") translation of the old
code, but unlike the preparatory patch this does actually does change
some semantics.
For example, right now the git loader code assumes that if the git save
data contains a o2pressure entry, it just hardcodes the O2 sensor index
to 1.
In fact, one issue is going to simply be that our file formats do not
have that multiple sensor format, but instead had very clearly encoded
things as being the CCR O2 pressure sensor.
But this is hopefully close to usable, and I will need feedback (and
maybe test cases) from people who have existing CCR dives with pressure
data.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2017-07-21 02:49:45 +00:00
|
|
|
if (SENSOR_PRESSURE(gasChangeEntry) == 0 || displayed_dive.cylinder[gasChangeEntry->sensor[0]].sample_start.mbar == 0) {
|
2014-11-20 00:15:10 +00:00
|
|
|
// if we have no sensorpressure or if we have no pressure from samples we can assume that
|
|
|
|
// we only have interpolated pressure (the pressure in the entry may be stored in the sensor
|
|
|
|
// pressure field if this is the first or last entry for this tank... see details in gaspressures.c
|
|
|
|
pressure_t pressure;
|
|
|
|
pressure.mbar = INTERPOLATED_PRESSURE(gasChangeEntry) ? : SENSOR_PRESSURE(gasChangeEntry);
|
2017-11-18 08:35:06 +00:00
|
|
|
QAction *adjustOldPressure = m.addAction(tr("Adjust pressure of cyl. %1 (currently interpolated as %2)")
|
Start cleaning up sensor indexing for multiple sensors
This is a very timid start at making us actually use multiple sensors
without the magical special case for just CCR oxygen tracking.
It mainly does:
- turn the "sample->sensor" index into an array of two indexes, to
match the pressures themselves.
- get rid of dive->{oxygen_cylinder_index,diluent_cylinder_index},
since a CCR dive should now simply set the sample->sensor[] indices
correctly instead.
- in a couple of places, start actually looping over the sensors rather
than special-case the O2 case (although often the small "loops" are
just unrolled, since it's just two cases.
but in many cases we still end up only covering the zero sensor case,
because the CCR O2 sensor code coverage was fairly limited.
It's entirely possible (even likely) that this migth break some existing
case: it tries to be a fairly direct ("stupid") translation of the old
code, but unlike the preparatory patch this does actually does change
some semantics.
For example, right now the git loader code assumes that if the git save
data contains a o2pressure entry, it just hardcodes the O2 sensor index
to 1.
In fact, one issue is going to simply be that our file formats do not
have that multiple sensor format, but instead had very clearly encoded
things as being the CCR O2 pressure sensor.
But this is hopefully close to usable, and I will need feedback (and
maybe test cases) from people who have existing CCR dives with pressure
data.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2017-07-21 02:49:45 +00:00
|
|
|
.arg(gasChangeEntry->sensor[0] + 1).arg(get_pressure_string(pressure)));
|
2014-11-20 00:15:10 +00:00
|
|
|
}
|
Start cleaning up sensor indexing for multiple sensors
This is a very timid start at making us actually use multiple sensors
without the magical special case for just CCR oxygen tracking.
It mainly does:
- turn the "sample->sensor" index into an array of two indexes, to
match the pressures themselves.
- get rid of dive->{oxygen_cylinder_index,diluent_cylinder_index},
since a CCR dive should now simply set the sample->sensor[] indices
correctly instead.
- in a couple of places, start actually looping over the sensors rather
than special-case the O2 case (although often the small "loops" are
just unrolled, since it's just two cases.
but in many cases we still end up only covering the zero sensor case,
because the CCR O2 sensor code coverage was fairly limited.
It's entirely possible (even likely) that this migth break some existing
case: it tries to be a fairly direct ("stupid") translation of the old
code, but unlike the preparatory patch this does actually does change
some semantics.
For example, right now the git loader code assumes that if the git save
data contains a o2pressure entry, it just hardcodes the O2 sensor index
to 1.
In fact, one issue is going to simply be that our file formats do not
have that multiple sensor format, but instead had very clearly encoded
things as being the CCR O2 pressure sensor.
But this is hopefully close to usable, and I will need feedback (and
maybe test cases) from people who have existing CCR dives with pressure
data.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2017-07-21 02:49:45 +00:00
|
|
|
if (SENSOR_PRESSURE(newGasEntry) == 0 || displayed_dive.cylinder[newGasEntry->sensor[0]].sample_start.mbar == 0) {
|
2014-11-20 00:15:10 +00:00
|
|
|
// we only have interpolated press -- see commend above
|
|
|
|
pressure_t pressure;
|
|
|
|
pressure.mbar = INTERPOLATED_PRESSURE(newGasEntry) ? : SENSOR_PRESSURE(newGasEntry);
|
2017-11-18 08:35:06 +00:00
|
|
|
QAction *adjustOldPressure = m.addAction(tr("Adjust pressure of cyl. %1 (currently interpolated as %2)")
|
Start cleaning up sensor indexing for multiple sensors
This is a very timid start at making us actually use multiple sensors
without the magical special case for just CCR oxygen tracking.
It mainly does:
- turn the "sample->sensor" index into an array of two indexes, to
match the pressures themselves.
- get rid of dive->{oxygen_cylinder_index,diluent_cylinder_index},
since a CCR dive should now simply set the sample->sensor[] indices
correctly instead.
- in a couple of places, start actually looping over the sensors rather
than special-case the O2 case (although often the small "loops" are
just unrolled, since it's just two cases.
but in many cases we still end up only covering the zero sensor case,
because the CCR O2 sensor code coverage was fairly limited.
It's entirely possible (even likely) that this migth break some existing
case: it tries to be a fairly direct ("stupid") translation of the old
code, but unlike the preparatory patch this does actually does change
some semantics.
For example, right now the git loader code assumes that if the git save
data contains a o2pressure entry, it just hardcodes the O2 sensor index
to 1.
In fact, one issue is going to simply be that our file formats do not
have that multiple sensor format, but instead had very clearly encoded
things as being the CCR O2 pressure sensor.
But this is hopefully close to usable, and I will need feedback (and
maybe test cases) from people who have existing CCR dives with pressure
data.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2017-07-21 02:49:45 +00:00
|
|
|
.arg(newGasEntry->sensor[0] + 1).arg(get_pressure_string(pressure)));
|
2014-11-20 00:15:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2014-02-17 22:15:40 +00:00
|
|
|
}
|
|
|
|
bool some_hidden = false;
|
|
|
|
for (int i = 0; i < evn_used; i++) {
|
|
|
|
if (ev_namelist[i].plot_ev == false) {
|
|
|
|
some_hidden = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (some_hidden) {
|
|
|
|
action = m.addAction(tr("Unhide all events"), this, SLOT(unhideEvents()));
|
|
|
|
action->setData(event->globalPos());
|
|
|
|
}
|
|
|
|
m.exec(event->globalPos());
|
|
|
|
}
|
|
|
|
|
2014-06-11 20:56:33 +00:00
|
|
|
void ProfileWidget2::deleteCurrentDC()
|
|
|
|
{
|
|
|
|
delete_current_divecomputer();
|
|
|
|
mark_divelist_changed(true);
|
|
|
|
// we need to force it since it's likely the same dive and same dc_number - but that's a different dive computer now
|
2018-05-16 13:25:57 +00:00
|
|
|
plotDive(0, true, false);
|
2015-11-05 20:51:05 +00:00
|
|
|
|
|
|
|
emit refreshDisplay(true);
|
2014-06-11 20:56:33 +00:00
|
|
|
}
|
|
|
|
|
2014-05-19 05:39:34 +00:00
|
|
|
void ProfileWidget2::makeFirstDC()
|
|
|
|
{
|
|
|
|
make_first_dc();
|
|
|
|
mark_divelist_changed(true);
|
|
|
|
// this is now the first DC, so we need to redraw the profile and refresh the dive list
|
|
|
|
// (and no, it's not just enough to rewrite the text - the first DC is special so values in the
|
|
|
|
// dive list may change).
|
|
|
|
// As a side benefit, this returns focus to the dive list.
|
|
|
|
dc_number = 0;
|
2015-11-05 20:51:05 +00:00
|
|
|
emit refreshDisplay(true);
|
2014-05-19 05:39:34 +00:00
|
|
|
}
|
|
|
|
|
2014-03-16 04:12:34 +00:00
|
|
|
void ProfileWidget2::hideEvents()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
DiveEventItem *item = static_cast<DiveEventItem *>(action->data().value<void *>());
|
|
|
|
struct event *event = item->getEvent();
|
|
|
|
|
2015-11-05 20:58:24 +00:00
|
|
|
if (QMessageBox::question(this,
|
2014-03-16 04:12:34 +00:00
|
|
|
TITLE_OR_TEXT(tr("Hide events"), tr("Hide all %1 events?").arg(event->name)),
|
|
|
|
QMessageBox::Ok | QMessageBox::Cancel) == QMessageBox::Ok) {
|
2018-01-07 10:12:48 +00:00
|
|
|
if (!empty_string(event->name)) {
|
2014-03-16 04:12:34 +00:00
|
|
|
for (int i = 0; i < evn_used; i++) {
|
2014-05-15 00:45:39 +00:00
|
|
|
if (same_string(event->name, ev_namelist[i].ev_name)) {
|
2014-03-16 04:12:34 +00:00
|
|
|
ev_namelist[i].plot_ev = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-05-15 00:45:39 +00:00
|
|
|
Q_FOREACH (DiveEventItem *evItem, eventItems) {
|
2014-05-22 18:40:22 +00:00
|
|
|
if (same_string(evItem->getEvent()->name, event->name))
|
2014-05-15 00:45:39 +00:00
|
|
|
evItem->hide();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
item->hide();
|
2014-03-16 04:12:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::unhideEvents()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < evn_used; i++) {
|
|
|
|
ev_namelist[i].plot_ev = true;
|
|
|
|
}
|
2014-05-15 00:51:45 +00:00
|
|
|
Q_FOREACH (DiveEventItem *item, eventItems)
|
|
|
|
item->show();
|
2014-03-16 04:12:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::removeEvent()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
DiveEventItem *item = static_cast<DiveEventItem *>(action->data().value<void *>());
|
|
|
|
struct event *event = item->getEvent();
|
|
|
|
|
2015-11-05 20:58:24 +00:00
|
|
|
if (QMessageBox::question(this, TITLE_OR_TEXT(
|
|
|
|
tr("Remove the selected event?"),
|
|
|
|
tr("%1 @ %2:%3").arg(event->name).arg(event->time.seconds / 60).arg(event->time.seconds % 60, 2, 10, QChar('0'))),
|
2014-03-16 04:12:34 +00:00
|
|
|
QMessageBox::Ok | QMessageBox::Cancel) == QMessageBox::Ok) {
|
2014-05-21 15:06:02 +00:00
|
|
|
remove_event(event);
|
2016-04-03 22:31:59 +00:00
|
|
|
invalidate_dive_cache(current_dive);
|
2014-03-16 04:12:34 +00:00
|
|
|
mark_divelist_changed(true);
|
|
|
|
replot();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::addBookmark()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
QPointF scenePos = mapToScene(mapFromGlobal(action->data().toPoint()));
|
2017-03-23 01:13:49 +00:00
|
|
|
add_event(current_dc, lrint(timeAxis->valueAt(scenePos)), SAMPLE_EVENT_BOOKMARK, 0, 0, "bookmark");
|
2016-04-03 22:31:59 +00:00
|
|
|
invalidate_dive_cache(current_dive);
|
2014-07-12 05:37:05 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-03-16 04:12:34 +00:00
|
|
|
replot();
|
|
|
|
}
|
|
|
|
|
2018-04-08 20:09:26 +00:00
|
|
|
void ProfileWidget2::addDivemodeSwith()
|
2018-04-07 18:39:41 +00:00
|
|
|
{
|
2018-04-08 20:09:26 +00:00
|
|
|
int i;
|
2018-04-07 18:39:41 +00:00
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
QPointF scenePos = mapToScene(mapFromGlobal(action->data().toPoint()));
|
2018-04-08 20:09:26 +00:00
|
|
|
for (i = 0; i < UNDEF_COMP_TYPE; i++)
|
2018-05-17 08:04:41 +00:00
|
|
|
if (QString(divemode_text_ui[i]) == action->text())
|
2018-04-08 20:09:26 +00:00
|
|
|
add_event(current_dc, lrint(timeAxis->valueAt(scenePos)), 8, 0, i, "modechange");
|
2018-04-07 18:39:41 +00:00
|
|
|
invalidate_dive_cache(current_dive);
|
|
|
|
mark_divelist_changed(true);
|
|
|
|
replot();
|
|
|
|
}
|
|
|
|
|
2014-11-26 13:22:41 +00:00
|
|
|
void ProfileWidget2::addSetpointChange()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
QPointF scenePos = mapToScene(mapFromGlobal(action->data().toPoint()));
|
2017-03-23 01:13:49 +00:00
|
|
|
SetpointDialog::instance()->setpointData(current_dc, lrint(timeAxis->valueAt(scenePos)));
|
2014-11-26 13:22:41 +00:00
|
|
|
SetpointDialog::instance()->show();
|
|
|
|
}
|
|
|
|
|
2014-02-17 22:15:40 +00:00
|
|
|
void ProfileWidget2::changeGas()
|
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
2014-02-17 22:15:40 +00:00
|
|
|
QPointF scenePos = mapToScene(mapFromGlobal(action->data().toPoint()));
|
|
|
|
QString gas = action->text();
|
2014-10-29 23:25:00 +00:00
|
|
|
gas.remove(QRegExp(" \\(.*\\)"));
|
|
|
|
|
2014-02-17 22:15:40 +00:00
|
|
|
// backup the things on the dataModel, since we will clear that out.
|
2014-06-01 21:17:06 +00:00
|
|
|
struct gasmix gasmix;
|
2015-03-13 00:32:51 +00:00
|
|
|
qreal sec_val = timeAxis->valueAt(scenePos);
|
2014-02-17 22:15:40 +00:00
|
|
|
|
2015-03-11 18:23:50 +00:00
|
|
|
// no gas changes before the dive starts
|
2017-11-18 20:26:04 +00:00
|
|
|
int seconds = (sec_val < 0.0) ? 0 : (int)sec_val;
|
2015-03-11 18:23:50 +00:00
|
|
|
|
2015-03-11 18:24:30 +00:00
|
|
|
// if there is a gas change at this time stamp, remove it before adding the new one
|
|
|
|
struct event *gasChangeEvent = current_dc->events;
|
|
|
|
while ((gasChangeEvent = get_next_event(gasChangeEvent, "gaschange")) != NULL) {
|
|
|
|
if (gasChangeEvent->time.seconds == seconds) {
|
|
|
|
remove_event(gasChangeEvent);
|
|
|
|
gasChangeEvent = current_dc->events;
|
|
|
|
} else {
|
|
|
|
gasChangeEvent = gasChangeEvent->next;
|
2015-03-11 00:00:18 +00:00
|
|
|
}
|
|
|
|
}
|
2018-02-25 12:51:41 +00:00
|
|
|
validate_gas(qPrintable(gas), &gasmix);
|
2014-10-29 23:25:00 +00:00
|
|
|
QRegExp rx("\\(\\D*(\\d+)");
|
|
|
|
int tank;
|
|
|
|
if (rx.indexIn(action->text()) > -1) {
|
|
|
|
tank = rx.cap(1).toInt() - 1; // we display the tank 1 based
|
|
|
|
} else {
|
|
|
|
qDebug() << "failed to parse tank number";
|
|
|
|
tank = get_gasidx(&displayed_dive, &gasmix);
|
|
|
|
}
|
2014-10-30 21:08:10 +00:00
|
|
|
// add this both to the displayed dive and the current dive
|
|
|
|
add_gas_switch_event(current_dive, current_dc, seconds, tank);
|
|
|
|
add_gas_switch_event(&displayed_dive, get_dive_dc(&displayed_dive, dc_number), seconds, tank);
|
2014-02-17 22:15:40 +00:00
|
|
|
// this means we potentially have a new tank that is being used and needs to be shown
|
2014-07-03 21:34:24 +00:00
|
|
|
fixup_dive(&displayed_dive);
|
2016-04-03 22:31:59 +00:00
|
|
|
invalidate_dive_cache(current_dive);
|
2014-07-03 21:34:24 +00:00
|
|
|
|
|
|
|
// FIXME - this no longer gets written to the dive list - so we need to enableEdition() here
|
|
|
|
|
2015-11-05 21:11:26 +00:00
|
|
|
emit updateDiveInfo(false);
|
2014-02-17 22:15:40 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-03-09 13:59:31 +00:00
|
|
|
replot();
|
2014-02-20 01:18:26 +00:00
|
|
|
}
|
2015-11-06 00:05:44 +00:00
|
|
|
#endif
|
2014-03-25 21:34:09 +00:00
|
|
|
|
2014-07-09 20:09:52 +00:00
|
|
|
bool ProfileWidget2::getPrintMode()
|
|
|
|
{
|
|
|
|
return printMode;
|
|
|
|
}
|
|
|
|
|
2014-03-25 21:34:09 +00:00
|
|
|
void ProfileWidget2::setPrintMode(bool mode, bool grayscale)
|
|
|
|
{
|
|
|
|
printMode = mode;
|
2015-10-11 11:21:13 +00:00
|
|
|
resetZoom();
|
2015-10-11 11:21:14 +00:00
|
|
|
|
|
|
|
// set printMode for axes
|
|
|
|
profileYAxis->setPrintMode(mode);
|
|
|
|
gasYAxis->setPrintMode(mode);
|
|
|
|
temperatureAxis->setPrintMode(mode);
|
|
|
|
timeAxis->setPrintMode(mode);
|
|
|
|
cylinderPressureAxis->setPrintMode(mode);
|
2016-02-06 21:25:58 +00:00
|
|
|
isGrayscale = mode ? grayscale : false;
|
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2015-10-11 11:21:14 +00:00
|
|
|
heartBeatAxis->setPrintMode(mode);
|
|
|
|
percentageAxis->setPrintMode(mode);
|
|
|
|
|
2015-01-02 00:28:37 +00:00
|
|
|
mouseFollowerHorizontal->setVisible(!mode);
|
|
|
|
mouseFollowerVertical->setVisible(!mode);
|
2015-11-15 21:02:02 +00:00
|
|
|
toolTipItem->setVisible(!mode);
|
2016-02-06 21:25:58 +00:00
|
|
|
#endif
|
2014-03-25 21:34:09 +00:00
|
|
|
}
|
2014-04-03 19:16:15 +00:00
|
|
|
|
2014-07-09 20:09:52 +00:00
|
|
|
void ProfileWidget2::setFontPrintScale(double scale)
|
|
|
|
{
|
|
|
|
fontPrintScale = scale;
|
2015-07-29 19:21:27 +00:00
|
|
|
emit fontPrintScaleChanged(scale);
|
2014-07-09 20:09:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double ProfileWidget2::getFontPrintScale()
|
|
|
|
{
|
|
|
|
if (printMode)
|
|
|
|
return fontPrintScale;
|
|
|
|
else
|
|
|
|
return 1.0;
|
|
|
|
}
|
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-04-03 19:16:15 +00:00
|
|
|
void ProfileWidget2::editName()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
DiveEventItem *item = static_cast<DiveEventItem *>(action->data().value<void *>());
|
|
|
|
struct event *event = item->getEvent();
|
|
|
|
bool ok;
|
2015-11-05 20:58:24 +00:00
|
|
|
QString newName = QInputDialog::getText(this, tr("Edit name of bookmark"),
|
2014-04-03 19:16:15 +00:00
|
|
|
tr("Custom name:"), QLineEdit::Normal,
|
|
|
|
event->name, &ok);
|
2014-04-03 19:52:05 +00:00
|
|
|
if (ok && !newName.isEmpty()) {
|
|
|
|
if (newName.length() > 22) { //longer names will display as garbage.
|
2014-04-03 19:16:15 +00:00
|
|
|
QMessageBox lengthWarning;
|
2014-07-28 14:12:51 +00:00
|
|
|
lengthWarning.setText(tr("Name is too long!"));
|
2014-04-03 19:16:15 +00:00
|
|
|
lengthWarning.exec();
|
|
|
|
return;
|
|
|
|
}
|
2014-07-12 19:48:27 +00:00
|
|
|
// order is important! first update the current dive (by matching the unchanged event),
|
|
|
|
// then update the displayed dive (as event is part of the events on displayed dive
|
|
|
|
// and will be freed as part of changing the name!
|
2018-02-25 12:51:41 +00:00
|
|
|
update_event_name(current_dive, event, qPrintable(newName));
|
|
|
|
update_event_name(&displayed_dive, event, qPrintable(newName));
|
2016-04-03 22:31:59 +00:00
|
|
|
invalidate_dive_cache(current_dive);
|
2014-07-12 19:48:27 +00:00
|
|
|
mark_divelist_changed(true);
|
|
|
|
replot();
|
2014-04-03 19:16:15 +00:00
|
|
|
}
|
|
|
|
}
|
2015-11-06 00:05:44 +00:00
|
|
|
#endif
|
2014-05-22 00:18:10 +00:00
|
|
|
|
2014-05-22 00:23:19 +00:00
|
|
|
void ProfileWidget2::disconnectTemporaryConnections()
|
|
|
|
{
|
2018-01-09 18:32:41 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2014-05-22 00:23:19 +00:00
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
disconnect(plannerModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), this, SLOT(replot()));
|
|
|
|
disconnect(plannerModel, SIGNAL(cylinderModelEdited()), this, SLOT(replot()));
|
|
|
|
|
|
|
|
disconnect(plannerModel, SIGNAL(rowsInserted(const QModelIndex &, int, int)),
|
2014-05-22 18:40:22 +00:00
|
|
|
this, SLOT(pointInserted(const QModelIndex &, int, int)));
|
2014-05-22 00:23:19 +00:00
|
|
|
disconnect(plannerModel, SIGNAL(rowsRemoved(const QModelIndex &, int, int)),
|
2014-05-22 18:40:22 +00:00
|
|
|
this, SLOT(pointsRemoved(const QModelIndex &, int, int)));
|
2015-11-12 00:37:18 +00:00
|
|
|
#endif
|
2014-06-03 22:28:06 +00:00
|
|
|
Q_FOREACH (QAction *action, actionsForKeys.values()) {
|
2014-05-24 15:39:40 +00:00
|
|
|
action->setShortcut(QKeySequence());
|
2014-06-03 22:28:06 +00:00
|
|
|
action->setShortcutContext(Qt::WidgetShortcut);
|
|
|
|
}
|
2014-05-22 00:23:19 +00:00
|
|
|
}
|
|
|
|
|
2015-11-06 00:05:44 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
2018-05-21 16:20:29 +00:00
|
|
|
void ProfileWidget2::pointInserted(const QModelIndex&, int, int)
|
2014-05-22 00:18:10 +00:00
|
|
|
{
|
|
|
|
DiveHandler *item = new DiveHandler();
|
|
|
|
scene()->addItem(item);
|
|
|
|
handles << item;
|
|
|
|
|
2014-05-23 23:51:30 +00:00
|
|
|
connect(item, SIGNAL(moved()), this, SLOT(recreatePlannedDive()));
|
2014-06-30 22:08:16 +00:00
|
|
|
connect(item, SIGNAL(clicked()), this, SLOT(divePlannerHandlerClicked()));
|
|
|
|
connect(item, SIGNAL(released()), this, SLOT(divePlannerHandlerReleased()));
|
2014-05-22 00:18:10 +00:00
|
|
|
QGraphicsSimpleTextItem *gasChooseBtn = new QGraphicsSimpleTextItem();
|
|
|
|
scene()->addItem(gasChooseBtn);
|
|
|
|
gasChooseBtn->setZValue(10);
|
|
|
|
gasChooseBtn->setFlag(QGraphicsItem::ItemIgnoresTransformations);
|
|
|
|
gases << gasChooseBtn;
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
if (plannerModel->recalcQ())
|
|
|
|
replot();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::pointsRemoved(const QModelIndex &, int start, int end)
|
|
|
|
{ // start and end are inclusive.
|
|
|
|
int num = (end - start) + 1;
|
|
|
|
for (int i = num; i != 0; i--) {
|
|
|
|
delete handles.back();
|
|
|
|
handles.pop_back();
|
|
|
|
delete gases.back();
|
|
|
|
gases.pop_back();
|
|
|
|
}
|
|
|
|
scene()->clearSelection();
|
|
|
|
replot();
|
|
|
|
}
|
2014-05-22 02:31:26 +00:00
|
|
|
|
|
|
|
void ProfileWidget2::repositionDiveHandlers()
|
|
|
|
{
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2018-01-10 15:36:45 +00:00
|
|
|
hideAll(gases);
|
2014-05-22 02:31:26 +00:00
|
|
|
// Re-position the user generated dive handlers
|
|
|
|
for (int i = 0; i < plannerModel->rowCount(); i++) {
|
|
|
|
struct divedatapoint datapoint = plannerModel->at(i);
|
|
|
|
if (datapoint.time == 0) // those are the magic entries for tanks
|
|
|
|
continue;
|
|
|
|
DiveHandler *h = handles.at(i);
|
2014-05-30 02:47:03 +00:00
|
|
|
h->setVisible(datapoint.entered);
|
2017-03-10 12:37:54 +00:00
|
|
|
h->setPos(timeAxis->posAtValue(datapoint.time), profileYAxis->posAtValue(datapoint.depth.mm));
|
2014-10-21 21:27:50 +00:00
|
|
|
QPointF p1;
|
|
|
|
if (i == 0) {
|
|
|
|
if (prefs.drop_stone_mode)
|
|
|
|
// place the text on the straight line from the drop to stone position
|
2017-03-10 12:37:54 +00:00
|
|
|
p1 = QPointF(timeAxis->posAtValue(datapoint.depth.mm / prefs.descrate),
|
|
|
|
profileYAxis->posAtValue(datapoint.depth.mm));
|
2014-10-21 21:27:50 +00:00
|
|
|
else
|
|
|
|
// place the text on the straight line from the origin to the first position
|
|
|
|
p1 = QPointF(timeAxis->posAtValue(0), profileYAxis->posAtValue(0));
|
|
|
|
} else {
|
|
|
|
// place the text on the line from the last position
|
|
|
|
p1 = handles[i - 1]->pos();
|
|
|
|
}
|
2014-05-22 02:31:26 +00:00
|
|
|
QPointF p2 = handles[i]->pos();
|
|
|
|
QLineF line(p1, p2);
|
|
|
|
QPointF pos = line.pointAt(0.5);
|
|
|
|
gases[i]->setPos(pos);
|
2016-07-06 12:40:28 +00:00
|
|
|
gases[i]->setText(get_gas_string(displayed_dive.cylinder[datapoint.cylinderid].gasmix));
|
2015-05-07 21:40:34 +00:00
|
|
|
gases[i]->setVisible(datapoint.entered &&
|
|
|
|
(i == 0 || gases[i]->text() != gases[i-1]->text()));
|
2014-05-22 02:31:26 +00:00
|
|
|
}
|
|
|
|
}
|
2014-05-23 23:51:30 +00:00
|
|
|
|
2014-05-24 01:22:02 +00:00
|
|
|
int ProfileWidget2::fixHandlerIndex(DiveHandler *activeHandler)
|
|
|
|
{
|
|
|
|
int index = handles.indexOf(activeHandler);
|
|
|
|
if (index > 0 && index < handles.count() - 1) {
|
|
|
|
DiveHandler *before = handles[index - 1];
|
|
|
|
if (before->pos().x() > activeHandler->pos().x()) {
|
2014-05-24 15:39:40 +00:00
|
|
|
handles.swap(index, index - 1);
|
2014-05-24 01:22:02 +00:00
|
|
|
return index - 1;
|
|
|
|
}
|
|
|
|
DiveHandler *after = handles[index + 1];
|
|
|
|
if (after->pos().x() < activeHandler->pos().x()) {
|
2014-05-24 15:39:40 +00:00
|
|
|
handles.swap(index, index + 1);
|
2014-05-24 01:22:02 +00:00
|
|
|
return index + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
2014-05-23 23:51:30 +00:00
|
|
|
void ProfileWidget2::recreatePlannedDive()
|
|
|
|
{
|
2014-05-24 15:39:40 +00:00
|
|
|
DiveHandler *activeHandler = qobject_cast<DiveHandler *>(sender());
|
2014-05-23 23:51:30 +00:00
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2014-05-24 01:22:02 +00:00
|
|
|
int index = fixHandlerIndex(activeHandler);
|
2017-05-17 20:22:50 +00:00
|
|
|
int mintime = 0;
|
|
|
|
int maxtime = plannerModel->at(plannerModel->size() - 1).time * 3 / 2;
|
2014-05-23 23:51:30 +00:00
|
|
|
if (index > 0)
|
|
|
|
mintime = plannerModel->at(index - 1).time;
|
|
|
|
if (index < plannerModel->size() - 1)
|
|
|
|
maxtime = plannerModel->at(index + 1).time;
|
|
|
|
|
2017-03-08 06:41:41 +00:00
|
|
|
int minutes = lrint(timeAxis->valueAt(activeHandler->pos()) / 60);
|
2014-05-23 23:51:30 +00:00
|
|
|
if (minutes * 60 <= mintime || minutes * 60 >= maxtime)
|
|
|
|
return;
|
2017-12-01 11:46:51 +00:00
|
|
|
if (minutes * 60 > timeAxis->maximum() * 0.9)
|
|
|
|
timeAxis->setMaximum(timeAxis->maximum() * 1.02);
|
2014-05-23 23:51:30 +00:00
|
|
|
|
|
|
|
divedatapoint data = plannerModel->at(index);
|
2017-12-01 11:49:38 +00:00
|
|
|
depth_t oldDepth = data.depth;
|
|
|
|
int oldtime = data.time;
|
2017-03-11 16:41:41 +00:00
|
|
|
data.depth.mm = lrint(profileYAxis->valueAt(activeHandler->pos()) / M_OR_FT(1, 1)) * M_OR_FT(1, 1);
|
2017-03-08 06:41:41 +00:00
|
|
|
data.time = lrint(timeAxis->valueAt(activeHandler->pos()));
|
2014-05-23 23:51:30 +00:00
|
|
|
|
2017-12-01 11:49:38 +00:00
|
|
|
if (data.depth.mm != oldDepth.mm || data.time != oldtime)
|
|
|
|
plannerModel->editStop(index, data);
|
2014-05-23 23:51:30 +00:00
|
|
|
}
|
2014-05-24 15:39:40 +00:00
|
|
|
|
|
|
|
void ProfileWidget2::keyDownAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2017-12-01 11:49:38 +00:00
|
|
|
bool oldRecalc = plannerModel->setRecalc(false);
|
|
|
|
|
2014-05-24 15:39:40 +00:00
|
|
|
Q_FOREACH (QGraphicsItem *i, scene()->selectedItems()) {
|
|
|
|
if (DiveHandler *handler = qgraphicsitem_cast<DiveHandler *>(i)) {
|
|
|
|
int row = handles.indexOf(handler);
|
|
|
|
divedatapoint dp = plannerModel->at(row);
|
2017-03-10 12:37:54 +00:00
|
|
|
if (dp.depth.mm >= profileYAxis->maximum())
|
2014-05-24 15:39:40 +00:00
|
|
|
continue;
|
|
|
|
|
2017-03-10 12:37:54 +00:00
|
|
|
dp.depth.mm += M_OR_FT(1, 5);
|
2014-05-24 15:39:40 +00:00
|
|
|
plannerModel->editStop(row, dp);
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 11:49:38 +00:00
|
|
|
plannerModel->setRecalc(oldRecalc);
|
|
|
|
replot();
|
2014-05-24 15:39:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::keyUpAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2017-12-01 11:49:38 +00:00
|
|
|
bool oldRecalc = plannerModel->setRecalc(false);
|
2014-05-24 15:39:40 +00:00
|
|
|
Q_FOREACH (QGraphicsItem *i, scene()->selectedItems()) {
|
|
|
|
if (DiveHandler *handler = qgraphicsitem_cast<DiveHandler *>(i)) {
|
|
|
|
int row = handles.indexOf(handler);
|
|
|
|
divedatapoint dp = plannerModel->at(row);
|
|
|
|
|
2017-03-10 12:37:54 +00:00
|
|
|
if (dp.depth.mm <= 0)
|
2014-05-24 15:39:40 +00:00
|
|
|
continue;
|
|
|
|
|
2017-03-10 12:37:54 +00:00
|
|
|
dp.depth.mm -= M_OR_FT(1, 5);
|
2014-05-24 15:39:40 +00:00
|
|
|
plannerModel->editStop(row, dp);
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 11:49:38 +00:00
|
|
|
plannerModel->setRecalc(oldRecalc);
|
|
|
|
replot();
|
2014-05-24 15:39:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::keyLeftAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2017-12-01 11:49:38 +00:00
|
|
|
bool oldRecalc = plannerModel->setRecalc(false);
|
2014-05-24 15:39:40 +00:00
|
|
|
Q_FOREACH (QGraphicsItem *i, scene()->selectedItems()) {
|
|
|
|
if (DiveHandler *handler = qgraphicsitem_cast<DiveHandler *>(i)) {
|
|
|
|
int row = handles.indexOf(handler);
|
|
|
|
divedatapoint dp = plannerModel->at(row);
|
|
|
|
|
|
|
|
if (dp.time / 60 <= 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// don't overlap positions.
|
|
|
|
// maybe this is a good place for a 'goto'?
|
|
|
|
double xpos = timeAxis->posAtValue((dp.time - 60) / 60);
|
|
|
|
bool nextStep = false;
|
|
|
|
Q_FOREACH (DiveHandler *h, handles) {
|
|
|
|
if (IS_FP_SAME(h->pos().x(), xpos)) {
|
|
|
|
nextStep = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nextStep)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
dp.time -= 60;
|
|
|
|
plannerModel->editStop(row, dp);
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 11:49:38 +00:00
|
|
|
plannerModel->setRecalc(oldRecalc);
|
|
|
|
replot();
|
2014-05-24 15:39:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::keyRightAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
2017-12-01 11:49:38 +00:00
|
|
|
bool oldRecalc = plannerModel->setRecalc(false);
|
2014-05-24 15:39:40 +00:00
|
|
|
Q_FOREACH (QGraphicsItem *i, scene()->selectedItems()) {
|
|
|
|
if (DiveHandler *handler = qgraphicsitem_cast<DiveHandler *>(i)) {
|
|
|
|
int row = handles.indexOf(handler);
|
|
|
|
divedatapoint dp = plannerModel->at(row);
|
2015-06-22 05:09:33 +00:00
|
|
|
if (dp.time / 60.0 >= timeAxis->maximum())
|
2014-05-24 15:39:40 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// don't overlap positions.
|
|
|
|
// maybe this is a good place for a 'goto'?
|
|
|
|
double xpos = timeAxis->posAtValue((dp.time + 60) / 60);
|
|
|
|
bool nextStep = false;
|
|
|
|
Q_FOREACH (DiveHandler *h, handles) {
|
|
|
|
if (IS_FP_SAME(h->pos().x(), xpos)) {
|
|
|
|
nextStep = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nextStep)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
dp.time += 60;
|
|
|
|
plannerModel->editStop(row, dp);
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 11:49:38 +00:00
|
|
|
plannerModel->setRecalc(oldRecalc);
|
|
|
|
replot();
|
2014-05-24 15:39:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::keyDeleteAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
int selCount = scene()->selectedItems().count();
|
|
|
|
if (selCount) {
|
|
|
|
QVector<int> selectedIndexes;
|
|
|
|
Q_FOREACH (QGraphicsItem *i, scene()->selectedItems()) {
|
|
|
|
if (DiveHandler *handler = qgraphicsitem_cast<DiveHandler *>(i)) {
|
|
|
|
selectedIndexes.push_back(handles.indexOf(handler));
|
2014-07-30 16:31:22 +00:00
|
|
|
handler->hide();
|
2014-05-24 15:39:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
plannerModel->removeSelectedPoints(selectedIndexes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::keyEscAction()
|
|
|
|
{
|
|
|
|
if (currentState != ADD && currentState != PLAN)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (scene()->selectedItems().count()) {
|
|
|
|
scene()->clearSelection();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
DivePlannerPointsModel *plannerModel = DivePlannerPointsModel::instance();
|
|
|
|
if (plannerModel->isPlanner())
|
|
|
|
plannerModel->cancelPlan();
|
|
|
|
}
|
2014-06-03 22:24:01 +00:00
|
|
|
|
2018-04-06 15:58:16 +00:00
|
|
|
void ProfileWidget2::clearPictures()
|
2014-06-03 22:24:01 +00:00
|
|
|
{
|
2014-07-21 21:54:28 +00:00
|
|
|
pictures.clear();
|
2018-04-06 15:58:16 +00:00
|
|
|
}
|
2014-07-21 21:54:28 +00:00
|
|
|
|
2018-04-06 15:58:16 +00:00
|
|
|
void ProfileWidget2::updatePictures(const QModelIndex &from, const QModelIndex &to)
|
|
|
|
{
|
|
|
|
DivePictureModel *m = DivePictureModel::instance();
|
|
|
|
for (int picNr = from.row(); picNr <= to.row(); ++picNr) {
|
|
|
|
int picItemNr = picNr - m->rowDDStart;
|
|
|
|
if (picItemNr < 0 || (size_t)picItemNr >= pictures.size())
|
|
|
|
return;
|
|
|
|
if (!pictures[picItemNr])
|
|
|
|
return;
|
2014-07-02 20:58:06 +00:00
|
|
|
|
2018-04-06 15:58:16 +00:00
|
|
|
pictures[picItemNr]->setPixmap(m->index(picNr, 0).data(Qt::UserRole).value<QPixmap>());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::plotPictures()
|
|
|
|
{
|
2014-06-03 22:24:01 +00:00
|
|
|
DivePictureModel *m = DivePictureModel::instance();
|
2018-04-06 15:58:16 +00:00
|
|
|
pictures.resize(m->rowDDEnd - m->rowDDStart);
|
|
|
|
|
|
|
|
double x, y, lastX = -1.0, lastY = -1.0;
|
2017-12-13 18:32:29 +00:00
|
|
|
for (int i = m->rowDDStart; i < m->rowDDEnd; i++) {
|
2018-04-06 15:58:16 +00:00
|
|
|
int picItemNr = i - m->rowDDStart;
|
2015-01-02 00:28:37 +00:00
|
|
|
int offsetSeconds = m->index(i, 1).data(Qt::UserRole).value<int>();
|
2014-06-03 22:42:47 +00:00
|
|
|
// it's a correct picture, but doesn't have a timestamp: only show on the widget near the
|
2018-04-06 15:58:16 +00:00
|
|
|
// information area. A null pointer in the pictures array indicates that this picture is not
|
|
|
|
// shown.
|
|
|
|
if (!offsetSeconds) {
|
|
|
|
pictures[picItemNr].reset();
|
2014-06-03 22:42:47 +00:00
|
|
|
continue;
|
2018-04-06 15:58:16 +00:00
|
|
|
}
|
|
|
|
DivePictureItem *item = pictures[picItemNr].get();
|
|
|
|
if (!item) {
|
|
|
|
item = new DivePictureItem;
|
|
|
|
pictures[picItemNr].reset(item);
|
|
|
|
scene()->addItem(item);
|
|
|
|
}
|
2017-12-17 15:17:38 +00:00
|
|
|
item->setPixmap(m->index(i, 0).data(Qt::UserRole).value<QPixmap>());
|
2015-01-02 00:28:37 +00:00
|
|
|
item->setFileUrl(m->index(i, 1).data().toString());
|
2014-06-08 18:56:35 +00:00
|
|
|
// let's put the picture at the correct time, but at a fixed "depth" on the profile
|
|
|
|
// not sure this is ideal, but it seems to look right.
|
2014-08-05 19:37:14 +00:00
|
|
|
x = timeAxis->posAtValue(offsetSeconds);
|
2014-06-08 19:36:27 +00:00
|
|
|
if (i == 0)
|
|
|
|
y = 10;
|
2017-12-11 20:28:33 +00:00
|
|
|
else if (fabs(x - lastX) < 3 && lastY <= (10 + 14 * 3))
|
2014-06-27 21:00:42 +00:00
|
|
|
y = lastY + 3;
|
2015-06-22 13:42:02 +00:00
|
|
|
else
|
|
|
|
y = 10;
|
2014-06-08 19:36:27 +00:00
|
|
|
lastX = x;
|
|
|
|
lastY = y;
|
2014-06-08 18:56:35 +00:00
|
|
|
item->setPos(x, y);
|
2014-06-03 22:24:01 +00:00
|
|
|
}
|
|
|
|
}
|
2018-05-19 19:18:39 +00:00
|
|
|
|
|
|
|
void ProfileWidget2::removePictures(const QModelIndex &, int first, int last)
|
|
|
|
{
|
|
|
|
DivePictureModel *m = DivePictureModel::instance();
|
|
|
|
first = std::max(0, first - m->rowDDStart);
|
|
|
|
// Note that last points *to* the last item and not *past* the last item,
|
|
|
|
// therefore we add 1 to achieve conventional C++ semantics.
|
|
|
|
last = std::min((int)pictures.size(), last + 1 - m->rowDDStart);
|
|
|
|
if (first >= (int)pictures.size() || last <= first)
|
|
|
|
return;
|
|
|
|
pictures.erase(pictures.begin() + first, pictures.begin() + last);
|
|
|
|
}
|
|
|
|
|
2018-01-10 12:55:29 +00:00
|
|
|
#endif
|
2015-11-03 20:17:50 +00:00
|
|
|
|
|
|
|
void ProfileWidget2::dropEvent(QDropEvent *event)
|
|
|
|
{
|
|
|
|
if (event->mimeData()->hasFormat("application/x-subsurfaceimagedrop")) {
|
|
|
|
QByteArray itemData = event->mimeData()->data("application/x-subsurfaceimagedrop");
|
|
|
|
QDataStream dataStream(&itemData, QIODevice::ReadOnly);
|
|
|
|
|
|
|
|
QString filename;
|
|
|
|
QPoint offset;
|
|
|
|
dataStream >> filename >> offset;
|
|
|
|
|
|
|
|
QPointF mappedPos = mapToScene(event->pos());
|
|
|
|
|
|
|
|
FOR_EACH_PICTURE(current_dive) {
|
|
|
|
if (QString(picture->filename) == filename) {
|
2017-03-23 01:13:49 +00:00
|
|
|
picture->offset.seconds = lrint(timeAxis->valueAt(mappedPos));
|
2015-11-03 20:17:50 +00:00
|
|
|
mark_divelist_changed(true);
|
2018-05-01 10:35:18 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
DivePictureModel::instance()->updateDivePictureOffset(filename, picture->offset.seconds);
|
|
|
|
plotPictures();
|
|
|
|
#endif
|
2015-11-03 20:17:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
copy_dive(current_dive, &displayed_dive);
|
|
|
|
|
|
|
|
if (event->source() == this) {
|
|
|
|
event->setDropAction(Qt::MoveAction);
|
|
|
|
event->accept();
|
|
|
|
} else {
|
|
|
|
event->acceptProposedAction();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
event->ignore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::dragEnterEvent(QDragEnterEvent *event)
|
|
|
|
{
|
|
|
|
if (event->mimeData()->hasFormat("application/x-subsurfaceimagedrop")) {
|
|
|
|
if (event->source() == this) {
|
|
|
|
event->setDropAction(Qt::MoveAction);
|
|
|
|
event->accept();
|
|
|
|
} else {
|
|
|
|
event->acceptProposedAction();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
event->ignore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileWidget2::dragMoveEvent(QDragMoveEvent *event)
|
|
|
|
{
|
|
|
|
if (event->mimeData()->hasFormat("application/x-subsurfaceimagedrop")) {
|
|
|
|
if (event->source() == this) {
|
|
|
|
event->setDropAction(Qt::MoveAction);
|
|
|
|
event->accept();
|
|
|
|
} else {
|
|
|
|
event->acceptProposedAction();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
event->ignore();
|
|
|
|
}
|
|
|
|
}
|