2013-04-14 03:44:02 +00:00
|
|
|
/*
|
|
|
|
* maintab.cpp
|
|
|
|
*
|
|
|
|
* classes for the "notebook" area of the main window of Subsurface
|
|
|
|
*
|
|
|
|
*/
|
2013-04-07 22:20:43 +00:00
|
|
|
#include "maintab.h"
|
2013-05-19 03:09:36 +00:00
|
|
|
#include "mainwindow.h"
|
2013-05-07 03:36:37 +00:00
|
|
|
#include "../helpers.h"
|
|
|
|
#include "../statistics.h"
|
2013-05-19 03:09:36 +00:00
|
|
|
#include "divelistview.h"
|
2013-05-22 17:11:49 +00:00
|
|
|
#include "modeldelegates.h"
|
2013-06-05 02:39:40 +00:00
|
|
|
#include "globe.h"
|
2013-08-13 13:49:59 +00:00
|
|
|
#include "completionmodels.h"
|
2013-09-19 04:33:39 +00:00
|
|
|
#include "diveplanner.h"
|
2013-11-19 22:16:33 +00:00
|
|
|
#include "divelist.h"
|
2013-10-05 16:48:26 +00:00
|
|
|
#include "qthelper.h"
|
2014-03-07 15:42:13 +00:00
|
|
|
#include "display.h"
|
2013-04-13 13:17:59 +00:00
|
|
|
|
|
|
|
#include <QLabel>
|
2013-08-13 13:49:59 +00:00
|
|
|
#include <QCompleter>
|
2013-05-18 23:42:59 +00:00
|
|
|
#include <QDebug>
|
2013-06-14 16:17:46 +00:00
|
|
|
#include <QSet>
|
2013-11-28 11:17:30 +00:00
|
|
|
#include <QSettings>
|
2013-09-02 19:21:08 +00:00
|
|
|
#include <QTableView>
|
2013-08-16 18:38:18 +00:00
|
|
|
#include <QPalette>
|
2013-12-12 01:08:56 +00:00
|
|
|
#include <QScrollBar>
|
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)),
|
|
|
|
cylindersModel(CylindersModel::instance()),
|
|
|
|
editMode(NONE)
|
2013-04-07 22:20:43 +00:00
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.setupUi(this);
|
2014-02-09 18:49:15 +00:00
|
|
|
|
|
|
|
memset(&multiEditEquipmentPlaceholder, 0, sizeof(multiEditEquipmentPlaceholder));
|
|
|
|
|
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
|
|
|
|
2013-12-02 17:33:00 +00:00
|
|
|
QAction *action = new QAction(tr("Save"), this);
|
|
|
|
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
|
|
|
|
|
|
|
action = new QAction(tr("Cancel"), this);
|
|
|
|
connect(action, SIGNAL(triggered(bool)), this, SLOT(rejectChanges()));
|
2013-12-03 20:44:48 +00:00
|
|
|
addMessageAction(action);
|
2013-12-02 17:33:00 +00:00
|
|
|
|
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.location->installEventFilter(this);
|
|
|
|
ui.coordinates->installEventFilter(this);
|
|
|
|
ui.divemaster->installEventFilter(this);
|
|
|
|
ui.buddy->installEventFilter(this);
|
|
|
|
ui.suit->installEventFilter(this);
|
|
|
|
ui.notes->viewport()->installEventFilter(this);
|
|
|
|
ui.rating->installEventFilter(this);
|
|
|
|
ui.visibility->installEventFilter(this);
|
|
|
|
ui.airtemp->installEventFilter(this);
|
|
|
|
ui.watertemp->installEventFilter(this);
|
|
|
|
ui.dateTimeEdit->installEventFilter(this);
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->installEventFilter(this);
|
2013-10-03 18:54:25 +00:00
|
|
|
|
|
|
|
QList<QObject *> statisticsTabWidgets = ui.statisticsTab->children();
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (QObject *obj, statisticsTabWidgets) {
|
2014-02-28 04:09:57 +00:00
|
|
|
QLabel *label = qobject_cast<QLabel *>(obj);
|
2013-05-10 22:56:05 +00:00
|
|
|
if (label)
|
|
|
|
label->setAlignment(Qt::AlignHCenter);
|
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.cylinders->setTitle(tr("Cylinders"));
|
|
|
|
ui.cylinders->setBtnToolTip(tr("Add Cylinder"));
|
|
|
|
connect(ui.cylinders, SIGNAL(addButtonClicked()), this, SLOT(addCylinder_clicked()));
|
|
|
|
|
|
|
|
ui.weights->setTitle(tr("Weights"));
|
|
|
|
ui.weights->setBtnToolTip(tr("Add Weight System"));
|
|
|
|
connect(ui.weights, SIGNAL(addButtonClicked()), this, SLOT(addWeight_clicked()));
|
|
|
|
|
|
|
|
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));
|
|
|
|
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.location = new QCompleter(&locationModel, ui.location);
|
|
|
|
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.location->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.location->setCompleter(completers.location);
|
|
|
|
ui.suit->setCompleter(completers.suit);
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->setCompleter(completers.tags);
|
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);
|
|
|
|
ui.scrollArea_3->viewport()->setPalette(p);
|
|
|
|
ui.scrollArea_4->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
|
|
|
}
|
2013-11-28 11:17:30 +00:00
|
|
|
ui.cylinders->view()->horizontalHeader()->setContextMenuPolicy(Qt::ActionsContextMenu);
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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.diveEquipmentMessage->addAction(action);
|
|
|
|
ui.diveNotesMessage->addAction(action);
|
|
|
|
ui.diveInfoMessage->addAction(action);
|
|
|
|
ui.diveStatisticsMessage->addAction(action);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::hideMessage()
|
|
|
|
{
|
|
|
|
ui.diveNotesMessage->animatedHide();
|
|
|
|
ui.diveEquipmentMessage->animatedHide();
|
|
|
|
ui.diveInfoMessage->animatedHide();
|
|
|
|
ui.diveStatisticsMessage->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);
|
|
|
|
ui.diveEquipmentMessage->setCloseButtonVisible(false);
|
|
|
|
ui.diveInfoMessage->setCloseButtonVisible(false);
|
|
|
|
ui.diveStatisticsMessage->setCloseButtonVisible(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::displayMessage(QString str)
|
|
|
|
{
|
|
|
|
ui.diveNotesMessage->setText(str);
|
|
|
|
ui.diveNotesMessage->animatedShow();
|
|
|
|
ui.diveEquipmentMessage->setText(str);
|
|
|
|
ui.diveEquipmentMessage->animatedShow();
|
|
|
|
ui.diveInfoMessage->setText(str);
|
|
|
|
ui.diveInfoMessage->animatedShow();
|
|
|
|
ui.diveStatisticsMessage->setText(str);
|
|
|
|
ui.diveStatisticsMessage->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-03-17 21:52:42 +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-03-17 21:52:42 +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
|
|
|
{
|
2013-12-27 16:18:53 +00:00
|
|
|
if (current_dive == NULL || editMode != NONE)
|
2013-08-13 11:34:04 +00:00
|
|
|
return;
|
2013-12-27 16:18:53 +00:00
|
|
|
if ((newEditMode == DIVE || newEditMode == NONE) &&
|
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-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->editCurrentDive();
|
2013-12-27 16:18:53 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->setEnabled(false);
|
2014-05-01 17:47:52 +00:00
|
|
|
if (amount_selected == 1)
|
2014-05-07 21:01:05 +00:00
|
|
|
MainWindow::instance()->globe()->prepareForGetDiveCoordinates();
|
2013-09-19 00:56:53 +00:00
|
|
|
// We may be editing one or more dives here. backup everything.
|
|
|
|
notesBackup.clear();
|
2014-02-12 14:22:54 +00:00
|
|
|
if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
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."));
|
2013-10-03 18:54:25 +00:00
|
|
|
notesBackup[NULL].notes = ui.notes->toPlainText();
|
|
|
|
notesBackup[NULL].location = ui.location->text();
|
2013-09-19 00:56:53 +00:00
|
|
|
editMode = TRIP;
|
|
|
|
} else {
|
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-09-19 00:56:53 +00:00
|
|
|
|
|
|
|
// We may be editing one or more dives here. backup everything.
|
|
|
|
struct dive *mydive;
|
2014-05-19 21:37:19 +00:00
|
|
|
int i;
|
2014-05-22 18:40:22 +00:00
|
|
|
for_each_dive (i, mydive) {
|
2013-09-19 00:56:53 +00:00
|
|
|
if (!mydive->selected)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
notesBackup[mydive].buddy = QString(mydive->buddy);
|
|
|
|
notesBackup[mydive].suit = QString(mydive->suit);
|
|
|
|
notesBackup[mydive].notes = QString(mydive->notes);
|
|
|
|
notesBackup[mydive].divemaster = QString(mydive->divemaster);
|
|
|
|
notesBackup[mydive].location = QString(mydive->location);
|
|
|
|
notesBackup[mydive].rating = mydive->rating;
|
|
|
|
notesBackup[mydive].visibility = mydive->visibility;
|
|
|
|
notesBackup[mydive].latitude = mydive->latitude;
|
|
|
|
notesBackup[mydive].longitude = mydive->longitude;
|
2014-02-28 04:09:57 +00:00
|
|
|
notesBackup[mydive].coordinates = ui.coordinates->text();
|
2013-09-20 23:41:42 +00:00
|
|
|
notesBackup[mydive].airtemp = get_temperature_string(mydive->airtemp, true);
|
|
|
|
notesBackup[mydive].watertemp = get_temperature_string(mydive->watertemp, true);
|
2013-11-30 07:35:15 +00:00
|
|
|
notesBackup[mydive].datetime = QDateTime::fromTime_t(mydive->when).toUTC().toString();
|
2013-11-02 01:20:02 +00:00
|
|
|
char buf[1024];
|
|
|
|
taglist_get_tagstring(mydive->tag_list, buf, 1024);
|
|
|
|
notesBackup[mydive].tags = QString(buf);
|
2013-09-25 17:10:15 +00:00
|
|
|
|
|
|
|
// maybe this is a place for memset?
|
2014-02-12 14:07:32 +00:00
|
|
|
for (int j = 0; j < MAX_CYLINDERS; j++) {
|
|
|
|
notesBackup[mydive].cylinders[j] = mydive->cylinder[j];
|
2013-09-25 17:10:15 +00:00
|
|
|
}
|
2014-02-12 14:07:32 +00:00
|
|
|
for (int j = 0; j < MAX_WEIGHTSYSTEMS; j++) {
|
|
|
|
notesBackup[mydive].weightsystem[j] = mydive->weightsystem[j];
|
2013-09-25 17:10:15 +00:00
|
|
|
}
|
2013-09-19 00:56:53 +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
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
bool MainTab::eventFilter(QObject *object, QEvent *event)
|
2013-05-19 14:45:01 +00:00
|
|
|
{
|
2013-12-19 21:45:54 +00:00
|
|
|
if (!isEnabled())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (editMode != NONE)
|
|
|
|
return false;
|
2013-12-14 05:15:07 +00:00
|
|
|
// for the dateTimeEdit widget we need to ignore Wheel events as well (as long as we aren't editing)
|
2013-12-19 21:45:54 +00:00
|
|
|
if (object->objectName() == "dateTimeEdit" &&
|
2013-12-14 05:15:07 +00:00
|
|
|
(event->type() == QEvent::FocusIn || event->type() == QEvent::Wheel))
|
|
|
|
return true;
|
|
|
|
// MouseButtonPress in any widget (not all will ever get this), KeyPress in the dateTimeEdit,
|
|
|
|
// FocusIn for the starWidgets or RequestSoftwareInputPanel for tagWidget start the editing
|
2013-12-19 21:45:54 +00:00
|
|
|
if ((event->type() == QEvent::MouseButtonPress) ||
|
2013-12-14 05:15:07 +00:00
|
|
|
(event->type() == QEvent::KeyPress && object == ui.dateTimeEdit) ||
|
2014-01-19 17:36:51 +00:00
|
|
|
(event->type() == QEvent::FocusIn && (object == ui.rating || object == ui.visibility || object == ui.buddy || object == ui.tagWidget || object || ui.divemaster))) {
|
2013-09-19 01:38:38 +00:00
|
|
|
tabBar()->setTabIcon(currentIndex(), QIcon(":warning"));
|
2013-08-13 11:34:04 +00:00
|
|
|
enableEdition();
|
2013-05-19 14:45:01 +00:00
|
|
|
}
|
2013-08-13 11:34:04 +00:00
|
|
|
return false; // don't "eat" the event.
|
2013-05-19 14:45:01 +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);
|
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
void MainTab::clearInfo()
|
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.sacText->clear();
|
|
|
|
ui.otuText->clear();
|
|
|
|
ui.oxygenHeliumText->clear();
|
|
|
|
ui.gasUsedText->clear();
|
|
|
|
ui.dateText->clear();
|
|
|
|
ui.diveTimeText->clear();
|
|
|
|
ui.surfaceIntervalText->clear();
|
|
|
|
ui.maximumDepthText->clear();
|
|
|
|
ui.averageDepthText->clear();
|
|
|
|
ui.waterTemperatureText->clear();
|
|
|
|
ui.airTemperatureText->clear();
|
|
|
|
ui.airPressureText->clear();
|
2013-10-10 08:16:32 +00:00
|
|
|
ui.salinityText->clear();
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->clear();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::clearStats()
|
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.depthLimits->clear();
|
|
|
|
ui.sacLimits->clear();
|
|
|
|
ui.divesAllText->clear();
|
|
|
|
ui.tempLimits->clear();
|
|
|
|
ui.totalTimeAllText->clear();
|
|
|
|
ui.timeLimits->clear();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
#define UPDATE_TEXT(d, field) \
|
|
|
|
if (!d || !d->field) \
|
|
|
|
ui.field->setText(""); \
|
|
|
|
else \
|
|
|
|
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) \
|
|
|
|
if (!d || d->field.mkelvin == 0) \
|
|
|
|
ui.field->setText(""); \
|
|
|
|
else \
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2013-05-06 16:23:14 +00:00
|
|
|
void MainTab::updateDiveInfo(int dive)
|
|
|
|
{
|
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;
|
2013-09-22 15:19:05 +00:00
|
|
|
if (!isEnabled() && dive != -1)
|
2013-09-19 00:56:53 +00:00
|
|
|
setEnabled(true);
|
2013-09-22 15:19:05 +00:00
|
|
|
if (isEnabled() && dive == -1)
|
|
|
|
setEnabled(false);
|
2013-08-16 16:31:52 +00:00
|
|
|
editMode = NONE;
|
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-20 05:42:24 +00:00
|
|
|
temperature_t temp;
|
2013-05-19 15:08:29 +00:00
|
|
|
struct dive *prevd;
|
2013-05-06 16:23:14 +00:00
|
|
|
struct dive *d = get_dive(dive);
|
2013-12-28 01:26:22 +00:00
|
|
|
char buf[1024];
|
2013-05-19 15:08:29 +00:00
|
|
|
|
2013-05-20 05:42:24 +00:00
|
|
|
process_selected_dives();
|
2013-05-19 15:08:29 +00:00
|
|
|
process_all_dives(d, &prevd);
|
2013-08-16 16:31:52 +00:00
|
|
|
|
2013-05-06 17:16:16 +00:00
|
|
|
UPDATE_TEXT(d, notes);
|
|
|
|
UPDATE_TEXT(d, location);
|
|
|
|
UPDATE_TEXT(d, suit);
|
|
|
|
UPDATE_TEXT(d, divemaster);
|
|
|
|
UPDATE_TEXT(d, buddy);
|
2013-09-20 23:41:42 +00:00
|
|
|
UPDATE_TEMP(d, airtemp);
|
|
|
|
UPDATE_TEMP(d, watertemp);
|
2013-05-08 19:08:00 +00:00
|
|
|
if (d) {
|
2013-11-11 23:21:45 +00:00
|
|
|
updateGpsCoordinates(d);
|
2013-11-30 07:35:15 +00:00
|
|
|
ui.dateTimeEdit->setDateTime(QDateTime::fromTime_t(d->when).toUTC());
|
2014-02-12 14:22:54 +00:00
|
|
|
if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2013-11-25 19:05:09 +00:00
|
|
|
setTabText(0, tr("Trip Notes"));
|
2013-06-14 16:17:46 +00:00
|
|
|
// only use trip relevant fields
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.coordinates->setVisible(false);
|
2013-11-04 19:28:03 +00:00
|
|
|
ui.CoordinatedLabel->setVisible(false);
|
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);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.waterTempLabel->setVisible(false);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.watertemp->setVisible(false);
|
2013-06-14 16:17:46 +00:00
|
|
|
// rename the remaining fields and fill data from selected trip
|
2014-02-12 14:22:54 +00:00
|
|
|
dive_trip_t *currentTrip = *MainWindow::instance()->dive_list()->selectedTrips().begin();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.LocationLabel->setText(tr("Trip Location"));
|
|
|
|
ui.location->setText(currentTrip->location);
|
|
|
|
ui.NotesLabel->setText(tr("Trip Notes"));
|
|
|
|
ui.notes->setText(currentTrip->notes);
|
2013-12-28 01:26:22 +00:00
|
|
|
clearEquipment();
|
|
|
|
ui.equipmentTab->setEnabled(false);
|
2013-06-14 16:17:46 +00:00
|
|
|
} else {
|
2013-11-25 19:05:09 +00:00
|
|
|
setTabText(0, tr("Dive Notes"));
|
2013-06-14 16:17:46 +00:00
|
|
|
// make all the fields visible writeable
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.coordinates->setVisible(true);
|
2013-11-04 19:28:03 +00:00
|
|
|
ui.CoordinatedLabel->setVisible(true);
|
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);
|
2013-11-30 10:13:40 +00:00
|
|
|
ui.waterTempLabel->setVisible(true);
|
2013-11-07 03:38:35 +00:00
|
|
|
ui.watertemp->setVisible(true);
|
2013-06-14 16:17:46 +00:00
|
|
|
/* and fill them from the dive */
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.rating->setCurrentStars(d->rating);
|
|
|
|
ui.visibility->setCurrentStars(d->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);
|
|
|
|
multiEditEquipmentPlaceholder = *d;
|
|
|
|
cylindersModel->setDive(&multiEditEquipmentPlaceholder);
|
|
|
|
weightModel->setDive(&multiEditEquipmentPlaceholder);
|
|
|
|
taglist_get_tagstring(d->tag_list, buf, 1024);
|
|
|
|
ui.tagWidget->setText(QString(buf));
|
2013-06-14 16:17:46 +00:00
|
|
|
}
|
2014-01-15 08:30:42 +00:00
|
|
|
ui.maximumDepthText->setText(get_depth_string(d->maxdepth, true));
|
|
|
|
ui.averageDepthText->setText(get_depth_string(d->meandepth, true));
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.otuText->setText(QString("%1").arg(d->otu));
|
2014-01-15 08:30:42 +00:00
|
|
|
ui.waterTemperatureText->setText(get_temperature_string(d->watertemp, true));
|
|
|
|
ui.airTemperatureText->setText(get_temperature_string(d->airtemp, true));
|
2013-12-09 21:37:49 +00:00
|
|
|
volume_t gases[MAX_CYLINDERS] = {};
|
2013-11-19 23:29:32 +00:00
|
|
|
get_gas_used(d, gases);
|
2014-01-15 08:30:42 +00:00
|
|
|
QString volumes = get_volume_string(gases[0], true);
|
2013-11-20 06:50:02 +00:00
|
|
|
int mean[MAX_CYLINDERS], duration[MAX_CYLINDERS];
|
2014-03-17 15:19:09 +00:00
|
|
|
per_cylinder_mean_depth(d, select_dc(d), mean, duration);
|
2013-11-20 06:50:02 +00:00
|
|
|
volume_t sac;
|
|
|
|
QString SACs;
|
|
|
|
if (mean[0] && duration[0]) {
|
2014-02-11 21:08:29 +00:00
|
|
|
sac.mliter = gases[0].mliter / (depth_to_atm(mean[0], d) * duration[0] / 60.0);
|
2014-01-15 08:30:42 +00:00
|
|
|
SACs = get_volume_string(sac, true).append(tr("/min"));
|
2013-11-20 06:50:02 +00:00
|
|
|
} else {
|
|
|
|
SACs = QString(tr("unknown"));
|
|
|
|
}
|
2014-02-28 04:09:57 +00:00
|
|
|
for (int i = 1; i < MAX_CYLINDERS && gases[i].mliter != 0; i++) {
|
2014-01-15 08:30:42 +00:00
|
|
|
volumes.append("\n" + get_volume_string(gases[i], true));
|
2013-11-20 06:50:02 +00:00
|
|
|
if (duration[i]) {
|
2014-02-11 21:08:29 +00:00
|
|
|
sac.mliter = gases[i].mliter / (depth_to_atm(mean[i], d) * duration[i] / 60);
|
2014-01-15 08:30:42 +00:00
|
|
|
SACs.append("\n" + get_volume_string(sac, true).append(tr("/min")));
|
2013-11-20 06:50:02 +00:00
|
|
|
} else {
|
|
|
|
SACs.append("\n");
|
|
|
|
}
|
|
|
|
}
|
2013-11-19 23:29:32 +00:00
|
|
|
ui.gasUsedText->setText(volumes);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.oxygenHeliumText->setText(get_gaslist(d));
|
|
|
|
ui.dateText->setText(get_short_dive_date_string(d->when));
|
|
|
|
ui.diveTimeText->setText(QString::number((int)((d->duration.seconds + 30) / 60)));
|
2013-05-19 15:08:29 +00:00
|
|
|
if (prevd)
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.surfaceIntervalText->setText(get_time_string(d->when - (prevd->when + prevd->duration.seconds), 4));
|
2013-11-14 10:36:41 +00:00
|
|
|
else
|
|
|
|
ui.surfaceIntervalText->clear();
|
2013-11-20 06:50:02 +00:00
|
|
|
if (mean[0])
|
|
|
|
ui.sacText->setText(SACs);
|
2013-05-08 19:08:00 +00:00
|
|
|
else
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.sacText->clear();
|
2013-05-08 19:08:00 +00:00
|
|
|
if (d->surface_pressure.mbar)
|
|
|
|
/* this is ALWAYS displayed in mbar */
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.airPressureText->setText(QString("%1mbar").arg(d->surface_pressure.mbar));
|
2013-05-08 19:08:00 +00:00
|
|
|
else
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.airPressureText->clear();
|
2013-10-10 08:16:32 +00:00
|
|
|
if (d->salinity)
|
2014-02-28 04:09:57 +00:00
|
|
|
ui.salinityText->setText(QString("%1g/l").arg(d->salinity / 10.0));
|
2013-10-10 08:16:32 +00:00
|
|
|
else
|
|
|
|
ui.salinityText->clear();
|
2014-01-15 08:30:42 +00:00
|
|
|
ui.depthLimits->setMaximum(get_depth_string(stats_selection.max_depth, true));
|
|
|
|
ui.depthLimits->setMinimum(get_depth_string(stats_selection.min_depth, true));
|
|
|
|
ui.depthLimits->setAverage(get_depth_string(stats_selection.avg_depth, true));
|
|
|
|
ui.sacLimits->setMaximum(get_volume_string(stats_selection.max_sac, true).append(tr("/min")));
|
|
|
|
ui.sacLimits->setMinimum(get_volume_string(stats_selection.min_sac, true).append(tr("/min")));
|
|
|
|
ui.sacLimits->setAverage(get_volume_string(stats_selection.avg_sac, true).append(tr("/min")));
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.divesAllText->setText(QString::number(stats_selection.selection_size));
|
2013-05-20 05:42:24 +00:00
|
|
|
temp.mkelvin = stats_selection.max_temp;
|
2014-01-15 08:30:42 +00:00
|
|
|
ui.tempLimits->setMaximum(get_temperature_string(temp, true));
|
2013-05-20 05:42:24 +00:00
|
|
|
temp.mkelvin = stats_selection.min_temp;
|
2014-01-15 08:30:42 +00:00
|
|
|
ui.tempLimits->setMinimum(get_temperature_string(temp, true));
|
2013-05-20 05:42:24 +00:00
|
|
|
if (stats_selection.combined_temp && stats_selection.combined_count) {
|
|
|
|
const char *unit;
|
|
|
|
get_temp_units(0, &unit);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.tempLimits->setAverage(QString("%1%2").arg(stats_selection.combined_temp / stats_selection.combined_count, 0, 'f', 1).arg(unit));
|
2013-05-20 05:42:24 +00:00
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.totalTimeAllText->setText(get_time_string(stats_selection.total_time.seconds, 0));
|
2013-05-20 05:42:24 +00:00
|
|
|
int seconds = stats_selection.total_time.seconds;
|
|
|
|
if (stats_selection.selection_size)
|
|
|
|
seconds /= stats_selection.selection_size;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.timeLimits->setAverage(get_time_string(seconds, 0));
|
|
|
|
ui.timeLimits->setMaximum(get_time_string(stats_selection.longest_time.seconds, 0));
|
|
|
|
ui.timeLimits->setMinimum(get_time_string(stats_selection.shortest_time.seconds, 0));
|
2013-05-08 19:08:00 +00:00
|
|
|
} else {
|
2013-05-20 00:38:20 +00:00
|
|
|
/* clear the fields */
|
2013-10-10 08:16:32 +00:00
|
|
|
clearInfo();
|
|
|
|
clearStats();
|
|
|
|
clearEquipment();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.rating->setCurrentStars(0);
|
|
|
|
ui.coordinates->clear();
|
|
|
|
ui.visibility->setCurrentStars(0);
|
2013-09-22 14:33:09 +00:00
|
|
|
/* turns out this is non-trivial for a dateTimeEdit... this is a partial hack */
|
2014-02-28 04:09:57 +00:00
|
|
|
QLineEdit *le = ui.dateTimeEdit->findChild<QLineEdit *>();
|
2013-09-22 14:33:09 +00:00
|
|
|
le->setText("");
|
2013-05-08 19:08:00 +00:00
|
|
|
}
|
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();
|
|
|
|
locationModel.updateModel();
|
|
|
|
diveMasterModel.updateModel();
|
|
|
|
tagModel.updateModel();
|
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
|
|
|
|
#define EDIT_SELECTED_DIVES(WHAT) \
|
|
|
|
do { \
|
|
|
|
struct dive *mydive = NULL; \
|
2014-05-19 21:37:19 +00:00
|
|
|
int _i; \
|
2014-05-07 20:00:33 +00:00
|
|
|
if (editMode == NONE) \
|
|
|
|
return; \
|
|
|
|
\
|
2014-05-19 21:37:19 +00:00
|
|
|
for_each_dive (_i, mydive) { \
|
|
|
|
if (!mydive->selected || mydive == current_dive) \
|
2014-05-07 20:00:33 +00:00
|
|
|
continue; \
|
|
|
|
\
|
|
|
|
WHAT; \
|
|
|
|
} \
|
|
|
|
mydive = current_dive; \
|
|
|
|
WHAT; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
// this macro is rather fragile and is intended to be used as WHAT inside
|
|
|
|
// an invocation of EDIT_SELECTED_DIVES(WHAT)
|
|
|
|
#define EDIT_TEXT(what, text) \
|
|
|
|
if (same_string(mydive->what, current_dive->what)) { \
|
|
|
|
QByteArray textByteArray = text.toUtf8(); \
|
|
|
|
free(mydive->what); \
|
|
|
|
mydive->what = strdup(textByteArray.data()); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define EDIT_VALUE(what, value) \
|
|
|
|
if (mydive->what == current_dive->what) { \
|
|
|
|
mydive->what = value; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define EDIT_TRIP_TEXT(what, text) \
|
|
|
|
QByteArray textByteArray = text.toUtf8(); \
|
|
|
|
free(what); \
|
|
|
|
what = strdup(textByteArray.data());
|
|
|
|
|
2013-09-19 00:56:53 +00:00
|
|
|
void MainTab::acceptChanges()
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
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-12-28 01:26:22 +00:00
|
|
|
ui.equipmentTab->setEnabled(true);
|
2013-09-19 00:56:53 +00:00
|
|
|
/* now figure out if things have changed */
|
2014-02-12 14:22:54 +00:00
|
|
|
if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2013-10-03 18:54:25 +00:00
|
|
|
if (notesBackup[NULL].notes != ui.notes->toPlainText() ||
|
2014-02-28 04:09:57 +00:00
|
|
|
notesBackup[NULL].location != ui.location->text())
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2013-05-19 16:00:57 +00:00
|
|
|
} else {
|
2013-09-19 00:56:53 +00:00
|
|
|
struct dive *curr = current_dive;
|
2013-09-19 16:37:44 +00:00
|
|
|
//Reset coordinates field, in case it contains garbage.
|
2013-11-11 23:21:45 +00:00
|
|
|
updateGpsCoordinates(curr);
|
2013-10-03 18:54:25 +00:00
|
|
|
if (notesBackup[curr].buddy != ui.buddy->text() ||
|
2014-02-28 04:09:57 +00:00
|
|
|
notesBackup[curr].suit != ui.suit->text() ||
|
|
|
|
notesBackup[curr].notes != ui.notes->toPlainText() ||
|
|
|
|
notesBackup[curr].divemaster != ui.divemaster->text() ||
|
|
|
|
notesBackup[curr].location != ui.location->text() ||
|
|
|
|
notesBackup[curr].coordinates != ui.coordinates->text() ||
|
|
|
|
notesBackup[curr].rating != ui.visibility->currentStars() ||
|
|
|
|
notesBackup[curr].airtemp != ui.airtemp->text() ||
|
|
|
|
notesBackup[curr].watertemp != ui.watertemp->text() ||
|
|
|
|
notesBackup[curr].datetime != ui.dateTimeEdit->dateTime().toString() ||
|
|
|
|
notesBackup[curr].visibility != ui.rating->currentStars() ||
|
|
|
|
notesBackup[curr].tags != ui.tagWidget->text()) {
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2013-09-20 23:41:42 +00:00
|
|
|
}
|
2014-05-07 20:00:33 +00:00
|
|
|
if (notesBackup[curr].location != ui.location->text()) {
|
|
|
|
EDIT_SELECTED_DIVES(EDIT_TEXT(location, ui.location->text()));
|
|
|
|
// if we have a location text and haven't edited the coordinates, try to fill the coordinates
|
|
|
|
// from the existing dives
|
|
|
|
if (!ui.location->text().trimmed().isEmpty() &&
|
|
|
|
(!ui.coordinates->isModified() ||
|
|
|
|
ui.coordinates->text().trimmed().isEmpty())) {
|
|
|
|
struct dive *dive;
|
|
|
|
int i = 0;
|
2014-05-22 18:40:22 +00:00
|
|
|
for_each_dive (i, dive) {
|
2014-05-07 20:00:33 +00:00
|
|
|
QString location(dive->location);
|
|
|
|
if (location == ui.location->text() &&
|
|
|
|
(dive->latitude.udeg || dive->longitude.udeg)) {
|
|
|
|
EDIT_SELECTED_DIVES(if (same_string(mydive->location, dive->location)) {
|
|
|
|
mydive->latitude = dive->latitude;
|
|
|
|
mydive->longitude = dive->longitude;
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
if (notesBackup[curr].location != ui.location->text() ||
|
2014-02-28 04:09:57 +00:00
|
|
|
notesBackup[curr].coordinates != ui.coordinates->text()) {
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->globe()->reload();
|
2013-06-14 16:17:46 +00:00
|
|
|
}
|
2013-09-25 17:57:41 +00:00
|
|
|
|
2013-11-02 01:20:02 +00:00
|
|
|
if (notesBackup[curr].tags != ui.tagWidget->text())
|
|
|
|
saveTags();
|
2013-11-14 09:36:21 +00:00
|
|
|
if (editMode == MANUALLY_ADDED_DIVE) {
|
|
|
|
DivePlannerPointsModel::instance()->copyCylinders(curr);
|
|
|
|
} else if (editMode != ADD && cylindersModel->changed) {
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (dive *d, notesBackup.keys()) {
|
2013-09-25 17:59:11 +00:00
|
|
|
for (int i = 0; i < MAX_CYLINDERS; i++) {
|
2013-12-14 17:20:37 +00:00
|
|
|
if (notesBackup.keys().count() > 1)
|
|
|
|
// only copy the cylinder type, none of the other values
|
|
|
|
d->cylinder[i].type = multiEditEquipmentPlaceholder.cylinder[i].type;
|
|
|
|
else
|
|
|
|
d->cylinder[i] = multiEditEquipmentPlaceholder.cylinder[i];
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
|
|
|
}
|
2014-03-24 18:15:19 +00:00
|
|
|
MainWindow::instance()->graphics()->replot();
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (weightModel->changed) {
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(true);
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (dive *d, notesBackup.keys()) {
|
2013-10-01 13:52:23 +00:00
|
|
|
for (int i = 0; i < MAX_WEIGHTSYSTEMS; i++) {
|
|
|
|
d->weightsystem[i] = multiEditEquipmentPlaceholder.weightsystem[i];
|
2013-09-25 17:57:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-19 14:45:01 +00:00
|
|
|
}
|
2013-11-19 22:16:33 +00:00
|
|
|
if (current_dive->divetrip) {
|
|
|
|
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) {
|
2013-09-19 04:33:39 +00:00
|
|
|
// clean up the dive data (get duration, depth information from samples)
|
|
|
|
fixup_dive(current_dive);
|
2013-09-22 14:00:14 +00:00
|
|
|
if (dive_table.nr == 1)
|
|
|
|
current_dive->number = 1;
|
2013-11-12 05:21:50 +00:00
|
|
|
else if (selected_dive == dive_table.nr - 1 && get_dive(dive_table.nr - 2)->number)
|
2013-09-20 12:14:25 +00:00
|
|
|
current_dive->number = get_dive(dive_table.nr - 2)->number + 1;
|
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
|
|
|
}
|
2013-12-02 04:17:17 +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
|
2014-05-22 18:40:22 +00:00
|
|
|
Q_FOREACH (dive *d, notesBackup.keys()) {
|
2013-12-03 23:17:38 +00:00
|
|
|
if (d)
|
|
|
|
fixup_dive(d);
|
2013-12-02 04:17:17 +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-04-03 17:11:39 +00:00
|
|
|
// it's tricky to keep the right dive selected;
|
|
|
|
// first remember which one is selected in the current sort order
|
|
|
|
// and unselect all dives
|
|
|
|
int rememberSelected = selected_dive;
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->unselectDives();
|
2014-04-03 17:11:39 +00:00
|
|
|
struct dive *d = get_dive(rememberSelected);
|
|
|
|
// mark the previously selected dive as remembered (abusing the selected flag)
|
2013-12-10 05:17:15 +00:00
|
|
|
// and then clear that flag out on the other side of the sort_table()
|
2013-12-03 22:34:05 +00:00
|
|
|
d->selected = true;
|
|
|
|
sort_table(&dive_table);
|
2014-05-22 18:40:22 +00:00
|
|
|
for_each_dive (rememberSelected, d) {
|
2013-12-10 05:17:15 +00:00
|
|
|
if (d->selected) {
|
|
|
|
d->selected = false;
|
|
|
|
break;
|
|
|
|
}
|
2013-12-03 22:34:05 +00:00
|
|
|
}
|
2014-04-03 17:11:39 +00:00
|
|
|
// refreshDisplay() will select the top dive if no dive was
|
|
|
|
// selected - but that may not be the right one, so select the one
|
|
|
|
// we remembered instead
|
|
|
|
MainWindow::instance()->dive_list()->selectDive(rememberSelected, 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-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-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->refreshDisplay();
|
|
|
|
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();
|
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.coordinates->setPalette(p);
|
|
|
|
ui.divemaster->setPalette(p);
|
|
|
|
ui.suit->setPalette(p);
|
|
|
|
ui.airtemp->setPalette(p);
|
|
|
|
ui.watertemp->setPalette(p);
|
|
|
|
ui.dateTimeEdit->setPalette(p);
|
2013-11-02 01:20:02 +00:00
|
|
|
ui.tagWidget->setPalette(p);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
#define EDIT_TEXT2(what, text) \
|
2013-12-16 22:50:42 +00:00
|
|
|
textByteArray = text.toUtf8(); \
|
2014-02-28 04:09:57 +00:00
|
|
|
free(what); \
|
2013-08-16 16:31:52 +00:00
|
|
|
what = strdup(textByteArray.data());
|
|
|
|
|
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-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-09-19 02:23:04 +00:00
|
|
|
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->setEnabled(true);
|
|
|
|
if (MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2014-02-28 04:09:57 +00:00
|
|
|
ui.notes->setText(notesBackup[NULL].notes);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.location->setText(notesBackup[NULL].location);
|
2013-11-08 09:15:04 +00:00
|
|
|
} else {
|
2013-11-12 05:41:29 +00:00
|
|
|
if (lastMode == ADD) {
|
2013-09-24 19:32:18 +00:00
|
|
|
// clean up
|
|
|
|
DivePlannerPointsModel::instance()->cancelPlan();
|
2014-05-28 22:03:10 +00:00
|
|
|
hideMessage();
|
|
|
|
resetPallete();
|
|
|
|
return;
|
2014-02-28 04:09:57 +00:00
|
|
|
} else if (lastMode == MANUALLY_ADDED_DIVE) {
|
2013-11-18 19:55:56 +00:00
|
|
|
// when we tried to edit a manually added dive, we destroyed
|
|
|
|
// the dive we edited, so let's just restore it from backup
|
|
|
|
DivePlannerPointsModel::instance()->restoreBackupDive();
|
2013-11-01 18:06:03 +00:00
|
|
|
}
|
2013-08-16 16:31:52 +00:00
|
|
|
struct dive *curr = current_dive;
|
2013-11-13 13:23:59 +00:00
|
|
|
if (curr) {
|
2014-05-28 22:03:10 +00:00
|
|
|
ui.notes->setText(notesBackup[curr].notes);
|
|
|
|
ui.location->setText(notesBackup[curr].location);
|
|
|
|
ui.buddy->setText(notesBackup[curr].buddy);
|
|
|
|
ui.suit->setText(notesBackup[curr].suit);
|
|
|
|
ui.divemaster->setText(notesBackup[curr].divemaster);
|
|
|
|
ui.rating->setCurrentStars(notesBackup[curr].rating);
|
|
|
|
ui.visibility->setCurrentStars(notesBackup[curr].visibility);
|
|
|
|
ui.airtemp->setText(notesBackup[curr].airtemp);
|
|
|
|
ui.watertemp->setText(notesBackup[curr].watertemp);
|
|
|
|
ui.tagWidget->setText(notesBackup[curr].tags);
|
|
|
|
// it's a little harder to do the right thing for the date time widget
|
2013-11-17 11:31:09 +00:00
|
|
|
ui.dateTimeEdit->setDateTime(QDateTime::fromString(notesBackup[curr].datetime));
|
2013-11-13 13:23:59 +00:00
|
|
|
} else {
|
2014-02-28 04:09:57 +00:00
|
|
|
QLineEdit *le = ui.dateTimeEdit->findChild<QLineEdit *>();
|
2013-11-13 13:23:59 +00:00
|
|
|
le->setText("");
|
|
|
|
}
|
2013-08-16 16:31:52 +00:00
|
|
|
|
|
|
|
struct dive *mydive;
|
2014-05-19 21:37:19 +00:00
|
|
|
int i;
|
|
|
|
for_each_dive (i, mydive) {
|
2013-08-16 16:31:52 +00:00
|
|
|
if (!mydive->selected)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
QByteArray textByteArray;
|
|
|
|
EDIT_TEXT2(mydive->buddy, notesBackup[mydive].buddy);
|
|
|
|
EDIT_TEXT2(mydive->suit, notesBackup[mydive].suit);
|
|
|
|
EDIT_TEXT2(mydive->notes, notesBackup[mydive].notes);
|
|
|
|
EDIT_TEXT2(mydive->divemaster, notesBackup[mydive].divemaster);
|
|
|
|
EDIT_TEXT2(mydive->location, notesBackup[mydive].location);
|
2013-09-17 19:50:15 +00:00
|
|
|
mydive->latitude = notesBackup[mydive].latitude;
|
|
|
|
mydive->longitude = notesBackup[mydive].longitude;
|
2013-08-16 16:31:52 +00:00
|
|
|
mydive->rating = notesBackup[mydive].rating;
|
|
|
|
mydive->visibility = notesBackup[mydive].visibility;
|
2013-09-25 17:10:15 +00:00
|
|
|
|
|
|
|
// maybe this is a place for memset?
|
2014-02-12 14:07:32 +00:00
|
|
|
for (int j = 0; j < MAX_CYLINDERS; j++) {
|
|
|
|
mydive->cylinder[j] = notesBackup[mydive].cylinders[j];
|
2013-09-25 17:10:15 +00:00
|
|
|
}
|
2014-02-12 14:07:32 +00:00
|
|
|
for (int j = 0; j < MAX_WEIGHTSYSTEMS; j++) {
|
|
|
|
mydive->weightsystem[j] = notesBackup[mydive].weightsystem[j];
|
2013-09-25 17:10:15 +00:00
|
|
|
}
|
2013-08-16 16:31:52 +00:00
|
|
|
}
|
2013-11-11 23:21:45 +00:00
|
|
|
updateGpsCoordinates(curr);
|
2013-11-06 01:15:19 +00:00
|
|
|
if (selected_dive >= 0) {
|
2013-10-09 06:24:44 +00:00
|
|
|
multiEditEquipmentPlaceholder = *get_dive(selected_dive);
|
|
|
|
cylindersModel->setDive(&multiEditEquipmentPlaceholder);
|
|
|
|
weightModel->setDive(&multiEditEquipmentPlaceholder);
|
|
|
|
} else {
|
|
|
|
cylindersModel->clear();
|
|
|
|
weightModel->clear();
|
2013-11-13 13:23:59 +00:00
|
|
|
setEnabled(false);
|
2013-10-09 06:24:44 +00:00
|
|
|
}
|
2013-06-14 16:17:46 +00:00
|
|
|
}
|
2013-09-19 02:23:04 +00:00
|
|
|
|
2013-12-03 20:44:48 +00:00
|
|
|
hideMessage();
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->setEnabled(true);
|
2013-09-25 17:36:59 +00:00
|
|
|
resetPallete();
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->globe()->reload();
|
2014-05-28 22:03:10 +00:00
|
|
|
if (lastMode == MANUALLY_ADDED_DIVE) {
|
2013-09-24 19:32:18 +00:00
|
|
|
// more clean up
|
|
|
|
updateDiveInfo(selected_dive);
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->showProfile();
|
2013-11-14 08:52:03 +00:00
|
|
|
// we already reloaded the divelist above, so don't recreate it or we'll lose the selection
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->refreshDisplay(false);
|
2013-09-20 03:58:53 +00:00
|
|
|
}
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::instance()->dive_list()->setFocus();
|
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
|
2014-05-07 20:58:07 +00:00
|
|
|
MainWindow::instance()->globe()->centerOnCurrentDive();
|
2014-05-28 23:50:01 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::NOTHING);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
2013-08-16 16:31:52 +00:00
|
|
|
#undef EDIT_TEXT2
|
2013-08-16 18:52:40 +00:00
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void markChangedWidget(QWidget *w)
|
|
|
|
{
|
2013-08-16 18:38:18 +00:00
|
|
|
QPalette p;
|
2013-12-24 11:49:38 +00:00
|
|
|
qreal h, s, l, a;
|
|
|
|
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);
|
|
|
|
}
|
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
|
|
|
{
|
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-05-06 17:19:43 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_TEXT(buddy, 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
|
|
|
{
|
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-05-06 17:19:43 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_TEXT(divemaster, text));
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.divemaster);
|
2013-05-18 23:42:59 +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
|
|
|
{
|
2014-05-06 20:56:46 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_VALUE(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
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_watertemp_textChanged(const QString &text)
|
2013-09-20 23:41:42 +00:00
|
|
|
{
|
2014-05-06 20:56:46 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_VALUE(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
|
|
|
|
|
|
|
// changing the time stamp on multiple dives really needs to be a relative shift
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_dateTimeEdit_dateTimeChanged(const QDateTime &datetime)
|
2013-09-20 23:41:42 +00:00
|
|
|
{
|
2013-11-30 07:35:15 +00:00
|
|
|
QDateTime dateTimeUtc(datetime);
|
|
|
|
dateTimeUtc.setTimeSpec(Qt::UTC);
|
2014-05-06 20:56:46 +00:00
|
|
|
time_t offset = current_dive->when - dateTimeUtc.toTime_t();
|
|
|
|
EDIT_SELECTED_DIVES(mydive->when -= offset);
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.dateTimeEdit);
|
2013-09-20 23:41:42 +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?
|
2013-11-02 01:20:02 +00:00
|
|
|
void MainTab::saveTags()
|
|
|
|
{
|
|
|
|
EDIT_SELECTED_DIVES(
|
2014-05-22 18:40:22 +00:00
|
|
|
QString tag;
|
|
|
|
taglist_free(mydive->tag_list);
|
|
|
|
mydive->tag_list = NULL;
|
|
|
|
foreach (tag, ui.tagWidget->getBlockStringList())
|
|
|
|
taglist_add_tag(&mydive->tag_list, tag.toUtf8().data()););
|
2013-11-02 01:20:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainTab::on_tagWidget_textChanged()
|
|
|
|
{
|
|
|
|
markChangedWidget(ui.tagWidget);
|
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_location_textChanged(const QString &text)
|
2013-05-18 23:42:59 +00:00
|
|
|
{
|
2014-05-07 21:51:39 +00:00
|
|
|
if (editMode == NONE)
|
|
|
|
return;
|
2014-05-07 21:53:52 +00:00
|
|
|
if (editMode == TRIP && MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
|
|
|
// we are editing a trip
|
|
|
|
dive_trip_t *currentTrip = *MainWindow::instance()->dive_list()->selectedTrips().begin();
|
|
|
|
EDIT_TRIP_TEXT(currentTrip->location, text);
|
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.location);
|
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
|
|
|
{
|
2014-05-06 17:19:43 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_TEXT(suit, 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()
|
|
|
|
{
|
2013-08-16 16:31:52 +00:00
|
|
|
if (editMode == NONE)
|
|
|
|
return;
|
2014-02-12 14:22:54 +00:00
|
|
|
if (editMode == TRIP && MainWindow::instance() && MainWindow::instance()->dive_list()->selectedTrips().count() == 1) {
|
2013-06-14 16:17:46 +00:00
|
|
|
// we are editing a trip
|
2014-02-12 14:22:54 +00:00
|
|
|
dive_trip_t *currentTrip = *MainWindow::instance()->dive_list()->selectedTrips().begin();
|
2014-05-06 17:19:43 +00:00
|
|
|
EDIT_TRIP_TEXT(currentTrip->notes, ui.notes->toPlainText());
|
2013-12-27 16:18:53 +00:00
|
|
|
} else if (editMode == DIVE || editMode == ADD || editMode == MANUALLY_ADDED_DIVE) {
|
2014-05-06 17:19:43 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_TEXT(notes, ui.notes->toPlainText()));
|
2013-06-14 16:17:46 +00:00
|
|
|
}
|
2013-10-03 18:54:25 +00:00
|
|
|
markChangedWidget(ui.notes);
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void MainTab::on_coordinates_textChanged(const QString &text)
|
2013-09-17 19:50:15 +00:00
|
|
|
{
|
2013-12-23 22:56:40 +00:00
|
|
|
bool gpsChanged = false;
|
|
|
|
bool parsed = false;
|
2014-05-07 22:14:01 +00:00
|
|
|
QPalette p;
|
|
|
|
ui.coordinates->setPalette(p); // reset palette
|
2013-12-23 22:56:40 +00:00
|
|
|
EDIT_SELECTED_DIVES(gpsChanged |= gpsHasChanged(mydive, current_dive, text, &parsed));
|
2014-05-07 22:14:01 +00:00
|
|
|
if (gpsChanged)
|
|
|
|
markChangedWidget(ui.coordinates); // marks things yellow
|
|
|
|
if (!parsed) {
|
2013-09-19 16:37:44 +00:00
|
|
|
p.setBrush(QPalette::Base, QColor(Qt::red).lighter());
|
2014-05-07 22:14:01 +00:00
|
|
|
ui.coordinates->setPalette(p); // marks things red
|
2013-09-19 16:37:44 +00:00
|
|
|
}
|
2013-09-17 19:50:15 +00:00
|
|
|
}
|
|
|
|
|
2013-05-18 23:42:59 +00:00
|
|
|
void MainTab::on_rating_valueChanged(int value)
|
|
|
|
{
|
2014-05-06 20:56:46 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_VALUE(rating, value));
|
2013-05-18 23:42:59 +00:00
|
|
|
}
|
2013-05-20 13:25:16 +00:00
|
|
|
|
|
|
|
void MainTab::on_visibility_valueChanged(int value)
|
|
|
|
{
|
2014-05-06 20:56:46 +00:00
|
|
|
EDIT_SELECTED_DIVES(EDIT_VALUE(visibility, value));
|
2013-05-20 13:25:16 +00:00
|
|
|
}
|
2013-06-16 17:36:23 +00:00
|
|
|
|
2014-05-06 20:56:46 +00:00
|
|
|
#undef EDIT_SELECTED_DIVES
|
|
|
|
#undef EDIT_TEXT
|
|
|
|
#undef EDIT_TRIP_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-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;
|
|
|
|
}
|
|
|
|
if (index.isValid() && index.column() != CylindersModel::REMOVE) {
|
|
|
|
if (editMode == NONE)
|
|
|
|
enableEdition();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.cylinders->edit(index);
|
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
|
|
|
|
2013-11-19 12:12:31 +00:00
|
|
|
void MainTab::updateCoordinatesText(qreal lat, qreal lon)
|
|
|
|
{
|
|
|
|
int ulat = rint(lat * 1000000);
|
|
|
|
int ulon = rint(lon * 1000000);
|
|
|
|
ui.coordinates->setText(printGPSCoords(ulat, ulon));
|
|
|
|
}
|
|
|
|
|
2013-11-11 23:21:45 +00:00
|
|
|
void MainTab::updateGpsCoordinates(const struct dive *dive)
|
|
|
|
{
|
2013-11-13 13:23:59 +00:00
|
|
|
if (dive) {
|
|
|
|
ui.coordinates->setText(printGPSCoords(dive->latitude.udeg, dive->longitude.udeg));
|
|
|
|
ui.coordinates->setModified(dive->latitude.udeg || dive->longitude.udeg);
|
|
|
|
} else {
|
|
|
|
ui.coordinates->clear();
|
|
|
|
}
|
2013-11-11 23:21:45 +00:00
|
|
|
}
|
2014-05-07 22:12:45 +00:00
|
|
|
|
|
|
|
QString MainTab::trHemisphere(const char *orig)
|
|
|
|
{
|
|
|
|
return tr(orig);
|
|
|
|
}
|