2017-04-27 18:26:05 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2013-04-14 03:44:02 +00:00
|
|
|
/*
|
|
|
|
* maintab.cpp
|
|
|
|
*
|
|
|
|
* classes for the "notebook" area of the main window of Subsurface
|
|
|
|
*
|
|
|
|
*/
|
2017-04-04 17:21:30 +00:00
|
|
|
#include "desktop-widgets/tab-widgets/maintab.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/mainwindow.h"
|
2017-07-15 20:37:02 +00:00
|
|
|
#include "desktop-widgets/mapwidget.h"
|
2018-06-03 20:15:19 +00:00
|
|
|
#include "core/qthelper.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "core/statistics.h"
|
|
|
|
#include "desktop-widgets/modeldelegates.h"
|
|
|
|
#include "qt-models/diveplannermodel.h"
|
|
|
|
#include "desktop-widgets/divelistview.h"
|
|
|
|
#include "core/display.h"
|
2015-09-03 18:56:37 +00:00
|
|
|
#include "profile-widget/profilewidget2.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/diveplanner.h"
|
|
|
|
#include "core/divesitehelpers.h"
|
|
|
|
#include "qt-models/cylindermodel.h"
|
|
|
|
#include "qt-models/weightmodel.h"
|
|
|
|
#include "qt-models/divecomputerextradatamodel.h"
|
|
|
|
#include "qt-models/divelocationmodel.h"
|
2017-11-26 21:21:58 +00:00
|
|
|
#include "qt-models/filtermodels.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "core/divesite.h"
|
2018-05-11 15:25:41 +00:00
|
|
|
#include "core/subsurface-string.h"
|
2018-06-17 06:48:54 +00:00
|
|
|
#include "core/gettextfromc.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/locationinformation.h"
|
2015-06-01 19:58:23 +00:00
|
|
|
|
2017-04-04 17:21:30 +00:00
|
|
|
#include "TabDiveExtraInfo.h"
|
|
|
|
#include "TabDiveInformation.h"
|
|
|
|
#include "TabDivePhotos.h"
|
|
|
|
#include "TabDiveStatistics.h"
|
|
|
|
|
2013-08-13 13:49:59 +00:00
|
|
|
#include <QCompleter>
|
2013-11-28 11:17:30 +00:00
|
|
|
#include <QSettings>
|
2013-12-12 01:08:56 +00:00
|
|
|
#include <QScrollBar>
|
2014-06-03 22:29:28 +00:00
|
|
|
#include <QShortcut>
|
2014-06-04 01:01:00 +00:00
|
|
|
#include <QMessageBox>
|
2014-06-27 12:17:33 +00:00
|
|
|
#include <QDesktopServices>
|
2015-05-11 17:38:50 +00:00
|
|
|
#include <QStringList>
|
2013-04-07 22:20:43 +00:00
|
|
|
|
|
|
|
MainTab::MainTab(QWidget *parent) : QTabWidget(parent),
|
2014-02-28 04:09:57 +00:00
|
|
|
weightModel(new WeightModel(this)),
|
2017-02-21 13:37:18 +00:00
|
|
|
cylindersModel(new CylindersModel(this)),
|
2014-06-02 22:50:42 +00:00
|
|
|
editMode(NONE),
|
2014-11-08 13:11:06 +00:00
|
|
|
copyPaste(false),
|
2018-03-15 21:41:59 +00:00
|
|
|
lastSelectedDive(true),
|
2017-11-30 19:33:43 +00:00
|
|
|
lastTabSelectedDive(0),
|
2017-12-19 22:05:42 +00:00
|
|
|
lastTabSelectedDiveTrip(0),
|
|
|
|
currentTrip(0)
|
2013-04-07 22:20:43 +00:00
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.setupUi(this);
|
2017-04-04 17:21:30 +00:00
|
|
|
|
2017-04-04 18:01:47 +00:00
|
|
|
extraWidgets << new TabDiveInformation();
|
2017-04-13 18:23:07 +00:00
|
|
|
ui.tabWidget->addTab(extraWidgets.last(), tr("Information"));
|
2017-04-04 18:01:47 +00:00
|
|
|
extraWidgets << new TabDiveStatistics();
|
2017-04-13 18:23:07 +00:00
|
|
|
ui.tabWidget->addTab(extraWidgets.last(), tr("Statistics"));
|
2017-04-04 18:01:47 +00:00
|
|
|
extraWidgets << new TabDivePhotos();
|
2017-04-13 18:23:07 +00:00
|
|
|
ui.tabWidget->addTab(extraWidgets.last(), tr("Photos"));
|
2017-04-21 16:33:38 +00:00
|
|
|
extraWidgets << new TabDiveExtraInfo();
|
|
|
|
ui.tabWidget->addTab(extraWidgets.last(), tr("Extra Info"));
|
2017-04-04 17:21:30 +00:00
|
|
|
|
2015-10-29 23:57:43 +00:00
|
|
|
ui.dateEdit->setDisplayFormat(prefs.date_format);
|
2018-05-04 03:58:47 +00:00
|
|
|
ui.timeEdit->setDisplayFormat(prefs.time_format);
|
2014-02-09 18:49:15 +00:00
|
|
|
|
2014-07-02 18:50:28 +00:00
|
|
|
memset(&displayed_dive, 0, sizeof(displayed_dive));
|
2014-07-30 15:52:11 +00:00
|
|
|
memset(&displayedTrip, 0, sizeof(displayedTrip));
|
2014-02-09 18:49:15 +00:00
|
|
|
|
2017-04-26 15:30:08 +00:00
|
|
|
// This makes sure we only delete the models
|
|
|
|
// after the destructor of the tables,
|
|
|
|
// this is needed to save the column sizes.
|
|
|
|
cylindersModel->setParent(ui.cylinders);
|
|
|
|
weightModel->setParent(ui.weights);
|
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.cylinders->setModel(cylindersModel);
|
|
|
|
ui.weights->setModel(weightModel);
|
2013-12-03 20:44:48 +00:00
|
|
|
closeMessage();
|
2013-09-26 20:02:27 +00:00
|
|
|
|
2015-08-25 21:27:19 +00:00
|
|
|
connect(ui.editDiveSiteButton, SIGNAL(clicked()), MainWindow::instance(), SIGNAL(startDiveSiteEdit()));
|
2017-07-15 20:37:02 +00:00
|
|
|
connect(ui.location, &DiveLocationLineEdit::entered, MapWidget::instance(), &MapWidget::centerOnIndex);
|
|
|
|
connect(ui.location, &DiveLocationLineEdit::currentChanged, MapWidget::instance(), &MapWidget::centerOnIndex);
|
2015-02-11 15:32:29 +00:00
|
|
|
|
2014-11-23 15:41:25 +00:00
|
|
|
QAction *action = new QAction(tr("Apply changes"), this);
|
2013-12-02 17:33:00 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(acceptChanges()));
|
2013-12-03 20:44:48 +00:00
|
|
|
addMessageAction(action);
|
2013-12-02 17:33:00 +00:00
|
|
|
|
2014-11-23 15:41:25 +00:00
|
|
|
action = new QAction(tr("Discard changes"), this);
|
2013-12-02 17:33:00 +00:00
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(rejectChanges()));
|
2015-02-11 15:58:23 +00:00
|
|
|
addMessageAction(action);
|
2014-06-03 22:29:28 +00:00
|
|
|
|
|
|
|
QShortcut *closeKey = new QShortcut(QKeySequence(Qt::Key_Escape), this);
|
|
|
|
connect(closeKey, SIGNAL(activated()), this, SLOT(escDetected()));
|
|
|
|
|
2013-09-26 20:02:27 +00:00
|
|
|
if (qApp->style()->objectName() == "oxygen")
|
2013-09-26 19:51:11 +00:00
|
|
|
setDocumentMode(true);
|
2013-09-26 20:02:27 +00:00
|
|
|
else
|
|
|
|
setDocumentMode(false);
|
|
|
|
|
2013-05-20 00:38:20 +00:00
|
|
|
// we start out with the fields read-only; once things are
|
|
|
|
// filled from a dive, they are made writeable
|
2013-09-19 00:56:53 +00:00
|
|
|
setEnabled(false);
|
2013-05-19 14:45:01 +00:00
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.cylinders->setTitle(tr("Cylinders"));
|
2014-11-25 15:47:24 +00:00
|
|
|
ui.cylinders->setBtnToolTip(tr("Add cylinder"));
|
2013-10-03 18:54:25 +00:00
|
|
|
connect(ui.cylinders, SIGNAL(addButtonClicked()), this, SLOT(addCylinder_clicked()));
|
|
|
|
|
|
|
|
ui.weights->setTitle(tr("Weights"));
|
2014-11-25 15:47:24 +00:00
|
|
|
ui.weights->setBtnToolTip(tr("Add weight system"));
|
2013-10-03 18:54:25 +00:00
|
|
|
connect(ui.weights, SIGNAL(addButtonClicked()), this, SLOT(addWeight_clicked()));
|
|
|
|
|
2015-01-01 16:00:46 +00:00
|
|
|
// This needs to be the same order as enum dive_comp_type in dive.h!
|
2018-05-17 08:04:41 +00:00
|
|
|
QStringList types = QStringList();
|
|
|
|
for (int i = 0; i < NUM_DIVEMODE; i++)
|
2018-06-17 15:55:47 +00:00
|
|
|
types.append(gettextFromC::tr(divemode_text_ui[i]));
|
2018-05-17 08:04:41 +00:00
|
|
|
ui.DiveType->insertItems(0, types);
|
2015-01-08 13:42:07 +00:00
|
|
|
connect(ui.DiveType, SIGNAL(currentIndexChanged(int)), this, SLOT(divetype_Changed(int)));
|
2015-01-01 16:00:46 +00:00
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
connect(ui.cylinders->view(), SIGNAL(clicked(QModelIndex)), this, SLOT(editCylinderWidget(QModelIndex)));
|
2013-10-03 23:04:51 +00:00
|
|
|
connect(ui.weights->view(), SIGNAL(clicked(QModelIndex)), this, SLOT(editWeightWidget(QModelIndex)));
|
2013-10-03 18:54:25 +00:00
|
|
|
|
2014-01-15 17:52:42 +00:00
|
|
|
ui.cylinders->view()->setItemDelegateForColumn(CylindersModel::TYPE, new TankInfoDelegate(this));
|
2014-11-17 14:03:37 +00:00
|
|
|
ui.cylinders->view()->setItemDelegateForColumn(CylindersModel::USE, new TankUseDelegate(this));
|
2014-01-15 17:52:42 +00:00
|
|
|
ui.weights->view()->setItemDelegateForColumn(WeightModel::TYPE, new WSInfoDelegate(this));
|
2014-03-22 14:13:58 +00:00
|
|
|
ui.cylinders->view()->setColumnHidden(CylindersModel::DEPTH, true);
|
2014-02-11 17:46:14 +00:00
|
|
|
completers.buddy = new QCompleter(&buddyModel, ui.buddy);
|
|
|
|
completers.divemaster = new QCompleter(&diveMasterModel, ui.divemaster);
|
|
|
|
completers.suit = new QCompleter(&suitModel, ui.suit);
|
|
|
|
completers.tags = new QCompleter(&tagModel, ui.tagWidget);
|
2014-01-06 22:00:10 +00:00
|
|
|
completers.buddy->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
completers.divemaster->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
completers.suit->setCaseSensitivity(Qt::CaseInsensitive);
|
2013-11-15 00:39:02 +00:00
|
|
|
completers.tags->setCaseSensitivity(Qt::CaseInsensitive);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.buddy->setCompleter(completers.buddy);
|
|
|
|
ui.divemaster->setCompleter(completers.divemaster);
|
|
|
|
ui.suit->setCompleter(completers.suit);
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->setCompleter(completers.tags);
|
2015-06-04 01:08:57 +00:00
|
|
|
ui.diveNotesMessage->hide();
|
2017-02-04 21:12:50 +00:00
|
|
|
ui.depth->hide();
|
|
|
|
ui.depthLabel->hide();
|
|
|
|
ui.duration->hide();
|
|
|
|
ui.durationLabel->hide();
|
2013-09-26 21:14:09 +00:00
|
|
|
setMinimumHeight(0);
|
|
|
|
setMinimumWidth(0);
|
2013-09-27 15:52:01 +00:00
|
|
|
|
|
|
|
// Current display of things on Gnome3 looks like shit, so
|
|
|
|
// let`s fix that.
|
|
|
|
if (isGnome3Session()) {
|
|
|
|
QPalette p;
|
|
|
|
p.setColor(QPalette::Window, QColor(Qt::white));
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.scrollArea->viewport()->setPalette(p);
|
|
|
|
ui.scrollArea_2->viewport()->setPalette(p);
|
2013-12-04 20:24:37 +00:00
|
|
|
|
|
|
|
// GroupBoxes in Gnome3 looks like I'v drawn them...
|
|
|
|
static const QString gnomeCss(
|
|
|
|
"QGroupBox {"
|
|
|
|
" background-color: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,"
|
|
|
|
" stop: 0 #E0E0E0, stop: 1 #FFFFFF);"
|
|
|
|
" border: 2px solid gray;"
|
|
|
|
" border-radius: 5px;"
|
|
|
|
" margin-top: 1ex;"
|
|
|
|
"}"
|
|
|
|
"QGroupBox::title {"
|
|
|
|
" subcontrol-origin: margin;"
|
|
|
|
" subcontrol-position: top center;"
|
|
|
|
" padding: 0 3px;"
|
|
|
|
" background-color: qlineargradient(x1: 0, y1: 0, x2: 0, y2: 1,"
|
2013-12-07 16:13:55 +00:00
|
|
|
" stop: 0 #E0E0E0, stop: 1 #FFFFFF);"
|
2013-12-04 20:24:37 +00:00
|
|
|
"}");
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (QGroupBox *box, findChildren<QGroupBox *>()) {
|
2013-12-04 20:24:37 +00:00
|
|
|
box->setStyleSheet(gnomeCss);
|
|
|
|
}
|
2013-09-27 15:52:01 +00:00
|
|
|
}
|
2014-12-23 18:16:11 +00:00
|
|
|
// QLineEdit and QLabels should have minimal margin on the left and right but not waste vertical space
|
2015-01-02 20:42:02 +00:00
|
|
|
QMargins margins(3, 2, 1, 0);
|
2014-12-23 18:16:11 +00:00
|
|
|
Q_FOREACH (QLabel *label, findChildren<QLabel *>()) {
|
|
|
|
label->setContentsMargins(margins);
|
|
|
|
}
|
2013-11-28 11:17:30 +00:00
|
|
|
ui.cylinders->view()->horizontalHeader()->setContextMenuPolicy(Qt::ActionsContextMenu);
|
2015-10-08 10:52:37 +00:00
|
|
|
ui.weights->view()->horizontalHeader()->setContextMenuPolicy(Qt::ActionsContextMenu);
|
2013-11-28 11:17:30 +00:00
|
|
|
|
|
|
|
QSettings s;
|
|
|
|
s.beginGroup("cylinders_dialog");
|
2014-02-28 04:09:57 +00:00
|
|
|
for (int i = 0; i < CylindersModel::COLUMNS; i++) {
|
2013-11-28 11:17:30 +00:00
|
|
|
if ((i == CylindersModel::REMOVE) || (i == CylindersModel::TYPE))
|
2014-02-28 04:09:57 +00:00
|
|
|
continue;
|
2013-11-28 11:17:30 +00:00
|
|
|
bool checked = s.value(QString("column%1_hidden").arg(i)).toBool();
|
2014-02-12 14:07:32 +00:00
|
|
|
action = new QAction(cylindersModel->headerData(i, Qt::Horizontal, Qt::DisplayRole).toString(), ui.cylinders->view());
|
2013-11-28 11:17:30 +00:00
|
|
|
action->setCheckable(true);
|
|
|
|
action->setData(i);
|
|
|
|
action->setChecked(!checked);
|
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(toggleTriggeredColumn()));
|
|
|
|
ui.cylinders->view()->setColumnHidden(i, checked);
|
|
|
|
ui.cylinders->view()->horizontalHeader()->addAction(action);
|
|
|
|
}
|
2014-08-06 21:05:54 +00:00
|
|
|
|
2015-05-17 21:50:22 +00:00
|
|
|
ui.waitingSpinner->setRoundness(70.0);
|
|
|
|
ui.waitingSpinner->setMinimumTrailOpacity(15.0);
|
|
|
|
ui.waitingSpinner->setTrailFadePercentage(70.0);
|
|
|
|
ui.waitingSpinner->setNumberOfLines(8);
|
|
|
|
ui.waitingSpinner->setLineLength(5);
|
|
|
|
ui.waitingSpinner->setLineWidth(3);
|
|
|
|
ui.waitingSpinner->setInnerRadius(5);
|
|
|
|
ui.waitingSpinner->setRevolutionsPerSecond(1);
|
|
|
|
|
2015-05-30 01:22:24 +00:00
|
|
|
connect(ReverseGeoLookupThread::instance(), SIGNAL(finished()),
|
|
|
|
LocationInformationModel::instance(), SLOT(update()));
|
|
|
|
|
|
|
|
connect(ReverseGeoLookupThread::instance(), &QThread::finished,
|
|
|
|
this, &MainTab::setCurrentLocationIndex);
|
|
|
|
|
2015-09-23 19:03:28 +00:00
|
|
|
connect(ui.diveNotesMessage, &KMessageWidget::showAnimationFinished,
|
|
|
|
ui.location, &DiveLocationLineEdit::fixPopupPosition);
|
2017-11-26 21:21:58 +00:00
|
|
|
connect(this, SIGNAL(diveSiteAdded(const QString &)), LocationFilterModel::instance(), SLOT(addName(const QString &)));
|
2015-09-23 19:03:28 +00:00
|
|
|
|
2015-11-16 04:56:24 +00:00
|
|
|
// enable URL clickability in notes:
|
|
|
|
new TextHyperlinkEventFilter(ui.notes);//destroyed when ui.notes is destroyed
|
|
|
|
|
2015-01-05 07:17:01 +00:00
|
|
|
acceptingEdit = false;
|
2015-10-01 20:54:13 +00:00
|
|
|
|
|
|
|
ui.diveTripLocation->hide();
|
2017-04-04 17:21:30 +00:00
|
|
|
|
|
|
|
|
2013-11-28 11:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MainTab::~MainTab()
|
|
|
|
{
|
|
|
|
QSettings s;
|
|
|
|
s.beginGroup("cylinders_dialog");
|
2014-02-28 04:09:57 +00:00
|
|
|
for (int i = 0; i < CylindersModel::COLUMNS; i++) {
|
2013-11-28 11:17:30 +00:00
|
|
|
if ((i == CylindersModel::REMOVE) || (i == CylindersModel::TYPE))
|
2014-02-28 04:09:57 +00:00
|
|
|
continue;
|
2013-11-28 11:17:30 +00:00
|
|
|
s.setValue(QString("column%1_hidden").arg(i), ui.cylinders->view()->isColumnHidden(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-30 01:22:24 +00:00
|
|
|
void MainTab::setCurrentLocationIndex()
|
|
|
|
{
|
2015-05-31 23:12:56 +00:00
|
|
|
if (current_dive) {
|
|
|
|
struct dive_site *ds = get_dive_site_by_uuid(current_dive->dive_site_uuid);
|
|
|
|
if (ds)
|
2015-09-25 17:51:10 +00:00
|
|
|
ui.location->setCurrentDiveSiteUuid(ds->uuid);
|
2015-05-31 23:12:56 +00:00
|
|
|
else
|
2015-06-25 17:24:22 +00:00
|
|
|
ui.location->clear();
|
2015-05-31 23:12:56 +00:00
|
|
|
}
|
2015-05-30 01:22:24 +00:00
|
|
|
}
|
|
|
|
|
2015-05-15 21:27:08 +00:00
|
|
|
void MainTab::enableGeoLookupEdition()
|
2015-05-16 21:51:43 +00:00
|
|
|
{
|
2015-05-17 21:50:22 +00:00
|
|
|
ui.waitingSpinner->stop();
|
2015-05-16 21:51:43 +00:00
|
|
|
}
|
|
|
|
|
2015-05-15 21:27:08 +00:00
|
|
|
void MainTab::disableGeoLookupEdition()
|
2015-05-16 21:51:43 +00:00
|
|
|
{
|
2015-05-17 21:50:22 +00:00
|
|
|
ui.waitingSpinner->start();
|
2015-05-16 21:51:43 +00:00
|
|
|
}
|
|
|
|
|
2013-11-28 11:17:30 +00:00
|
|
|
void MainTab::toggleTriggeredColumn()
|
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
2013-11-28 11:17:30 +00:00
|
|
|
int col = action->data().toInt();
|
|
|
|
QTableView *view = ui.cylinders->view();
|
|
|
|
|
2014-01-16 04:50:56 +00:00
|
|
|
if (action->isChecked()) {
|
2013-11-28 11:17:30 +00:00
|
|
|
view->showColumn(col);
|
2014-01-16 04:50:56 +00:00
|
|
|
if (view->columnWidth(col) <= 15)
|
2013-11-28 11:17:30 +00:00
|
|
|
view->setColumnWidth(col, 80);
|
2014-02-28 04:09:57 +00:00
|
|
|
} else
|
2013-11-28 11:17:30 +00:00
|
|
|
view->hideColumn(col);
|
2013-06-27 12:33:43 +00:00
|
|
|
}
|
2013-05-22 12:17:18 +00:00
|
|
|
|
2013-09-19 04:33:39 +00:00
|
|
|
void MainTab::addDiveStarted()
|
|
|
|
{
|
2013-12-27 16:18:53 +00:00
|
|
|
enableEdition(ADD);
|
2013-09-19 04:33:39 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::addMessageAction(QAction *action)
|
2013-12-03 20:44:48 +00:00
|
|
|
{
|
|
|
|
ui.diveNotesMessage->addAction(action);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::hideMessage()
|
|
|
|
{
|
|
|
|
ui.diveNotesMessage->animatedHide();
|
2014-04-05 16:03:45 +00:00
|
|
|
updateTextLabels(false);
|
2013-12-03 20:44:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::closeMessage()
|
|
|
|
{
|
|
|
|
hideMessage();
|
|
|
|
ui.diveNotesMessage->setCloseButtonVisible(false);
|
2017-04-04 17:50:46 +00:00
|
|
|
}
|
2013-12-03 20:44:48 +00:00
|
|
|
|
|
|
|
void MainTab::displayMessage(QString str)
|
|
|
|
{
|
2014-08-05 05:45:17 +00:00
|
|
|
ui.diveNotesMessage->setCloseButtonVisible(false);
|
2013-12-03 20:44:48 +00:00
|
|
|
ui.diveNotesMessage->setText(str);
|
|
|
|
ui.diveNotesMessage->animatedShow();
|
2014-04-05 16:03:45 +00:00
|
|
|
updateTextLabels();
|
2014-03-11 03:49:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::updateTextLabels(bool showUnits)
|
|
|
|
{
|
2014-04-05 16:03:45 +00:00
|
|
|
if (showUnits) {
|
2014-07-10 23:06:51 +00:00
|
|
|
ui.airTempLabel->setText(tr("Air temp. [%1]").arg(get_temp_unit()));
|
|
|
|
ui.waterTempLabel->setText(tr("Water temp. [%1]").arg(get_temp_unit()));
|
2014-03-11 03:49:08 +00:00
|
|
|
} else {
|
2014-07-10 23:06:51 +00:00
|
|
|
ui.airTempLabel->setText(tr("Air temp."));
|
|
|
|
ui.waterTempLabel->setText(tr("Water temp."));
|
2014-03-11 03:49:08 +00:00
|
|
|
}
|
2013-12-03 20:44:48 +00:00
|
|
|
}
|
|
|
|
|
2013-11-01 15:48:34 +00:00
|
|
|
void MainTab::enableEdition(EditMode newEditMode)
|
2013-08-13 11:34:04 +00:00
|
|
|
{
|
2015-03-26 00:23:43 +00:00
|
|
|
const bool isTripEdit = MainWindow::instance() &&
|
|
|
|
MainWindow::instance()->dive_list()->selectedTrips().count() == 1;
|
|
|
|
|
2014-07-03 21:45:01 +00:00
|
|
|
if (((newEditMode == DIVE || newEditMode == NONE) && current_dive == NULL) || editMode != NONE)
|
2013-08-13 11:34:04 +00:00
|
|
|
return;
|
2014-07-03 04:33:03 +00:00
|
|
|
modified = false;
|
2014-11-05 14:34:29 +00:00
|
|
|
copyPaste = false;
|
2013-12-27 16:18:53 +00:00
|
|
|
if ((newEditMode == DIVE || newEditMode == NONE) &&
|
2015-03-26 00:23:43 +00:00
|
|
|
!isTripEdit &&
|
2014-01-03 12:49:17 +00:00
|
|
|
current_dive->dc.model &&
|
2013-12-27 16:18:53 +00:00
|
|
|
strcmp(current_dive->dc.model, "manually added dive") == 0) {
|
|
|
|
// editCurrentDive will call enableEdition with newEditMode == MANUALLY_ADDED_DIVE
|
|
|
|
// so exit this function here after editCurrentDive() returns
|
2014-07-03 21:45:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// FIXME : can we get rid of this recursive crap?
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->editCurrentDive();
|
2013-12-27 16:18:53 +00:00
|
|
|
return;
|
|
|
|
}
|
2015-09-23 20:11:43 +00:00
|
|
|
|
|
|
|
ui.editDiveSiteButton->setEnabled(false);
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->setEnabled(false);
|
2014-08-04 15:58:21 +00:00
|
|
|
MainWindow::instance()->setEnabledToolbar(false);
|
2018-05-03 17:19:54 +00:00
|
|
|
MainWindow::instance()->enterEditState();
|
2014-07-03 21:45:01 +00:00
|
|
|
|
2015-03-26 00:23:43 +00:00
|
|
|
if (isTripEdit) {
|
2013-09-19 00:56:53 +00:00
|
|
|
// we are editing trip location and notes
|
2013-12-03 20:44:48 +00:00
|
|
|
displayMessage(tr("This trip is being edited."));
|
2014-07-30 15:52:11 +00:00
|
|
|
currentTrip = current_dive->divetrip;
|
2014-06-19 16:52:34 +00:00
|
|
|
ui.dateEdit->setEnabled(false);
|
2013-09-19 00:56:53 +00:00
|
|
|
editMode = TRIP;
|
|
|
|
} else {
|
2014-08-23 14:49:28 +00:00
|
|
|
ui.dateEdit->setEnabled(true);
|
2013-11-05 06:57:24 +00:00
|
|
|
if (amount_selected > 1) {
|
2013-12-03 20:44:48 +00:00
|
|
|
displayMessage(tr("Multiple dives are being edited."));
|
2013-11-05 06:57:24 +00:00
|
|
|
} else {
|
2013-12-03 20:44:48 +00:00
|
|
|
displayMessage(tr("This dive is being edited."));
|
2013-11-05 06:57:24 +00:00
|
|
|
}
|
2013-11-01 15:48:34 +00:00
|
|
|
editMode = newEditMode != NONE ? newEditMode : DIVE;
|
2013-09-19 00:56:53 +00:00
|
|
|
}
|
2013-08-13 11:34:04 +00:00
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
void MainTab::clearEquipment()
|
|
|
|
{
|
2013-10-10 08:16:32 +00:00
|
|
|
cylindersModel->clear();
|
|
|
|
weightModel->clear();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2014-05-16 06:12:46 +00:00
|
|
|
void MainTab::nextInputField(QKeyEvent *event)
|
|
|
|
{
|
|
|
|
keyPressEvent(event);
|
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
#define UPDATE_TEXT(d, field) \
|
2014-07-03 05:15:08 +00:00
|
|
|
if (clear || !d.field) \
|
2014-07-21 23:34:15 +00:00
|
|
|
ui.field->setText(QString()); \
|
2014-02-28 04:09:57 +00:00
|
|
|
else \
|
2014-07-02 21:15:17 +00:00
|
|
|
ui.field->setText(d.field)
|
2013-05-06 17:16:16 +00:00
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
#define UPDATE_TEMP(d, field) \
|
2014-07-02 21:15:17 +00:00
|
|
|
if (clear || d.field.mkelvin == 0) \
|
2014-02-28 04:09:57 +00:00
|
|
|
ui.field->setText(""); \
|
|
|
|
else \
|
2014-07-02 21:15:17 +00:00
|
|
|
ui.field->setText(get_temperature_string(d.field, true))
|
2013-09-20 23:41:42 +00:00
|
|
|
|
2013-11-10 21:23:18 +00:00
|
|
|
bool MainTab::isEditing()
|
|
|
|
{
|
|
|
|
return editMode != NONE;
|
|
|
|
}
|
|
|
|
|
2015-02-14 07:49:05 +00:00
|
|
|
void MainTab::showLocation()
|
|
|
|
{
|
2015-06-02 02:13:51 +00:00
|
|
|
if (get_dive_site_by_uuid(displayed_dive.dive_site_uuid))
|
2015-09-25 17:51:10 +00:00
|
|
|
ui.location->setCurrentDiveSiteUuid(displayed_dive.dive_site_uuid);
|
2015-06-02 02:13:51 +00:00
|
|
|
else
|
2015-06-25 17:24:22 +00:00
|
|
|
ui.location->clear();
|
2015-02-14 07:49:05 +00:00
|
|
|
}
|
|
|
|
|
2017-02-26 20:05:23 +00:00
|
|
|
void MainTab::updateDepthDuration()
|
|
|
|
{
|
|
|
|
ui.depth->setVisible(true);
|
|
|
|
ui.depthLabel->setVisible(true);
|
|
|
|
ui.duration->setVisible(true);
|
|
|
|
ui.durationLabel->setVisible(true);
|
2018-02-04 19:07:32 +00:00
|
|
|
ui.duration->setText(render_seconds_to_string(displayed_dive.duration.seconds));
|
2017-02-26 20:05:23 +00:00
|
|
|
ui.depth->setText(get_depth_string(displayed_dive.maxdepth, true));
|
|
|
|
}
|
|
|
|
|
2014-07-02 21:15:17 +00:00
|
|
|
void MainTab::updateDiveInfo(bool clear)
|
2013-05-06 16:23:14 +00:00
|
|
|
{
|
2015-09-21 17:01:58 +00:00
|
|
|
ui.location->refreshDiveSiteCache();
|
2015-07-16 17:42:56 +00:00
|
|
|
EditMode rememberEM = editMode;
|
2014-05-26 22:17:34 +00:00
|
|
|
// don't execute this while adding / planning a dive
|
|
|
|
if (editMode == ADD || editMode == MANUALLY_ADDED_DIVE || MainWindow::instance()->graphics()->isPlanner())
|
2013-11-12 07:33:27 +00:00
|
|
|
return;
|
2015-05-17 21:50:22 +00:00
|
|
|
if (!isEnabled() && !clear )
|
2013-09-19 00:56:53 +00:00
|
|
|
setEnabled(true);
|
2014-07-02 21:15:17 +00:00
|
|
|
if (isEnabled() && clear)
|
2013-09-22 15:19:05 +00:00
|
|
|
setEnabled(false);
|
2014-07-03 05:15:08 +00:00
|
|
|
editMode = IGNORE; // don't trigger on changes to the widgets
|
|
|
|
|
2013-06-14 16:17:46 +00:00
|
|
|
// This method updates ALL tabs whenever a new dive or trip is
|
|
|
|
// selected.
|
|
|
|
// If exactly one trip has been selected, we show the location / notes
|
|
|
|
// for the trip in the Info tab, otherwise we show the info of the
|
|
|
|
// selected_dive
|
2013-05-19 15:08:29 +00:00
|
|
|
struct dive *prevd;
|
|
|
|
|
2013-05-20 05:42:24 +00:00
|
|
|
process_selected_dives();
|
2014-07-02 21:15:17 +00:00
|
|
|
process_all_dives(&displayed_dive, &prevd);
|
|
|
|
|
2017-04-04 17:21:30 +00:00
|
|
|
for (auto widget : extraWidgets) {
|
|
|
|
widget->updateData();
|
|
|
|
}
|
2014-07-21 23:34:15 +00:00
|
|
|
|
|
|
|
ui.notes->setText(QString());
|
|
|
|
if (!clear) {
|
|
|
|
QString tmp(displayed_dive.notes);
|
2017-11-28 17:10:37 +00:00
|
|
|
if (tmp.indexOf("<table") != -1) {
|
|
|
|
tmp.replace(QString("\n"), QString("<br>"));
|
2014-07-21 23:34:15 +00:00
|
|
|
ui.notes->setHtml(tmp);
|
2017-11-28 17:10:37 +00:00
|
|
|
} else {
|
2014-07-21 23:34:15 +00:00
|
|
|
ui.notes->setPlainText(tmp);
|
2017-11-28 17:10:37 +00:00
|
|
|
}
|
2014-07-21 23:34:15 +00:00
|
|
|
}
|
2014-07-02 21:15:17 +00:00
|
|
|
UPDATE_TEXT(displayed_dive, suit);
|
|
|
|
UPDATE_TEXT(displayed_dive, divemaster);
|
|
|
|
UPDATE_TEXT(displayed_dive, buddy);
|
|
|
|
UPDATE_TEMP(displayed_dive, airtemp);
|
|
|
|
UPDATE_TEMP(displayed_dive, watertemp);
|
2015-07-09 09:06:11 +00:00
|
|
|
ui.DiveType->setCurrentIndex(get_dive_dc(&displayed_dive, dc_number)->divemode);
|
2014-07-03 05:15:08 +00:00
|
|
|
|
2014-07-02 21:15:17 +00:00
|
|
|
if (!clear) {
|
2015-07-16 17:43:37 +00:00
|
|
|
struct dive_site *ds = NULL;
|
|
|
|
// if we are showing a dive and editing it, let's refer to the displayed_dive_site as that
|
|
|
|
// already may contain changes, otherwise start with the dive site referred to by the displayed
|
|
|
|
// dive
|
|
|
|
if (rememberEM == DIVE) {
|
|
|
|
ds = &displayed_dive_site;
|
|
|
|
} else {
|
|
|
|
ds = get_dive_site_by_uuid(displayed_dive.dive_site_uuid);
|
|
|
|
if (ds)
|
|
|
|
copy_dive_site(ds, &displayed_dive_site);
|
|
|
|
}
|
2015-08-29 23:00:22 +00:00
|
|
|
|
2015-06-26 18:03:34 +00:00
|
|
|
if (ds) {
|
2015-09-25 17:51:10 +00:00
|
|
|
ui.location->setCurrentDiveSiteUuid(ds->uuid);
|
2017-10-16 13:52:13 +00:00
|
|
|
ui.locationTags->setText(constructLocationTags(ds, true));
|
2015-06-26 18:03:34 +00:00
|
|
|
} else {
|
2015-06-25 17:24:22 +00:00
|
|
|
ui.location->clear();
|
2017-11-26 10:23:00 +00:00
|
|
|
ui.locationTags->clear();
|
2015-06-26 18:03:34 +00:00
|
|
|
clear_dive_site(&displayed_dive_site);
|
|
|
|
}
|
|
|
|
|
2014-07-29 17:58:35 +00:00
|
|
|
// Subsurface always uses "local time" as in "whatever was the local time at the location"
|
|
|
|
// so all time stamps have no time zone information and are in UTC
|
2016-04-28 23:31:37 +00:00
|
|
|
QDateTime localTime = QDateTime::fromMSecsSinceEpoch(1000*displayed_dive.when, Qt::UTC);
|
2014-07-29 17:58:35 +00:00
|
|
|
localTime.setTimeSpec(Qt::UTC);
|
2014-06-26 23:05:48 +00:00
|
|
|
ui.dateEdit->setDate(localTime.date());
|
|
|
|
ui.timeEdit->setTime(localTime.time());
|
2014-02-12 14:22:54 +00:00
|
|
|
if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2017-11-30 19:33:43 +00:00
|
|
|
// Remember the tab selected for last dive
|
|
|
|
if (lastSelectedDive)
|
|
|
|
lastTabSelectedDive = ui.tabWidget->currentIndex();
|
2017-11-30 18:36:05 +00:00
|
|
|
ui.tabWidget->setTabText(0, tr("Trip notes"));
|
2017-11-30 19:33:43 +00:00
|
|
|
ui.tabWidget->setTabEnabled(1, false);
|
|
|
|
ui.tabWidget->setTabEnabled(2, false);
|
|
|
|
ui.tabWidget->setTabEnabled(5, false);
|
|
|
|
// Recover the tab selected for last dive trip
|
|
|
|
if (lastSelectedDive)
|
|
|
|
ui.tabWidget->setCurrentIndex(lastTabSelectedDiveTrip);
|
|
|
|
lastSelectedDive = false;
|
2014-07-30 15:52:11 +00:00
|
|
|
currentTrip = *MainWindow::instance()->dive_list()->selectedTrips().begin();
|
2013-06-14 16:17:46 +00:00
|
|
|
// only use trip relevant fields
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.divemaster->setVisible(false);
|
|
|
|
ui.DivemasterLabel->setVisible(false);
|
|
|
|
ui.buddy->setVisible(false);
|
|
|
|
ui.BuddyLabel->setVisible(false);
|
|
|
|
ui.suit->setVisible(false);
|
|
|
|
ui.SuitLabel->setVisible(false);
|
|
|
|
ui.rating->setVisible(false);
|
|
|
|
ui.RatingLabel->setVisible(false);
|
|
|
|
ui.visibility->setVisible(false);
|
|
|
|
ui.visibilityLabel->setVisible(false);
|
2013-11-04 19:28:03 +00:00
|
|
|
ui.tagWidget->setVisible(false);
|
|
|
|
ui.TagLabel->setVisible(false);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.airTempLabel->setVisible(false);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.airtemp->setVisible(false);
|
2015-01-01 16:00:46 +00:00
|
|
|
ui.DiveType->setVisible(false);
|
|
|
|
ui.TypeLabel->setVisible(false);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.waterTempLabel->setVisible(false);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.watertemp->setVisible(false);
|
2016-01-17 10:53:50 +00:00
|
|
|
ui.dateEdit->setReadOnly(true);
|
2017-11-16 06:35:57 +00:00
|
|
|
ui.timeLabel->setVisible(false);
|
2016-01-17 10:53:51 +00:00
|
|
|
ui.timeEdit->setVisible(false);
|
2015-10-01 20:41:38 +00:00
|
|
|
ui.diveTripLocation->show();
|
|
|
|
ui.location->hide();
|
|
|
|
ui.editDiveSiteButton->hide();
|
2013-06-14 16:17:46 +00:00
|
|
|
// rename the remaining fields and fill data from selected trip
|
2014-07-10 23:06:42 +00:00
|
|
|
ui.LocationLabel->setText(tr("Trip location"));
|
2015-10-01 21:11:03 +00:00
|
|
|
ui.diveTripLocation->setText(currentTrip->location);
|
2015-07-01 23:38:22 +00:00
|
|
|
ui.locationTags->clear();
|
2015-09-25 17:51:10 +00:00
|
|
|
//TODO: Fix this.
|
|
|
|
//ui.location->setText(currentTrip->location);
|
2014-07-10 23:06:42 +00:00
|
|
|
ui.NotesLabel->setText(tr("Trip notes"));
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.notes->setText(currentTrip->notes);
|
2013-12-28 01:26:22 +00:00
|
|
|
clearEquipment();
|
|
|
|
ui.equipmentTab->setEnabled(false);
|
2017-02-04 21:44:10 +00:00
|
|
|
ui.depth->setVisible(false);
|
|
|
|
ui.depthLabel->setVisible(false);
|
|
|
|
ui.duration->setVisible(false);
|
|
|
|
ui.durationLabel->setVisible(false);
|
2013-06-14 16:17:46 +00:00
|
|
|
} else {
|
2017-11-30 19:33:43 +00:00
|
|
|
// Remember the tab selected for last dive trip
|
|
|
|
if (!lastSelectedDive)
|
|
|
|
lastTabSelectedDiveTrip = ui.tabWidget->currentIndex();
|
2017-11-30 18:36:05 +00:00
|
|
|
ui.tabWidget->setTabText(0, tr("Notes"));
|
2017-11-30 19:33:43 +00:00
|
|
|
ui.tabWidget->setTabEnabled(1, true);
|
|
|
|
ui.tabWidget->setTabEnabled(2, true);
|
|
|
|
ui.tabWidget->setTabEnabled(5, true);
|
|
|
|
// Recover the tab selected for last dive
|
|
|
|
if (!lastSelectedDive)
|
|
|
|
ui.tabWidget->setCurrentIndex(lastTabSelectedDive);
|
|
|
|
lastSelectedDive = true;
|
2014-07-30 15:52:11 +00:00
|
|
|
currentTrip = NULL;
|
2013-06-14 16:17:46 +00:00
|
|
|
// make all the fields visible writeable
|
2015-10-01 20:41:38 +00:00
|
|
|
ui.diveTripLocation->hide();
|
|
|
|
ui.location->show();
|
|
|
|
ui.editDiveSiteButton->show();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.divemaster->setVisible(true);
|
|
|
|
ui.buddy->setVisible(true);
|
|
|
|
ui.suit->setVisible(true);
|
|
|
|
ui.SuitLabel->setVisible(true);
|
|
|
|
ui.rating->setVisible(true);
|
|
|
|
ui.RatingLabel->setVisible(true);
|
|
|
|
ui.visibility->setVisible(true);
|
|
|
|
ui.visibilityLabel->setVisible(true);
|
|
|
|
ui.BuddyLabel->setVisible(true);
|
|
|
|
ui.DivemasterLabel->setVisible(true);
|
2013-11-04 19:28:03 +00:00
|
|
|
ui.TagLabel->setVisible(true);
|
|
|
|
ui.tagWidget->setVisible(true);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.airTempLabel->setVisible(true);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.airtemp->setVisible(true);
|
2015-01-01 16:00:46 +00:00
|
|
|
ui.TypeLabel->setVisible(true);
|
|
|
|
ui.DiveType->setVisible(true);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.waterTempLabel->setVisible(true);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.watertemp->setVisible(true);
|
2016-01-17 10:53:50 +00:00
|
|
|
ui.dateEdit->setReadOnly(false);
|
2017-11-16 06:35:57 +00:00
|
|
|
ui.timeLabel->setVisible(true);
|
2016-01-17 10:53:51 +00:00
|
|
|
ui.timeEdit->setVisible(true);
|
2013-06-14 16:17:46 +00:00
|
|
|
/* and fill them from the dive */
|
2014-07-02 21:15:17 +00:00
|
|
|
ui.rating->setCurrentStars(displayed_dive.rating);
|
|
|
|
ui.visibility->setCurrentStars(displayed_dive.visibility);
|
2013-06-14 16:17:46 +00:00
|
|
|
// reset labels in case we last displayed trip notes
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.LocationLabel->setText(tr("Location"));
|
|
|
|
ui.NotesLabel->setText(tr("Notes"));
|
2013-12-28 01:26:22 +00:00
|
|
|
ui.equipmentTab->setEnabled(true);
|
2014-07-03 05:38:08 +00:00
|
|
|
cylindersModel->updateDive();
|
|
|
|
weightModel->updateDive();
|
2018-04-09 08:09:34 +00:00
|
|
|
ui.tagWidget->setText(get_taglist_string(displayed_dive.tag_list));
|
2017-11-28 19:55:50 +00:00
|
|
|
if (current_dive) {
|
|
|
|
bool isManual = same_string(current_dive->dc.model, "manually added dive");
|
|
|
|
ui.depth->setVisible(isManual);
|
|
|
|
ui.depthLabel->setVisible(isManual);
|
|
|
|
ui.duration->setVisible(isManual);
|
|
|
|
ui.durationLabel->setVisible(isManual);
|
|
|
|
}
|
2013-06-14 16:17:46 +00:00
|
|
|
}
|
2018-02-04 19:07:32 +00:00
|
|
|
ui.duration->setText(render_seconds_to_string(displayed_dive.duration.seconds));
|
2017-02-04 21:44:10 +00:00
|
|
|
ui.depth->setText(get_depth_string(displayed_dive.maxdepth, true));
|
2015-01-11 15:29:29 +00:00
|
|
|
ui.DiveType->setCurrentIndex(get_dive_dc(&displayed_dive, dc_number)->divemode);
|
2015-06-25 18:42:28 +00:00
|
|
|
|
2013-12-09 21:37:49 +00:00
|
|
|
volume_t gases[MAX_CYLINDERS] = {};
|
2014-07-02 21:15:17 +00:00
|
|
|
get_gas_used(&displayed_dive, gases);
|
2013-11-20 06:50:02 +00:00
|
|
|
int mean[MAX_CYLINDERS], duration[MAX_CYLINDERS];
|
2014-07-02 21:15:17 +00:00
|
|
|
per_cylinder_mean_depth(&displayed_dive, select_dc(&displayed_dive), mean, duration);
|
2017-04-04 17:21:30 +00:00
|
|
|
|
2014-06-10 05:45:44 +00:00
|
|
|
// now let's get some gas use statistics
|
|
|
|
QVector<QPair<QString, int> > gasUsed;
|
|
|
|
QString gasUsedString;
|
|
|
|
volume_t vol;
|
|
|
|
selectedDivesGasUsed(gasUsed);
|
2017-11-27 13:45:02 +00:00
|
|
|
for (int j = 0; j < MAX_CYLINDERS; j++) {
|
2014-06-10 05:45:44 +00:00
|
|
|
if (gasUsed.isEmpty())
|
|
|
|
break;
|
|
|
|
QPair<QString, int> gasPair = gasUsed.last();
|
|
|
|
gasUsed.pop_back();
|
|
|
|
vol.mliter = gasPair.second;
|
|
|
|
gasUsedString.append(gasPair.first).append(": ").append(get_volume_string(vol, true)).append("\n");
|
|
|
|
}
|
|
|
|
if (!gasUsed.isEmpty())
|
|
|
|
gasUsedString.append("...");
|
2014-06-29 18:01:56 +00:00
|
|
|
volume_t o2_tot = {}, he_tot = {};
|
|
|
|
selected_dives_gas_parts(&o2_tot, &he_tot);
|
2014-07-06 02:34:43 +00:00
|
|
|
|
2015-06-25 18:42:28 +00:00
|
|
|
if(ui.locationTags->text().isEmpty())
|
|
|
|
ui.locationTags->hide();
|
|
|
|
else
|
|
|
|
ui.locationTags->show();
|
2017-10-06 14:51:02 +00:00
|
|
|
ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty());
|
2015-10-08 06:38:26 +00:00
|
|
|
/* unset the special value text for date and time, just in case someone dove at midnight */
|
|
|
|
ui.dateEdit->setSpecialValueText(QString(""));
|
|
|
|
ui.timeEdit->setSpecialValueText(QString(""));
|
|
|
|
|
2013-05-08 19:08:00 +00:00
|
|
|
} else {
|
2013-05-20 00:38:20 +00:00
|
|
|
/* clear the fields */
|
2017-04-04 17:21:30 +00:00
|
|
|
clearTabs();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.rating->setCurrentStars(0);
|
|
|
|
ui.visibility->setCurrentStars(0);
|
2015-02-12 20:46:21 +00:00
|
|
|
ui.location->clear();
|
2015-10-08 06:38:26 +00:00
|
|
|
/* set date and time to minimums which triggers showing the special value text */
|
|
|
|
ui.dateEdit->setSpecialValueText(QString("-"));
|
|
|
|
ui.dateEdit->setMinimumDate(QDate(1, 1, 1));
|
|
|
|
ui.dateEdit->setDate(QDate(1, 1, 1));
|
|
|
|
ui.timeEdit->setSpecialValueText(QString("-"));
|
|
|
|
ui.timeEdit->setMinimumTime(QTime(0, 0, 0, 0));
|
|
|
|
ui.timeEdit->setTime(QTime(0, 0, 0, 0));
|
2018-01-01 07:49:40 +00:00
|
|
|
ui.tagWidget->clear();
|
2013-05-08 19:08:00 +00:00
|
|
|
}
|
2015-07-16 17:42:56 +00:00
|
|
|
editMode = rememberEM;
|
2014-08-05 20:40:50 +00:00
|
|
|
ui.cylinders->view()->hideColumn(CylindersModel::DEPTH);
|
2015-01-10 23:01:15 +00:00
|
|
|
if (get_dive_dc(&displayed_dive, dc_number)->divemode == CCR)
|
2014-11-17 14:03:37 +00:00
|
|
|
ui.cylinders->view()->showColumn(CylindersModel::USE);
|
|
|
|
else
|
|
|
|
ui.cylinders->view()->hideColumn(CylindersModel::USE);
|
2015-06-04 01:44:16 +00:00
|
|
|
|
2015-09-23 19:20:21 +00:00
|
|
|
if (verbose)
|
|
|
|
qDebug() << "Set the current dive site:" << displayed_dive.dive_site_uuid;
|
2015-07-16 19:41:04 +00:00
|
|
|
emit diveSiteChanged(get_dive_site_by_uuid(displayed_dive.dive_site_uuid));
|
2013-05-06 16:23:14 +00:00
|
|
|
}
|
|
|
|
|
2013-05-22 12:40:26 +00:00
|
|
|
void MainTab::addCylinder_clicked()
|
2013-04-13 13:17:59 +00:00
|
|
|
{
|
2014-01-16 04:50:56 +00:00
|
|
|
if (editMode == NONE)
|
2013-09-25 18:05:48 +00:00
|
|
|
enableEdition();
|
2013-05-22 17:52:38 +00:00
|
|
|
cylindersModel->add();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2013-05-22 12:40:26 +00:00
|
|
|
void MainTab::addWeight_clicked()
|
2013-05-01 21:30:34 +00:00
|
|
|
{
|
2014-01-16 04:50:56 +00:00
|
|
|
if (editMode == NONE)
|
2013-09-25 18:05:48 +00:00
|
|
|
enableEdition();
|
2013-05-22 17:52:38 +00:00
|
|
|
weightModel->add();
|
2013-05-01 21:30:34 +00:00
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
void MainTab::reload()
|
|
|
|
{
|
2014-02-11 17:46:14 +00:00
|
|
|
suitModel.updateModel();
|
|
|
|
buddyModel.updateModel();
|
|
|
|
diveMasterModel.updateModel();
|
|
|
|
tagModel.updateModel();
|
2015-07-01 22:31:35 +00:00
|
|
|
LocationInformationModel::instance()->update();
|
2013-04-07 22:20:43 +00:00
|
|
|
}
|
2013-05-18 23:42:59 +00:00
|
|
|
|
2014-05-07 20:00:33 +00:00
|
|
|
// tricky little macro to edit all the selected dives
|
|
|
|
// loop over all dives, for each selected dive do WHAT, but do it
|
|
|
|
// last for the current dive; this is required in case the invocation
|
|
|
|
// wants to compare things to the original value in current_dive like it should
|
2014-07-02 18:50:28 +00:00
|
|
|
#define MODIFY_SELECTED_DIVES(WHAT) \
|
2014-05-07 20:00:33 +00:00
|
|
|
do { \
|
|
|
|
struct dive *mydive = NULL; \
|
2014-05-19 21:37:19 +00:00
|
|
|
int _i; \
|
|
|
|
for_each_dive (_i, mydive) { \
|
2014-06-03 01:13:50 +00:00
|
|
|
if (!mydive->selected || mydive == cd) \
|
2014-05-07 20:00:33 +00:00
|
|
|
continue; \
|
|
|
|
\
|
|
|
|
WHAT; \
|
|
|
|
} \
|
2014-06-03 01:13:50 +00:00
|
|
|
mydive = cd; \
|
2014-05-07 20:00:33 +00:00
|
|
|
WHAT; \
|
2014-06-03 01:13:50 +00:00
|
|
|
mark_divelist_changed(true); \
|
2014-05-07 20:00:33 +00:00
|
|
|
} while (0)
|
|
|
|
|
2014-11-05 14:34:29 +00:00
|
|
|
#define EDIT_TEXT(what) \
|
|
|
|
if (same_string(mydive->what, cd->what) || copyPaste) { \
|
|
|
|
free(mydive->what); \
|
|
|
|
mydive->what = copy_string(displayed_dive.what); \
|
2014-05-07 20:00:33 +00:00
|
|
|
}
|
|
|
|
|
2015-08-26 00:08:05 +00:00
|
|
|
MainTab::EditMode MainTab::getEditMode() const
|
|
|
|
{
|
|
|
|
return editMode;
|
|
|
|
}
|
|
|
|
|
2014-11-05 14:34:29 +00:00
|
|
|
#define EDIT_VALUE(what) \
|
|
|
|
if (mydive->what == cd->what || copyPaste) { \
|
|
|
|
mydive->what = displayed_dive.what; \
|
2014-05-07 20:00:33 +00:00
|
|
|
}
|
|
|
|
|
2015-08-29 23:00:22 +00:00
|
|
|
void MainTab::refreshDisplayedDiveSite()
|
|
|
|
{
|
|
|
|
if (displayed_dive_site.uuid) {
|
|
|
|
copy_dive_site(get_dive_site_by_uuid(displayed_dive_site.uuid), &displayed_dive_site);
|
2015-09-25 17:51:10 +00:00
|
|
|
ui.location->setCurrentDiveSiteUuid(displayed_dive_site.uuid);
|
2015-08-29 23:00:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-15 22:40:27 +00:00
|
|
|
// when this is called we already have updated the current_dive and know that it exists
|
|
|
|
// there is no point in calling this function if there is no current dive
|
2015-09-30 22:58:38 +00:00
|
|
|
uint32_t MainTab::updateDiveSite(uint32_t pickedUuid, int divenr)
|
2015-07-15 22:40:27 +00:00
|
|
|
{
|
|
|
|
struct dive *cd = get_dive(divenr);
|
|
|
|
if (!cd)
|
2015-09-30 22:58:38 +00:00
|
|
|
return 0;
|
2015-07-15 22:40:27 +00:00
|
|
|
|
2015-10-06 18:50:49 +00:00
|
|
|
if (ui.location->text().isEmpty())
|
2015-09-30 22:58:38 +00:00
|
|
|
return 0;
|
2015-09-23 19:55:48 +00:00
|
|
|
|
2015-09-24 03:03:53 +00:00
|
|
|
if (pickedUuid == 0)
|
2015-09-30 22:58:38 +00:00
|
|
|
return 0;
|
2015-09-24 03:03:53 +00:00
|
|
|
|
2015-07-15 22:40:27 +00:00
|
|
|
const uint32_t origUuid = cd->dive_site_uuid;
|
|
|
|
struct dive_site *origDs = get_dive_site_by_uuid(origUuid);
|
2015-09-23 20:05:31 +00:00
|
|
|
struct dive_site *newDs = NULL;
|
2015-10-06 18:59:51 +00:00
|
|
|
bool createdNewDive = false;
|
2015-09-23 20:05:31 +00:00
|
|
|
|
2015-10-06 18:50:49 +00:00
|
|
|
if (pickedUuid == origUuid)
|
2015-09-30 22:58:38 +00:00
|
|
|
return origUuid;
|
2015-09-23 19:55:48 +00:00
|
|
|
|
|
|
|
if (pickedUuid == RECENTLY_ADDED_DIVESITE) {
|
2017-11-26 21:21:58 +00:00
|
|
|
QString name = ui.location->text().isEmpty() ? tr("New dive site") : ui.location->text();
|
|
|
|
pickedUuid = create_dive_site(qPrintable(name), displayed_dive.when);
|
2015-10-06 18:59:51 +00:00
|
|
|
createdNewDive = true;
|
2017-11-26 21:21:58 +00:00
|
|
|
emit diveSiteAdded(name);
|
2015-09-23 19:55:48 +00:00
|
|
|
}
|
|
|
|
|
2015-09-23 20:05:31 +00:00
|
|
|
newDs = get_dive_site_by_uuid(pickedUuid);
|
2015-09-30 22:29:03 +00:00
|
|
|
|
|
|
|
// Copy everything from the displayed_dive_site, so we have the latitude, longitude, notes, etc.
|
|
|
|
// The user *might* be using wrongly the 'choose dive site' just to edit the name of it, sigh.
|
2015-10-06 18:59:51 +00:00
|
|
|
if (origDs) {
|
|
|
|
if(createdNewDive) {
|
|
|
|
copy_dive_site(origDs, newDs);
|
|
|
|
free(newDs->name);
|
2018-02-28 22:37:09 +00:00
|
|
|
newDs->name = copy_qstring(ui.location->text());
|
2015-10-06 18:59:51 +00:00
|
|
|
newDs->uuid = pickedUuid;
|
|
|
|
qDebug() << "Creating and copying dive site";
|
|
|
|
} else if (newDs->latitude.udeg == 0 && newDs->longitude.udeg == 0) {
|
|
|
|
newDs->latitude.udeg = origDs->latitude.udeg;
|
|
|
|
newDs->longitude.udeg = origDs->longitude.udeg;
|
|
|
|
qDebug() << "Copying GPS information";
|
|
|
|
}
|
2015-09-30 22:29:03 +00:00
|
|
|
}
|
2015-09-23 20:05:31 +00:00
|
|
|
|
2015-09-23 19:55:48 +00:00
|
|
|
if (origDs && pickedUuid != origDs->uuid && same_string(origDs->notes, "SubsurfaceWebservice")) {
|
|
|
|
if (!is_dive_site_used(origDs->uuid, false)) {
|
2015-09-30 00:10:29 +00:00
|
|
|
if (verbose)
|
|
|
|
qDebug() << "delete the autogenerated dive site" << origDs->name;
|
2015-09-23 19:55:48 +00:00
|
|
|
delete_dive_site(origDs->uuid);
|
2015-07-13 18:14:46 +00:00
|
|
|
}
|
|
|
|
}
|
2015-09-23 19:55:48 +00:00
|
|
|
|
|
|
|
cd->dive_site_uuid = pickedUuid;
|
|
|
|
qDebug() << "Setting the dive site id on the dive:" << pickedUuid;
|
2015-09-30 22:58:38 +00:00
|
|
|
return pickedUuid;
|
2015-07-13 18:14:46 +00:00
|
|
|
}
|
|
|
|
|
2013-09-19 00:56:53 +00:00
|
|
|
void MainTab::acceptChanges()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2014-07-03 21:45:01 +00:00
|
|
|
int i, addedId = -1;
|
2014-06-03 01:13:50 +00:00
|
|
|
struct dive *d;
|
2014-11-03 17:47:24 +00:00
|
|
|
bool do_replot = false;
|
2015-01-04 18:06:33 +00:00
|
|
|
|
2015-07-13 18:38:46 +00:00
|
|
|
if(ui.location->hasFocus()) {
|
|
|
|
this->setFocus();
|
|
|
|
}
|
|
|
|
|
2015-01-04 18:06:33 +00:00
|
|
|
acceptingEdit = true;
|
2013-09-19 01:38:38 +00:00
|
|
|
tabBar()->setTabIcon(0, QIcon()); // Notes
|
|
|
|
tabBar()->setTabIcon(1, QIcon()); // Equipment
|
2014-08-23 14:49:28 +00:00
|
|
|
ui.dateEdit->setEnabled(true);
|
2013-12-03 20:44:48 +00:00
|
|
|
hideMessage();
|
2013-12-28 01:26:22 +00:00
|
|
|
ui.equipmentTab->setEnabled(true);
|
2014-07-03 21:45:01 +00:00
|
|
|
if (editMode == ADD) {
|
2014-07-03 23:12:47 +00:00
|
|
|
// We need to add the dive we just created to the dive list and select it.
|
2014-07-03 21:45:01 +00:00
|
|
|
// Easy, right?
|
|
|
|
struct dive *added_dive = clone_dive(&displayed_dive);
|
|
|
|
record_dive(added_dive);
|
|
|
|
addedId = added_dive->id;
|
2015-07-15 22:40:27 +00:00
|
|
|
// make sure that the dive site is handled as well
|
2015-09-30 22:58:38 +00:00
|
|
|
updateDiveSite(ui.location->currDiveSiteUuid(), get_idx_by_uniq_id(added_dive->id));
|
2015-07-13 18:14:46 +00:00
|
|
|
|
2014-08-22 00:58:15 +00:00
|
|
|
// unselect everything as far as the UI is concerned and select the new
|
|
|
|
// dive - we'll have to undo/redo this later after we resort the dive_table
|
|
|
|
// but we need the dive selected for the middle part of this function - this
|
|
|
|
// way we can reuse the code used for editing dives
|
2014-07-03 21:45:01 +00:00
|
|
|
MainWindow::instance()->dive_list()->unselectDives();
|
|
|
|
selected_dive = get_divenr(added_dive);
|
|
|
|
amount_selected = 1;
|
2017-05-06 19:46:58 +00:00
|
|
|
// finally, make sure we get the tags
|
|
|
|
saveTags();
|
2014-07-24 15:20:34 +00:00
|
|
|
} else if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2014-07-03 21:45:01 +00:00
|
|
|
/* now figure out if things have changed */
|
2015-01-04 00:06:00 +00:00
|
|
|
if (displayedTrip.notes && !same_string(displayedTrip.notes, currentTrip->notes)) {
|
2014-11-11 19:41:47 +00:00
|
|
|
currentTrip->notes = copy_string(displayedTrip.notes);
|
2014-06-03 01:13:50 +00:00
|
|
|
mark_divelist_changed(true);
|
|
|
|
}
|
2015-01-04 00:06:00 +00:00
|
|
|
if (displayedTrip.location && !same_string(displayedTrip.location, currentTrip->location)) {
|
2014-11-11 19:41:47 +00:00
|
|
|
currentTrip->location = copy_string(displayedTrip.location);
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-06-03 01:13:50 +00:00
|
|
|
}
|
2014-07-30 15:52:11 +00:00
|
|
|
currentTrip = NULL;
|
2014-06-19 16:52:34 +00:00
|
|
|
ui.dateEdit->setEnabled(true);
|
2013-05-19 16:00:57 +00:00
|
|
|
} else {
|
2014-07-31 21:41:30 +00:00
|
|
|
if (editMode == MANUALLY_ADDED_DIVE) {
|
|
|
|
// preserve any changes to the profile
|
|
|
|
free(current_dive->dc.sample);
|
|
|
|
copy_samples(&displayed_dive.dc, ¤t_dive->dc);
|
2015-06-07 13:29:17 +00:00
|
|
|
addedId = displayed_dive.id;
|
2014-07-31 21:41:30 +00:00
|
|
|
}
|
2014-06-03 01:13:50 +00:00
|
|
|
struct dive *cd = current_dive;
|
|
|
|
// now check if something has changed and if yes, edit the selected dives that
|
|
|
|
// were identical with the master dive shown (and mark the divelist as changed)
|
2014-07-02 18:50:28 +00:00
|
|
|
if (!same_string(displayed_dive.suit, cd->suit))
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_TEXT(suit));
|
|
|
|
if (!same_string(displayed_dive.notes, cd->notes))
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_TEXT(notes));
|
2015-05-28 22:10:09 +00:00
|
|
|
if (displayed_dive.rating != cd->rating)
|
2014-07-02 18:50:28 +00:00
|
|
|
MODIFY_SELECTED_DIVES(EDIT_VALUE(rating));
|
|
|
|
if (displayed_dive.visibility != cd->visibility)
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_VALUE(visibility));
|
|
|
|
if (displayed_dive.airtemp.mkelvin != cd->airtemp.mkelvin)
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_VALUE(airtemp.mkelvin));
|
2015-07-09 09:06:11 +00:00
|
|
|
if (displayed_dc->divemode != current_dc->divemode) {
|
|
|
|
MODIFY_SELECTED_DIVES(
|
|
|
|
if (get_dive_dc(mydive, dc_number)->divemode == current_dc->divemode || copyPaste) {
|
|
|
|
get_dive_dc(mydive, dc_number)->divemode = displayed_dc->divemode;
|
|
|
|
}
|
|
|
|
);
|
2016-04-02 20:06:54 +00:00
|
|
|
MODIFY_SELECTED_DIVES(update_setpoint_events(mydive, get_dive_dc(mydive, dc_number)));
|
2015-01-10 22:42:21 +00:00
|
|
|
do_replot = true;
|
2015-01-08 13:42:07 +00:00
|
|
|
}
|
2014-07-02 18:50:28 +00:00
|
|
|
if (displayed_dive.watertemp.mkelvin != cd->watertemp.mkelvin)
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_VALUE(watertemp.mkelvin));
|
|
|
|
if (displayed_dive.when != cd->when) {
|
2014-07-09 16:35:27 +00:00
|
|
|
time_t offset = cd->when - displayed_dive.when;
|
2014-07-02 18:50:28 +00:00
|
|
|
MODIFY_SELECTED_DIVES(mydive->when -= offset;);
|
2013-09-20 23:41:42 +00:00
|
|
|
}
|
2015-07-13 18:14:46 +00:00
|
|
|
|
2015-04-22 16:35:58 +00:00
|
|
|
if (displayed_dive.dive_site_uuid != cd->dive_site_uuid)
|
|
|
|
MODIFY_SELECTED_DIVES(EDIT_VALUE(dive_site_uuid));
|
2014-07-09 16:35:27 +00:00
|
|
|
|
2015-05-11 17:38:50 +00:00
|
|
|
// three text fields are somewhat special and are represented as tags
|
|
|
|
// in the UI - they need somewhat smarter handling
|
|
|
|
saveTaggedStrings();
|
2014-07-08 21:18:33 +00:00
|
|
|
saveTags();
|
2014-07-03 21:45:01 +00:00
|
|
|
|
|
|
|
if (editMode != ADD && cylindersModel->changed) {
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-07-02 18:50:28 +00:00
|
|
|
MODIFY_SELECTED_DIVES(
|
2013-09-25 17:59:11 +00:00
|
|
|
for (int i = 0; i < MAX_CYLINDERS; i++) {
|
2014-06-03 01:13:50 +00:00
|
|
|
if (mydive != cd) {
|
2014-11-05 14:34:29 +00:00
|
|
|
if (same_string(mydive->cylinder[i].type.description, cd->cylinder[i].type.description) || copyPaste) {
|
|
|
|
// if we started out with the same cylinder description (for multi-edit) or if we do copt & paste
|
|
|
|
// make sure that we have the same cylinder type and copy the gasmix, but DON'T copy the start
|
|
|
|
// and end pressures (those are per dive after all)
|
2014-11-04 22:18:27 +00:00
|
|
|
if (!same_string(mydive->cylinder[i].type.description, displayed_dive.cylinder[i].type.description)) {
|
|
|
|
free((void*)mydive->cylinder[i].type.description);
|
|
|
|
mydive->cylinder[i].type.description = copy_string(displayed_dive.cylinder[i].type.description);
|
|
|
|
}
|
|
|
|
mydive->cylinder[i].type.size = displayed_dive.cylinder[i].type.size;
|
|
|
|
mydive->cylinder[i].type.workingpressure = displayed_dive.cylinder[i].type.workingpressure;
|
|
|
|
mydive->cylinder[i].gasmix = displayed_dive.cylinder[i].gasmix;
|
2014-11-17 10:45:14 +00:00
|
|
|
mydive->cylinder[i].cylinder_use = displayed_dive.cylinder[i].cylinder_use;
|
|
|
|
mydive->cylinder[i].depth = displayed_dive.cylinder[i].depth;
|
2014-11-04 22:18:27 +00:00
|
|
|
}
|
2014-06-03 01:13:50 +00:00
|
|
|
}
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
2014-06-03 01:13:50 +00:00
|
|
|
);
|
2014-11-04 22:18:27 +00:00
|
|
|
for (int i = 0; i < MAX_CYLINDERS; i++) {
|
|
|
|
// copy the cylinder but make sure we have our own copy of the strings
|
|
|
|
free((void*)cd->cylinder[i].type.description);
|
2014-07-16 03:29:14 +00:00
|
|
|
cd->cylinder[i] = displayed_dive.cylinder[i];
|
2014-11-04 22:18:27 +00:00
|
|
|
cd->cylinder[i].type.description = copy_string(displayed_dive.cylinder[i].type.description);
|
|
|
|
}
|
2015-04-23 22:24:02 +00:00
|
|
|
/* if cylinders changed we may have changed gas change events
|
2017-11-27 15:59:58 +00:00
|
|
|
* and sensor idx in samples as well
|
2015-04-23 22:24:02 +00:00
|
|
|
* - so far this is ONLY supported for a single selected dive */
|
|
|
|
struct divecomputer *tdc = ¤t_dive->dc;
|
|
|
|
struct divecomputer *sdc = &displayed_dive.dc;
|
|
|
|
while(tdc && sdc) {
|
|
|
|
free_events(tdc->events);
|
|
|
|
copy_events(sdc, tdc);
|
2017-11-27 15:59:58 +00:00
|
|
|
free(tdc->sample);
|
|
|
|
copy_samples(sdc, tdc);
|
2015-04-23 22:24:02 +00:00
|
|
|
tdc = tdc->next;
|
|
|
|
sdc = sdc->next;
|
|
|
|
}
|
2014-11-03 17:47:24 +00:00
|
|
|
do_replot = true;
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (weightModel->changed) {
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-07-02 18:50:28 +00:00
|
|
|
MODIFY_SELECTED_DIVES(
|
2013-10-01 13:52:23 +00:00
|
|
|
for (int i = 0; i < MAX_WEIGHTSYSTEMS; i++) {
|
2014-11-17 10:41:02 +00:00
|
|
|
if (mydive != cd && (copyPaste || same_string(mydive->weightsystem[i].description, cd->weightsystem[i].description))) {
|
2014-07-02 18:50:28 +00:00
|
|
|
mydive->weightsystem[i] = displayed_dive.weightsystem[i];
|
2014-11-17 10:41:02 +00:00
|
|
|
mydive->weightsystem[i].description = copy_string(displayed_dive.weightsystem[i].description);
|
|
|
|
}
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
2014-06-03 01:13:50 +00:00
|
|
|
);
|
2014-11-17 10:41:02 +00:00
|
|
|
for (int i = 0; i < MAX_WEIGHTSYSTEMS; i++) {
|
2014-07-16 03:29:14 +00:00
|
|
|
cd->weightsystem[i] = displayed_dive.weightsystem[i];
|
2014-11-17 10:41:02 +00:00
|
|
|
cd->weightsystem[i].description = copy_string(displayed_dive.weightsystem[i].description);
|
|
|
|
}
|
2014-06-03 01:13:50 +00:00
|
|
|
}
|
2015-07-13 18:14:46 +00:00
|
|
|
|
2015-07-15 22:40:27 +00:00
|
|
|
// update the dive site for the selected dives that had the same dive site as the current dive
|
2015-09-30 00:58:57 +00:00
|
|
|
uint32_t oldUuid = cd->dive_site_uuid;
|
2015-09-30 22:58:38 +00:00
|
|
|
uint32_t newUuid = 0;
|
2015-07-15 22:40:27 +00:00
|
|
|
MODIFY_SELECTED_DIVES(
|
2015-09-30 22:58:38 +00:00
|
|
|
if (mydive->dive_site_uuid == current_dive->dive_site_uuid) {
|
|
|
|
newUuid = updateDiveSite(newUuid == 0 ? ui.location->currDiveSiteUuid() : newUuid, get_idx_by_uniq_id(mydive->id));
|
|
|
|
}
|
2015-07-15 22:40:27 +00:00
|
|
|
);
|
2015-09-30 00:58:57 +00:00
|
|
|
if (!is_dive_site_used(oldUuid, false)) {
|
|
|
|
if (verbose) {
|
|
|
|
struct dive_site *ds = get_dive_site_by_uuid(oldUuid);
|
|
|
|
qDebug() << "delete now unused dive site" << ((ds && ds->name) ? ds->name : "without name");
|
|
|
|
}
|
|
|
|
delete_dive_site(oldUuid);
|
2017-07-15 20:37:02 +00:00
|
|
|
MapWidget::instance()->reload();
|
2015-09-30 00:58:57 +00:00
|
|
|
}
|
2015-07-16 13:40:16 +00:00
|
|
|
// the code above can change the correct uuid for the displayed dive site - and the
|
|
|
|
// code below triggers an update of the display without re-initializing displayed_dive
|
|
|
|
// so let's make sure here that our data is consistent now that we have handled the
|
|
|
|
// dive sites
|
|
|
|
displayed_dive.dive_site_uuid = current_dive->dive_site_uuid;
|
|
|
|
struct dive_site *ds = get_dive_site_by_uuid(displayed_dive.dive_site_uuid);
|
|
|
|
if (ds)
|
|
|
|
copy_dive_site(ds, &displayed_dive_site);
|
|
|
|
|
2014-06-03 01:13:50 +00:00
|
|
|
// each dive that was selected might have had the temperatures in its active divecomputer changed
|
|
|
|
// so re-populate the temperatures - easiest way to do this is by calling fixup_dive
|
|
|
|
for_each_dive (i, d) {
|
2016-04-03 22:31:59 +00:00
|
|
|
if (d->selected) {
|
2014-06-03 01:13:50 +00:00
|
|
|
fixup_dive(d);
|
2016-04-03 22:31:59 +00:00
|
|
|
invalidate_dive_cache(d);
|
|
|
|
}
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
2013-05-19 14:45:01 +00:00
|
|
|
}
|
2015-07-05 22:09:19 +00:00
|
|
|
if (editMode != TRIP && current_dive->divetrip) {
|
2013-11-19 22:16:33 +00:00
|
|
|
current_dive->divetrip->when = current_dive->when;
|
|
|
|
find_new_trip_start_time(current_dive->divetrip);
|
|
|
|
}
|
2013-11-01 15:48:34 +00:00
|
|
|
if (editMode == ADD || editMode == MANUALLY_ADDED_DIVE) {
|
2015-09-11 17:54:05 +00:00
|
|
|
// we just added or edited the dive, let fixup_dive() make
|
2017-03-06 12:27:39 +00:00
|
|
|
// sure we get the max. depth right
|
2015-09-11 17:54:05 +00:00
|
|
|
current_dive->maxdepth.mm = current_dc->maxdepth.mm = 0;
|
2013-09-19 04:33:39 +00:00
|
|
|
fixup_dive(current_dive);
|
2014-08-18 19:12:05 +00:00
|
|
|
set_dive_nr_for_current_dive();
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->showProfile();
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2013-11-09 00:09:46 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::NOTHING);
|
2013-09-19 04:33:39 +00:00
|
|
|
}
|
2014-02-12 14:22:54 +00:00
|
|
|
int scrolledBy = MainWindow::instance()->dive_list()->verticalScrollBar()->sliderPosition();
|
2013-09-25 17:36:59 +00:00
|
|
|
resetPallete();
|
2014-01-16 04:50:56 +00:00
|
|
|
if (editMode == ADD || editMode == MANUALLY_ADDED_DIVE) {
|
2014-08-22 00:58:15 +00:00
|
|
|
// since a newly added dive could be in the middle of the dive_table we need
|
|
|
|
// to resort the dive list and make sure the newly added dive gets selected again
|
2014-07-24 15:20:34 +00:00
|
|
|
sort_table(&dive_table);
|
2014-07-04 04:05:08 +00:00
|
|
|
MainWindow::instance()->dive_list()->reload(DiveTripModel::CURRENT, true);
|
2014-07-03 21:45:01 +00:00
|
|
|
int newDiveNr = get_divenr(get_dive_by_uniq_id(addedId));
|
2014-08-22 00:58:15 +00:00
|
|
|
MainWindow::instance()->dive_list()->unselectDives();
|
2014-07-03 21:45:01 +00:00
|
|
|
MainWindow::instance()->dive_list()->selectDive(newDiveNr, true);
|
2013-12-03 23:25:20 +00:00
|
|
|
editMode = NONE;
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->refreshDisplay();
|
2014-03-09 13:59:31 +00:00
|
|
|
MainWindow::instance()->graphics()->replot();
|
2014-05-21 23:34:06 +00:00
|
|
|
emit addDiveFinished();
|
2013-12-10 05:17:15 +00:00
|
|
|
} else {
|
2013-12-03 23:25:20 +00:00
|
|
|
editMode = NONE;
|
2014-11-03 17:47:24 +00:00
|
|
|
if (do_replot)
|
|
|
|
MainWindow::instance()->graphics()->replot();
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->rememberSelection();
|
2013-12-03 22:34:05 +00:00
|
|
|
sort_table(&dive_table);
|
2014-08-07 17:23:08 +00:00
|
|
|
MainWindow::instance()->refreshDisplay();
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->restoreSelection();
|
2013-12-03 22:34:05 +00:00
|
|
|
}
|
2014-02-19 17:43:34 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::NOTHING);
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->verticalScrollBar()->setSliderPosition(scrolledBy);
|
|
|
|
MainWindow::instance()->dive_list()->setFocus();
|
2018-05-03 17:19:54 +00:00
|
|
|
MainWindow::instance()->exitEditState();
|
2014-07-16 03:33:48 +00:00
|
|
|
cylindersModel->changed = false;
|
|
|
|
weightModel->changed = false;
|
2014-08-04 15:58:21 +00:00
|
|
|
MainWindow::instance()->setEnabledToolbar(true);
|
2015-01-04 18:06:33 +00:00
|
|
|
acceptingEdit = false;
|
2017-10-06 14:51:02 +00:00
|
|
|
ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty());
|
2013-09-25 17:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::resetPallete()
|
|
|
|
{
|
2013-08-16 18:38:18 +00:00
|
|
|
QPalette p;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.buddy->setPalette(p);
|
|
|
|
ui.notes->setPalette(p);
|
|
|
|
ui.location->setPalette(p);
|
|
|
|
ui.divemaster->setPalette(p);
|
|
|
|
ui.suit->setPalette(p);
|
|
|
|
ui.airtemp->setPalette(p);
|
2015-01-01 16:00:46 +00:00
|
|
|
ui.DiveType->setPalette(p);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.watertemp->setPalette(p);
|
2014-06-19 16:52:34 +00:00
|
|
|
ui.dateEdit->setPalette(p);
|
2014-06-26 17:54:16 +00:00
|
|
|
ui.timeEdit->setPalette(p);
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->setPalette(p);
|
2015-10-01 21:11:03 +00:00
|
|
|
ui.diveTripLocation->setPalette(p);
|
2017-02-26 04:06:47 +00:00
|
|
|
ui.duration->setPalette(p);
|
|
|
|
ui.depth->setPalette(p);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2013-09-19 00:56:53 +00:00
|
|
|
void MainTab::rejectChanges()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2013-11-12 05:41:29 +00:00
|
|
|
EditMode lastMode = editMode;
|
2014-07-03 04:33:03 +00:00
|
|
|
|
|
|
|
if (lastMode != NONE && current_dive &&
|
|
|
|
(modified ||
|
|
|
|
memcmp(¤t_dive->cylinder[0], &displayed_dive.cylinder[0], sizeof(cylinder_t) * MAX_CYLINDERS) ||
|
2018-05-10 15:12:06 +00:00
|
|
|
memcmp(¤t_dive->weightsystem[0], &displayed_dive.weightsystem[0], sizeof(weightsystem_t) * MAX_WEIGHTSYSTEMS))) {
|
2014-07-11 08:21:38 +00:00
|
|
|
if (QMessageBox::warning(MainWindow::instance(), TITLE_OR_TEXT(tr("Discard the changes?"),
|
2014-06-04 01:01:00 +00:00
|
|
|
tr("You are about to discard your changes.")),
|
|
|
|
QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Discard) != QMessageBox::Discard) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2014-08-23 14:49:28 +00:00
|
|
|
ui.dateEdit->setEnabled(true);
|
2014-05-28 03:31:18 +00:00
|
|
|
editMode = NONE;
|
2013-09-19 01:38:38 +00:00
|
|
|
tabBar()->setTabIcon(0, QIcon()); // Notes
|
|
|
|
tabBar()->setTabIcon(1, QIcon()); // Equipment
|
2013-12-03 20:44:48 +00:00
|
|
|
hideMessage();
|
2013-09-25 17:36:59 +00:00
|
|
|
resetPallete();
|
2014-07-06 19:36:25 +00:00
|
|
|
// no harm done to call cancelPlan even if we were not in ADD or PLAN mode...
|
|
|
|
DivePlannerPointsModel::instance()->cancelPlan();
|
2014-07-16 22:44:06 +00:00
|
|
|
if(lastMode == ADD)
|
|
|
|
MainWindow::instance()->dive_list()->restoreSelection();
|
2014-07-06 19:36:25 +00:00
|
|
|
|
|
|
|
// now make sure that the correct dive is displayed
|
|
|
|
if (selected_dive >= 0)
|
|
|
|
copy_dive(current_dive, &displayed_dive);
|
|
|
|
else
|
|
|
|
clear_dive(&displayed_dive);
|
|
|
|
updateDiveInfo(selected_dive < 0);
|
2017-04-04 17:21:30 +00:00
|
|
|
|
|
|
|
for (auto widget : extraWidgets) {
|
|
|
|
widget->updateData();
|
|
|
|
}
|
2014-05-02 04:26:18 +00:00
|
|
|
// the user could have edited the location and then canceled the edit
|
|
|
|
// let's get the correct location back in view
|
2017-07-15 20:37:02 +00:00
|
|
|
MapWidget::instance()->centerOnDiveSite(get_dive_site_by_uuid(displayed_dive.dive_site_uuid));
|
2014-07-06 19:36:25 +00:00
|
|
|
// show the profile and dive info
|
|
|
|
MainWindow::instance()->graphics()->replot();
|
2014-08-04 15:58:21 +00:00
|
|
|
MainWindow::instance()->setEnabledToolbar(true);
|
2018-05-03 17:19:54 +00:00
|
|
|
MainWindow::instance()->exitEditState();
|
2014-07-16 03:33:48 +00:00
|
|
|
cylindersModel->changed = false;
|
|
|
|
weightModel->changed = false;
|
2014-07-06 19:36:25 +00:00
|
|
|
cylindersModel->updateDive();
|
|
|
|
weightModel->updateDive();
|
2017-10-06 14:51:02 +00:00
|
|
|
ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty());
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
2013-08-16 18:52:40 +00:00
|
|
|
|
2014-07-03 04:33:03 +00:00
|
|
|
void MainTab::markChangedWidget(QWidget *w)
|
2014-02-28 04:09:57 +00:00
|
|
|
{
|
2013-08-16 18:38:18 +00:00
|
|
|
QPalette p;
|
2013-12-24 11:49:38 +00:00
|
|
|
qreal h, s, l, a;
|
2014-07-03 05:15:08 +00:00
|
|
|
enableEdition();
|
2013-12-24 11:49:38 +00:00
|
|
|
qApp->palette().color(QPalette::Text).getHslF(&h, &s, &l, &a);
|
2014-02-28 04:09:57 +00:00
|
|
|
p.setBrush(QPalette::Base, (l <= 0.3) ? QColor(Qt::yellow).lighter() : (l <= 0.6) ? QColor(Qt::yellow).light() : /* else */ QColor(Qt::yellow).darker(300));
|
2013-08-16 18:52:40 +00:00
|
|
|
w->setPalette(p);
|
2015-03-09 18:12:48 +00:00
|
|
|
modified = true;
|
2013-08-16 18:52:40 +00:00
|
|
|
}
|
2013-05-18 23:42:59 +00:00
|
|
|
|
2014-01-15 20:35:14 +00:00
|
|
|
void MainTab::on_buddy_textChanged()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2015-02-01 19:25:27 +00:00
|
|
|
|
2018-02-25 12:51:41 +00:00
|
|
|
if (same_string(displayed_dive.buddy, qPrintable(ui.buddy->toPlainText())))
|
2015-02-01 19:25:27 +00:00
|
|
|
return;
|
|
|
|
|
2014-03-26 20:09:01 +00:00
|
|
|
QStringList text_list = ui.buddy->toPlainText().split(",", QString::SkipEmptyParts);
|
|
|
|
for (int i = 0; i < text_list.size(); i++)
|
|
|
|
text_list[i] = text_list[i].trimmed();
|
|
|
|
QString text = text_list.join(", ");
|
2014-07-03 04:31:55 +00:00
|
|
|
free(displayed_dive.buddy);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.buddy = copy_qstring(text);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.buddy);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2014-01-19 17:30:47 +00:00
|
|
|
void MainTab::on_divemaster_textChanged()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2015-02-01 19:25:27 +00:00
|
|
|
|
2018-02-25 12:51:41 +00:00
|
|
|
if (same_string(displayed_dive.divemaster, qPrintable(ui.divemaster->toPlainText())))
|
2015-02-01 19:25:27 +00:00
|
|
|
return;
|
|
|
|
|
2014-03-26 20:09:01 +00:00
|
|
|
QStringList text_list = ui.divemaster->toPlainText().split(",", QString::SkipEmptyParts);
|
|
|
|
for (int i = 0; i < text_list.size(); i++)
|
|
|
|
text_list[i] = text_list[i].trimmed();
|
|
|
|
QString text = text_list.join(", ");
|
2014-07-03 04:31:55 +00:00
|
|
|
free(displayed_dive.divemaster);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.divemaster = copy_qstring(text);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.divemaster);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2017-02-04 21:59:20 +00:00
|
|
|
void MainTab::on_duration_textChanged(const QString &text)
|
|
|
|
{
|
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
|
|
|
return;
|
2017-02-26 05:00:09 +00:00
|
|
|
// parse this
|
|
|
|
MainWindow::instance()->graphics()->setReplot(false);
|
|
|
|
if (!isEditing())
|
|
|
|
enableEdition();
|
|
|
|
displayed_dive.dc.duration.seconds = parseDurationToSeconds(text);
|
|
|
|
displayed_dive.duration = displayed_dive.dc.duration;
|
|
|
|
displayed_dive.dc.meandepth.mm = 0;
|
|
|
|
displayed_dive.dc.samples = 0;
|
|
|
|
DivePlannerPointsModel::instance()->loadFromDive(&displayed_dive);
|
|
|
|
markChangedWidget(ui.duration);
|
|
|
|
MainWindow::instance()->graphics()->setReplot(true);
|
|
|
|
MainWindow::instance()->graphics()->plotDive();
|
|
|
|
|
2017-02-04 21:59:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::on_depth_textChanged(const QString &text)
|
|
|
|
{
|
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
|
|
|
return;
|
2017-02-26 04:06:47 +00:00
|
|
|
// don't replot until we set things up the way we want them
|
|
|
|
MainWindow::instance()->graphics()->setReplot(false);
|
|
|
|
if (!isEditing())
|
|
|
|
enableEdition();
|
|
|
|
displayed_dive.dc.maxdepth.mm = parseLengthToMm(text);
|
|
|
|
displayed_dive.maxdepth = displayed_dive.dc.maxdepth;
|
|
|
|
displayed_dive.dc.meandepth.mm = 0;
|
|
|
|
displayed_dive.dc.samples = 0;
|
|
|
|
DivePlannerPointsModel::instance()->loadFromDive(&displayed_dive);
|
2017-02-04 21:59:20 +00:00
|
|
|
markChangedWidget(ui.depth);
|
2017-02-26 04:06:47 +00:00
|
|
|
MainWindow::instance()->graphics()->setReplot(true);
|
|
|
|
MainWindow::instance()->graphics()->plotDive();
|
2017-02-04 21:59:20 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_airtemp_textChanged(const QString &text)
|
2013-09-20 23:41:42 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2014-07-02 18:50:28 +00:00
|
|
|
displayed_dive.airtemp.mkelvin = parseTemperatureToMkelvin(text);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.airtemp);
|
2014-04-13 16:19:15 +00:00
|
|
|
validate_temp_field(ui.airtemp, text);
|
2013-09-20 23:41:42 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 13:42:07 +00:00
|
|
|
void MainTab::divetype_Changed(int index)
|
2015-01-01 16:00:46 +00:00
|
|
|
{
|
|
|
|
if (editMode == IGNORE)
|
|
|
|
return;
|
2018-05-08 14:24:51 +00:00
|
|
|
displayed_dc->divemode = (enum divemode_t) index;
|
2016-04-02 20:06:54 +00:00
|
|
|
update_setpoint_events(&displayed_dive, displayed_dc);
|
2015-01-01 16:00:46 +00:00
|
|
|
markChangedWidget(ui.DiveType);
|
2015-01-30 10:16:55 +00:00
|
|
|
MainWindow::instance()->graphics()->recalcCeiling();
|
2015-01-01 16:00:46 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_watertemp_textChanged(const QString &text)
|
2013-09-20 23:41:42 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2014-07-02 18:50:28 +00:00
|
|
|
displayed_dive.watertemp.mkelvin = parseTemperatureToMkelvin(text);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.watertemp);
|
2014-04-13 16:19:15 +00:00
|
|
|
validate_temp_field(ui.watertemp, text);
|
2013-09-20 23:41:42 +00:00
|
|
|
}
|
|
|
|
|
2014-05-22 18:40:22 +00:00
|
|
|
void MainTab::validate_temp_field(QLineEdit *tempField, const QString &text)
|
2014-04-13 16:19:15 +00:00
|
|
|
{
|
|
|
|
static bool missing_unit = false;
|
|
|
|
static bool missing_precision = false;
|
|
|
|
if (!text.contains(QRegExp("^[-+]{0,1}[0-9]+([,.][0-9]+){0,1}(°[CF]){0,1}$")) &&
|
|
|
|
!text.isEmpty() &&
|
|
|
|
!text.contains(QRegExp("^[-+]$"))) {
|
|
|
|
if (text.contains(QRegExp("^[-+]{0,1}[0-9]+([,.][0-9]+){0,1}(°)$")) && !missing_unit) {
|
|
|
|
if (!missing_unit) {
|
|
|
|
missing_unit = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (text.contains(QRegExp("^[-+]{0,1}[0-9]+([,.]){0,1}(°[CF]){0,1}$")) && !missing_precision) {
|
|
|
|
if (!missing_precision) {
|
|
|
|
missing_precision = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QPalette p;
|
|
|
|
p.setBrush(QPalette::Base, QColor(Qt::red).lighter());
|
|
|
|
tempField->setPalette(p);
|
|
|
|
} else {
|
|
|
|
missing_unit = false;
|
|
|
|
missing_precision = false;
|
|
|
|
}
|
|
|
|
}
|
2014-05-06 20:56:46 +00:00
|
|
|
|
2014-06-26 17:57:25 +00:00
|
|
|
void MainTab::on_dateEdit_dateChanged(const QDate &date)
|
2013-09-20 23:41:42 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2014-10-21 00:25:13 +00:00
|
|
|
markChangedWidget(ui.dateEdit);
|
2016-04-28 23:31:37 +00:00
|
|
|
QDateTime dateTime = QDateTime::fromMSecsSinceEpoch(1000*displayed_dive.when, Qt::UTC);
|
2014-06-26 17:57:25 +00:00
|
|
|
dateTime.setTimeSpec(Qt::UTC);
|
|
|
|
dateTime.setDate(date);
|
2014-10-21 00:25:13 +00:00
|
|
|
DivePlannerPointsModel::instance()->getDiveplan().when = displayed_dive.when = dateTime.toTime_t();
|
2014-10-21 00:36:24 +00:00
|
|
|
emit dateTimeChanged();
|
2013-09-20 23:41:42 +00:00
|
|
|
}
|
|
|
|
|
2014-06-26 17:54:16 +00:00
|
|
|
void MainTab::on_timeEdit_timeChanged(const QTime &time)
|
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-26 17:54:16 +00:00
|
|
|
return;
|
2014-10-21 00:25:13 +00:00
|
|
|
markChangedWidget(ui.timeEdit);
|
2016-04-28 23:31:37 +00:00
|
|
|
QDateTime dateTime = QDateTime::fromMSecsSinceEpoch(1000*displayed_dive.when, Qt::UTC);
|
2014-06-26 17:57:25 +00:00
|
|
|
dateTime.setTimeSpec(Qt::UTC);
|
2014-06-26 17:54:16 +00:00
|
|
|
dateTime.setTime(time);
|
2014-10-21 00:25:13 +00:00
|
|
|
DivePlannerPointsModel::instance()->getDiveplan().when = displayed_dive.when = dateTime.toTime_t();
|
2014-10-21 00:36:24 +00:00
|
|
|
emit dateTimeChanged();
|
2014-06-26 17:54:16 +00:00
|
|
|
}
|
|
|
|
|
2014-05-06 20:56:46 +00:00
|
|
|
// changing the tags on multiple dives is semantically strange - what's the right thing to do?
|
2015-05-02 19:18:12 +00:00
|
|
|
// here's what I think... add the tags that were added to the displayed dive and remove the tags
|
|
|
|
// that were removed from it
|
2013-11-02 01:20:02 +00:00
|
|
|
void MainTab::saveTags()
|
|
|
|
{
|
2014-06-03 01:13:50 +00:00
|
|
|
struct dive *cd = current_dive;
|
2015-05-02 19:18:12 +00:00
|
|
|
struct tag_entry *added_list = NULL;
|
|
|
|
struct tag_entry *removed_list = NULL;
|
|
|
|
struct tag_entry *tl;
|
|
|
|
|
2014-10-30 20:01:09 +00:00
|
|
|
taglist_free(displayed_dive.tag_list);
|
|
|
|
displayed_dive.tag_list = NULL;
|
2014-07-10 16:39:51 +00:00
|
|
|
Q_FOREACH (const QString& tag, ui.tagWidget->getBlockStringList())
|
2018-02-25 12:51:41 +00:00
|
|
|
taglist_add_tag(&displayed_dive.tag_list, qPrintable(tag));
|
2014-07-10 16:39:51 +00:00
|
|
|
taglist_cleanup(&displayed_dive.tag_list);
|
2015-05-02 19:18:12 +00:00
|
|
|
|
|
|
|
// figure out which tags were added and which tags were removed
|
|
|
|
added_list = taglist_added(cd->tag_list, displayed_dive.tag_list);
|
|
|
|
removed_list = taglist_added(displayed_dive.tag_list, cd->tag_list);
|
|
|
|
// dump_taglist("added tags:", added_list);
|
|
|
|
// dump_taglist("removed tags:", removed_list);
|
|
|
|
|
2015-02-10 14:31:37 +00:00
|
|
|
// we need to check if the tags were changed before just overwriting them
|
2015-05-02 19:18:12 +00:00
|
|
|
if (added_list == NULL && removed_list == NULL)
|
2015-02-10 14:31:37 +00:00
|
|
|
return;
|
2015-05-02 19:18:12 +00:00
|
|
|
|
2014-07-02 18:50:28 +00:00
|
|
|
MODIFY_SELECTED_DIVES(
|
2015-05-02 19:18:12 +00:00
|
|
|
// create a new tag list and all the existing tags that were not
|
|
|
|
// removed and then all the added tags
|
|
|
|
struct tag_entry *new_tag_list;
|
|
|
|
new_tag_list = NULL;
|
|
|
|
tl = mydive->tag_list;
|
|
|
|
while (tl) {
|
|
|
|
if (!taglist_contains(removed_list, tl->tag->name))
|
|
|
|
taglist_add_tag(&new_tag_list, tl->tag->name);
|
|
|
|
tl = tl->next;
|
|
|
|
}
|
|
|
|
tl = added_list;
|
|
|
|
while (tl) {
|
|
|
|
taglist_add_tag(&new_tag_list, tl->tag->name);
|
|
|
|
tl = tl->next;
|
|
|
|
}
|
2014-05-22 18:40:22 +00:00
|
|
|
taglist_free(mydive->tag_list);
|
2015-05-02 19:18:12 +00:00
|
|
|
mydive->tag_list = new_tag_list;
|
2014-07-08 21:45:20 +00:00
|
|
|
);
|
2015-06-22 03:24:07 +00:00
|
|
|
taglist_free(added_list);
|
|
|
|
taglist_free(removed_list);
|
2013-11-02 01:20:02 +00:00
|
|
|
}
|
|
|
|
|
2015-05-11 17:38:50 +00:00
|
|
|
// buddy and divemaster are represented in the UI just like the tags, but the internal
|
|
|
|
// representation is just a string (with commas as delimiters). So we need to do the same
|
|
|
|
// thing we did for tags, just differently
|
|
|
|
void MainTab::saveTaggedStrings()
|
|
|
|
{
|
|
|
|
QStringList addedList, removedList;
|
|
|
|
struct dive *cd = current_dive;
|
|
|
|
|
|
|
|
diffTaggedStrings(cd->buddy, displayed_dive.buddy, addedList, removedList);
|
|
|
|
MODIFY_SELECTED_DIVES(
|
|
|
|
QStringList oldList = QString(mydive->buddy).split(QRegExp("\\s*,\\s*"), QString::SkipEmptyParts);
|
|
|
|
QString newString;
|
|
|
|
QString comma;
|
|
|
|
Q_FOREACH (const QString tag, oldList) {
|
|
|
|
if (!removedList.contains(tag, Qt::CaseInsensitive)) {
|
|
|
|
newString += comma + tag;
|
|
|
|
comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Q_FOREACH (const QString tag, addedList) {
|
|
|
|
if (!oldList.contains(tag, Qt::CaseInsensitive)) {
|
|
|
|
newString += comma + tag;
|
|
|
|
comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(mydive->buddy);
|
2018-02-28 22:37:09 +00:00
|
|
|
mydive->buddy = copy_qstring(newString);
|
2015-05-11 17:38:50 +00:00
|
|
|
);
|
|
|
|
addedList.clear();
|
|
|
|
removedList.clear();
|
|
|
|
diffTaggedStrings(cd->divemaster, displayed_dive.divemaster, addedList, removedList);
|
|
|
|
MODIFY_SELECTED_DIVES(
|
|
|
|
QStringList oldList = QString(mydive->divemaster).split(QRegExp("\\s*,\\s*"), QString::SkipEmptyParts);
|
|
|
|
QString newString;
|
|
|
|
QString comma;
|
|
|
|
Q_FOREACH (const QString tag, oldList) {
|
|
|
|
if (!removedList.contains(tag, Qt::CaseInsensitive)) {
|
|
|
|
newString += comma + tag;
|
|
|
|
comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Q_FOREACH (const QString tag, addedList) {
|
|
|
|
if (!oldList.contains(tag, Qt::CaseInsensitive)) {
|
|
|
|
newString += comma + tag;
|
|
|
|
comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(mydive->divemaster);
|
2018-02-28 22:37:09 +00:00
|
|
|
mydive->divemaster = copy_qstring(newString);
|
2015-05-11 17:38:50 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::diffTaggedStrings(QString currentString, QString displayedString, QStringList &addedList, QStringList &removedList)
|
|
|
|
{
|
|
|
|
QStringList displayedList, currentList;
|
|
|
|
currentList = currentString.split(',', QString::SkipEmptyParts);
|
|
|
|
displayedList = displayedString.split(',', QString::SkipEmptyParts);
|
|
|
|
Q_FOREACH ( const QString tag, currentList) {
|
|
|
|
if (!displayedList.contains(tag, Qt::CaseInsensitive))
|
|
|
|
removedList << tag.trimmed();
|
|
|
|
}
|
|
|
|
Q_FOREACH (const QString tag, displayedList) {
|
|
|
|
if (!currentList.contains(tag, Qt::CaseInsensitive))
|
|
|
|
addedList << tag.trimmed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-02 01:20:02 +00:00
|
|
|
void MainTab::on_tagWidget_textChanged()
|
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2015-02-01 19:25:27 +00:00
|
|
|
|
2018-04-09 08:09:34 +00:00
|
|
|
if (get_taglist_string(displayed_dive.tag_list) == ui.tagWidget->toPlainText())
|
2015-02-01 19:25:27 +00:00
|
|
|
return;
|
|
|
|
|
2014-07-08 21:45:20 +00:00
|
|
|
markChangedWidget(ui.tagWidget);
|
2013-11-02 01:20:02 +00:00
|
|
|
}
|
|
|
|
|
2015-06-27 13:50:20 +00:00
|
|
|
void MainTab::on_location_textChanged()
|
|
|
|
{
|
|
|
|
if (editMode == IGNORE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// we don't want to act on the edit until editing is finished,
|
|
|
|
// but we want to mark the field so it's obvious it is being edited
|
|
|
|
QString currentLocation;
|
|
|
|
struct dive_site *ds = get_dive_site_by_uuid(displayed_dive.dive_site_uuid);
|
|
|
|
if (ds)
|
|
|
|
currentLocation = ds->name;
|
|
|
|
if (ui.location->text() != currentLocation)
|
|
|
|
markChangedWidget(ui.location);
|
|
|
|
}
|
|
|
|
|
2015-09-23 17:46:29 +00:00
|
|
|
void MainTab::on_location_diveSiteSelected()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-05-07 21:51:39 +00:00
|
|
|
return;
|
2015-06-04 01:32:13 +00:00
|
|
|
|
2015-06-27 01:07:04 +00:00
|
|
|
if (ui.location->text().isEmpty()) {
|
|
|
|
displayed_dive.dive_site_uuid = 0;
|
|
|
|
markChangedWidget(ui.location);
|
|
|
|
emit diveSiteChanged(0);
|
2015-06-26 23:42:14 +00:00
|
|
|
return;
|
2015-09-23 17:46:29 +00:00
|
|
|
} else {
|
|
|
|
if (ui.location->currDiveSiteUuid() != displayed_dive.dive_site_uuid) {
|
|
|
|
markChangedWidget(ui.location);
|
|
|
|
} else {
|
|
|
|
QPalette p;
|
|
|
|
ui.location->setPalette(p);
|
|
|
|
}
|
2015-06-27 01:07:04 +00:00
|
|
|
}
|
2015-10-01 20:52:00 +00:00
|
|
|
}
|
|
|
|
|
2015-10-01 21:11:03 +00:00
|
|
|
void MainTab::on_diveTripLocation_textEdited(const QString& text)
|
2015-10-01 20:52:00 +00:00
|
|
|
{
|
2014-07-30 15:52:11 +00:00
|
|
|
if (currentTrip) {
|
|
|
|
free(displayedTrip.location);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayedTrip.location = copy_qstring(text);
|
2015-10-01 20:52:00 +00:00
|
|
|
markChangedWidget(ui.diveTripLocation);
|
2015-06-25 17:24:22 +00:00
|
|
|
}
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_suit_textChanged(const QString &text)
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2014-06-03 01:13:50 +00:00
|
|
|
return;
|
2014-07-03 04:31:55 +00:00
|
|
|
free(displayed_dive.suit);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.suit = copy_qstring(text);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.suit);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::on_notes_textChanged()
|
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (editMode == IGNORE || acceptingEdit == true)
|
2013-08-16 16:31:52 +00:00
|
|
|
return;
|
2014-07-30 15:52:11 +00:00
|
|
|
if (currentTrip) {
|
2018-02-25 12:51:41 +00:00
|
|
|
if (same_string(displayedTrip.notes, qPrintable(ui.notes->toPlainText())))
|
2015-01-30 20:27:16 +00:00
|
|
|
return;
|
2014-07-30 15:52:11 +00:00
|
|
|
free(displayedTrip.notes);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayedTrip.notes = copy_qstring(ui.notes->toPlainText());
|
2014-07-30 15:52:11 +00:00
|
|
|
} else {
|
2018-02-25 12:51:41 +00:00
|
|
|
if (same_string(displayed_dive.notes, qPrintable(ui.notes->toPlainText())))
|
2015-01-30 20:27:16 +00:00
|
|
|
return;
|
2014-07-30 15:52:11 +00:00
|
|
|
free(displayed_dive.notes);
|
|
|
|
if (ui.notes->toHtml().indexOf("<table") != -1)
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.notes = copy_qstring(ui.notes->toHtml());
|
2014-07-30 15:52:11 +00:00
|
|
|
else
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.notes = copy_qstring(ui.notes->toPlainText());
|
2014-07-30 15:52:11 +00:00
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.notes);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::on_rating_valueChanged(int value)
|
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (acceptingEdit == true)
|
|
|
|
return;
|
2014-07-03 04:33:03 +00:00
|
|
|
if (displayed_dive.rating != value) {
|
|
|
|
displayed_dive.rating = value;
|
|
|
|
modified = true;
|
2014-07-03 05:15:08 +00:00
|
|
|
enableEdition();
|
2014-07-03 04:33:03 +00:00
|
|
|
}
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
2013-05-20 13:25:16 +00:00
|
|
|
|
|
|
|
void MainTab::on_visibility_valueChanged(int value)
|
|
|
|
{
|
2015-01-04 18:06:33 +00:00
|
|
|
if (acceptingEdit == true)
|
|
|
|
return;
|
2014-07-03 04:33:03 +00:00
|
|
|
if (displayed_dive.visibility != value) {
|
|
|
|
displayed_dive.visibility = value;
|
|
|
|
modified = true;
|
2014-07-03 05:15:08 +00:00
|
|
|
enableEdition();
|
2014-07-03 04:33:03 +00:00
|
|
|
}
|
2013-05-20 13:25:16 +00:00
|
|
|
}
|
2013-06-16 17:36:23 +00:00
|
|
|
|
2014-07-03 04:33:03 +00:00
|
|
|
#undef MODIFY_SELECTED_DIVES
|
2014-05-06 20:56:46 +00:00
|
|
|
#undef EDIT_TEXT
|
|
|
|
#undef EDIT_VALUE
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::editCylinderWidget(const QModelIndex &index)
|
2013-07-17 15:13:04 +00:00
|
|
|
{
|
2014-07-01 17:18:21 +00:00
|
|
|
// we need a local copy or bad things happen when enableEdition() is called
|
|
|
|
QModelIndex editIndex = index;
|
2014-03-15 22:14:59 +00:00
|
|
|
if (cylindersModel->changed && editMode == NONE) {
|
2013-09-25 17:11:38 +00:00
|
|
|
enableEdition();
|
2014-03-15 22:14:59 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-07-01 17:18:21 +00:00
|
|
|
if (editIndex.isValid() && editIndex.column() != CylindersModel::REMOVE) {
|
2014-03-15 22:14:59 +00:00
|
|
|
if (editMode == NONE)
|
|
|
|
enableEdition();
|
2014-07-01 17:18:21 +00:00
|
|
|
ui.cylinders->edit(editIndex);
|
2014-03-15 22:14:59 +00:00
|
|
|
}
|
2013-07-17 15:13:04 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::editWeightWidget(const QModelIndex &index)
|
2013-07-17 15:13:04 +00:00
|
|
|
{
|
2013-09-25 17:11:38 +00:00
|
|
|
if (editMode == NONE)
|
|
|
|
enableEdition();
|
|
|
|
|
2013-09-02 19:21:08 +00:00
|
|
|
if (index.isValid() && index.column() != WeightModel::REMOVE)
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.weights->edit(index);
|
2013-06-16 17:36:23 +00:00
|
|
|
}
|
2013-10-07 18:48:14 +00:00
|
|
|
|
2014-06-03 22:29:28 +00:00
|
|
|
void MainTab::escDetected()
|
|
|
|
{
|
|
|
|
if (editMode != NONE)
|
|
|
|
rejectChanges();
|
|
|
|
}
|
2014-06-27 12:17:33 +00:00
|
|
|
|
2017-04-04 17:21:30 +00:00
|
|
|
void MainTab::clearTabs() {
|
|
|
|
for (auto widget : extraWidgets) {
|
|
|
|
widget->clear();
|
2015-10-20 19:08:59 +00:00
|
|
|
}
|
2017-04-04 17:21:30 +00:00
|
|
|
clearEquipment();
|
2015-10-20 20:36:59 +00:00
|
|
|
}
|
|
|
|
|
2014-08-17 14:26:09 +00:00
|
|
|
#define SHOW_SELECTIVE(_component) \
|
|
|
|
if (what._component) \
|
|
|
|
ui._component->setText(displayed_dive._component);
|
|
|
|
|
|
|
|
void MainTab::showAndTriggerEditSelective(struct dive_components what)
|
|
|
|
{
|
|
|
|
// take the data in our copyPasteDive and apply it to selected dives
|
|
|
|
enableEdition();
|
2014-11-05 14:34:29 +00:00
|
|
|
copyPaste = true;
|
2014-08-17 14:26:09 +00:00
|
|
|
SHOW_SELECTIVE(buddy);
|
|
|
|
SHOW_SELECTIVE(divemaster);
|
|
|
|
SHOW_SELECTIVE(suit);
|
|
|
|
if (what.notes) {
|
|
|
|
QString tmp(displayed_dive.notes);
|
2017-11-28 17:10:37 +00:00
|
|
|
if (tmp.contains("<table")) {
|
|
|
|
tmp.replace(QString("\n"), QString("<br>"));
|
2014-08-17 14:26:09 +00:00
|
|
|
ui.notes->setHtml(tmp);
|
2017-11-28 17:10:37 +00:00
|
|
|
} else {
|
2014-08-17 14:26:09 +00:00
|
|
|
ui.notes->setPlainText(tmp);
|
2017-11-28 17:10:37 +00:00
|
|
|
}
|
2014-08-17 14:26:09 +00:00
|
|
|
}
|
|
|
|
if (what.rating)
|
|
|
|
ui.rating->setCurrentStars(displayed_dive.rating);
|
|
|
|
if (what.visibility)
|
|
|
|
ui.visibility->setCurrentStars(displayed_dive.visibility);
|
2015-02-13 04:58:03 +00:00
|
|
|
if (what.divesite)
|
2015-09-25 17:51:10 +00:00
|
|
|
ui.location->setCurrentDiveSiteUuid(displayed_dive.dive_site_uuid);
|
2014-08-17 14:26:09 +00:00
|
|
|
if (what.tags) {
|
2018-04-09 08:09:34 +00:00
|
|
|
ui.tagWidget->setText(get_taglist_string(displayed_dive.tag_list));
|
2014-08-17 14:26:09 +00:00
|
|
|
}
|
|
|
|
if (what.cylinders) {
|
|
|
|
cylindersModel->updateDive();
|
|
|
|
cylindersModel->changed = true;
|
|
|
|
}
|
|
|
|
if (what.weights) {
|
|
|
|
weightModel->updateDive();
|
|
|
|
weightModel->changed = true;
|
|
|
|
}
|
|
|
|
}
|