2017-04-27 18:26:05 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2013-04-14 03:44:02 +00:00
|
|
|
/*
|
|
|
|
* mainwindow.cpp
|
|
|
|
*
|
|
|
|
* classes for the main UI window in Subsurface
|
|
|
|
*/
|
2013-04-07 22:20:43 +00:00
|
|
|
#include "mainwindow.h"
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
#include <QFileDialog>
|
|
|
|
#include <QMessageBox>
|
2013-10-04 15:28:40 +00:00
|
|
|
#include <QDesktopWidget>
|
2014-02-13 21:48:07 +00:00
|
|
|
#include <QSettings>
|
2014-04-25 04:37:47 +00:00
|
|
|
#include <QShortcut>
|
2014-08-23 01:26:07 +00:00
|
|
|
#include <QToolBar>
|
2017-10-02 13:25:44 +00:00
|
|
|
#include <QStatusBar>
|
2018-09-13 15:12:36 +00:00
|
|
|
#include <QNetworkProxy>
|
|
|
|
#include <QUndoStack>
|
|
|
|
#include <QtConcurrentRun>
|
2015-10-09 23:15:26 +00:00
|
|
|
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "core/color.h"
|
|
|
|
#include "core/divecomputer.h"
|
|
|
|
#include "core/divesitehelpers.h"
|
2019-03-03 21:29:40 +00:00
|
|
|
#include "core/file.h"
|
2018-06-17 21:28:44 +00:00
|
|
|
#include "core/gettextfromc.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "core/git-access.h"
|
2018-09-28 10:09:18 +00:00
|
|
|
#include "core/import-csv.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "core/planner.h"
|
|
|
|
#include "core/qthelper.h"
|
2018-05-11 15:25:41 +00:00
|
|
|
#include "core/subsurface-string.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "core/version.h"
|
|
|
|
#include "core/windowtitleupdate.h"
|
|
|
|
|
|
|
|
#include "core/settings/qPrefCloudStorage.h"
|
|
|
|
#include "core/settings/qPrefDisplay.h"
|
|
|
|
#include "core/settings/qPrefPartialPressureGas.h"
|
|
|
|
#include "core/settings/qPrefTechnicalDetails.h"
|
|
|
|
|
Dive list: implement proper Qt-model semantics for DiveTripModel
Previously, each dive-list modifying function would lead to a
full model reset. Instead, implement proper Qt-model semantics
using beginInsertRows()/endInsertRows(), beginRemoveRows()/
endRemoveRows(), dataChange().
To do so, a DiveListNotifer singleton is generatated, which
broadcasts all changes to the dive-list. Signals are sent by
the commands and received by the DiveTripModel. Signals are
batched by dive-trip. This seems to be an adequate compromise
for the two kinds of list-views (tree and list). In the common
usecase mostly dives of a single trip are affected.
Thus, batching of dives is performed in two positions:
- At command-level to batch by trip
- In DiveTripModel to feed batches of contiguous elements
to Qt's begin*/end*-functions.
This is conceptually simple, but rather complex code. To avoid
repetition of complex loops, the batching is implemented in
templated-functions, which are passed lambda-functions, which
are called for each batch.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2018-07-30 07:20:25 +00:00
|
|
|
#include "core/subsurface-qt/DiveListNotifier.h"
|
|
|
|
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/about.h"
|
2018-07-23 21:41:23 +00:00
|
|
|
#include "desktop-widgets/command.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/divecomputermanagementdialog.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/divelistview.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/divelogexportdialog.h"
|
|
|
|
#include "desktop-widgets/divelogimportdialog.h"
|
|
|
|
#include "desktop-widgets/diveplanner.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/downloadfromdivecomputer.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/findmovedimagesdialog.h"
|
|
|
|
#include "desktop-widgets/locationinformation.h"
|
|
|
|
#include "desktop-widgets/mapwidget.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/subsurfacewebservices.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/tab-widgets/maintab.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/updatemanager.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
#include "desktop-widgets/usersurvey.h"
|
2018-10-21 16:00:02 +00:00
|
|
|
#include "desktop-widgets/simplewidgets.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
|
2015-09-03 18:56:37 +00:00
|
|
|
#include "profile-widget/profilewidget2.h"
|
2018-09-13 15:12:36 +00:00
|
|
|
|
2014-03-26 22:36:06 +00:00
|
|
|
#ifndef NO_PRINTING
|
2014-06-23 13:43:51 +00:00
|
|
|
#include <QPrintDialog>
|
2015-11-15 21:02:02 +00:00
|
|
|
#include <QBuffer>
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "desktop-widgets/printdialog.h"
|
2014-03-26 22:36:06 +00:00
|
|
|
#endif
|
2018-09-13 15:12:36 +00:00
|
|
|
|
|
|
|
#include "qt-models/cylindermodel.h"
|
|
|
|
#include "qt-models/divepicturemodel.h"
|
|
|
|
#include "qt-models/diveplannermodel.h"
|
|
|
|
#include "qt-models/filtermodels.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "qt-models/tankinfomodel.h"
|
2018-05-10 15:35:30 +00:00
|
|
|
#include "qt-models/weightsysteminfomodel.h"
|
2016-04-05 05:02:03 +00:00
|
|
|
#include "qt-models/yearlystatisticsmodel.h"
|
2015-09-17 20:16:40 +00:00
|
|
|
#include "preferences/preferencesdialog.h"
|
2015-07-25 15:30:20 +00:00
|
|
|
|
2014-03-26 22:35:24 +00:00
|
|
|
#ifndef NO_USERMANUAL
|
|
|
|
#include "usermanual.h"
|
|
|
|
#endif
|
2013-04-13 13:17:59 +00:00
|
|
|
|
2018-09-13 15:55:47 +00:00
|
|
|
namespace {
|
|
|
|
QProgressDialog *progressDialog = nullptr;
|
|
|
|
bool progressDialogCanceled = false;
|
|
|
|
int progressCounter = 0;
|
2018-09-13 15:45:14 +00:00
|
|
|
|
|
|
|
int round_int (double value) {
|
|
|
|
return static_cast<int>(lrint(value));
|
|
|
|
};
|
2018-09-13 15:55:47 +00:00
|
|
|
}
|
2015-09-09 20:02:39 +00:00
|
|
|
|
2017-06-18 08:46:49 +00:00
|
|
|
|
2017-06-18 06:22:37 +00:00
|
|
|
extern "C" int updateProgress(const char *text)
|
2015-09-09 20:02:39 +00:00
|
|
|
{
|
2016-04-03 23:13:22 +00:00
|
|
|
if (verbose)
|
2017-06-18 06:22:37 +00:00
|
|
|
qDebug() << "git storage:" << text;
|
|
|
|
if (progressDialog) {
|
2018-06-16 03:54:12 +00:00
|
|
|
// apparently we don't always get enough space to show the full label
|
|
|
|
// so let's manually make enough space (but don't shrink the existing size)
|
|
|
|
int width = QFontMetrics(qApp->font()).width(text) + 100;
|
|
|
|
if (width > progressDialog->width())
|
|
|
|
progressDialog->resize(width + 20, progressDialog->height());
|
2017-06-18 06:22:37 +00:00
|
|
|
progressDialog->setLabelText(text);
|
2017-06-18 08:46:49 +00:00
|
|
|
progressDialog->setValue(++progressCounter);
|
|
|
|
if (progressCounter == 100)
|
|
|
|
progressCounter = 0; // yes this is silly, but we really don't know how long it will take
|
2017-06-18 06:22:37 +00:00
|
|
|
}
|
2016-04-06 06:01:16 +00:00
|
|
|
qApp->processEvents();
|
2015-09-09 20:02:39 +00:00
|
|
|
return progressDialogCanceled;
|
|
|
|
}
|
|
|
|
|
2018-09-15 15:25:10 +00:00
|
|
|
MainWindow *MainWindow::m_Instance = nullptr;
|
2013-05-19 03:09:36 +00:00
|
|
|
|
2018-01-26 09:40:04 +00:00
|
|
|
extern "C" void showErrorFromC(char *buf)
|
2017-10-26 13:50:19 +00:00
|
|
|
{
|
2018-01-26 09:40:04 +00:00
|
|
|
QString error(buf);
|
|
|
|
free(buf);
|
|
|
|
emit MainWindow::instance()->showError(error);
|
2017-10-31 20:28:59 +00:00
|
|
|
}
|
2017-10-26 13:50:19 +00:00
|
|
|
|
2014-02-09 19:04:21 +00:00
|
|
|
MainWindow::MainWindow() : QMainWindow(),
|
2018-09-15 15:25:10 +00:00
|
|
|
actionNextDive(nullptr),
|
|
|
|
actionPreviousDive(nullptr),
|
2018-05-31 14:36:34 +00:00
|
|
|
#ifndef NO_USERMANUAL
|
2014-02-09 19:04:21 +00:00
|
|
|
helpView(0),
|
2018-05-31 14:36:34 +00:00
|
|
|
#endif
|
2014-04-02 19:56:14 +00:00
|
|
|
state(VIEWALL),
|
2018-09-15 15:25:10 +00:00
|
|
|
survey(nullptr),
|
|
|
|
findMovedImagesDialog(nullptr)
|
2013-04-07 22:20:43 +00:00
|
|
|
{
|
2014-02-19 00:47:21 +00:00
|
|
|
Q_ASSERT_X(m_Instance == NULL, "MainWindow", "MainWindow recreated!");
|
2014-02-12 14:22:54 +00:00
|
|
|
m_Instance = this;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.setupUi(this);
|
2015-02-26 13:39:42 +00:00
|
|
|
read_hashes();
|
2015-02-09 18:23:30 +00:00
|
|
|
// Define the States of the Application Here, Currently the states are situations where the different
|
|
|
|
// widgets will change on the mainwindow.
|
|
|
|
|
|
|
|
// for the "default" mode
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab = new MainTab(this);
|
|
|
|
diveList = new DiveListView(this);
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics = new ProfileWidget2(this);
|
2017-07-14 22:09:23 +00:00
|
|
|
MapWidget *mapWidget = MapWidget::instance();
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerSettingsWidget = new PlannerSettingsWidget(this);
|
|
|
|
divePlannerWidget = new DivePlannerWidget(this);
|
|
|
|
plannerDetails = new PlannerDetails(this);
|
2015-02-09 18:28:33 +00:00
|
|
|
|
2015-01-04 15:30:09 +00:00
|
|
|
// what is a sane order for those icons? we should have the ones the user is
|
|
|
|
// most likely to want towards the top so they are always visible
|
|
|
|
// and the ones that someone likely sets and then never touches again towards the bottom
|
2018-09-14 14:47:23 +00:00
|
|
|
profileToolbarActions = { ui.profCalcCeiling, ui.profCalcAllTissues, // start with various ceilings
|
|
|
|
ui.profIncrement3m, ui.profDcCeiling,
|
|
|
|
ui.profPhe, ui.profPn2, ui.profPO2, // partial pressure graphs
|
|
|
|
ui.profRuler, ui.profScaled, // measuring and scaling
|
|
|
|
ui.profTogglePicture, ui.profTankbar,
|
2019-01-29 13:27:27 +00:00
|
|
|
ui.profMod, ui.profDeco, ui.profNdl_tts, // various values that a user is either interested in or not
|
2018-09-14 14:47:23 +00:00
|
|
|
ui.profEad, ui.profSAC,
|
|
|
|
ui.profHR, // very few dive computers support this
|
|
|
|
ui.profTissues}; // maybe less frequently used
|
2015-02-17 02:33:25 +00:00
|
|
|
|
|
|
|
QToolBar *toolBar = new QToolBar();
|
|
|
|
Q_FOREACH (QAction *a, profileToolbarActions)
|
|
|
|
toolBar->addAction(a);
|
|
|
|
toolBar->setOrientation(Qt::Vertical);
|
|
|
|
toolBar->setIconSize(QSize(24,24));
|
|
|
|
QWidget *profileContainer = new QWidget();
|
|
|
|
QHBoxLayout *profLayout = new QHBoxLayout();
|
2015-05-25 19:14:59 +00:00
|
|
|
profLayout->setSpacing(0);
|
|
|
|
profLayout->setMargin(0);
|
|
|
|
profLayout->setContentsMargins(0,0,0,0);
|
2015-02-17 02:33:25 +00:00
|
|
|
profLayout->addWidget(toolBar);
|
2018-10-12 13:07:40 +00:00
|
|
|
profLayout->addWidget(graphics);
|
2015-02-17 02:33:25 +00:00
|
|
|
profileContainer->setLayout(profLayout);
|
2015-07-25 15:30:20 +00:00
|
|
|
|
2018-10-13 06:57:46 +00:00
|
|
|
diveSiteEdit = new LocationInformationWidget(this);
|
2015-08-29 23:00:22 +00:00
|
|
|
|
2015-08-21 01:24:49 +00:00
|
|
|
std::pair<QByteArray, QVariant> enabled = std::make_pair("enabled", QVariant(true));
|
|
|
|
std::pair<QByteArray, QVariant> disabled = std::make_pair("enabled", QVariant(false));
|
|
|
|
PropertyList enabledList;
|
|
|
|
PropertyList disabledList;
|
|
|
|
enabledList.push_back(enabled);
|
|
|
|
disabledList.push_back(disabled);
|
|
|
|
|
2018-10-12 14:13:42 +00:00
|
|
|
registerApplicationState("Default", mainTab, profileContainer, diveList, mapWidget );
|
|
|
|
registerApplicationState("EditDive", mainTab, profileContainer, diveList, mapWidget );
|
2018-10-12 13:07:40 +00:00
|
|
|
registerApplicationState("PlanDive", divePlannerWidget, profileContainer, divePlannerSettingsWidget, plannerDetails );
|
2018-10-12 14:13:42 +00:00
|
|
|
registerApplicationState("EditPlannedDive", divePlannerWidget, profileContainer, diveList, mapWidget );
|
|
|
|
registerApplicationState("EditDiveSite", diveSiteEdit, profileContainer, diveList, mapWidget);
|
Desktop: Make filterWidget2 a subobject of MainWindow
In commit b0556abdd35f96b816ba11e40bf5707abe0c3ebf, the filter-widget
and the filter were connected by a direct function call. This led
to a rather obscure crash on application-close with Qt 5.12. The
crash is due to the Ui::MainWindow class being a sub-object of MainWindow,
but the FilterWidget2 being *not* a subobject.
What happens is that after calling the MainWindow destructor, the
subobjects are destructed, notably the Ui class. Then the base-class
destructor is called (which makes sense, as destructors are called
in reverse order of constructors).
But: the QObject destructor calls hide() on all still existing child-objects
according to Qt's object hierarchy, notably the visible FilterWidget2.
Now the FilterWidget2, on hiding, updates the MainWindow, which has already
destructed all its subobjects. Crash.
Prevent this crash by making FilterWidget2 a subobject of MainWindow
and thus have it destructed before running the QObject destructor.
Alternative ways would be:
1) Use signal/slot() instead of function calls, as these are automatically
removed if an object is destroyed.
2) Make the FilterWidget2 subobject a smart-pointer. Thus, we probably
wouldn't have to include the corresponding header.
3) Make the FilterWidget2 subobject a plain pointer and delete it
explicitly in the constructor.
Reported-by: Jan Mulder <jlmulder@xs4all.nl>
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2019-01-17 21:44:09 +00:00
|
|
|
registerApplicationState("FilterDive", mainTab, profileContainer, diveList, &filterWidget2);
|
2015-02-17 02:33:25 +00:00
|
|
|
|
2019-02-09 23:31:05 +00:00
|
|
|
setStateProperties("Default", enabledList, enabledList, enabledList, enabledList);
|
|
|
|
setStateProperties("EditDive", enabledList, enabledList, enabledList, enabledList);
|
|
|
|
setStateProperties("PlanDive", enabledList, enabledList, enabledList, enabledList);
|
|
|
|
setStateProperties("EditPlannedDive", enabledList, enabledList, enabledList, enabledList);
|
2015-08-25 21:12:02 +00:00
|
|
|
setStateProperties("EditDiveSite", enabledList, disabledList, disabledList, enabledList);
|
2018-10-13 10:47:10 +00:00
|
|
|
setStateProperties("FilterDive", enabledList, enabledList, enabledList, enabledList);
|
2015-02-17 02:33:25 +00:00
|
|
|
setApplicationState("Default");
|
|
|
|
|
2013-04-25 23:44:06 +00:00
|
|
|
setWindowIcon(QIcon(":subsurface-icon"));
|
2014-06-28 08:14:36 +00:00
|
|
|
if (!QIcon::hasThemeIcon("window-close")) {
|
|
|
|
QIcon::setThemeName("subsurface");
|
|
|
|
}
|
Dive list view: replace signal-magic by flag
In DiveListView, we have a very fundamental problem: When
On the one hand, we get informed of user-selection in the
DiveListView::selectionChanged() slot. This has to set the
correct flags in the C-backend.
On the other hand, sometimes we have to set the selection
programatically, e.g. when selecting a trip. This is done
by calling QItemSelectionModel::select().
But: this will *also* call into the above slot, in which
we can't tell whether it was a user interaction or an
internal call. This can lead to either infinite loops or
very inefficient behavior, because the current dive
is set numerous times.
The current code is aware of that and disconnects the
corresponding signal. This is scary, as these signals are
set internally by the model and view. Replace this
by a global "command executing" flag in DiveListNotifier.
The flag is set using a "marker" class, which resets the flag
once it goes out of scope (cf. RAII pattern).
In DiveListView, only process a selection if the flag is not
set. Otherwise simply call the QTreeView base class, to reflect
the new selection in the UI.
To have a common point for notifications of selection changes,
add such a signal to DiveListNotifier. This signal will be
used by the DiveListView as well as the Command-objects.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2018-08-01 17:23:43 +00:00
|
|
|
connect(&diveListNotifier, &DiveListNotifier::selectionChanged, this, &MainWindow::selectionChanged);
|
2013-05-26 18:33:45 +00:00
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), this, SLOT(readSettings()));
|
2018-10-12 14:13:42 +00:00
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), diveList, SLOT(update()));
|
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), diveList, SLOT(reloadHeaderActions()));
|
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), mainTab, SLOT(updateDiveInfo()));
|
2018-10-12 13:07:40 +00:00
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), divePlannerWidget, SLOT(settingsChanged()));
|
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), divePlannerSettingsWidget, SLOT(settingsChanged()));
|
2014-01-11 14:57:06 +00:00
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), TankInfoModel::instance(), SLOT(update()));
|
2017-11-30 16:44:32 +00:00
|
|
|
for (int i = 0; i < NUM_RECENT_FILES; i++) {
|
|
|
|
actionsRecent[i] = new QAction(this);
|
2017-11-30 16:56:16 +00:00
|
|
|
actionsRecent[i]->setData(i);
|
2017-11-30 16:44:32 +00:00
|
|
|
ui.menuFile->insertAction(ui.actionQuit, actionsRecent[i]);
|
|
|
|
connect(actionsRecent[i], SIGNAL(triggered(bool)), this, SLOT(recentFileTriggered(bool)));
|
|
|
|
}
|
|
|
|
ui.menuFile->insertSeparator(ui.actionQuit);
|
2014-07-16 22:23:02 +00:00
|
|
|
connect(DivePlannerPointsModel::instance(), SIGNAL(planCreated()), this, SLOT(planCreated()));
|
|
|
|
connect(DivePlannerPointsModel::instance(), SIGNAL(planCanceled()), this, SLOT(planCanceled()));
|
2017-11-27 16:36:21 +00:00
|
|
|
connect(DivePlannerPointsModel::instance(), SIGNAL(variationsComputed(QString)), this, SLOT(updateVariations(QString)));
|
2018-10-12 13:07:40 +00:00
|
|
|
connect(plannerDetails->printPlan(), SIGNAL(pressed()), divePlannerWidget, SLOT(printDecoPlan()));
|
2018-10-12 14:13:42 +00:00
|
|
|
connect(mainTab, &MainTab::diveSiteChanged, mapWidget, &MapWidget::centerOnSelectedDiveSite);
|
2018-01-28 21:08:30 +00:00
|
|
|
connect(this, &MainWindow::showError, ui.mainErrorMessage, &NotificationWidget::showError, Qt::AutoConnection);
|
2017-07-14 22:31:16 +00:00
|
|
|
|
2018-07-30 18:59:07 +00:00
|
|
|
connect(&windowTitleUpdate, &WindowTitleUpdate::updateTitle, this, &MainWindow::setAutomaticTitle);
|
2014-06-26 15:04:39 +00:00
|
|
|
#ifdef NO_PRINTING
|
2015-03-09 08:14:01 +00:00
|
|
|
plannerDetails->printPlan()->hide();
|
2015-02-09 17:44:53 +00:00
|
|
|
ui.menuFile->removeAction(ui.actionPrint);
|
2014-06-26 15:04:39 +00:00
|
|
|
#endif
|
2015-06-14 22:42:28 +00:00
|
|
|
enableDisableCloudActions();
|
2014-06-26 15:04:39 +00:00
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.mainErrorMessage->hide();
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setEmptyState();
|
2013-06-04 12:40:09 +00:00
|
|
|
initialUiSetup();
|
2013-05-22 01:29:23 +00:00
|
|
|
readSettings();
|
2018-11-04 16:54:38 +00:00
|
|
|
diveList->reload();
|
2018-10-12 14:13:42 +00:00
|
|
|
diveList->reloadHeaderActions();
|
|
|
|
diveList->setFocus();
|
2017-07-15 19:58:16 +00:00
|
|
|
MapWidget::instance()->reload();
|
2018-10-12 14:13:42 +00:00
|
|
|
diveList->expand(diveList->model()->index(0, 0));
|
|
|
|
diveList->scrollTo(diveList->model()->index(0, 0), QAbstractItemView::PositionAtCenter);
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->settingsChanged();
|
|
|
|
divePlannerSettingsWidget->settingsChanged();
|
2014-03-26 22:35:24 +00:00
|
|
|
#ifdef NO_USERMANUAL
|
|
|
|
ui.menuHelp->removeAction(ui.actionUserManual);
|
2014-03-26 22:36:06 +00:00
|
|
|
#endif
|
2014-08-16 15:32:23 +00:00
|
|
|
memset(©PasteDive, 0, sizeof(copyPasteDive));
|
|
|
|
memset(&what, 0, sizeof(what));
|
2014-08-23 01:26:07 +00:00
|
|
|
|
2015-01-02 04:49:24 +00:00
|
|
|
updateManager = new UpdateManager(this);
|
2018-07-30 09:15:08 +00:00
|
|
|
undoAction = Command::undoAction(this);
|
|
|
|
redoAction = Command::redoAction(this);
|
2015-02-11 06:26:17 +00:00
|
|
|
undoAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_Z));
|
|
|
|
redoAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_Z));
|
2018-07-30 09:15:08 +00:00
|
|
|
ui.menu_Edit->addActions({ undoAction, redoAction });
|
2015-05-10 15:44:35 +00:00
|
|
|
|
2015-07-24 07:26:25 +00:00
|
|
|
#ifndef NO_PRINTING
|
2017-11-24 20:54:54 +00:00
|
|
|
// copy the bundled print templates to the user path
|
|
|
|
QStringList templateBackupList;
|
|
|
|
QString templatePathUser(getPrintingTemplatePathUser());
|
|
|
|
copy_bundled_templates(getPrintingTemplatePathBundle(), templatePathUser, &templateBackupList);
|
|
|
|
if (templateBackupList.length()) {
|
|
|
|
QMessageBox msgBox(this);
|
|
|
|
templatePathUser.replace("\\", "/");
|
|
|
|
templateBackupList.replaceInStrings(templatePathUser + "/", "");
|
|
|
|
msgBox.setWindowTitle(tr("Template backup created"));
|
|
|
|
msgBox.setText(tr("The following backup printing templates were created:\n\n%1\n\n"
|
|
|
|
"Location:\n%2\n\n"
|
|
|
|
"Please note that as of this version of Subsurface the default templates\n"
|
|
|
|
"are read-only and should not be edited directly, since the application\n"
|
|
|
|
"can overwrite them on startup.").arg(templateBackupList.join("\n")).arg(templatePathUser));
|
|
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
|
|
msgBox.exec();
|
|
|
|
}
|
2017-11-22 23:59:26 +00:00
|
|
|
set_bundled_templates_as_read_only();
|
2015-07-24 07:26:25 +00:00
|
|
|
find_all_templates();
|
|
|
|
#endif
|
2015-08-25 08:56:07 +00:00
|
|
|
|
2015-11-09 20:05:30 +00:00
|
|
|
setupSocialNetworkMenu();
|
2015-09-09 20:02:39 +00:00
|
|
|
set_git_update_cb(&updateProgress);
|
2017-10-26 13:50:19 +00:00
|
|
|
set_error_cb(&showErrorFromC);
|
2016-01-25 17:54:23 +00:00
|
|
|
|
|
|
|
// Toolbar Connections related to the Profile Update
|
2018-08-15 09:56:17 +00:00
|
|
|
auto tec = qPrefTechnicalDetails::instance();
|
|
|
|
connect(ui.profCalcAllTissues, &QAction::triggered, tec, &qPrefTechnicalDetails::set_calcalltissues);
|
|
|
|
connect(ui.profCalcCeiling, &QAction::triggered, tec, &qPrefTechnicalDetails::set_calcceiling);
|
|
|
|
connect(ui.profDcCeiling, &QAction::triggered, tec, &qPrefTechnicalDetails::set_dcceiling);
|
|
|
|
connect(ui.profEad, &QAction::triggered, tec, &qPrefTechnicalDetails::set_ead);
|
|
|
|
connect(ui.profIncrement3m, &QAction::triggered, tec, &qPrefTechnicalDetails::set_calcceiling3m);
|
|
|
|
connect(ui.profMod, &QAction::triggered, tec, &qPrefTechnicalDetails::set_mod);
|
|
|
|
connect(ui.profNdl_tts, &QAction::triggered, tec, &qPrefTechnicalDetails::set_calcndltts);
|
2019-01-29 13:27:27 +00:00
|
|
|
connect(ui.profDeco, &QAction::triggered, tec, &qPrefTechnicalDetails::set_decoinfo);
|
2018-08-15 09:56:17 +00:00
|
|
|
connect(ui.profHR, &QAction::triggered, tec, &qPrefTechnicalDetails::set_hrgraph);
|
|
|
|
connect(ui.profRuler, &QAction::triggered, tec, &qPrefTechnicalDetails::set_rulergraph);
|
|
|
|
connect(ui.profSAC, &QAction::triggered, tec, &qPrefTechnicalDetails::set_show_sac);
|
|
|
|
connect(ui.profScaled, &QAction::triggered, tec, &qPrefTechnicalDetails::set_zoomed_plot);
|
|
|
|
connect(ui.profTogglePicture, &QAction::triggered, tec, &qPrefTechnicalDetails::set_show_pictures_in_profile);
|
|
|
|
connect(ui.profTankbar, &QAction::triggered, tec, &qPrefTechnicalDetails::set_tankbar);
|
|
|
|
connect(ui.profTissues, &QAction::triggered, tec, &qPrefTechnicalDetails::set_percentagegraph);
|
2016-01-25 17:54:23 +00:00
|
|
|
|
2018-03-17 09:48:45 +00:00
|
|
|
connect(ui.profTissues, &QAction::triggered, this, &MainWindow::unsetProfHR);
|
|
|
|
connect(ui.profHR, &QAction::triggered, this, &MainWindow::unsetProfTissues);
|
|
|
|
|
2018-08-11 17:55:41 +00:00
|
|
|
auto pp_gas = qPrefPartialPressureGas::instance();
|
|
|
|
connect(ui.profPhe, &QAction::triggered, pp_gas, &qPrefPartialPressureGas::set_phe);
|
|
|
|
connect(ui.profPn2, &QAction::triggered, pp_gas, &qPrefPartialPressureGas::set_pn2);
|
|
|
|
connect(ui.profPO2, &QAction::triggered, pp_gas, &qPrefPartialPressureGas::set_po2);
|
2016-01-25 17:54:23 +00:00
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
connect(tec, &qPrefTechnicalDetails::calcalltissuesChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::calcceilingChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::dcceilingChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::eadChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::calcceiling3mChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::modChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::calcndlttsChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::hrgraphChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::rulergraphChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::show_sacChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::zoomed_plotChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::show_pictures_in_profileChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::tankbarChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(tec, &qPrefTechnicalDetails::percentagegraphChanged , graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
|
|
|
|
connect(pp_gas, &qPrefPartialPressureGas::pheChanged, graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(pp_gas, &qPrefPartialPressureGas::pn2Changed, graphics, &ProfileWidget2::actionRequestedReplot);
|
|
|
|
connect(pp_gas, &qPrefPartialPressureGas::po2Changed, graphics, &ProfileWidget2::actionRequestedReplot);
|
2016-01-25 17:54:23 +00:00
|
|
|
|
|
|
|
// now let's set up some connections
|
2018-10-12 13:07:40 +00:00
|
|
|
connect(graphics, &ProfileWidget2::enableToolbar ,this, &MainWindow::setEnabledToolbar);
|
|
|
|
connect(graphics, &ProfileWidget2::disableShortcuts, this, &MainWindow::disableShortcuts);
|
|
|
|
connect(graphics, &ProfileWidget2::enableShortcuts, this, &MainWindow::enableShortcuts);
|
|
|
|
connect(graphics, &ProfileWidget2::refreshDisplay, this, &MainWindow::refreshDisplay);
|
|
|
|
connect(graphics, &ProfileWidget2::editCurrentDive, this, &MainWindow::editCurrentDive);
|
2018-10-12 14:13:42 +00:00
|
|
|
connect(graphics, &ProfileWidget2::updateDiveInfo, mainTab, &MainTab::updateDiveInfo);
|
2016-01-25 17:54:23 +00:00
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
connect(PreferencesDialog::instance(), SIGNAL(settingsChanged()), graphics, SLOT(settingsChanged()));
|
2016-01-25 17:54:23 +00:00
|
|
|
|
2018-08-15 09:56:17 +00:00
|
|
|
ui.profCalcAllTissues->setChecked(qPrefTechnicalDetails::calcalltissues());
|
|
|
|
ui.profCalcCeiling->setChecked(qPrefTechnicalDetails::calcceiling());
|
|
|
|
ui.profDcCeiling->setChecked(qPrefTechnicalDetails::dcceiling());
|
|
|
|
ui.profEad->setChecked(qPrefTechnicalDetails::ead());
|
|
|
|
ui.profIncrement3m->setChecked(qPrefTechnicalDetails::calcceiling3m());
|
|
|
|
ui.profMod->setChecked(qPrefTechnicalDetails::mod());
|
|
|
|
ui.profNdl_tts->setChecked(qPrefTechnicalDetails::calcndltts());
|
2019-01-29 13:27:27 +00:00
|
|
|
ui.profDeco->setChecked(qPrefTechnicalDetails::decoinfo());
|
2018-08-11 17:55:41 +00:00
|
|
|
ui.profPhe->setChecked(pp_gas->phe());
|
|
|
|
ui.profPn2->setChecked(pp_gas->pn2());
|
|
|
|
ui.profPO2->setChecked(pp_gas->po2());
|
2018-08-15 09:56:17 +00:00
|
|
|
ui.profHR->setChecked(qPrefTechnicalDetails::hrgraph());
|
|
|
|
ui.profRuler->setChecked(qPrefTechnicalDetails::rulergraph());
|
|
|
|
ui.profSAC->setChecked(qPrefTechnicalDetails::show_sac());
|
|
|
|
ui.profTogglePicture->setChecked(qPrefTechnicalDetails::show_pictures_in_profile());
|
|
|
|
ui.profTankbar->setChecked(qPrefTechnicalDetails::tankbar());
|
|
|
|
ui.profTissues->setChecked(qPrefTechnicalDetails::percentagegraph());
|
|
|
|
ui.profScaled->setChecked(qPrefTechnicalDetails::zoomed_plot());
|
2017-10-21 20:25:42 +00:00
|
|
|
|
|
|
|
// full screen support is buggy on Windows and Ubuntu.
|
|
|
|
// require the FULLSCREEN_SUPPORT macro to enable it!
|
|
|
|
#ifndef FULLSCREEN_SUPPORT
|
|
|
|
ui.actionFullScreen->setEnabled(false);
|
|
|
|
ui.actionFullScreen->setVisible(false);
|
|
|
|
setWindowState(windowState() & ~Qt::WindowFullScreen);
|
|
|
|
#endif
|
2013-05-14 11:18:26 +00:00
|
|
|
}
|
|
|
|
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow::~MainWindow()
|
|
|
|
{
|
2015-02-26 13:39:42 +00:00
|
|
|
write_hashes();
|
2018-09-15 15:25:10 +00:00
|
|
|
m_Instance = nullptr;
|
2014-02-12 14:22:54 +00:00
|
|
|
}
|
|
|
|
|
2015-11-09 20:05:30 +00:00
|
|
|
void MainWindow::setupSocialNetworkMenu()
|
|
|
|
{
|
2015-11-08 23:50:12 +00:00
|
|
|
}
|
|
|
|
|
2015-08-21 01:24:49 +00:00
|
|
|
void MainWindow::setStateProperties(const QByteArray& state, const PropertyList& tl, const PropertyList& tr, const PropertyList& bl, const PropertyList& br)
|
|
|
|
{
|
|
|
|
stateProperties[state] = PropertiesForQuadrant(tl, tr, bl, br);
|
|
|
|
}
|
|
|
|
|
2019-03-16 10:35:44 +00:00
|
|
|
void MainWindow::editDiveSite(dive_site *ds)
|
2018-10-13 06:57:46 +00:00
|
|
|
{
|
2019-03-16 10:35:44 +00:00
|
|
|
if (!ds)
|
|
|
|
return;
|
|
|
|
diveSiteEdit->initFields(ds);
|
2015-07-25 15:30:20 +00:00
|
|
|
setApplicationState("EditDiveSite");
|
|
|
|
}
|
|
|
|
|
2019-03-16 10:51:42 +00:00
|
|
|
void MainWindow::startDiveSiteEdit()
|
2019-03-16 10:35:44 +00:00
|
|
|
{
|
2019-03-20 20:46:58 +00:00
|
|
|
if (current_dive)
|
|
|
|
editDiveSite(get_dive_site_for_dive(current_dive));
|
2019-03-16 10:35:44 +00:00
|
|
|
}
|
|
|
|
|
2015-06-14 22:42:28 +00:00
|
|
|
void MainWindow::enableDisableCloudActions()
|
|
|
|
{
|
2018-09-02 14:22:16 +00:00
|
|
|
ui.actionCloudstorageopen->setEnabled(prefs.cloud_verification_status == qPrefCloudStorage::CS_VERIFIED);
|
|
|
|
ui.actionCloudstoragesave->setEnabled(prefs.cloud_verification_status == qPrefCloudStorage::CS_VERIFIED);
|
2015-06-14 22:42:28 +00:00
|
|
|
}
|
|
|
|
|
2018-09-17 07:08:06 +00:00
|
|
|
void MainWindow::enableDisableOtherDCsActions()
|
|
|
|
{
|
|
|
|
bool nr = number_of_computers(current_dive) > 1;
|
|
|
|
ui.actionNextDC->setEnabled(nr);
|
|
|
|
ui.actionPreviousDC->setEnabled(nr);
|
|
|
|
}
|
|
|
|
|
2019-02-23 17:31:02 +00:00
|
|
|
void MainWindow::setDefaultState()
|
|
|
|
{
|
2015-02-11 15:58:23 +00:00
|
|
|
setApplicationState("Default");
|
2019-02-23 17:31:02 +00:00
|
|
|
if (mainTab->getEditMode() != MainTab::NONE)
|
2015-08-26 00:08:05 +00:00
|
|
|
ui.bottomLeft->currentWidget()->setEnabled(false);
|
2015-02-11 15:58:23 +00:00
|
|
|
}
|
|
|
|
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow *MainWindow::instance()
|
|
|
|
{
|
|
|
|
return m_Instance;
|
|
|
|
}
|
|
|
|
|
2017-11-25 15:15:57 +00:00
|
|
|
// This gets called after one or more dives were added, edited or downloaded for a dive computer
|
2014-05-18 10:08:58 +00:00
|
|
|
void MainWindow::refreshDisplay(bool doRecreateDiveList)
|
2013-05-30 08:58:59 +00:00
|
|
|
{
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->reload();
|
2013-12-28 22:56:01 +00:00
|
|
|
TankInfoModel::instance()->update();
|
2017-07-15 19:58:16 +00:00
|
|
|
MapWidget::instance()->reload();
|
2014-05-18 10:08:58 +00:00
|
|
|
if (doRecreateDiveList)
|
|
|
|
recreateDiveList();
|
2015-02-09 22:14:08 +00:00
|
|
|
|
|
|
|
setApplicationState("Default");
|
2018-10-12 14:13:42 +00:00
|
|
|
diveList->setEnabled(true);
|
|
|
|
diveList->setFocus();
|
2013-11-16 20:58:31 +00:00
|
|
|
WSInfoModel::instance()->updateInfo();
|
2018-01-01 20:54:38 +00:00
|
|
|
ui.actionAutoGroup->setChecked(autogroup);
|
2013-05-30 08:58:59 +00:00
|
|
|
}
|
|
|
|
|
2014-05-18 10:08:58 +00:00
|
|
|
void MainWindow::recreateDiveList()
|
|
|
|
{
|
2018-11-04 16:54:38 +00:00
|
|
|
diveList->reload();
|
2017-11-25 15:15:57 +00:00
|
|
|
MultiFilterSortModel::instance()->myInvalidate();
|
2014-05-18 10:08:58 +00:00
|
|
|
}
|
|
|
|
|
2019-02-23 17:31:02 +00:00
|
|
|
void MainWindow::configureToolbar()
|
|
|
|
{
|
Undo: select dives after add, remove, merge, split dive commands
Select the proper dives after the add, remove, split and merge
dives commands on undo *and* redo. Generally, select the added
dives. For undo of add, remember the pre-addition selection.
For redo of remove, select the closest dive to the first removed
dive.
The biggest part of the commit is the signal-interface between
the dive commands and the dive-list model and dive-list view.
This is done in two steps:
1) To the DiveTripModel in batches of trips. The dive trip model
transforms the dives into indices.
2) To the DiveListView. The DiveListView has to translate the
DiveTripModel indexes to actual indexes via its QSortFilterProxy-
model.
For code-reuse, derive all divelist-changing commands from a new base-class,
which has a flag that describes whether the divelist changed. The helper
functions which add and remove dives are made members of the base class and
set the flag is a selected dive is added or removed.
To properly detect when the current dive was deleted it
became necessary to turn the current dive from an index
to a pointer, because indices are not stable.
Unfortunately, in some cases an index was expected and these
places now have to transform the dive into an index. These
should be converted in due course.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2018-08-03 09:35:43 +00:00
|
|
|
if (current_dive) {
|
2018-09-14 14:48:51 +00:00
|
|
|
bool freeDiveMode = current_dive->dc.divemode == FREEDIVE;
|
|
|
|
ui.profCalcCeiling->setDisabled(freeDiveMode);
|
|
|
|
ui.profCalcCeiling->setDisabled(freeDiveMode);
|
|
|
|
ui.profCalcAllTissues ->setDisabled(freeDiveMode);
|
|
|
|
ui.profIncrement3m->setDisabled(freeDiveMode);
|
|
|
|
ui.profDcCeiling->setDisabled(freeDiveMode);
|
|
|
|
ui.profPhe->setDisabled(freeDiveMode);
|
|
|
|
ui.profPn2->setDisabled(freeDiveMode); //TODO is the same as scuba?
|
|
|
|
ui.profPO2->setDisabled(freeDiveMode); //TODO is the same as scuba?
|
|
|
|
ui.profTankbar->setDisabled(freeDiveMode);
|
|
|
|
ui.profMod->setDisabled(freeDiveMode);
|
|
|
|
ui.profNdl_tts->setDisabled(freeDiveMode);
|
2019-01-29 13:27:27 +00:00
|
|
|
ui.profDeco->setDisabled(freeDiveMode);
|
2018-09-14 14:48:51 +00:00
|
|
|
ui.profEad->setDisabled(freeDiveMode);
|
|
|
|
ui.profSAC->setDisabled(freeDiveMode);
|
|
|
|
ui.profTissues->setDisabled(freeDiveMode);
|
|
|
|
|
|
|
|
ui.profRuler->setDisabled(false);
|
|
|
|
ui.profScaled->setDisabled(false); // measuring and scaling
|
|
|
|
ui.profTogglePicture->setDisabled(false);
|
|
|
|
ui.profHR->setDisabled(false);
|
2015-10-11 20:24:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Dive list view: replace signal-magic by flag
In DiveListView, we have a very fundamental problem: When
On the one hand, we get informed of user-selection in the
DiveListView::selectionChanged() slot. This has to set the
correct flags in the C-backend.
On the other hand, sometimes we have to set the selection
programatically, e.g. when selecting a trip. This is done
by calling QItemSelectionModel::select().
But: this will *also* call into the above slot, in which
we can't tell whether it was a user interaction or an
internal call. This can lead to either infinite loops or
very inefficient behavior, because the current dive
is set numerous times.
The current code is aware of that and disconnects the
corresponding signal. This is scary, as these signals are
set internally by the model and view. Replace this
by a global "command executing" flag in DiveListNotifier.
The flag is set using a "marker" class, which resets the flag
once it goes out of scope (cf. RAII pattern).
In DiveListView, only process a selection if the flag is not
set. Otherwise simply call the QTreeView base class, to reflect
the new selection in the UI.
To have a common point for notifications of selection changes,
add such a signal to DiveListNotifier. This signal will be
used by the DiveListView as well as the Command-objects.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2018-08-01 17:23:43 +00:00
|
|
|
void MainWindow::selectionChanged()
|
2013-05-14 11:18:26 +00:00
|
|
|
{
|
2018-08-14 13:12:21 +00:00
|
|
|
if (!current_dive) {
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->clearTabs();
|
|
|
|
mainTab->updateDiveInfo(true);
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setEmptyState();
|
2018-08-14 13:12:21 +00:00
|
|
|
} else {
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->plotDive(nullptr, false, true);
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->updateDiveInfo();
|
2018-08-14 13:12:21 +00:00
|
|
|
configureToolbar();
|
2018-09-17 07:08:06 +00:00
|
|
|
enableDisableOtherDCsActions();
|
2018-08-14 13:12:21 +00:00
|
|
|
MapWidget::instance()->reload();
|
|
|
|
}
|
2014-01-15 20:13:20 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 08:35:44 +00:00
|
|
|
void MainWindow::on_actionNew_triggered()
|
|
|
|
{
|
2013-06-26 12:07:50 +00:00
|
|
|
on_actionClose_triggered();
|
2013-04-09 08:35:44 +00:00
|
|
|
}
|
2013-04-09 16:26:23 +00:00
|
|
|
|
2018-06-09 14:24:39 +00:00
|
|
|
static QString lastUsedDir()
|
|
|
|
{
|
|
|
|
QString lastDir = QDir::homePath();
|
|
|
|
|
2018-08-18 18:00:55 +00:00
|
|
|
if (QDir(qPrefDisplay::lastDir()).exists())
|
|
|
|
lastDir = qPrefDisplay::lastDir();
|
2018-06-09 14:24:39 +00:00
|
|
|
return lastDir;
|
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionOpen_triggered()
|
|
|
|
{
|
2014-06-09 02:46:43 +00:00
|
|
|
if (!okToClose(tr("Please save or cancel the current dive edit before opening a new file.")))
|
2013-11-15 21:42:49 +00:00
|
|
|
return;
|
2014-06-09 02:46:43 +00:00
|
|
|
|
2014-11-21 17:46:24 +00:00
|
|
|
// yes, this look wrong to use getSaveFileName() for the open dialog, but we need to be able
|
|
|
|
// to enter file names that don't exist in order to use our git syntax /path/to/dir[branch]
|
|
|
|
// with is a potentially valid input, but of course won't exist. So getOpenFileName() wouldn't work
|
2017-10-27 12:52:27 +00:00
|
|
|
QFileDialog dialog(this, tr("Open file"), lastUsedDir(), filter_open());
|
2014-11-23 00:06:01 +00:00
|
|
|
dialog.setFileMode(QFileDialog::AnyFile);
|
|
|
|
dialog.setViewMode(QFileDialog::Detail);
|
|
|
|
dialog.setLabelText(QFileDialog::Accept, tr("Open"));
|
|
|
|
dialog.setLabelText(QFileDialog::Reject, tr("Cancel"));
|
2015-08-07 16:12:28 +00:00
|
|
|
dialog.setAcceptMode(QFileDialog::AcceptOpen);
|
2014-11-23 00:06:01 +00:00
|
|
|
QStringList filenames;
|
|
|
|
if (dialog.exec())
|
|
|
|
filenames = dialog.selectedFiles();
|
|
|
|
if (filenames.isEmpty())
|
2013-04-13 13:17:59 +00:00
|
|
|
return;
|
2014-11-23 00:06:01 +00:00
|
|
|
updateLastUsedDir(QFileInfo(filenames.first()).dir().path());
|
2014-06-09 02:46:43 +00:00
|
|
|
closeCurrentFile();
|
2015-06-21 23:28:38 +00:00
|
|
|
// some file dialogs decide to add the default extension to a filename without extension
|
|
|
|
// so we would get dir[branch].ssrf when trying to select dir[branch].
|
|
|
|
// let's detect that and remove the incorrect extension
|
|
|
|
QStringList cleanFilenames;
|
|
|
|
QRegularExpression reg(".*\\[[^]]+]\\.ssrf", QRegularExpression::CaseInsensitiveOption);
|
|
|
|
|
|
|
|
Q_FOREACH (QString filename, filenames) {
|
|
|
|
if (reg.match(filename).hasMatch())
|
|
|
|
filename.remove(QRegularExpression("\\.ssrf$", QRegularExpression::CaseInsensitiveOption));
|
|
|
|
cleanFilenames << filename;
|
|
|
|
}
|
|
|
|
loadFiles(cleanFilenames);
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionSave_triggered()
|
|
|
|
{
|
2013-05-19 22:25:47 +00:00
|
|
|
file_save();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionSaveAs_triggered()
|
|
|
|
{
|
2013-05-19 22:25:47 +00:00
|
|
|
file_save_as();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
2013-09-23 05:24:28 +00:00
|
|
|
|
2015-06-01 05:11:27 +00:00
|
|
|
void MainWindow::on_actionCloudstorageopen_triggered()
|
|
|
|
{
|
|
|
|
if (!okToClose(tr("Please save or cancel the current dive edit before opening a new file.")))
|
|
|
|
return;
|
|
|
|
|
|
|
|
QString filename;
|
2017-10-26 13:55:49 +00:00
|
|
|
if (getCloudURL(filename))
|
2015-06-01 05:11:27 +00:00
|
|
|
return;
|
2017-10-26 13:55:49 +00:00
|
|
|
|
2017-04-22 19:29:46 +00:00
|
|
|
if (verbose)
|
|
|
|
qDebug() << "Opening cloud storage from:" << filename;
|
2015-06-01 05:11:27 +00:00
|
|
|
|
|
|
|
closeCurrentFile();
|
|
|
|
|
2015-09-09 20:02:39 +00:00
|
|
|
showProgressBar();
|
2015-06-01 05:11:27 +00:00
|
|
|
QByteArray fileNamePtr = QFile::encodeName(filename);
|
2019-02-28 21:45:17 +00:00
|
|
|
if (!parse_file(fileNamePtr.data(), &dive_table, &trip_table, &dive_site_table))
|
2018-01-03 16:11:52 +00:00
|
|
|
setCurrentFile(fileNamePtr.data());
|
2018-09-23 10:53:35 +00:00
|
|
|
process_loaded_dives();
|
2018-07-23 21:41:23 +00:00
|
|
|
Command::clear();
|
2015-09-09 20:02:39 +00:00
|
|
|
hideProgressBar();
|
2015-06-01 05:11:27 +00:00
|
|
|
refreshDisplay();
|
|
|
|
}
|
|
|
|
|
2018-05-14 18:17:08 +00:00
|
|
|
// Return whether saving to cloud is OK. If it isn't, show an error return false.
|
|
|
|
static bool saveToCloudOK()
|
|
|
|
{
|
|
|
|
if (!dive_table.nr) {
|
2018-07-03 14:52:20 +00:00
|
|
|
report_error(qPrintable(gettextFromC::tr("Don't save an empty log to the cloud")));
|
2018-05-14 18:17:08 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-06-01 05:11:27 +00:00
|
|
|
void MainWindow::on_actionCloudstoragesave_triggered()
|
|
|
|
{
|
|
|
|
QString filename;
|
2018-05-14 18:17:08 +00:00
|
|
|
if (!saveToCloudOK())
|
2017-10-19 11:28:40 +00:00
|
|
|
return;
|
2017-10-26 13:55:49 +00:00
|
|
|
if (getCloudURL(filename))
|
2015-06-01 05:11:27 +00:00
|
|
|
return;
|
2017-10-26 13:55:49 +00:00
|
|
|
|
2017-04-22 19:29:46 +00:00
|
|
|
if (verbose)
|
|
|
|
qDebug() << "Saving cloud storage to:" << filename;
|
2018-10-12 14:13:42 +00:00
|
|
|
if (mainTab->isEditing())
|
|
|
|
mainTab->acceptChanges();
|
2015-06-01 05:11:27 +00:00
|
|
|
|
2015-09-09 20:02:39 +00:00
|
|
|
showProgressBar();
|
2018-02-25 12:51:41 +00:00
|
|
|
int error = save_dives(qPrintable(filename));
|
2015-09-09 20:02:39 +00:00
|
|
|
hideProgressBar();
|
2017-12-01 17:20:43 +00:00
|
|
|
if (error)
|
|
|
|
return;
|
2015-08-25 19:49:48 +00:00
|
|
|
|
2018-02-25 12:51:41 +00:00
|
|
|
setCurrentFile(qPrintable(filename));
|
2015-06-01 05:11:27 +00:00
|
|
|
mark_divelist_changed(false);
|
|
|
|
}
|
|
|
|
|
2018-01-03 10:46:21 +00:00
|
|
|
void MainWindow::on_actionCloudOnline_triggered()
|
2016-07-30 20:09:31 +00:00
|
|
|
{
|
2018-01-03 10:46:21 +00:00
|
|
|
bool isOffline = !ui.actionCloudOnline->isChecked();
|
2018-09-10 13:30:01 +00:00
|
|
|
if (isOffline == git_local_only)
|
2018-01-03 10:46:21 +00:00
|
|
|
return;
|
|
|
|
|
2018-01-02 20:56:03 +00:00
|
|
|
// Refuse to go online if there is an edit in progress
|
2018-01-03 10:46:21 +00:00
|
|
|
if (!isOffline &&
|
|
|
|
(DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING ||
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->isEditing())) {
|
2018-01-02 20:56:03 +00:00
|
|
|
QMessageBox::warning(this, tr("Warning"), tr("Please save or cancel the current dive edit before going online"));
|
2018-01-03 10:46:21 +00:00
|
|
|
// We didn't switch to online, therefore uncheck the checkbox
|
|
|
|
ui.actionCloudOnline->setChecked(false);
|
2018-01-02 20:56:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-09-10 13:30:01 +00:00
|
|
|
git_local_only = isOffline;
|
2018-01-03 10:46:21 +00:00
|
|
|
if (!isOffline) {
|
|
|
|
// User requests to go online. Try to sync cloud storage
|
|
|
|
if (unsaved_changes()) {
|
|
|
|
// If there are unsaved changes, ask the user if they want to save them.
|
|
|
|
// If they don't, they have to sync manually.
|
|
|
|
if (QMessageBox::warning(this, tr("Save changes?"),
|
|
|
|
tr("You have unsaved changes. Do you want to commit them to the cloud storage?\n"
|
|
|
|
"If answering no, the cloud will only be synced on next call to "
|
|
|
|
"\"Open cloud storage\" or \"Save to cloud storage\"."),
|
|
|
|
QMessageBox::Yes|QMessageBox::No) == QMessageBox::Yes)
|
|
|
|
on_actionCloudstoragesave_triggered();
|
|
|
|
} else {
|
|
|
|
// If there are no unsaved changes, let's just try to load the remote cloud
|
|
|
|
on_actionCloudstorageopen_triggered();
|
|
|
|
}
|
2018-09-10 13:30:01 +00:00
|
|
|
if (git_local_only)
|
2018-01-03 10:46:21 +00:00
|
|
|
report_error(qPrintable(tr("Failure taking cloud storage online")));
|
2018-01-02 20:56:03 +00:00
|
|
|
}
|
2018-01-03 10:46:21 +00:00
|
|
|
|
|
|
|
setTitle();
|
2018-01-03 16:11:52 +00:00
|
|
|
updateCloudOnlineStatus();
|
2016-07-30 20:09:31 +00:00
|
|
|
}
|
|
|
|
|
2013-09-23 05:24:28 +00:00
|
|
|
void MainWindow::cleanUpEmpty()
|
|
|
|
{
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->clearTabs();
|
|
|
|
mainTab->updateDiveInfo(true);
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setEmptyState();
|
2018-11-04 16:54:38 +00:00
|
|
|
diveList->reload();
|
2018-12-27 09:06:11 +00:00
|
|
|
diveList->setSortOrder(DiveTripModelBase::NR, Qt::DescendingOrder);
|
2017-07-15 19:58:16 +00:00
|
|
|
MapWidget::instance()->reload();
|
2013-12-05 12:38:12 +00:00
|
|
|
if (!existing_filename)
|
2017-12-11 16:43:53 +00:00
|
|
|
setTitle();
|
2015-02-03 15:30:08 +00:00
|
|
|
disableShortcuts();
|
2013-09-23 05:24:28 +00:00
|
|
|
}
|
|
|
|
|
2014-06-09 02:46:43 +00:00
|
|
|
bool MainWindow::okToClose(QString message)
|
2013-04-09 16:26:23 +00:00
|
|
|
{
|
2014-01-16 04:50:56 +00:00
|
|
|
if (DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING ||
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->isEditing() ) {
|
2014-06-09 02:46:43 +00:00
|
|
|
QMessageBox::warning(this, tr("Warning"), message);
|
|
|
|
return false;
|
2013-11-15 21:42:49 +00:00
|
|
|
}
|
2014-06-09 02:46:43 +00:00
|
|
|
if (unsaved_changes() && askSaveChanges() == false)
|
|
|
|
return false;
|
2013-04-13 13:17:59 +00:00
|
|
|
|
2014-06-09 02:46:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::closeCurrentFile()
|
|
|
|
{
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setEmptyState();
|
2013-04-13 13:17:59 +00:00
|
|
|
/* free the dives and trips */
|
2014-03-13 22:42:45 +00:00
|
|
|
clear_git_id();
|
2015-07-24 20:18:30 +00:00
|
|
|
clear_dive_file_data();
|
2018-09-15 15:25:10 +00:00
|
|
|
setCurrentFile(nullptr);
|
2013-09-23 05:24:28 +00:00
|
|
|
cleanUpEmpty();
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(false);
|
2013-04-13 13:17:59 +00:00
|
|
|
|
|
|
|
clear_events();
|
2014-12-30 20:32:10 +00:00
|
|
|
|
2018-06-16 12:06:35 +00:00
|
|
|
dcList.dcs.clear();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2018-01-03 16:11:52 +00:00
|
|
|
void MainWindow::updateCloudOnlineStatus()
|
|
|
|
{
|
2018-09-02 14:22:16 +00:00
|
|
|
bool is_cloud = existing_filename && prefs.cloud_git_url && prefs.cloud_verification_status == qPrefCloudStorage::CS_VERIFIED &&
|
2018-01-03 16:11:52 +00:00
|
|
|
strstr(existing_filename, prefs.cloud_git_url);
|
|
|
|
ui.actionCloudOnline->setEnabled(is_cloud);
|
2018-09-10 13:30:01 +00:00
|
|
|
ui.actionCloudOnline->setChecked(is_cloud && !git_local_only);
|
2018-01-03 16:11:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::setCurrentFile(const char *f)
|
|
|
|
{
|
|
|
|
set_filename(f);
|
|
|
|
setTitle();
|
|
|
|
updateCloudOnlineStatus();
|
|
|
|
}
|
|
|
|
|
2014-06-09 02:46:43 +00:00
|
|
|
void MainWindow::on_actionClose_triggered()
|
|
|
|
{
|
2014-12-10 03:16:39 +00:00
|
|
|
if (okToClose(tr("Please save or cancel the current dive edit before closing the file."))) {
|
2014-06-09 02:46:43 +00:00
|
|
|
closeCurrentFile();
|
2014-12-10 04:15:52 +00:00
|
|
|
DivePictureModel::instance()->updateDivePictures();
|
2018-10-21 13:40:25 +00:00
|
|
|
setApplicationState("Default");
|
2014-12-10 03:16:39 +00:00
|
|
|
recreateDiveList();
|
|
|
|
}
|
2014-06-09 02:46:43 +00:00
|
|
|
}
|
|
|
|
|
2014-02-14 06:11:05 +00:00
|
|
|
void MainWindow::updateLastUsedDir(const QString &dir)
|
2013-11-14 15:42:26 +00:00
|
|
|
{
|
2018-08-18 18:00:55 +00:00
|
|
|
qPrefDisplay::set_lastDir(dir);
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionPrint_triggered()
|
|
|
|
{
|
2014-03-26 22:36:06 +00:00
|
|
|
#ifndef NO_PRINTING
|
2014-02-08 19:12:13 +00:00
|
|
|
PrintDialog dlg(this);
|
|
|
|
|
|
|
|
dlg.exec();
|
2014-03-26 22:36:06 +00:00
|
|
|
#endif
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 15:30:08 +00:00
|
|
|
void MainWindow::disableShortcuts(bool disablePaste)
|
2013-07-04 15:30:05 +00:00
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.actionPreviousDC->setShortcut(QKeySequence());
|
|
|
|
ui.actionNextDC->setShortcut(QKeySequence());
|
2015-02-03 15:30:08 +00:00
|
|
|
ui.copy->setShortcut(QKeySequence());
|
|
|
|
if (disablePaste)
|
|
|
|
ui.paste->setShortcut(QKeySequence());
|
2013-07-04 15:30:05 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 15:30:08 +00:00
|
|
|
void MainWindow::enableShortcuts()
|
2013-07-04 15:30:05 +00:00
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.actionPreviousDC->setShortcut(Qt::Key_Left);
|
|
|
|
ui.actionNextDC->setShortcut(Qt::Key_Right);
|
2015-02-03 15:30:08 +00:00
|
|
|
ui.copy->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
|
|
|
|
ui.paste->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_V));
|
2013-07-04 15:30:05 +00:00
|
|
|
}
|
|
|
|
|
2013-06-27 17:48:03 +00:00
|
|
|
void MainWindow::showProfile()
|
|
|
|
{
|
2015-02-03 15:30:08 +00:00
|
|
|
enableShortcuts();
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setProfileState();
|
2015-02-09 21:19:10 +00:00
|
|
|
setApplicationState("Default");
|
2013-06-20 21:48:21 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionPreferences_triggered()
|
|
|
|
{
|
Desktop UI: allow user defined cylinder as default
Currently, it was only possible to use a hard-coded default
cylinder in the preferences. At the same time, the user is
allowed to add own cylinders to a dive (by just typing in
a new name and cylinder properties). These own cylinders
could not be selected in the preferences, requiring the
user to manually add this every dive.
Not sure the reason for all this was intentional, or just
an overlooked aspect in the implementation. It appeared
that the selection list in the UI was constructed before
any logbook was parsed, so at that moment, there are only
hard-coded cylinders.
The fix is simple. Refresh the UI of the preferences just
before it is shown. While opening the logbook, a new
list of cylinders, including the own cylinders is
constructed, so the UI is refreshed to that.
While a simple change, there is a use case that might be
considered strange. Open a logbook, choose new logbook,
and change the default cylinder preference to an own
(from the previously opened logbook). Do not add a dive
with the new (own) cylinder and save this logbook.
Reopen this new logbook, and see that the default
cylinder in the preferences is empty. This is logical,
as the list of own possible cylinders is constructed
from the (new) logbook, that has no dive with that
specific own cylinder. I consider this acceptable
behavior, as it can be also be used to copy own
cylinders to a new logbook.
Fixes: #821
Proposed-by: Davide DB <dbdavide@gmail.com>
Signed-off-by: Jan Mulder <jlmulder@xs4all.nl>
2017-11-21 11:23:58 +00:00
|
|
|
// the refreshPages() is currently done for just one
|
|
|
|
// reason. Allow the user to define a default cylinder that
|
|
|
|
// is not hardcoded but coming from the logbook.
|
|
|
|
PreferencesDialog::instance()->refreshPages();
|
|
|
|
|
2013-05-24 18:19:48 +00:00
|
|
|
PreferencesDialog::instance()->show();
|
2015-10-06 22:02:38 +00:00
|
|
|
PreferencesDialog::instance()->raise();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionQuit_triggered()
|
|
|
|
{
|
2018-10-12 14:13:42 +00:00
|
|
|
if (mainTab->isEditing()) {
|
|
|
|
mainTab->rejectChanges();
|
|
|
|
if (mainTab->isEditing())
|
2014-06-04 01:17:32 +00:00
|
|
|
// didn't discard the edits
|
|
|
|
return;
|
2013-11-17 23:46:27 +00:00
|
|
|
}
|
2014-05-06 13:56:28 +00:00
|
|
|
if (DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING) {
|
|
|
|
DivePlannerPointsModel::instance()->cancelPlan();
|
|
|
|
if (DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING)
|
|
|
|
// The planned dive was not discarded
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-15 08:30:42 +00:00
|
|
|
if (unsaved_changes() && (askSaveChanges() == false))
|
2013-04-27 09:09:57 +00:00
|
|
|
return;
|
2013-05-03 23:30:36 +00:00
|
|
|
writeSettings();
|
|
|
|
QApplication::quit();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionDownloadDC_triggered()
|
|
|
|
{
|
2017-05-12 17:18:11 +00:00
|
|
|
DownloadFromDCWidget dlg(this);
|
|
|
|
dlg.exec();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2013-10-25 01:02:59 +00:00
|
|
|
void MainWindow::on_actionDivelogs_de_triggered()
|
|
|
|
{
|
2013-11-15 02:57:09 +00:00
|
|
|
DivelogsDeWebServices::instance()->downloadDives();
|
2013-10-25 01:02:59 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionEditDeviceNames_triggered()
|
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
DiveComputerManagementDialog::instance()->init();
|
2013-06-07 14:43:45 +00:00
|
|
|
DiveComputerManagementDialog::instance()->show();
|
2013-06-10 18:34:57 +00:00
|
|
|
}
|
2013-04-09 16:26:23 +00:00
|
|
|
|
2014-05-25 18:19:36 +00:00
|
|
|
bool MainWindow::plannerStateClean()
|
2014-05-25 18:15:57 +00:00
|
|
|
{
|
2016-03-23 00:19:20 +00:00
|
|
|
if (progressDialog)
|
|
|
|
// we are accessing the cloud, so let's not switch into Add or Plan mode
|
|
|
|
return false;
|
|
|
|
|
2014-05-25 18:15:57 +00:00
|
|
|
if (DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING ||
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->isEditing()) {
|
2014-05-25 18:19:36 +00:00
|
|
|
QMessageBox::warning(this, tr("Warning"), tr("Please save or cancel the current dive edit before trying to add a dive."));
|
|
|
|
return false;
|
2014-05-25 18:15:57 +00:00
|
|
|
}
|
2014-05-25 18:19:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-09-23 02:55:48 +00:00
|
|
|
void MainWindow::refreshProfile()
|
|
|
|
{
|
|
|
|
showProfile();
|
2015-10-11 20:24:35 +00:00
|
|
|
configureToolbar();
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->replot(current_dive);
|
2015-09-23 02:55:48 +00:00
|
|
|
DivePictureModel::instance()->updateDivePictures();
|
|
|
|
}
|
|
|
|
|
2014-05-28 18:43:32 +00:00
|
|
|
void MainWindow::planCanceled()
|
|
|
|
{
|
2014-08-19 20:53:36 +00:00
|
|
|
// while planning we might have modified the displayed_dive
|
|
|
|
// let's refresh what's shown on the profile
|
2015-09-23 02:55:48 +00:00
|
|
|
refreshProfile();
|
2014-07-06 19:36:25 +00:00
|
|
|
refreshDisplay(false);
|
2014-05-28 18:43:32 +00:00
|
|
|
}
|
|
|
|
|
2014-05-28 18:54:04 +00:00
|
|
|
void MainWindow::planCreated()
|
|
|
|
{
|
2014-10-27 19:35:19 +00:00
|
|
|
// make sure our UI is in a consistent state
|
2014-05-28 18:54:04 +00:00
|
|
|
showProfile();
|
Undo: make adding of planned dive undo-able
Planned dives were still added by directly calling core code.
This could confuse the undo-machinery, leading to crashes.
Instead, use the proper undo-command. The problem is that as
opposed to the other AddDive-commands, planned dives may
belong to a trip. Thus, the interface to the AddDive command
was changed to respect the divetrip field. Make sure that
the other callers reset that field (actually, it should never
be set). Add a comment describing the perhaps surprising
interface (the passed-in dive, usually displayed dive, is
reset).
Moreover, a dive cloned in the planner is not assigned a
new number. Thus, add an argument to the AddDive-command,
which expresses whether a new number should be generated
for the to-be-added dive.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2018-09-08 17:58:11 +00:00
|
|
|
setApplicationState("Default");
|
2018-10-12 14:13:42 +00:00
|
|
|
diveList->setEnabled(true);
|
|
|
|
diveList->setFocus();
|
2014-05-28 18:54:04 +00:00
|
|
|
}
|
|
|
|
|
2015-06-16 12:37:02 +00:00
|
|
|
void MainWindow::setPlanNotes()
|
2014-06-01 04:14:44 +00:00
|
|
|
{
|
2018-10-12 13:07:40 +00:00
|
|
|
plannerDetails->divePlanOutput()->setHtml(displayed_dive.notes);
|
2014-06-01 04:14:44 +00:00
|
|
|
}
|
|
|
|
|
2017-11-27 16:36:21 +00:00
|
|
|
void MainWindow::updateVariations(QString variations)
|
|
|
|
{
|
|
|
|
QString notes = QString(displayed_dive.notes);
|
|
|
|
free(displayed_dive.notes);
|
2018-02-28 22:37:09 +00:00
|
|
|
displayed_dive.notes = copy_qstring(notes.replace("VARIATIONS", variations));
|
2018-10-12 13:07:40 +00:00
|
|
|
plannerDetails->divePlanOutput()->setHtml(displayed_dive.notes);
|
2017-11-27 16:36:21 +00:00
|
|
|
}
|
|
|
|
|
2014-06-03 08:06:18 +00:00
|
|
|
void MainWindow::printPlan()
|
|
|
|
{
|
2014-06-23 13:43:51 +00:00
|
|
|
#ifndef NO_PRINTING
|
2018-10-12 13:07:40 +00:00
|
|
|
QString diveplan = plannerDetails->divePlanOutput()->toHtml();
|
2014-07-01 07:57:54 +00:00
|
|
|
QString withDisclaimer = QString("<img height=50 src=\":subsurface-icon\"> ") + diveplan + QString(disclaimer);
|
2014-06-04 21:34:09 +00:00
|
|
|
|
2014-06-03 08:06:18 +00:00
|
|
|
QPrinter printer;
|
|
|
|
QPrintDialog *dialog = new QPrintDialog(&printer, this);
|
|
|
|
dialog->setWindowTitle(tr("Print runtime table"));
|
|
|
|
if (dialog->exec() != QDialog::Accepted)
|
|
|
|
return;
|
|
|
|
|
2015-11-15 21:02:02 +00:00
|
|
|
/* render the profile as a pixmap that is inserted as base64 data into a HTML <img> tag
|
|
|
|
* make it fit a page width defined by 2 cm margins via QTextDocument->print() (cannot be changed?)
|
|
|
|
* the height of the profile is 40% of the page height.
|
|
|
|
*/
|
|
|
|
QSizeF renderSize = printer.pageRect(QPrinter::Inch).size();
|
|
|
|
const qreal marginsInch = 1.57480315; // = (2 x 2cm) / 2.45cm/inch
|
|
|
|
renderSize.setWidth((renderSize.width() - marginsInch) * printer.resolution());
|
|
|
|
renderSize.setHeight(((renderSize.height() - marginsInch) * printer.resolution()) / 2.5);
|
|
|
|
|
|
|
|
QPixmap pixmap(renderSize.toSize());
|
|
|
|
QPainter painter(&pixmap);
|
|
|
|
painter.setRenderHint(QPainter::Antialiasing);
|
|
|
|
painter.setRenderHint(QPainter::SmoothPixmapTransform);
|
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
ProfileWidget2 *profile = graphics;
|
2015-11-15 21:02:02 +00:00
|
|
|
QSize origSize = profile->size();
|
|
|
|
profile->resize(renderSize.toSize());
|
|
|
|
profile->setPrintMode(true);
|
|
|
|
profile->render(&painter);
|
|
|
|
profile->resize(origSize);
|
|
|
|
profile->setPrintMode(false);
|
|
|
|
|
|
|
|
QByteArray byteArray;
|
|
|
|
QBuffer buffer(&byteArray);
|
|
|
|
pixmap.save(&buffer, "PNG");
|
|
|
|
QString profileImage = QString("<img src=\"data:image/png;base64,") + byteArray.toBase64() + "\"/><br><br>";
|
|
|
|
withDisclaimer = profileImage + withDisclaimer;
|
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
plannerDetails->divePlanOutput()->setHtml(withDisclaimer);
|
|
|
|
plannerDetails->divePlanOutput()->print(&printer);
|
|
|
|
plannerDetails->divePlanOutput()->setHtml(displayed_dive.notes);
|
2014-06-23 13:43:51 +00:00
|
|
|
#endif
|
2014-06-03 08:06:18 +00:00
|
|
|
}
|
|
|
|
|
2014-07-04 14:14:16 +00:00
|
|
|
void MainWindow::setupForAddAndPlan(const char *model)
|
|
|
|
{
|
|
|
|
// clean out the dive and give it an id and the correct dc model
|
|
|
|
clear_dive(&displayed_dive);
|
2018-07-17 21:05:03 +00:00
|
|
|
displayed_dive.id = dive_getUniqID();
|
2014-07-04 14:14:16 +00:00
|
|
|
displayed_dive.when = QDateTime::currentMSecsSinceEpoch() / 1000L + gettimezoneoffset() + 3600;
|
2016-06-10 00:19:05 +00:00
|
|
|
displayed_dive.dc.model = strdup(model); // don't translate! this is stored in the XML file
|
2017-10-19 13:29:59 +00:00
|
|
|
dc_number = 1;
|
2014-07-04 14:14:16 +00:00
|
|
|
// setup the dive cylinders
|
|
|
|
DivePlannerPointsModel::instance()->clear();
|
|
|
|
DivePlannerPointsModel::instance()->setupCylinders();
|
|
|
|
}
|
|
|
|
|
2014-08-19 20:03:53 +00:00
|
|
|
void MainWindow::on_actionReplanDive_triggered()
|
|
|
|
{
|
2016-11-08 20:17:35 +00:00
|
|
|
if (!plannerStateClean() || !current_dive)
|
2014-08-19 20:03:53 +00:00
|
|
|
return;
|
2019-01-01 17:02:04 +00:00
|
|
|
else if (!is_dc_planner(¤t_dive->dc)) {
|
2015-09-09 09:53:30 +00:00
|
|
|
if (QMessageBox::warning(this, tr("Warning"), tr("Trying to replan a dive that's not a planned dive."),
|
2015-06-04 03:44:00 +00:00
|
|
|
QMessageBox::Ok | QMessageBox::Cancel) == QMessageBox::Cancel)
|
2015-05-07 20:59:12 +00:00
|
|
|
return;
|
2014-08-19 20:03:53 +00:00
|
|
|
}
|
|
|
|
// put us in PLAN mode
|
2014-08-19 20:49:18 +00:00
|
|
|
DivePlannerPointsModel::instance()->clear();
|
2014-08-19 20:03:53 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::PLAN);
|
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setPlanState();
|
|
|
|
graphics->clearHandlers();
|
2015-02-09 21:19:10 +00:00
|
|
|
setApplicationState("PlanDive");
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->setReplanButton(true);
|
|
|
|
divePlannerWidget->setupStartTime(QDateTime::fromMSecsSinceEpoch(1000 * current_dive->when, Qt::UTC));
|
2017-01-16 08:54:15 +00:00
|
|
|
if (current_dive->surface_pressure.mbar)
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->setSurfacePressure(current_dive->surface_pressure.mbar);
|
2017-01-16 08:54:15 +00:00
|
|
|
if (current_dive->salinity)
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->setSalinity(current_dive->salinity);
|
2014-08-19 20:03:53 +00:00
|
|
|
DivePlannerPointsModel::instance()->loadFromDive(current_dive);
|
|
|
|
reset_cylinders(&displayed_dive, true);
|
2017-11-27 11:37:25 +00:00
|
|
|
CylindersModel::instance()->updateDive();
|
2014-08-19 20:03:53 +00:00
|
|
|
}
|
|
|
|
|
2014-05-25 18:19:36 +00:00
|
|
|
void MainWindow::on_actionDivePlanner_triggered()
|
|
|
|
{
|
2014-08-19 20:03:53 +00:00
|
|
|
if (!plannerStateClean())
|
2014-05-25 18:19:36 +00:00
|
|
|
return;
|
|
|
|
|
2014-05-27 18:32:18 +00:00
|
|
|
// put us in PLAN mode
|
2014-05-26 20:43:52 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::PLAN);
|
2015-02-09 21:19:10 +00:00
|
|
|
setApplicationState("PlanDive");
|
2014-07-04 13:53:33 +00:00
|
|
|
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setPlanState();
|
2014-05-27 18:32:18 +00:00
|
|
|
|
2015-03-19 12:32:43 +00:00
|
|
|
// create a simple starting dive, using the first gas from the just copied cylinders
|
2014-07-04 14:14:16 +00:00
|
|
|
setupForAddAndPlan("planned dive"); // don't translate, stored in XML file
|
2014-06-28 15:07:28 +00:00
|
|
|
DivePlannerPointsModel::instance()->setupStartTime();
|
2014-05-27 18:28:42 +00:00
|
|
|
DivePlannerPointsModel::instance()->createSimpleDive();
|
2017-04-19 06:05:28 +00:00
|
|
|
// plan the dive in the same mode as the currently selected one
|
2017-11-04 08:11:16 +00:00
|
|
|
if (current_dive) {
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerSettingsWidget->setDiveMode(current_dive->dc.divemode);
|
2019-01-10 20:18:53 +00:00
|
|
|
divePlannerSettingsWidget->setBailoutVisibility(current_dive->dc.divemode);
|
2017-11-04 08:11:16 +00:00
|
|
|
if (current_dive->salinity)
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->setSalinity(current_dive->salinity);
|
2017-11-04 08:11:16 +00:00
|
|
|
}
|
2014-08-02 20:05:52 +00:00
|
|
|
DivePictureModel::instance()->updateDivePictures();
|
2018-10-12 13:07:40 +00:00
|
|
|
divePlannerWidget->setReplanButton(false);
|
2015-02-09 21:19:10 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionAddDive_triggered()
|
|
|
|
{
|
2014-09-17 22:39:49 +00:00
|
|
|
if (!plannerStateClean())
|
2013-11-09 00:09:46 +00:00
|
|
|
return;
|
2014-05-25 18:19:36 +00:00
|
|
|
|
2019-03-28 16:23:35 +00:00
|
|
|
// TODO: We (mis)use displayed_dive to construct a default dive using the dive planner.
|
|
|
|
// This means that we have to do all this in a setPlanState()/setProfileState() pair,
|
|
|
|
// to avoid the profile and planner going out of sync (which in turn can lead to crashes).
|
|
|
|
// This should all be simplified. There is no apparent no reason to go via the planner
|
|
|
|
// to create a default profile.
|
|
|
|
clear_dive(&displayed_dive);
|
|
|
|
graphics->setPlanState();
|
|
|
|
DivePlannerPointsModel::instance()->setupStartTime();
|
2013-09-20 12:36:14 +00:00
|
|
|
DivePlannerPointsModel::instance()->createSimpleDive();
|
2019-03-28 16:23:35 +00:00
|
|
|
displayed_dive.id = dive_getUniqID();
|
|
|
|
displayed_dive.when = QDateTime::currentMSecsSinceEpoch() / 1000L + gettimezoneoffset() + 3600;
|
|
|
|
displayed_dive.dc.model = strdup("manually added dive"); // don't translate! this is stored in the XML file
|
2017-02-26 20:05:23 +00:00
|
|
|
fixup_dc_duration(&displayed_dive.dc);
|
|
|
|
displayed_dive.duration = displayed_dive.dc.duration;
|
2019-03-28 16:23:35 +00:00
|
|
|
graphics->setProfileState();
|
2017-02-26 20:05:23 +00:00
|
|
|
|
2019-03-28 16:23:35 +00:00
|
|
|
Command::addDive(&displayed_dive, autogroup, true);
|
|
|
|
|
|
|
|
// Plot dive actually copies current_dive to displayed_dive and therefore ensures that the
|
|
|
|
// correct data are displayed!
|
|
|
|
graphics->plotDive(nullptr, false, true);
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionRenumber_triggered()
|
|
|
|
{
|
2014-05-19 21:11:32 +00:00
|
|
|
RenumberDialog::instance()->renumberOnlySelected(false);
|
2013-06-17 16:41:00 +00:00
|
|
|
RenumberDialog::instance()->show();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionAutoGroup_triggered()
|
|
|
|
{
|
2018-01-03 20:58:05 +00:00
|
|
|
set_autogroup(ui.actionAutoGroup->isChecked());
|
2013-10-17 23:30:32 +00:00
|
|
|
if (autogroup)
|
2018-07-23 21:41:23 +00:00
|
|
|
Command::autogroupDives();
|
2013-10-17 23:30:32 +00:00
|
|
|
else
|
2018-07-23 21:41:23 +00:00
|
|
|
Command::removeAutogenTrips();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionYearlyStatistics_triggered()
|
|
|
|
{
|
2014-08-25 18:46:08 +00:00
|
|
|
QDialog d;
|
|
|
|
QVBoxLayout *l = new QVBoxLayout(&d);
|
2014-08-25 18:48:26 +00:00
|
|
|
YearlyStatisticsModel *m = new YearlyStatisticsModel();
|
|
|
|
QTreeView *view = new QTreeView();
|
|
|
|
view->setModel(m);
|
|
|
|
l->addWidget(view);
|
2017-03-23 01:13:49 +00:00
|
|
|
d.resize(lrint(width() * .8), height() / 2);
|
|
|
|
d.move(lrint(width() * .1), height() / 4);
|
2014-11-17 18:19:51 +00:00
|
|
|
QShortcut *close = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_W), &d);
|
|
|
|
connect(close, SIGNAL(activated()), &d, SLOT(close()));
|
|
|
|
QShortcut *quit = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q), &d);
|
|
|
|
connect(quit, SIGNAL(activated()), this, SLOT(close()));
|
2014-11-20 15:34:49 +00:00
|
|
|
d.setWindowFlags(Qt::Window | Qt::CustomizeWindowHint
|
|
|
|
| Qt::WindowCloseButtonHint | Qt::WindowTitleHint);
|
2014-11-25 15:47:24 +00:00
|
|
|
d.setWindowTitle(tr("Yearly statistics"));
|
2017-11-28 07:37:01 +00:00
|
|
|
d.setWindowIcon(QIcon(":subsurface-icon"));
|
2014-08-25 18:46:08 +00:00
|
|
|
d.exec();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2017-12-29 11:01:21 +00:00
|
|
|
void MainWindow::toggleCollapsible(bool toggle)
|
|
|
|
{
|
|
|
|
ui.mainSplitter->setCollapsible(0, toggle);
|
|
|
|
ui.mainSplitter->setCollapsible(1, toggle);
|
|
|
|
ui.topSplitter->setCollapsible(0, toggle);
|
|
|
|
ui.topSplitter->setCollapsible(1, toggle);
|
|
|
|
ui.bottomSplitter->setCollapsible(0, toggle);
|
|
|
|
ui.bottomSplitter->setCollapsible(1, toggle);
|
|
|
|
}
|
2014-10-31 16:00:52 +00:00
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionViewList_triggered()
|
|
|
|
{
|
2017-12-29 11:01:21 +00:00
|
|
|
toggleCollapsible(true);
|
2013-11-07 16:37:27 +00:00
|
|
|
beginChangeState(LIST_MAXIMIZED);
|
2017-12-29 11:05:34 +00:00
|
|
|
ui.mainSplitter->setSizes({ COLLAPSED, EXPANDED });
|
2019-02-10 00:01:50 +00:00
|
|
|
showFilterIfEnabled();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionViewProfile_triggered()
|
|
|
|
{
|
2017-12-29 11:01:21 +00:00
|
|
|
toggleCollapsible(true);
|
2013-11-07 16:37:27 +00:00
|
|
|
beginChangeState(PROFILE_MAXIMIZED);
|
2017-12-29 11:05:34 +00:00
|
|
|
ui.topSplitter->setSizes({ COLLAPSED, EXPANDED });
|
|
|
|
ui.mainSplitter->setSizes({ EXPANDED, COLLAPSED });
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionViewInfo_triggered()
|
|
|
|
{
|
2017-12-29 11:01:21 +00:00
|
|
|
toggleCollapsible(true);
|
2013-11-07 16:37:27 +00:00
|
|
|
beginChangeState(INFO_MAXIMIZED);
|
2017-12-29 11:05:34 +00:00
|
|
|
ui.topSplitter->setSizes({ EXPANDED, COLLAPSED });
|
|
|
|
ui.mainSplitter->setSizes({ EXPANDED, COLLAPSED });
|
2013-06-12 18:53:23 +00:00
|
|
|
}
|
|
|
|
|
2017-07-14 22:48:53 +00:00
|
|
|
void MainWindow::on_actionViewMap_triggered()
|
2013-06-12 18:53:23 +00:00
|
|
|
{
|
2017-12-29 11:01:21 +00:00
|
|
|
toggleCollapsible(true);
|
2017-07-14 22:51:23 +00:00
|
|
|
beginChangeState(MAP_MAXIMIZED);
|
2017-12-29 11:05:34 +00:00
|
|
|
ui.mainSplitter->setSizes({ COLLAPSED, EXPANDED });
|
|
|
|
ui.bottomSplitter->setSizes({ COLLAPSED, EXPANDED });
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionViewAll_triggered()
|
|
|
|
{
|
2017-12-29 11:01:21 +00:00
|
|
|
toggleCollapsible(false);
|
2013-11-07 16:37:27 +00:00
|
|
|
beginChangeState(VIEWALL);
|
2018-09-13 15:45:14 +00:00
|
|
|
|
2013-11-22 01:52:21 +00:00
|
|
|
const int appH = qApp->desktop()->size().height();
|
|
|
|
const int appW = qApp->desktop()->size().width();
|
|
|
|
|
2018-09-13 15:45:14 +00:00
|
|
|
QList<int> mainSizes = { round_int(appH * 0.7), round_int(appH * 0.3) };
|
|
|
|
QList<int> infoProfileSizes = { round_int(appW * 0.3), round_int(appW * 0.7) };
|
|
|
|
QList<int> listGlobeSizes = { round_int(appW * 0.7), round_int(appW * 0.3) };
|
2013-11-22 01:52:21 +00:00
|
|
|
|
2018-08-29 10:08:47 +00:00
|
|
|
QSettings settings;
|
|
|
|
settings.beginGroup("MainWindow");
|
|
|
|
if (settings.value("mainSplitter").isValid()) {
|
|
|
|
ui.mainSplitter->restoreState(settings.value("mainSplitter").toByteArray());
|
|
|
|
ui.topSplitter->restoreState(settings.value("topSplitter").toByteArray());
|
|
|
|
ui.bottomSplitter->restoreState(settings.value("bottomSplitter").toByteArray());
|
2014-01-16 04:50:56 +00:00
|
|
|
if (ui.mainSplitter->sizes().first() == 0 || ui.mainSplitter->sizes().last() == 0)
|
2013-11-22 01:52:21 +00:00
|
|
|
ui.mainSplitter->setSizes(mainSizes);
|
2015-02-09 19:09:06 +00:00
|
|
|
if (ui.topSplitter->sizes().first() == 0 || ui.topSplitter->sizes().last() == 0)
|
|
|
|
ui.topSplitter->setSizes(infoProfileSizes);
|
|
|
|
if (ui.bottomSplitter->sizes().first() == 0 || ui.bottomSplitter->sizes().last() == 0)
|
|
|
|
ui.bottomSplitter->setSizes(listGlobeSizes);
|
2013-11-22 01:52:21 +00:00
|
|
|
|
2013-11-07 16:37:27 +00:00
|
|
|
} else {
|
2014-01-16 04:50:56 +00:00
|
|
|
ui.mainSplitter->setSizes(mainSizes);
|
2015-02-09 19:09:06 +00:00
|
|
|
ui.topSplitter->setSizes(infoProfileSizes);
|
|
|
|
ui.bottomSplitter->setSizes(listGlobeSizes);
|
2013-11-07 16:37:27 +00:00
|
|
|
}
|
2014-10-31 16:00:52 +00:00
|
|
|
ui.mainSplitter->setCollapsible(0, false);
|
|
|
|
ui.mainSplitter->setCollapsible(1, false);
|
2015-02-09 19:09:06 +00:00
|
|
|
ui.topSplitter->setCollapsible(0, false);
|
|
|
|
ui.topSplitter->setCollapsible(1, false);
|
|
|
|
ui.bottomSplitter->setCollapsible(0,false);
|
|
|
|
ui.bottomSplitter->setCollapsible(1,false);
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2018-05-03 17:19:54 +00:00
|
|
|
void MainWindow::enterEditState()
|
|
|
|
{
|
2018-07-30 09:15:08 +00:00
|
|
|
undoAction->setEnabled(false);
|
|
|
|
redoAction->setEnabled(false);
|
2018-05-03 17:19:54 +00:00
|
|
|
stateBeforeEdit = state;
|
|
|
|
if (state == VIEWALL || state == INFO_MAXIMIZED)
|
|
|
|
return;
|
|
|
|
toggleCollapsible(true);
|
|
|
|
beginChangeState(EDIT);
|
|
|
|
ui.topSplitter->setSizes({ EXPANDED, EXPANDED });
|
|
|
|
ui.mainSplitter->setSizes({ EXPANDED, COLLAPSED });
|
|
|
|
int appW = qApp->desktop()->size().width();
|
2018-09-13 15:45:14 +00:00
|
|
|
QList<int> infoProfileSizes { round_int(appW * 0.3), round_int(appW * 0.7) };
|
2018-05-03 17:19:54 +00:00
|
|
|
|
2018-08-29 10:08:47 +00:00
|
|
|
QSettings settings;
|
|
|
|
settings.beginGroup("MainWindow");
|
|
|
|
if (settings.value("mainSplitter").isValid()) {
|
|
|
|
ui.topSplitter->restoreState(settings.value("topSplitter").toByteArray());
|
2018-05-03 17:19:54 +00:00
|
|
|
if (ui.topSplitter->sizes().first() == 0 || ui.topSplitter->sizes().last() == 0)
|
|
|
|
ui.topSplitter->setSizes(infoProfileSizes);
|
|
|
|
} else {
|
|
|
|
ui.topSplitter->setSizes(infoProfileSizes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::exitEditState()
|
|
|
|
{
|
2018-07-30 09:15:08 +00:00
|
|
|
undoAction->setEnabled(true);
|
|
|
|
redoAction->setEnabled(true);
|
2018-05-03 17:19:54 +00:00
|
|
|
if (stateBeforeEdit == state)
|
|
|
|
return;
|
|
|
|
enterState(stateBeforeEdit);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::enterState(CurrentState newState)
|
|
|
|
{
|
|
|
|
state = newState;
|
|
|
|
switch (state) {
|
|
|
|
case VIEWALL:
|
|
|
|
on_actionViewAll_triggered();
|
|
|
|
break;
|
|
|
|
case MAP_MAXIMIZED:
|
|
|
|
on_actionViewMap_triggered();
|
|
|
|
break;
|
|
|
|
case INFO_MAXIMIZED:
|
|
|
|
on_actionViewInfo_triggered();
|
|
|
|
break;
|
|
|
|
case LIST_MAXIMIZED:
|
|
|
|
on_actionViewList_triggered();
|
|
|
|
break;
|
|
|
|
case PROFILE_MAXIMIZED:
|
|
|
|
on_actionViewProfile_triggered();
|
|
|
|
break;
|
|
|
|
case EDIT:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-14 06:11:05 +00:00
|
|
|
void MainWindow::beginChangeState(CurrentState s)
|
|
|
|
{
|
2014-01-16 04:50:56 +00:00
|
|
|
if (state == VIEWALL && state != s) {
|
2013-11-07 16:37:27 +00:00
|
|
|
saveSplitterSizes();
|
|
|
|
}
|
|
|
|
state = s;
|
|
|
|
}
|
|
|
|
|
2014-02-14 06:11:05 +00:00
|
|
|
void MainWindow::saveSplitterSizes()
|
|
|
|
{
|
2018-08-29 10:08:47 +00:00
|
|
|
QSettings settings;
|
|
|
|
settings.beginGroup("MainWindow");
|
|
|
|
settings.setValue("mainSplitter", ui.mainSplitter->saveState());
|
|
|
|
settings.setValue("topSplitter", ui.topSplitter->saveState());
|
|
|
|
settings.setValue("bottomSplitter", ui.bottomSplitter->saveState());
|
2013-11-07 16:37:27 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionPreviousDC_triggered()
|
|
|
|
{
|
2014-03-17 15:19:09 +00:00
|
|
|
unsigned nrdc = number_of_computers(current_dive);
|
|
|
|
dc_number = (dc_number + nrdc - 1) % nrdc;
|
2015-10-11 20:24:35 +00:00
|
|
|
configureToolbar();
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->plotDive(nullptr, false, true);
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->updateDiveInfo();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_actionNextDC_triggered()
|
|
|
|
{
|
2014-03-17 15:19:09 +00:00
|
|
|
unsigned nrdc = number_of_computers(current_dive);
|
|
|
|
dc_number = (dc_number + 1) % nrdc;
|
2015-10-11 20:24:35 +00:00
|
|
|
configureToolbar();
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->plotDive(nullptr, false, true);
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->updateDiveInfo();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2014-01-14 17:36:07 +00:00
|
|
|
void MainWindow::on_actionFullScreen_triggered(bool checked)
|
|
|
|
{
|
|
|
|
if (checked) {
|
|
|
|
setWindowState(windowState() | Qt::WindowFullScreen);
|
2014-01-16 04:50:56 +00:00
|
|
|
} else {
|
2014-01-14 17:36:07 +00:00
|
|
|
setWindowState(windowState() & ~Qt::WindowFullScreen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionAboutSubsurface_triggered()
|
|
|
|
{
|
2014-02-08 07:50:39 +00:00
|
|
|
SubsurfaceAbout dlg(this);
|
|
|
|
|
|
|
|
dlg.exec();
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
|
|
|
|
2014-04-02 19:41:39 +00:00
|
|
|
void MainWindow::on_action_Check_for_Updates_triggered()
|
|
|
|
{
|
2014-04-02 19:56:14 +00:00
|
|
|
if (!updateManager)
|
|
|
|
updateManager = new UpdateManager(this);
|
|
|
|
|
2014-04-02 19:41:39 +00:00
|
|
|
updateManager->checkForUpdates();
|
|
|
|
}
|
|
|
|
|
2013-04-09 16:26:23 +00:00
|
|
|
void MainWindow::on_actionUserManual_triggered()
|
|
|
|
{
|
2014-03-26 22:35:24 +00:00
|
|
|
#ifndef NO_USERMANUAL
|
2018-05-25 21:08:18 +00:00
|
|
|
if (!helpView)
|
|
|
|
helpView = new UserManual(this);
|
2013-05-30 13:54:06 +00:00
|
|
|
helpView->show();
|
2014-03-26 22:35:24 +00:00
|
|
|
#endif
|
2013-04-09 16:26:23 +00:00
|
|
|
}
|
2013-04-13 13:17:59 +00:00
|
|
|
|
2014-12-17 03:59:41 +00:00
|
|
|
void MainWindow::on_actionUserSurvey_triggered()
|
|
|
|
{
|
2018-05-25 21:08:18 +00:00
|
|
|
if(!survey)
|
2014-12-30 15:37:04 +00:00
|
|
|
survey = new UserSurvey(this);
|
2014-12-17 03:59:41 +00:00
|
|
|
survey->show();
|
|
|
|
}
|
|
|
|
|
2018-06-10 14:40:23 +00:00
|
|
|
void MainWindow::on_actionHash_images_triggered()
|
|
|
|
{
|
|
|
|
if(!findMovedImagesDialog)
|
|
|
|
findMovedImagesDialog = new FindMovedImagesDialog(this);
|
|
|
|
findMovedImagesDialog->show();
|
|
|
|
}
|
|
|
|
|
2017-10-27 12:52:27 +00:00
|
|
|
QString MainWindow::filter_open()
|
2013-04-13 13:17:59 +00:00
|
|
|
{
|
2018-09-15 15:26:35 +00:00
|
|
|
QString f = tr("Dive log files") +
|
|
|
|
" (*.ssrf"
|
|
|
|
" *.xml"
|
|
|
|
" *.can"
|
|
|
|
" *.db"
|
|
|
|
" *.sql"
|
|
|
|
" *.dld"
|
|
|
|
" *.jlb"
|
|
|
|
" *.lvd"
|
|
|
|
" *.sde"
|
|
|
|
" *.udcf"
|
|
|
|
" *.uddf"
|
|
|
|
" *.dlf"
|
|
|
|
" *.log"
|
|
|
|
" *.txt"
|
|
|
|
" *.apd"
|
|
|
|
" *.dive"
|
|
|
|
" *.zxu *.zxl"
|
|
|
|
");;";
|
2013-04-13 13:17:59 +00:00
|
|
|
|
2017-10-27 12:52:27 +00:00
|
|
|
f += tr("Subsurface files") + " (*.ssrf *.xml);;";
|
|
|
|
f += tr("Cochran") + " (*.can);;";
|
|
|
|
f += tr("DiveLogs.de") + " (*.dld);;";
|
|
|
|
f += tr("JDiveLog") + " (*.jlb);;";
|
|
|
|
f += tr("Liquivision") + " (*.lvd);;";
|
|
|
|
f += tr("Suunto") + " (*.sde *.db);;";
|
|
|
|
f += tr("UDCF") + " (*.udcf);;";
|
|
|
|
f += tr("UDDF") + " (*.uddf);;";
|
|
|
|
f += tr("XML") + " (*.xml);;";
|
|
|
|
f += tr("Divesoft") + " (*.dlf);;";
|
|
|
|
f += tr("Datatrak/WLog") + " (*.log);;";
|
|
|
|
f += tr("MkVI files") + " (*.txt);;";
|
|
|
|
f += tr("APD log viewer") + " (*.apd);;";
|
|
|
|
f += tr("OSTCtools") + " (*.dive);;";
|
|
|
|
f += tr("DAN DL7") + " (*.zxu *.zxl)";
|
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString MainWindow::filter_import()
|
|
|
|
{
|
2018-09-15 15:26:35 +00:00
|
|
|
QString f = tr("Dive log files") +
|
|
|
|
" (*.ssrf"
|
|
|
|
" *.xml"
|
|
|
|
" *.can"
|
|
|
|
" *.csv"
|
|
|
|
" *.db"
|
|
|
|
" *.sql"
|
|
|
|
" *.dld"
|
|
|
|
" *.jlb"
|
|
|
|
" *.lvd"
|
|
|
|
" *.sde"
|
|
|
|
" *.udcf"
|
|
|
|
" *.uddf"
|
|
|
|
" *.dlf"
|
|
|
|
" *.log"
|
|
|
|
" *.txt"
|
|
|
|
" *.apd"
|
|
|
|
" *.dive"
|
|
|
|
" *.zxu *.zxl"
|
|
|
|
");;";
|
2017-10-27 12:52:27 +00:00
|
|
|
|
|
|
|
f += tr("Subsurface files") + " (*.ssrf *.xml);;";
|
|
|
|
f += tr("Cochran") + " (*.can);;";
|
|
|
|
f += tr("CSV") + " (*.csv *.CSV);;";
|
|
|
|
f += tr("DiveLogs.de") + " (*.dld);;";
|
|
|
|
f += tr("JDiveLog") + " (*.jlb);;";
|
|
|
|
f += tr("Liquivision") + " (*.lvd);;";
|
|
|
|
f += tr("Suunto") + " (*.sde *.db);;";
|
|
|
|
f += tr("UDCF") + " (*.udcf);;";
|
|
|
|
f += tr("UDDF") + " (*.uddf);;";
|
|
|
|
f += tr("XML") + " (*.xml);;";
|
|
|
|
f += tr("Divesoft") + " (*.dlf);;";
|
|
|
|
f += tr("Datatrak/WLog") + " (*.log);;";
|
|
|
|
f += tr("MkVI files") + " (*.txt);;";
|
|
|
|
f += tr("APD log viewer") + " (*.apd);;";
|
|
|
|
f += tr("OSTCtools") + " (*.dive);;";
|
|
|
|
f += tr("DAN DL7") + " (*.zxu *.zxl);;";
|
|
|
|
f += tr("All files") + " (*.*)";
|
2013-04-13 13:17:59 +00:00
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MainWindow::askSaveChanges()
|
|
|
|
{
|
2014-07-16 22:23:02 +00:00
|
|
|
QMessageBox response(this);
|
2013-04-13 13:17:59 +00:00
|
|
|
|
2018-09-13 16:08:26 +00:00
|
|
|
QString message = existing_filename ?
|
|
|
|
tr("Do you want to save the changes that you made in the file %1?").arg(displayedFilename(existing_filename)) :
|
|
|
|
tr("Do you want to save the changes that you made in the data file?");
|
2013-05-24 07:28:48 +00:00
|
|
|
|
|
|
|
response.setStandardButtons(QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel);
|
|
|
|
response.setDefaultButton(QMessageBox::Save);
|
|
|
|
response.setText(message);
|
2014-07-10 23:06:48 +00:00
|
|
|
response.setWindowTitle(tr("Save changes?")); // Not displayed on MacOSX as described in Qt API
|
2013-05-24 07:28:48 +00:00
|
|
|
response.setInformativeText(tr("Changes will be lost if you don't save them."));
|
|
|
|
response.setIcon(QMessageBox::Warning);
|
2014-03-14 23:00:18 +00:00
|
|
|
response.setWindowModality(Qt::WindowModal);
|
2013-05-24 07:28:48 +00:00
|
|
|
int ret = response.exec();
|
|
|
|
|
|
|
|
switch (ret) {
|
|
|
|
case QMessageBox::Save:
|
2013-05-19 22:25:47 +00:00
|
|
|
file_save();
|
2013-04-13 13:17:59 +00:00
|
|
|
return true;
|
2013-05-24 07:28:48 +00:00
|
|
|
case QMessageBox::Discard:
|
2013-05-19 21:46:53 +00:00
|
|
|
return true;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2013-04-27 09:09:57 +00:00
|
|
|
|
2013-06-04 12:40:09 +00:00
|
|
|
void MainWindow::initialUiSetup()
|
2013-04-27 09:09:57 +00:00
|
|
|
{
|
2018-08-29 10:08:47 +00:00
|
|
|
QSettings settings;
|
|
|
|
settings.beginGroup("MainWindow");
|
|
|
|
if (settings.value("maximized", isMaximized()).value<bool>()) {
|
2013-12-02 10:32:27 +00:00
|
|
|
showMaximized();
|
2015-10-09 16:03:12 +00:00
|
|
|
} else {
|
2018-08-29 10:08:47 +00:00
|
|
|
restoreGeometry(settings.value("geometry").toByteArray());
|
|
|
|
restoreState(settings.value("windowState", 0).toByteArray());
|
2015-10-09 16:03:12 +00:00
|
|
|
}
|
2013-10-04 18:07:36 +00:00
|
|
|
|
2018-08-29 10:08:47 +00:00
|
|
|
enterState((CurrentState)settings.value("lastState", 0).toInt());
|
|
|
|
settings.endGroup();
|
2015-10-09 16:03:12 +00:00
|
|
|
show();
|
2013-06-04 12:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::readSettings()
|
|
|
|
{
|
2014-06-30 21:42:48 +00:00
|
|
|
static bool firstRun = true;
|
2015-11-14 17:38:18 +00:00
|
|
|
init_proxy();
|
2014-06-26 16:20:34 +00:00
|
|
|
|
2015-11-10 21:45:13 +00:00
|
|
|
// now make sure that the cloud menu items are enabled IFF cloud account is verified
|
|
|
|
enableDisableCloudActions();
|
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
loadRecentFiles();
|
2014-06-30 21:42:48 +00:00
|
|
|
if (firstRun) {
|
2017-11-30 14:57:26 +00:00
|
|
|
checkSurvey();
|
2014-06-30 21:42:48 +00:00
|
|
|
firstRun = false;
|
|
|
|
}
|
2013-04-27 09:09:57 +00:00
|
|
|
}
|
|
|
|
|
2014-03-02 21:37:29 +00:00
|
|
|
#undef TOOLBOX_PREF_BUTTON
|
|
|
|
|
2017-11-30 14:57:26 +00:00
|
|
|
void MainWindow::checkSurvey()
|
2014-06-13 17:56:46 +00:00
|
|
|
{
|
2017-11-30 14:57:26 +00:00
|
|
|
QSettings s;
|
|
|
|
s.beginGroup("UserSurvey");
|
|
|
|
if (!s.contains("FirstUse42")) {
|
2014-06-13 17:56:46 +00:00
|
|
|
QVariant value = QDate().currentDate();
|
2017-11-30 14:57:26 +00:00
|
|
|
s.setValue("FirstUse42", value);
|
2014-06-14 21:45:42 +00:00
|
|
|
}
|
|
|
|
// wait a week for production versions, but not at all for non-tagged builds
|
2015-01-02 06:50:03 +00:00
|
|
|
int waitTime = 7;
|
2017-11-30 14:57:26 +00:00
|
|
|
QDate firstUse42 = s.value("FirstUse42").toDate();
|
2018-08-29 10:08:47 +00:00
|
|
|
if (run_survey || (firstUse42.daysTo(QDate().currentDate()) > waitTime && !s.contains("SurveyDone"))) {
|
2014-06-14 21:45:42 +00:00
|
|
|
if (!survey)
|
|
|
|
survey = new UserSurvey(this);
|
|
|
|
survey->show();
|
2014-06-13 17:56:46 +00:00
|
|
|
}
|
2017-11-30 14:57:26 +00:00
|
|
|
s.endGroup();
|
2014-06-13 17:56:46 +00:00
|
|
|
}
|
|
|
|
|
2013-04-27 09:09:57 +00:00
|
|
|
void MainWindow::writeSettings()
|
|
|
|
{
|
2018-08-29 10:08:47 +00:00
|
|
|
QSettings settings;
|
|
|
|
|
|
|
|
settings.beginGroup("MainWindow");
|
|
|
|
settings.setValue("geometry", saveGeometry());
|
|
|
|
settings.setValue("windowState", saveState());
|
|
|
|
settings.setValue("maximized", isMaximized());
|
|
|
|
settings.setValue("lastState", (int)state);
|
2014-01-16 04:50:56 +00:00
|
|
|
if (state == VIEWALL)
|
2013-11-07 16:37:27 +00:00
|
|
|
saveSplitterSizes();
|
2018-08-29 10:08:47 +00:00
|
|
|
settings.endGroup();
|
2013-04-27 09:09:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::closeEvent(QCloseEvent *event)
|
|
|
|
{
|
2014-01-16 04:50:56 +00:00
|
|
|
if (DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING ||
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->isEditing()) {
|
2014-06-04 01:22:22 +00:00
|
|
|
on_actionQuit_triggered();
|
2013-12-17 00:37:44 +00:00
|
|
|
event->ignore();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-15 08:30:42 +00:00
|
|
|
if (unsaved_changes() && (askSaveChanges() == false)) {
|
2013-04-27 09:09:57 +00:00
|
|
|
event->ignore();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
event->accept();
|
|
|
|
writeSettings();
|
2014-03-11 16:31:01 +00:00
|
|
|
QApplication::closeAllWindows();
|
2013-04-28 03:47:47 +00:00
|
|
|
}
|
2013-05-19 03:09:36 +00:00
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
void MainWindow::loadRecentFiles()
|
2014-02-13 21:48:07 +00:00
|
|
|
{
|
2017-11-30 10:52:42 +00:00
|
|
|
recentFiles.clear();
|
|
|
|
QSettings s;
|
|
|
|
s.beginGroup("Recent_Files");
|
|
|
|
foreach (const QString &key, s.childKeys()) {
|
|
|
|
// TODO Sorting only correct up to 9 entries. Currently, only 4 used, so no problem.
|
|
|
|
if (!key.startsWith("File_"))
|
|
|
|
continue;
|
|
|
|
QString file = s.value(key).toString();
|
|
|
|
if (QFile::exists(file))
|
|
|
|
recentFiles.append(file);
|
|
|
|
if (recentFiles.count() > NUM_RECENT_FILES)
|
2014-02-13 21:48:07 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-11-30 10:52:42 +00:00
|
|
|
s.endGroup();
|
|
|
|
updateRecentFilesMenu();
|
|
|
|
}
|
2014-02-13 21:48:07 +00:00
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
void MainWindow::updateRecentFilesMenu()
|
|
|
|
{
|
|
|
|
for (int c = 0; c < NUM_RECENT_FILES; c++) {
|
2017-11-30 16:44:32 +00:00
|
|
|
QAction *action = actionsRecent[c];
|
2014-02-13 21:48:07 +00:00
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
if (recentFiles.count() > c) {
|
|
|
|
QFileInfo fi(recentFiles.at(c));
|
2014-02-13 21:48:07 +00:00
|
|
|
action->setText(fi.fileName());
|
|
|
|
action->setToolTip(fi.absoluteFilePath());
|
|
|
|
action->setVisible(true);
|
2014-02-14 06:11:05 +00:00
|
|
|
} else {
|
2014-02-13 21:48:07 +00:00
|
|
|
action->setVisible(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
void MainWindow::addRecentFile(const QString &file, bool update)
|
2014-02-13 21:48:07 +00:00
|
|
|
{
|
2017-11-30 10:52:42 +00:00
|
|
|
QString localFile = QDir::toNativeSeparators(file);
|
|
|
|
int index = recentFiles.indexOf(localFile);
|
|
|
|
if (index >= 0)
|
|
|
|
recentFiles.removeAt(index);
|
|
|
|
recentFiles.prepend(localFile);
|
|
|
|
while (recentFiles.count() > NUM_RECENT_FILES)
|
|
|
|
recentFiles.removeLast();
|
|
|
|
if (update)
|
|
|
|
updateRecentFiles();
|
2014-02-13 21:48:07 +00:00
|
|
|
}
|
|
|
|
|
2017-11-30 10:52:42 +00:00
|
|
|
void MainWindow::updateRecentFiles()
|
2014-02-28 09:04:00 +00:00
|
|
|
{
|
|
|
|
QSettings s;
|
|
|
|
|
|
|
|
s.beginGroup("Recent_Files");
|
2017-11-30 10:52:42 +00:00
|
|
|
s.remove(""); // Remove all old entries
|
|
|
|
for (int c = 1; c <= recentFiles.count(); c++) {
|
2014-02-28 09:04:00 +00:00
|
|
|
QString key = QString("File_%1").arg(c);
|
2017-11-30 10:52:42 +00:00
|
|
|
s.setValue(key, recentFiles.at(c - 1));
|
2014-02-28 09:04:00 +00:00
|
|
|
}
|
|
|
|
s.endGroup();
|
|
|
|
s.sync();
|
2017-11-30 10:52:42 +00:00
|
|
|
updateRecentFilesMenu();
|
2014-02-28 09:04:00 +00:00
|
|
|
}
|
|
|
|
|
2018-05-21 16:09:09 +00:00
|
|
|
void MainWindow::recentFileTriggered(bool)
|
2014-02-13 21:48:07 +00:00
|
|
|
{
|
2014-06-09 02:46:43 +00:00
|
|
|
if (!okToClose(tr("Please save or cancel the current dive edit before opening a new file.")))
|
|
|
|
return;
|
|
|
|
|
2017-11-30 16:56:16 +00:00
|
|
|
int filenr = ((QAction *)sender())->data().toInt();
|
|
|
|
if (filenr >= recentFiles.count())
|
|
|
|
return;
|
|
|
|
const QString &filename = recentFiles[filenr];
|
2014-02-13 21:48:07 +00:00
|
|
|
|
|
|
|
updateLastUsedDir(QFileInfo(filename).dir().path());
|
2014-06-09 02:46:43 +00:00
|
|
|
closeCurrentFile();
|
2014-02-13 21:48:07 +00:00
|
|
|
loadFiles(QStringList() << filename);
|
|
|
|
}
|
|
|
|
|
2014-03-14 17:19:23 +00:00
|
|
|
int MainWindow::file_save_as(void)
|
2013-05-19 22:25:47 +00:00
|
|
|
{
|
|
|
|
QString filename;
|
2014-05-13 21:13:03 +00:00
|
|
|
const char *default_filename = existing_filename;
|
2015-05-11 22:26:56 +00:00
|
|
|
|
2015-09-20 15:39:15 +00:00
|
|
|
// if the default is to save to cloud storage, pick something that will work as local file:
|
|
|
|
// simply extract the branch name which should be the users email address
|
|
|
|
if (default_filename && strstr(default_filename, prefs.cloud_git_url)) {
|
|
|
|
QString filename(default_filename);
|
|
|
|
filename.remove(prefs.cloud_git_url);
|
|
|
|
filename.remove(0, filename.indexOf("[") + 1);
|
|
|
|
filename.replace("]", ".ssrf");
|
2018-02-28 22:37:09 +00:00
|
|
|
default_filename = copy_qstring(filename);
|
2015-09-20 15:39:15 +00:00
|
|
|
}
|
2015-05-11 22:26:56 +00:00
|
|
|
// create a file dialog that allows us to save to a new file
|
2015-03-21 15:10:55 +00:00
|
|
|
QFileDialog selection_dialog(this, tr("Save file as"), default_filename,
|
2017-10-27 12:52:27 +00:00
|
|
|
tr("Subsurface files") + " (*.ssrf *.xml)");
|
2015-05-11 22:26:56 +00:00
|
|
|
selection_dialog.setAcceptMode(QFileDialog::AcceptSave);
|
|
|
|
selection_dialog.setFileMode(QFileDialog::AnyFile);
|
2015-05-27 18:14:26 +00:00
|
|
|
selection_dialog.setDefaultSuffix("");
|
2018-01-07 10:12:48 +00:00
|
|
|
if (empty_string(default_filename)) {
|
2015-10-05 02:34:27 +00:00
|
|
|
QFileInfo defaultFile(system_default_filename());
|
|
|
|
selection_dialog.setDirectory(qPrintable(defaultFile.absolutePath()));
|
|
|
|
}
|
2015-03-21 15:10:55 +00:00
|
|
|
/* if the exit/cancel button is pressed return */
|
|
|
|
if (!selection_dialog.exec())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* get the first selected file */
|
|
|
|
filename = selection_dialog.selectedFiles().at(0);
|
2015-05-27 18:14:26 +00:00
|
|
|
|
|
|
|
/* now for reasons I don't understand we appear to add a .ssrf to
|
|
|
|
* git style filenames <path>/directory[branch]
|
|
|
|
* so let's remove that */
|
2015-06-21 23:28:38 +00:00
|
|
|
QRegularExpression reg(".*\\[[^]]+]\\.ssrf", QRegularExpression::CaseInsensitiveOption);
|
|
|
|
if (reg.match(filename).hasMatch())
|
|
|
|
filename.remove(QRegularExpression("\\.ssrf$", QRegularExpression::CaseInsensitiveOption));
|
2014-03-14 17:19:23 +00:00
|
|
|
if (filename.isNull() || filename.isEmpty())
|
|
|
|
return report_error("No filename to save into");
|
2013-11-23 02:40:48 +00:00
|
|
|
|
2018-10-12 14:13:42 +00:00
|
|
|
if (mainTab->isEditing())
|
|
|
|
mainTab->acceptChanges();
|
2013-11-23 02:40:48 +00:00
|
|
|
|
2018-02-25 12:51:41 +00:00
|
|
|
if (save_dives(qPrintable(filename)))
|
2014-03-14 17:19:23 +00:00
|
|
|
return -1;
|
|
|
|
|
2018-02-25 12:51:41 +00:00
|
|
|
setCurrentFile(qPrintable(filename));
|
2014-03-14 17:19:23 +00:00
|
|
|
mark_divelist_changed(false);
|
2017-11-30 10:52:42 +00:00
|
|
|
addRecentFile(filename, true);
|
2014-03-14 17:19:23 +00:00
|
|
|
return 0;
|
2013-05-19 22:25:47 +00:00
|
|
|
}
|
|
|
|
|
2014-03-14 17:19:23 +00:00
|
|
|
int MainWindow::file_save(void)
|
2013-05-19 22:25:47 +00:00
|
|
|
{
|
|
|
|
const char *current_default;
|
2015-09-28 17:05:20 +00:00
|
|
|
bool is_cloud = false;
|
2013-05-19 22:25:47 +00:00
|
|
|
|
|
|
|
if (!existing_filename)
|
|
|
|
return file_save_as();
|
|
|
|
|
2015-09-28 17:05:20 +00:00
|
|
|
is_cloud = (strncmp(existing_filename, "http", 4) == 0);
|
2018-05-14 18:17:08 +00:00
|
|
|
if (is_cloud && !saveToCloudOK())
|
|
|
|
return -1;
|
2015-09-28 17:05:20 +00:00
|
|
|
|
2018-10-12 14:13:42 +00:00
|
|
|
if (mainTab->isEditing())
|
|
|
|
mainTab->acceptChanges();
|
2013-11-23 02:40:48 +00:00
|
|
|
|
2013-05-19 22:25:47 +00:00
|
|
|
current_default = prefs.default_filename;
|
2014-02-14 06:11:05 +00:00
|
|
|
if (strcmp(existing_filename, current_default) == 0) {
|
2013-05-19 22:25:47 +00:00
|
|
|
/* if we are using the default filename the directory
|
|
|
|
* that we are creating the file in may not exist */
|
2013-05-20 00:18:44 +00:00
|
|
|
QDir current_def_dir = QFileInfo(current_default).absoluteDir();
|
|
|
|
if (!current_def_dir.exists())
|
|
|
|
current_def_dir.mkpath(current_def_dir.absolutePath());
|
2013-05-19 22:25:47 +00:00
|
|
|
}
|
2015-09-28 17:05:20 +00:00
|
|
|
if (is_cloud)
|
|
|
|
showProgressBar();
|
2014-03-14 17:35:09 +00:00
|
|
|
if (save_dives(existing_filename)) {
|
2015-09-28 17:05:20 +00:00
|
|
|
if (is_cloud)
|
|
|
|
hideProgressBar();
|
2014-03-14 17:19:23 +00:00
|
|
|
return -1;
|
2014-03-14 17:35:09 +00:00
|
|
|
}
|
2015-09-28 17:05:20 +00:00
|
|
|
if (is_cloud)
|
|
|
|
hideProgressBar();
|
2014-01-15 08:30:42 +00:00
|
|
|
mark_divelist_changed(false);
|
2017-11-30 10:52:42 +00:00
|
|
|
addRecentFile(QString(existing_filename), true);
|
2014-03-14 17:19:23 +00:00
|
|
|
return 0;
|
2013-05-19 22:25:47 +00:00
|
|
|
}
|
2013-05-22 06:13:45 +00:00
|
|
|
|
2015-02-26 14:07:39 +00:00
|
|
|
NotificationWidget *MainWindow::getNotificationWidget()
|
|
|
|
{
|
|
|
|
return ui.mainErrorMessage;
|
2013-05-22 06:13:45 +00:00
|
|
|
}
|
2013-06-26 12:13:06 +00:00
|
|
|
|
2015-06-12 13:53:00 +00:00
|
|
|
QString MainWindow::displayedFilename(QString fullFilename)
|
|
|
|
{
|
|
|
|
QFile f(fullFilename);
|
|
|
|
QFileInfo fileInfo(f);
|
|
|
|
QString fileName(fileInfo.fileName());
|
|
|
|
|
2016-07-17 03:27:01 +00:00
|
|
|
if (fullFilename.contains(prefs.cloud_git_url)) {
|
|
|
|
QString email = fileName.left(fileName.indexOf('['));
|
2018-09-13 16:08:26 +00:00
|
|
|
return git_local_only ?
|
|
|
|
tr("[local cache for] %1").arg(email) :
|
|
|
|
tr("[cloud storage for] %1").arg(email);
|
2016-07-17 03:27:01 +00:00
|
|
|
} else {
|
2015-06-12 13:53:00 +00:00
|
|
|
return fileName;
|
2016-07-17 03:27:01 +00:00
|
|
|
}
|
2015-06-12 13:53:00 +00:00
|
|
|
}
|
|
|
|
|
2016-07-17 03:27:01 +00:00
|
|
|
|
2015-06-16 13:04:34 +00:00
|
|
|
void MainWindow::setAutomaticTitle()
|
|
|
|
{
|
|
|
|
setTitle();
|
|
|
|
}
|
|
|
|
|
2017-12-11 16:43:53 +00:00
|
|
|
void MainWindow::setTitle()
|
2013-06-26 12:13:06 +00:00
|
|
|
{
|
2018-01-07 14:42:28 +00:00
|
|
|
if (empty_string(existing_filename)) {
|
2013-06-26 12:13:06 +00:00
|
|
|
setWindowTitle("Subsurface");
|
2017-12-11 16:43:53 +00:00
|
|
|
return;
|
2013-06-26 12:13:06 +00:00
|
|
|
}
|
2017-12-11 16:43:53 +00:00
|
|
|
|
|
|
|
QString unsaved = (unsaved_changes() ? " *" : "");
|
2019-02-18 20:47:20 +00:00
|
|
|
QString shown = QString(" (%1)").arg(filterWidget2.shownText());
|
|
|
|
setWindowTitle("Subsurface: " + displayedFilename(existing_filename) + unsaved + shown);
|
2013-06-26 12:13:06 +00:00
|
|
|
}
|
2013-09-09 08:59:03 +00:00
|
|
|
|
|
|
|
void MainWindow::importFiles(const QStringList fileNames)
|
|
|
|
{
|
2013-10-04 15:12:46 +00:00
|
|
|
if (fileNames.isEmpty())
|
|
|
|
return;
|
|
|
|
|
2013-09-09 08:59:03 +00:00
|
|
|
QByteArray fileNamePtr;
|
2018-09-28 08:21:23 +00:00
|
|
|
struct dive_table table = { 0 };
|
2018-12-23 11:46:45 +00:00
|
|
|
struct trip_table trips = { 0 };
|
2019-03-03 14:12:22 +00:00
|
|
|
struct dive_site_table sites = { 0 };
|
2014-03-14 18:26:07 +00:00
|
|
|
|
2013-09-09 08:59:03 +00:00
|
|
|
for (int i = 0; i < fileNames.size(); ++i) {
|
2013-12-20 01:02:34 +00:00
|
|
|
fileNamePtr = QFile::encodeName(fileNames.at(i));
|
2019-03-03 14:12:22 +00:00
|
|
|
parse_file(fileNamePtr.data(), &table, &trips, &sites);
|
2013-09-09 08:59:03 +00:00
|
|
|
}
|
2018-12-23 22:45:12 +00:00
|
|
|
QString source = fileNames.size() == 1 ? fileNames[0] : tr("multiple files");
|
2019-03-03 14:12:22 +00:00
|
|
|
Command::importDives(&table, &trips, &sites, IMPORT_MERGE_ALL_TRIPS, source);
|
2013-09-09 08:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::loadFiles(const QStringList fileNames)
|
|
|
|
{
|
2015-10-08 06:40:29 +00:00
|
|
|
if (fileNames.isEmpty()) {
|
|
|
|
refreshDisplay();
|
2013-10-04 15:12:46 +00:00
|
|
|
return;
|
2015-10-08 06:40:29 +00:00
|
|
|
}
|
2013-09-09 08:59:03 +00:00
|
|
|
QByteArray fileNamePtr;
|
|
|
|
|
2015-09-23 19:16:45 +00:00
|
|
|
showProgressBar();
|
2013-09-09 08:59:03 +00:00
|
|
|
for (int i = 0; i < fileNames.size(); ++i) {
|
2014-03-14 18:26:07 +00:00
|
|
|
fileNamePtr = QFile::encodeName(fileNames.at(i));
|
2019-02-28 21:45:17 +00:00
|
|
|
if (!parse_file(fileNamePtr.data(), &dive_table, &trip_table, &dive_site_table)) {
|
2018-01-03 16:11:52 +00:00
|
|
|
setCurrentFile(fileNamePtr.data());
|
2017-11-30 10:52:42 +00:00
|
|
|
addRecentFile(fileNamePtr, false);
|
2013-09-09 08:59:03 +00:00
|
|
|
}
|
|
|
|
}
|
2015-09-23 19:16:45 +00:00
|
|
|
hideProgressBar();
|
2017-11-30 10:52:42 +00:00
|
|
|
updateRecentFiles();
|
2018-09-23 10:53:35 +00:00
|
|
|
process_loaded_dives();
|
2018-07-23 21:41:23 +00:00
|
|
|
Command::clear();
|
2013-09-09 08:59:03 +00:00
|
|
|
|
2013-11-01 13:43:41 +00:00
|
|
|
refreshDisplay();
|
2015-06-22 19:14:42 +00:00
|
|
|
|
2015-06-25 17:55:40 +00:00
|
|
|
int min_datafile_version = get_min_datafile_version();
|
|
|
|
if (min_datafile_version >0 && min_datafile_version < DATAFORMAT_VERSION) {
|
2015-06-22 19:14:42 +00:00
|
|
|
QMessageBox::warning(this, tr("Opening datafile from older version"),
|
2015-06-25 17:55:40 +00:00
|
|
|
tr("You opened a data file from an older version of Subsurface. We recommend "
|
2015-09-04 22:25:35 +00:00
|
|
|
"you read the manual to learn about the changes in the new version, especially "
|
|
|
|
"about dive site management which has changed significantly.\n"
|
|
|
|
"Subsurface has already tried to pre-populate the data but it might be worth "
|
|
|
|
"while taking a look at the new dive site management system and to make "
|
2015-06-22 19:14:42 +00:00
|
|
|
"sure that everything looks correct."));
|
|
|
|
}
|
2013-09-09 08:59:03 +00:00
|
|
|
}
|
2013-10-16 19:05:19 +00:00
|
|
|
|
2018-10-01 12:17:38 +00:00
|
|
|
static const char *csvExtensions[] = {
|
|
|
|
".csv", ".apd", ".zxu", ".zxl", ".txt"
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool isCsvFile(const QString &s)
|
|
|
|
{
|
|
|
|
for (const char *ext: csvExtensions) {
|
|
|
|
if (s.endsWith(ext, Qt::CaseInsensitive))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-12-29 16:11:20 +00:00
|
|
|
void MainWindow::on_actionImportDiveLog_triggered()
|
2013-10-16 19:05:19 +00:00
|
|
|
{
|
2017-10-27 12:52:27 +00:00
|
|
|
QStringList fileNames = QFileDialog::getOpenFileNames(this, tr("Open dive log file"), lastUsedDir(), filter_import());
|
2014-01-07 20:01:28 +00:00
|
|
|
|
|
|
|
if (fileNames.isEmpty())
|
|
|
|
return;
|
|
|
|
updateLastUsedDir(QFileInfo(fileNames[0]).dir().path());
|
|
|
|
|
2018-10-01 12:17:38 +00:00
|
|
|
QStringList logFiles;
|
|
|
|
QStringList csvFiles;
|
|
|
|
for (const QString &fn: fileNames) {
|
|
|
|
if (isCsvFile(fn))
|
|
|
|
csvFiles.append(fn);
|
|
|
|
else
|
|
|
|
logFiles.append(fn);
|
|
|
|
}
|
2014-10-28 20:34:33 +00:00
|
|
|
|
2014-01-07 20:01:28 +00:00
|
|
|
if (logFiles.size()) {
|
|
|
|
importFiles(logFiles);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (csvFiles.size()) {
|
2018-10-01 16:12:59 +00:00
|
|
|
DiveLogImportDialog diveLogImport(csvFiles, this);
|
|
|
|
diveLogImport.exec();
|
2014-01-07 20:01:28 +00:00
|
|
|
}
|
2013-10-16 19:05:19 +00:00
|
|
|
}
|
2013-11-01 15:48:34 +00:00
|
|
|
|
|
|
|
void MainWindow::editCurrentDive()
|
|
|
|
{
|
2018-05-03 14:50:46 +00:00
|
|
|
if (!current_dive)
|
|
|
|
return;
|
|
|
|
|
2018-10-12 14:13:42 +00:00
|
|
|
if (mainTab->isEditing() || DivePlannerPointsModel::instance()->currentMode() != DivePlannerPointsModel::NOTHING) {
|
2014-02-26 14:59:06 +00:00
|
|
|
QMessageBox::warning(this, tr("Warning"), tr("Please, first finish the current edition before trying to do another."));
|
2013-11-09 00:09:46 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-01 15:48:34 +00:00
|
|
|
struct dive *d = current_dive;
|
|
|
|
QString defaultDC(d->dc.model);
|
2013-11-09 00:09:46 +00:00
|
|
|
DivePlannerPointsModel::instance()->clear();
|
2018-05-03 14:50:46 +00:00
|
|
|
disableShortcuts();
|
2014-01-16 04:50:56 +00:00
|
|
|
if (defaultDC == "manually added dive") {
|
2013-11-09 00:09:46 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::ADD);
|
2018-10-12 13:07:40 +00:00
|
|
|
graphics->setAddState();
|
2015-02-09 21:19:10 +00:00
|
|
|
setApplicationState("EditDive");
|
2013-11-01 15:48:34 +00:00
|
|
|
DivePlannerPointsModel::instance()->loadFromDive(d);
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->enableEdition(MainTab::MANUALLY_ADDED_DIVE);
|
2014-01-16 04:50:56 +00:00
|
|
|
} else if (defaultDC == "planned dive") {
|
2013-12-07 15:25:43 +00:00
|
|
|
DivePlannerPointsModel::instance()->setPlanMode(DivePlannerPointsModel::PLAN);
|
2015-02-09 21:19:10 +00:00
|
|
|
setApplicationState("EditPlannedDive");
|
2013-12-07 15:25:43 +00:00
|
|
|
DivePlannerPointsModel::instance()->loadFromDive(d);
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->enableEdition(MainTab::MANUALLY_ADDED_DIVE);
|
2018-05-03 14:50:46 +00:00
|
|
|
} else {
|
|
|
|
setApplicationState("EditDive");
|
2018-10-12 14:13:42 +00:00
|
|
|
mainTab->enableEdition();
|
2013-11-01 15:48:34 +00:00
|
|
|
}
|
|
|
|
}
|
2014-02-06 13:38:28 +00:00
|
|
|
|
2014-06-04 20:41:50 +00:00
|
|
|
void MainWindow::turnOffNdlTts()
|
|
|
|
{
|
2018-08-15 09:56:17 +00:00
|
|
|
qPrefTechnicalDetails::set_calcndltts(false);
|
2014-07-10 16:50:49 +00:00
|
|
|
}
|
|
|
|
|
2014-02-08 07:50:39 +00:00
|
|
|
#undef TOOLBOX_PREF_PROFILE
|
2014-07-10 22:48:00 +00:00
|
|
|
#undef PERF_PROFILE
|
2014-05-20 16:33:32 +00:00
|
|
|
|
|
|
|
void MainWindow::on_actionExport_triggered()
|
|
|
|
{
|
2014-07-10 22:53:58 +00:00
|
|
|
DiveLogExportDialog diveLogExport;
|
|
|
|
diveLogExport.exec();
|
2014-05-20 16:33:32 +00:00
|
|
|
}
|
2014-08-04 15:58:21 +00:00
|
|
|
|
2014-05-29 15:54:19 +00:00
|
|
|
void MainWindow::on_actionConfigure_Dive_Computer_triggered()
|
|
|
|
{
|
|
|
|
ConfigureDiveComputerDialog *dcConfig = new ConfigureDiveComputerDialog(this);
|
|
|
|
dcConfig->show();
|
|
|
|
}
|
|
|
|
|
2014-08-04 15:58:21 +00:00
|
|
|
void MainWindow::setEnabledToolbar(bool arg1)
|
|
|
|
{
|
2014-09-17 22:39:49 +00:00
|
|
|
Q_FOREACH (QAction *b, profileToolbarActions)
|
2014-08-04 15:58:21 +00:00
|
|
|
b->setEnabled(arg1);
|
|
|
|
}
|
2014-08-16 15:32:23 +00:00
|
|
|
|
|
|
|
void MainWindow::on_copy_triggered()
|
|
|
|
{
|
|
|
|
// open dialog to select what gets copied
|
|
|
|
// copy the displayed dive
|
|
|
|
DiveComponentSelection dialog(this, ©PasteDive, &what);
|
|
|
|
dialog.exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::on_paste_triggered()
|
|
|
|
{
|
2019-02-23 17:17:20 +00:00
|
|
|
Command::pasteDives(©PasteDive, what);
|
2014-08-16 15:32:23 +00:00
|
|
|
}
|
2014-09-17 17:50:41 +00:00
|
|
|
|
|
|
|
void MainWindow::on_actionFilterTags_triggered()
|
|
|
|
{
|
2018-10-21 13:40:25 +00:00
|
|
|
setApplicationState(getCurrentAppState() == "FilterDive" ? "Default" : "FilterDive");
|
2019-02-10 00:01:50 +00:00
|
|
|
if (state == LIST_MAXIMIZED)
|
|
|
|
showFilterIfEnabled();
|
2017-10-27 15:52:49 +00:00
|
|
|
}
|
|
|
|
|
2019-02-10 00:01:50 +00:00
|
|
|
void MainWindow::showFilterIfEnabled()
|
|
|
|
{
|
|
|
|
if (getCurrentAppState() == "FilterDive") {
|
|
|
|
const int appW = qApp->desktop()->size().width();
|
|
|
|
QList<int> profileFilterSizes = { round_int(appW * 0.7), round_int(appW * 0.3) };
|
|
|
|
ui.bottomSplitter->setSizes(profileFilterSizes);
|
|
|
|
} else {
|
|
|
|
ui.bottomSplitter->setSizes({ EXPANDED, COLLAPSED });
|
|
|
|
}
|
|
|
|
}
|
2015-02-10 18:41:15 +00:00
|
|
|
void MainWindow::registerApplicationState(const QByteArray& state, QWidget *topLeft, QWidget *topRight, QWidget *bottomLeft, QWidget *bottomRight)
|
2015-02-09 17:42:32 +00:00
|
|
|
{
|
2015-02-10 18:41:15 +00:00
|
|
|
applicationState[state] = WidgetForQuadrant(topLeft, topRight, bottomLeft, bottomRight);
|
2015-02-10 19:36:16 +00:00
|
|
|
if (ui.topLeft->indexOf(topLeft) == -1 && topLeft) {
|
2015-02-10 17:51:30 +00:00
|
|
|
ui.topLeft->addWidget(topLeft);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
if (ui.topRight->indexOf(topRight) == -1 && topRight) {
|
2015-02-10 17:51:30 +00:00
|
|
|
ui.topRight->addWidget(topRight);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
if (ui.bottomLeft->indexOf(bottomLeft) == -1 && bottomLeft) {
|
2015-02-10 17:51:30 +00:00
|
|
|
ui.bottomLeft->addWidget(bottomLeft);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
if(ui.bottomRight->indexOf(bottomRight) == -1 && bottomRight) {
|
2015-02-10 17:51:30 +00:00
|
|
|
ui.bottomRight->addWidget(bottomRight);
|
|
|
|
}
|
2015-02-09 17:42:32 +00:00
|
|
|
}
|
2015-02-09 19:10:08 +00:00
|
|
|
|
2019-02-23 17:31:02 +00:00
|
|
|
void MainWindow::setApplicationState(const QByteArray &state)
|
|
|
|
{
|
2015-02-09 19:10:08 +00:00
|
|
|
if (!applicationState.keys().contains(state))
|
|
|
|
return;
|
|
|
|
|
2015-08-22 14:05:13 +00:00
|
|
|
if (getCurrentAppState() == state)
|
2015-02-10 17:22:14 +00:00
|
|
|
return;
|
|
|
|
|
2015-08-22 14:05:13 +00:00
|
|
|
setCurrentAppState(state);
|
2015-08-12 10:06:52 +00:00
|
|
|
|
2015-02-10 19:36:16 +00:00
|
|
|
#define SET_CURRENT_INDEX( X ) \
|
|
|
|
if (applicationState[state].X) { \
|
|
|
|
ui.X->setCurrentWidget( applicationState[state].X); \
|
|
|
|
ui.X->show(); \
|
|
|
|
} else { \
|
|
|
|
ui.X->hide(); \
|
|
|
|
}
|
|
|
|
|
|
|
|
SET_CURRENT_INDEX( topLeft )
|
2015-08-21 01:24:49 +00:00
|
|
|
Q_FOREACH(const WidgetProperty& p, stateProperties[state].topLeft) {
|
|
|
|
ui.topLeft->currentWidget()->setProperty( p.first.data(), p.second);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
SET_CURRENT_INDEX( topRight )
|
2015-08-21 01:24:49 +00:00
|
|
|
Q_FOREACH(const WidgetProperty& p, stateProperties[state].topRight) {
|
|
|
|
ui.topRight->currentWidget()->setProperty( p.first.data(), p.second);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
SET_CURRENT_INDEX( bottomLeft )
|
2015-08-21 01:24:49 +00:00
|
|
|
Q_FOREACH(const WidgetProperty& p, stateProperties[state].bottomLeft) {
|
|
|
|
ui.bottomLeft->currentWidget()->setProperty( p.first.data(), p.second);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
SET_CURRENT_INDEX( bottomRight )
|
2015-08-21 01:24:49 +00:00
|
|
|
Q_FOREACH(const WidgetProperty& p, stateProperties[state].bottomRight) {
|
|
|
|
ui.bottomRight->currentWidget()->setProperty( p.first.data(), p.second);
|
|
|
|
}
|
2015-02-10 19:36:16 +00:00
|
|
|
#undef SET_CURRENT_INDEX
|
2015-02-09 19:10:08 +00:00
|
|
|
}
|
2015-08-25 19:49:48 +00:00
|
|
|
|
2015-09-09 20:02:39 +00:00
|
|
|
void MainWindow::showProgressBar()
|
2015-08-25 19:49:48 +00:00
|
|
|
{
|
2015-10-09 21:19:10 +00:00
|
|
|
delete progressDialog;
|
2015-09-09 20:02:39 +00:00
|
|
|
|
|
|
|
progressDialog = new QProgressDialog(tr("Contacting cloud service..."), tr("Cancel"), 0, 100, this);
|
|
|
|
progressDialog->setWindowModality(Qt::WindowModal);
|
2017-06-18 06:22:37 +00:00
|
|
|
progressDialog->setMinimumDuration(0);
|
2015-09-09 20:02:39 +00:00
|
|
|
progressDialogCanceled = false;
|
2017-06-18 08:46:49 +00:00
|
|
|
progressCounter = 0;
|
2015-09-09 20:02:39 +00:00
|
|
|
connect(progressDialog, SIGNAL(canceled()), this, SLOT(cancelCloudStorageOperation()));
|
2015-08-25 19:49:48 +00:00
|
|
|
}
|
|
|
|
|
2015-09-09 20:02:39 +00:00
|
|
|
void MainWindow::cancelCloudStorageOperation()
|
2015-08-25 19:49:48 +00:00
|
|
|
{
|
2015-09-09 20:02:39 +00:00
|
|
|
progressDialogCanceled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::hideProgressBar()
|
|
|
|
{
|
|
|
|
if (progressDialog) {
|
|
|
|
progressDialog->setValue(100);
|
2018-05-25 21:08:18 +00:00
|
|
|
delete progressDialog;
|
2018-09-15 15:25:10 +00:00
|
|
|
progressDialog = nullptr;
|
2015-09-09 20:02:39 +00:00
|
|
|
}
|
2015-08-25 19:49:48 +00:00
|
|
|
}
|
2018-03-17 09:48:45 +00:00
|
|
|
|
|
|
|
void MainWindow::unsetProfHR()
|
|
|
|
{
|
|
|
|
ui.profHR->setChecked(false);
|
2018-08-15 09:56:17 +00:00
|
|
|
qPrefTechnicalDetails::set_hrgraph(false);
|
2018-03-17 09:48:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MainWindow::unsetProfTissues()
|
|
|
|
{
|
|
|
|
ui.profTissues->setChecked(false);
|
2018-08-15 09:56:17 +00:00
|
|
|
qPrefTechnicalDetails::set_percentagegraph(false);
|
2018-03-17 09:48:45 +00:00
|
|
|
}
|