2013-05-20 19:43:33 +00:00
|
|
|
#include "downloadfromdivecomputer.h"
|
2014-12-27 16:31:51 +00:00
|
|
|
#include "divecomputer.h"
|
|
|
|
#include "libdivecomputer.h"
|
|
|
|
#include "helpers.h"
|
|
|
|
#include "display.h"
|
|
|
|
#include "divelist.h"
|
2013-05-30 08:58:59 +00:00
|
|
|
#include "mainwindow.h"
|
2014-12-27 16:31:51 +00:00
|
|
|
|
2013-05-20 20:58:06 +00:00
|
|
|
#include <cstdlib>
|
2013-05-20 19:43:33 +00:00
|
|
|
#include <QThread>
|
|
|
|
#include <QDebug>
|
2013-05-20 20:58:06 +00:00
|
|
|
#include <QStringListModel>
|
2013-08-25 22:02:30 +00:00
|
|
|
#include <QTimer>
|
2013-12-25 00:26:00 +00:00
|
|
|
#include <QFileDialog>
|
2013-09-01 14:14:54 +00:00
|
|
|
#include <QMessageBox>
|
2014-04-25 17:44:23 +00:00
|
|
|
#include <QShortcut>
|
2013-05-20 20:58:06 +00:00
|
|
|
|
|
|
|
struct product {
|
|
|
|
const char *product;
|
|
|
|
dc_descriptor_t *descriptor;
|
|
|
|
struct product *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct vendor {
|
|
|
|
const char *vendor;
|
|
|
|
struct product *productlist;
|
|
|
|
struct vendor *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mydescriptor {
|
|
|
|
const char *vendor;
|
|
|
|
const char *product;
|
|
|
|
dc_family_t type;
|
|
|
|
unsigned int model;
|
|
|
|
};
|
2013-05-20 19:43:33 +00:00
|
|
|
|
2014-05-22 18:40:22 +00:00
|
|
|
namespace DownloadFromDcGlobal {
|
2013-05-20 19:43:33 +00:00
|
|
|
const char *err_string;
|
|
|
|
};
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
DownloadFromDCWidget::DownloadFromDCWidget(QWidget *parent, Qt::WindowFlags f) : QDialog(parent, f),
|
2014-02-09 18:38:26 +00:00
|
|
|
thread(0),
|
|
|
|
downloading(false),
|
|
|
|
previousLast(0),
|
|
|
|
vendorModel(0),
|
|
|
|
productModel(0),
|
|
|
|
timer(new QTimer(this)),
|
|
|
|
dumpWarningShown(false),
|
2014-12-29 23:25:54 +00:00
|
|
|
ostcFirmwareCheck(0),
|
|
|
|
currentState(INITIAL)
|
2013-05-20 19:43:33 +00:00
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.setupUi(this);
|
|
|
|
ui.progressBar->hide();
|
|
|
|
ui.progressBar->setMinimum(0);
|
|
|
|
ui.progressBar->setMaximum(100);
|
2015-01-08 12:12:11 +00:00
|
|
|
diveImportedModel = new DiveImportedModel(this);
|
2015-01-08 13:29:28 +00:00
|
|
|
ui.downloadedView->setModel(diveImportedModel);
|
2015-01-08 19:19:01 +00:00
|
|
|
ui.downloadedView->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
ui.downloadedView->setSelectionMode(QAbstractItemView::SingleSelection);
|
|
|
|
int startingWidth = defaultModelFont().pointSize();
|
|
|
|
ui.downloadedView->setColumnWidth(0, startingWidth * 20);
|
|
|
|
ui.downloadedView->setColumnWidth(1, startingWidth * 15);
|
|
|
|
ui.downloadedView->setColumnWidth(2, startingWidth * 10);
|
2015-01-08 19:39:34 +00:00
|
|
|
connect(ui.downloadedView, SIGNAL(clicked(QModelIndex)), diveImportedModel, SLOT(changeSelected(QModelIndex)));
|
2015-01-08 19:19:01 +00:00
|
|
|
QRect mainGeometry = parent->geometry();
|
|
|
|
int width = mainGeometry.width() * 0.8;
|
|
|
|
int height = mainGeometry.height() * 0.8;
|
|
|
|
resize(width, height);
|
2013-09-16 21:04:42 +00:00
|
|
|
|
2014-02-26 16:37:13 +00:00
|
|
|
progress_bar_text = "";
|
|
|
|
|
2013-05-20 20:58:06 +00:00
|
|
|
fill_computer_list();
|
|
|
|
|
2013-12-25 00:26:00 +00:00
|
|
|
ui.chooseDumpFile->setEnabled(ui.dumpToFile->isChecked());
|
|
|
|
connect(ui.chooseDumpFile, SIGNAL(clicked()), this, SLOT(pickDumpFile()));
|
|
|
|
connect(ui.dumpToFile, SIGNAL(stateChanged(int)), this, SLOT(checkDumpFile(int)));
|
|
|
|
ui.chooseLogFile->setEnabled(ui.logToFile->isChecked());
|
|
|
|
connect(ui.chooseLogFile, SIGNAL(clicked()), this, SLOT(pickLogFile()));
|
|
|
|
connect(ui.logToFile, SIGNAL(stateChanged(int)), this, SLOT(checkLogFile(int)));
|
2015-01-08 20:54:53 +00:00
|
|
|
ui.selectAllButton->setEnabled(false);
|
|
|
|
ui.unselectAllButton->setEnabled(false);
|
|
|
|
connect(ui.selectAllButton, SIGNAL(clicked()), diveImportedModel, SLOT(selectAll()));
|
|
|
|
connect(ui.unselectAllButton, SIGNAL(clicked()), diveImportedModel, SLOT(selectNone()));
|
2013-05-20 20:58:06 +00:00
|
|
|
vendorModel = new QStringListModel(vendorList);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.vendor->setModel(vendorModel);
|
2013-05-23 06:24:33 +00:00
|
|
|
if (default_dive_computer_vendor) {
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.vendor->setCurrentIndex(ui.vendor->findText(default_dive_computer_vendor));
|
2013-05-23 06:24:33 +00:00
|
|
|
productModel = new QStringListModel(productList[default_dive_computer_vendor]);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.product->setModel(productModel);
|
2013-05-23 06:24:33 +00:00
|
|
|
if (default_dive_computer_product)
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.product->setCurrentIndex(ui.product->findText(default_dive_computer_product));
|
2013-05-23 06:24:33 +00:00
|
|
|
}
|
|
|
|
if (default_dive_computer_device)
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.device->setEditText(default_dive_computer_device);
|
2013-08-25 22:02:30 +00:00
|
|
|
|
|
|
|
timer->setInterval(200);
|
|
|
|
connect(timer, SIGNAL(timeout()), this, SLOT(updateProgressBar()));
|
|
|
|
updateState(INITIAL);
|
2014-02-15 14:03:54 +00:00
|
|
|
memset(&data, 0, sizeof(data));
|
2014-04-25 17:44:23 +00:00
|
|
|
QShortcut *close = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_W), this);
|
|
|
|
connect(close, SIGNAL(activated()), this, SLOT(close()));
|
|
|
|
QShortcut *quit = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q), this);
|
|
|
|
connect(quit, SIGNAL(activated()), parent, SLOT(close()));
|
2015-01-08 17:39:47 +00:00
|
|
|
ui.ok->setEnabled(false);
|
2015-01-09 20:00:34 +00:00
|
|
|
ui.startDownload->setEnabled(true);
|
2013-05-20 20:58:06 +00:00
|
|
|
}
|
|
|
|
|
2013-08-25 22:02:30 +00:00
|
|
|
void DownloadFromDCWidget::updateProgressBar()
|
2013-05-30 09:21:08 +00:00
|
|
|
{
|
2014-01-27 20:47:40 +00:00
|
|
|
if (*progress_bar_text != '\0') {
|
|
|
|
ui.progressBar->setFormat(progress_bar_text);
|
|
|
|
} else {
|
|
|
|
ui.progressBar->setFormat("%p%");
|
|
|
|
}
|
2014-05-07 19:17:24 +00:00
|
|
|
ui.progressBar->setValue(progress_bar_fraction * 100);
|
2013-08-25 22:02:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadFromDCWidget::updateState(states state)
|
|
|
|
{
|
|
|
|
if (state == currentState)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (state == INITIAL) {
|
2014-05-18 04:29:40 +00:00
|
|
|
fill_device_list(DC_TYPE_OTHER);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.progressBar->hide();
|
2013-08-25 22:02:30 +00:00
|
|
|
markChildrenAsEnabled();
|
|
|
|
timer->stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// tries to cancel an on going download
|
|
|
|
else if (currentState == DOWNLOADING && state == CANCELLING) {
|
|
|
|
import_thread_cancelled = true;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.cancel->setEnabled(false);
|
2013-08-25 22:02:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// user pressed cancel but the application isn't doing anything.
|
|
|
|
// means close the window
|
2014-02-28 04:09:57 +00:00
|
|
|
else if ((currentState == INITIAL || currentState == CANCELLED || currentState == DONE || currentState == ERROR) && state == CANCELLING) {
|
2013-08-25 22:02:30 +00:00
|
|
|
timer->stop();
|
|
|
|
reject();
|
2013-11-09 06:57:36 +00:00
|
|
|
ui.ok->setText(tr("OK"));
|
2013-08-25 22:02:30 +00:00
|
|
|
}
|
|
|
|
|
2013-09-01 14:14:54 +00:00
|
|
|
// the cancelation process is finished
|
2013-08-25 22:02:30 +00:00
|
|
|
else if (currentState == CANCELLING && (state == DONE || state == CANCELLED)) {
|
|
|
|
timer->stop();
|
|
|
|
state = CANCELLED;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.progressBar->setValue(0);
|
|
|
|
ui.progressBar->hide();
|
2013-08-25 22:02:30 +00:00
|
|
|
markChildrenAsEnabled();
|
|
|
|
}
|
|
|
|
|
2014-01-27 20:47:40 +00:00
|
|
|
// DOWNLOAD is finally done, but we don't know if there was an error as libdivecomputer doesn't pass
|
2015-01-08 17:29:57 +00:00
|
|
|
// that information on to us.
|
|
|
|
// If we find an error, offer to retry, otherwise continue the interaction to pick the dives the user wants
|
2013-08-25 22:02:30 +00:00
|
|
|
else if (currentState == DOWNLOADING && state == DONE) {
|
|
|
|
timer->stop();
|
2014-01-27 20:47:40 +00:00
|
|
|
if (QString(progress_bar_text).contains("error", Qt::CaseInsensitive)) {
|
|
|
|
updateProgressBar();
|
|
|
|
markChildrenAsEnabled();
|
|
|
|
progress_bar_text = "";
|
|
|
|
ui.ok->setText(tr("Retry"));
|
|
|
|
} else {
|
|
|
|
ui.progressBar->setValue(100);
|
|
|
|
markChildrenAsEnabled();
|
|
|
|
ui.ok->setText(tr("OK"));
|
|
|
|
}
|
2013-08-25 22:02:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DOWNLOAD is started.
|
|
|
|
else if (state == DOWNLOADING) {
|
|
|
|
timer->start();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.progressBar->setValue(0);
|
2014-01-27 20:47:40 +00:00
|
|
|
updateProgressBar();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.progressBar->show();
|
2013-08-25 22:02:30 +00:00
|
|
|
markChildrenAsDisabled();
|
|
|
|
}
|
|
|
|
|
2013-09-01 14:14:54 +00:00
|
|
|
// got an error
|
|
|
|
else if (state == ERROR) {
|
2013-11-21 12:23:10 +00:00
|
|
|
QMessageBox::critical(this, TITLE_OR_TEXT(tr("Error"), this->thread->error), QMessageBox::Ok);
|
2013-09-01 14:14:54 +00:00
|
|
|
|
|
|
|
markChildrenAsEnabled();
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.progressBar->hide();
|
2013-11-09 06:57:36 +00:00
|
|
|
ui.ok->setText(tr("Retry"));
|
2013-09-01 14:14:54 +00:00
|
|
|
}
|
|
|
|
|
2013-08-25 22:02:30 +00:00
|
|
|
// properly updating the widget state
|
|
|
|
currentState = state;
|
2013-05-30 09:21:08 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
void DownloadFromDCWidget::on_vendor_currentIndexChanged(const QString &vendor)
|
2013-05-20 20:58:06 +00:00
|
|
|
{
|
2014-05-18 04:29:40 +00:00
|
|
|
int dcType = DC_TYPE_SERIAL;
|
2013-10-03 18:54:25 +00:00
|
|
|
QAbstractItemModel *currentModel = ui.product->model();
|
2013-05-20 20:58:06 +00:00
|
|
|
if (!currentModel)
|
|
|
|
return;
|
|
|
|
|
|
|
|
productModel = new QStringListModel(productList[vendor]);
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.product->setModel(productModel);
|
2013-05-20 20:58:06 +00:00
|
|
|
|
2014-05-18 04:29:40 +00:00
|
|
|
if (vendor == QString("Uemis"))
|
|
|
|
dcType = DC_TYPE_UEMIS;
|
|
|
|
fill_device_list(dcType);
|
|
|
|
|
2013-05-20 20:58:06 +00:00
|
|
|
// Memleak - but deleting gives me a crash.
|
|
|
|
//currentModel->deleteLater();
|
|
|
|
}
|
|
|
|
|
2014-05-25 22:06:56 +00:00
|
|
|
void DownloadFromDCWidget::on_product_currentIndexChanged(const QString &product)
|
2013-12-27 10:50:13 +00:00
|
|
|
{
|
|
|
|
// Set up the DC descriptor
|
|
|
|
dc_descriptor_t *descriptor = NULL;
|
2014-05-25 22:06:56 +00:00
|
|
|
descriptor = descriptorLookup[ui.vendor->currentText() + product];
|
2013-12-27 10:50:13 +00:00
|
|
|
|
|
|
|
// call dc_descriptor_get_transport to see if the dc_transport_t is DC_TRANSPORT_SERIAL
|
|
|
|
if (dc_descriptor_get_transport(descriptor) == DC_TRANSPORT_SERIAL) {
|
|
|
|
// if the dc_transport_t is DC_TRANSPORT_SERIAL, then enable the device node box.
|
|
|
|
ui.device->setEnabled(true);
|
|
|
|
} else {
|
|
|
|
// otherwise disable the device node box
|
|
|
|
ui.device->setEnabled(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-20 20:58:06 +00:00
|
|
|
void DownloadFromDCWidget::fill_computer_list()
|
|
|
|
{
|
|
|
|
dc_iterator_t *iterator = NULL;
|
|
|
|
dc_descriptor_t *descriptor = NULL;
|
|
|
|
struct mydescriptor *mydescriptor;
|
|
|
|
|
|
|
|
QStringList computer;
|
|
|
|
dc_descriptor_iterator(&iterator);
|
2014-02-28 04:09:57 +00:00
|
|
|
while (dc_iterator_next(iterator, &descriptor) == DC_STATUS_SUCCESS) {
|
2013-05-20 20:58:06 +00:00
|
|
|
const char *vendor = dc_descriptor_get_vendor(descriptor);
|
|
|
|
const char *product = dc_descriptor_get_product(descriptor);
|
|
|
|
|
2013-05-23 04:25:05 +00:00
|
|
|
if (!vendorList.contains(vendor))
|
|
|
|
vendorList.append(vendor);
|
2013-05-20 20:58:06 +00:00
|
|
|
|
2013-05-23 04:25:05 +00:00
|
|
|
if (!productList[vendor].contains(product))
|
|
|
|
productList[vendor].push_back(product);
|
2013-05-21 04:55:56 +00:00
|
|
|
|
|
|
|
descriptorLookup[QString(vendor) + QString(product)] = descriptor;
|
2013-05-20 20:58:06 +00:00
|
|
|
}
|
|
|
|
dc_iterator_free(iterator);
|
|
|
|
|
|
|
|
/* and add the Uemis Zurich which we are handling internally
|
|
|
|
THIS IS A HACK as we magically have a data structure here that
|
|
|
|
happens to match a data structure that is internal to libdivecomputer;
|
|
|
|
this WILL BREAK if libdivecomputer changes the dc_descriptor struct...
|
|
|
|
eventually the UEMIS code needs to move into libdivecomputer, I guess */
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
mydescriptor = (struct mydescriptor *)malloc(sizeof(struct mydescriptor));
|
2013-05-20 20:58:06 +00:00
|
|
|
mydescriptor->vendor = "Uemis";
|
|
|
|
mydescriptor->product = "Zurich";
|
|
|
|
mydescriptor->type = DC_FAMILY_NULL;
|
|
|
|
mydescriptor->model = 0;
|
|
|
|
|
2013-05-23 04:25:05 +00:00
|
|
|
if (!vendorList.contains("Uemis"))
|
2013-05-20 20:58:06 +00:00
|
|
|
vendorList.append("Uemis");
|
|
|
|
|
2013-05-23 04:25:05 +00:00
|
|
|
if (!productList["Uemis"].contains("Zurich"))
|
|
|
|
productList["Uemis"].push_back("Zurich");
|
2013-05-21 04:55:56 +00:00
|
|
|
|
2014-05-12 17:58:15 +00:00
|
|
|
descriptorLookup["UemisZurich"] = (dc_descriptor_t *)mydescriptor;
|
2014-01-14 14:05:41 +00:00
|
|
|
|
|
|
|
qSort(vendorList);
|
2013-05-20 19:43:33 +00:00
|
|
|
}
|
|
|
|
|
2014-02-19 01:20:35 +00:00
|
|
|
void DownloadFromDCWidget::on_search_clicked()
|
|
|
|
{
|
|
|
|
if (ui.vendor->currentText() == "Uemis") {
|
|
|
|
QString dirName = QFileDialog::getExistingDirectory(this,
|
2014-02-28 04:09:57 +00:00
|
|
|
tr("Find Uemis dive computer"),
|
|
|
|
QDir::homePath(),
|
|
|
|
QFileDialog::ShowDirsOnly);
|
2014-02-19 01:20:35 +00:00
|
|
|
if (ui.device->findText(dirName) == -1)
|
|
|
|
ui.device->addItem(dirName);
|
|
|
|
ui.device->setEditText(dirName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-20 19:43:33 +00:00
|
|
|
void DownloadFromDCWidget::on_cancel_clicked()
|
|
|
|
{
|
2013-08-25 22:02:30 +00:00
|
|
|
updateState(CANCELLING);
|
2013-05-20 19:43:33 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 13:29:28 +00:00
|
|
|
void DownloadFromDCWidget::on_startDownload_clicked()
|
2013-05-20 19:43:33 +00:00
|
|
|
{
|
2013-08-25 22:02:30 +00:00
|
|
|
updateState(DOWNLOADING);
|
2013-05-20 19:43:33 +00:00
|
|
|
|
2013-08-25 22:02:30 +00:00
|
|
|
// I don't really think that create/destroy the thread
|
|
|
|
// is really necessary.
|
2013-05-23 04:25:05 +00:00
|
|
|
if (thread) {
|
2013-05-20 19:43:33 +00:00
|
|
|
thread->deleteLater();
|
|
|
|
}
|
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
data.devname = strdup(ui.device->currentText().toUtf8().data());
|
|
|
|
data.vendor = strdup(ui.vendor->currentText().toUtf8().data());
|
|
|
|
data.product = strdup(ui.product->currentText().toUtf8().data());
|
2013-08-25 13:01:59 +00:00
|
|
|
|
2013-10-03 18:54:25 +00:00
|
|
|
data.descriptor = descriptorLookup[ui.vendor->currentText() + ui.product->currentText()];
|
|
|
|
data.force_download = ui.forceDownload->isChecked();
|
2014-07-24 20:59:11 +00:00
|
|
|
data.create_new_trip = ui.createNewTrip->isChecked();
|
|
|
|
data.trip = NULL;
|
2013-06-16 00:48:31 +00:00
|
|
|
data.deviceid = data.diveid = 0;
|
2013-05-23 06:24:33 +00:00
|
|
|
set_default_dive_computer(data.vendor, data.product);
|
|
|
|
set_default_dive_computer_device(data.devname);
|
2013-05-21 04:55:56 +00:00
|
|
|
|
2013-08-25 22:02:30 +00:00
|
|
|
thread = new DownloadThread(this, &data);
|
2013-05-20 20:02:17 +00:00
|
|
|
|
2013-08-25 22:02:30 +00:00
|
|
|
connect(thread, SIGNAL(finished()),
|
2014-02-28 04:09:57 +00:00
|
|
|
this, SLOT(onDownloadThreadFinished()), Qt::QueuedConnection);
|
2013-08-25 22:02:30 +00:00
|
|
|
|
2014-02-12 14:22:54 +00:00
|
|
|
MainWindow *w = MainWindow::instance();
|
2013-08-25 22:02:30 +00:00
|
|
|
connect(thread, SIGNAL(finished()), w, SLOT(refreshDisplay()));
|
2013-05-20 20:02:17 +00:00
|
|
|
|
2013-12-08 05:33:46 +00:00
|
|
|
// before we start, remember where the dive_table ended
|
|
|
|
previousLast = dive_table.nr;
|
|
|
|
|
2013-05-20 19:43:33 +00:00
|
|
|
thread->start();
|
2014-12-27 16:31:51 +00:00
|
|
|
|
2014-12-28 15:37:11 +00:00
|
|
|
QString product(ui.product->currentText());
|
|
|
|
if (product == "OSTC 3" || product == "OSTC Sport")
|
|
|
|
ostcFirmwareCheck = new OstcFirmwareCheck(product);
|
2013-05-20 19:43:33 +00:00
|
|
|
}
|
|
|
|
|
2013-05-30 08:58:59 +00:00
|
|
|
bool DownloadFromDCWidget::preferDownloaded()
|
|
|
|
{
|
2013-10-03 18:54:25 +00:00
|
|
|
return ui.preferDownloaded->isChecked();
|
2013-05-30 08:58:59 +00:00
|
|
|
}
|
|
|
|
|
2013-12-25 00:26:00 +00:00
|
|
|
void DownloadFromDCWidget::checkLogFile(int state)
|
|
|
|
{
|
|
|
|
ui.chooseLogFile->setEnabled(state == Qt::Checked);
|
|
|
|
data.libdc_log = (state == Qt::Checked);
|
|
|
|
if (state == Qt::Checked && logFile.isEmpty()) {
|
|
|
|
pickLogFile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadFromDCWidget::pickLogFile()
|
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
QString filename = existing_filename ?: prefs.default_filename;
|
2013-12-25 00:26:00 +00:00
|
|
|
QFileInfo fi(filename);
|
|
|
|
filename = fi.absolutePath().append(QDir::separator()).append("subsurface.log");
|
|
|
|
logFile = QFileDialog::getSaveFileName(this, tr("Choose file for divecomputer download logfile"),
|
|
|
|
filename, tr("Log files (*.log)"));
|
2014-01-07 14:41:21 +00:00
|
|
|
if (!logFile.isEmpty()) {
|
2014-05-12 17:58:15 +00:00
|
|
|
free(logfile_name);
|
2013-12-25 00:26:00 +00:00
|
|
|
logfile_name = strdup(logFile.toUtf8().data());
|
2014-01-07 14:41:21 +00:00
|
|
|
}
|
2013-12-25 00:26:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadFromDCWidget::checkDumpFile(int state)
|
|
|
|
{
|
|
|
|
ui.chooseDumpFile->setEnabled(state == Qt::Checked);
|
|
|
|
data.libdc_dump = (state == Qt::Checked);
|
2013-12-26 17:18:57 +00:00
|
|
|
if (state == Qt::Checked) {
|
|
|
|
if (dumpFile.isEmpty())
|
|
|
|
pickDumpFile();
|
|
|
|
if (!dumpWarningShown) {
|
|
|
|
QMessageBox::warning(this, tr("Warning"),
|
|
|
|
tr("Saving the libdivecomputer dump will NOT download dives to the dive list."));
|
|
|
|
dumpWarningShown = true;
|
|
|
|
}
|
2013-12-25 00:26:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadFromDCWidget::pickDumpFile()
|
|
|
|
{
|
2014-02-28 04:09:57 +00:00
|
|
|
QString filename = existing_filename ?: prefs.default_filename;
|
2013-12-25 00:26:00 +00:00
|
|
|
QFileInfo fi(filename);
|
|
|
|
filename = fi.absolutePath().append(QDir::separator()).append("subsurface.bin");
|
|
|
|
dumpFile = QFileDialog::getSaveFileName(this, tr("Choose file for divecomputer binary dump file"),
|
|
|
|
filename, tr("Dump files (*.bin)"));
|
2014-01-07 14:41:21 +00:00
|
|
|
if (!dumpFile.isEmpty()) {
|
2014-05-12 17:58:15 +00:00
|
|
|
free(dumpfile_name);
|
2013-12-25 00:26:00 +00:00
|
|
|
dumpfile_name = strdup(dumpFile.toUtf8().data());
|
2014-01-07 14:41:21 +00:00
|
|
|
}
|
2013-12-25 00:26:00 +00:00
|
|
|
}
|
|
|
|
|
2013-08-25 13:01:59 +00:00
|
|
|
void DownloadFromDCWidget::reject()
|
|
|
|
{
|
|
|
|
// we don't want the download window being able to close
|
|
|
|
// while we're still downloading.
|
2013-08-25 22:02:30 +00:00
|
|
|
if (currentState != DOWNLOADING && currentState != CANCELLING)
|
2013-08-25 13:01:59 +00:00
|
|
|
QDialog::reject();
|
|
|
|
}
|
|
|
|
|
2013-10-15 17:25:53 +00:00
|
|
|
void DownloadFromDCWidget::onDownloadThreadFinished()
|
2013-08-25 22:02:30 +00:00
|
|
|
{
|
2013-09-01 14:14:54 +00:00
|
|
|
if (currentState == DOWNLOADING) {
|
|
|
|
if (thread->error.isEmpty())
|
|
|
|
updateState(DONE);
|
|
|
|
else
|
|
|
|
updateState(ERROR);
|
2013-09-17 21:36:22 +00:00
|
|
|
|
|
|
|
// I'm not sure if we should really call process_dives even
|
2013-12-08 05:33:46 +00:00
|
|
|
// if there's an error
|
|
|
|
if (import_thread_cancelled) {
|
|
|
|
// walk backwards so we don't keep moving the dives
|
|
|
|
// down in the dive_table
|
|
|
|
for (int i = dive_table.nr - 1; i >= previousLast; i--)
|
|
|
|
delete_single_dive(i);
|
2015-01-09 19:28:15 +00:00
|
|
|
} else if (dive_table.nr && previousLast < dive_table.nr) {
|
2015-01-08 18:02:28 +00:00
|
|
|
diveImportedModel->setImportedDivesIndexes(previousLast, dive_table.nr - 1);
|
2013-12-08 05:33:46 +00:00
|
|
|
}
|
2015-01-09 20:00:34 +00:00
|
|
|
ui.startDownload->setEnabled(false);
|
2014-05-22 18:40:22 +00:00
|
|
|
} else if (currentState == CANCELLING || currentState == CANCELLED) {
|
2014-04-24 16:16:57 +00:00
|
|
|
if (import_thread_cancelled) {
|
|
|
|
// walk backwards so we don't keep moving the dives
|
|
|
|
// down in the dive_table
|
|
|
|
for (int i = dive_table.nr - 1; i >= previousLast; i--)
|
|
|
|
delete_single_dive(i);
|
|
|
|
}
|
2013-08-25 22:02:30 +00:00
|
|
|
updateState(CANCELLED);
|
2013-12-08 05:33:46 +00:00
|
|
|
}
|
2013-08-25 22:02:30 +00:00
|
|
|
}
|
2015-01-08 13:31:05 +00:00
|
|
|
|
|
|
|
void DownloadFromDCWidget::on_ok_clicked()
|
|
|
|
{
|
2015-01-08 13:56:00 +00:00
|
|
|
if (currentState != DONE)
|
|
|
|
return;
|
|
|
|
|
2015-01-08 13:52:12 +00:00
|
|
|
// remove all unselected dives from the dive-list.
|
|
|
|
diveImportedModel->removeUnused();
|
|
|
|
|
2015-01-08 12:12:11 +00:00
|
|
|
int uniqId, idx;
|
|
|
|
// remember the last downloaded dive (on most dive computers this will be the chronologically
|
|
|
|
// first new dive) and select it again after processing all the dives
|
|
|
|
MainWindow::instance()->dive_list()->unselectDives();
|
|
|
|
uniqId = get_dive(dive_table.nr - 1)->id;
|
|
|
|
process_dives(true, preferDownloaded());
|
|
|
|
// after process_dives does any merging or resorting needed, we need
|
|
|
|
// to recreate the model for the dive list so we can select the newest dive
|
|
|
|
MainWindow::instance()->recreateDiveList();
|
|
|
|
idx = get_idx_by_uniq_id(uniqId);
|
|
|
|
// this shouldn't be necessary - but there are reports that somehow existing dives stay selected
|
|
|
|
// (but not visible as selected)
|
|
|
|
MainWindow::instance()->dive_list()->unselectDives();
|
|
|
|
MainWindow::instance()->dive_list()->selectDive(idx, true);
|
2015-01-08 13:53:16 +00:00
|
|
|
|
2015-01-08 12:12:11 +00:00
|
|
|
if (ostcFirmwareCheck && currentState == DONE)
|
|
|
|
ostcFirmwareCheck->checkLatest(this, &data);
|
2015-01-08 17:59:01 +00:00
|
|
|
accept();
|
2015-01-08 13:31:05 +00:00
|
|
|
}
|
2013-08-25 22:02:30 +00:00
|
|
|
|
2013-08-25 13:01:59 +00:00
|
|
|
void DownloadFromDCWidget::markChildrenAsDisabled()
|
|
|
|
{
|
2015-01-08 20:56:17 +00:00
|
|
|
ui.device->setEnabled(false);
|
|
|
|
ui.vendor->setEnabled(false);
|
|
|
|
ui.product->setEnabled(false);
|
|
|
|
ui.forceDownload->setEnabled(false);
|
|
|
|
ui.createNewTrip->setEnabled(false);
|
|
|
|
ui.preferDownloaded->setEnabled(false);
|
|
|
|
ui.ok->setEnabled(false);
|
|
|
|
ui.search->setEnabled(false);
|
|
|
|
ui.logToFile->setEnabled(false);
|
|
|
|
ui.dumpToFile->setEnabled(false);
|
|
|
|
ui.chooseLogFile->setEnabled(false);
|
|
|
|
ui.chooseDumpFile->setEnabled(false);
|
|
|
|
ui.selectAllButton->setEnabled(false);
|
|
|
|
ui.unselectAllButton->setEnabled(false);
|
2013-08-25 13:01:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DownloadFromDCWidget::markChildrenAsEnabled()
|
|
|
|
{
|
2015-01-08 20:56:17 +00:00
|
|
|
ui.device->setEnabled(true);
|
|
|
|
ui.vendor->setEnabled(true);
|
|
|
|
ui.product->setEnabled(true);
|
|
|
|
ui.forceDownload->setEnabled(true);
|
|
|
|
ui.createNewTrip->setEnabled(true);
|
|
|
|
ui.preferDownloaded->setEnabled(true);
|
|
|
|
ui.ok->setEnabled(true);
|
|
|
|
ui.cancel->setEnabled(true);
|
|
|
|
ui.search->setEnabled(true);
|
|
|
|
ui.logToFile->setEnabled(true);
|
|
|
|
ui.dumpToFile->setEnabled(true);
|
|
|
|
ui.chooseLogFile->setEnabled(true);
|
|
|
|
ui.chooseDumpFile->setEnabled(true);
|
|
|
|
ui.selectAllButton->setEnabled(true);
|
|
|
|
ui.unselectAllButton->setEnabled(true);
|
2013-08-25 13:01:59 +00:00
|
|
|
}
|
|
|
|
|
2013-09-16 21:04:42 +00:00
|
|
|
static void fillDeviceList(const char *name, void *data)
|
|
|
|
{
|
|
|
|
QComboBox *comboBox = (QComboBox *)data;
|
|
|
|
comboBox->addItem(name);
|
|
|
|
}
|
|
|
|
|
2014-05-18 04:29:40 +00:00
|
|
|
void DownloadFromDCWidget::fill_device_list(int dc_type)
|
2013-09-16 21:04:42 +00:00
|
|
|
{
|
|
|
|
int deviceIndex;
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.device->clear();
|
2014-05-18 04:29:40 +00:00
|
|
|
deviceIndex = enumerate_devices(fillDeviceList, ui.device, dc_type);
|
2013-09-16 21:04:42 +00:00
|
|
|
if (deviceIndex >= 0)
|
2013-10-03 18:54:25 +00:00
|
|
|
ui.device->setCurrentIndex(deviceIndex);
|
2013-09-16 21:04:42 +00:00
|
|
|
}
|
|
|
|
|
2014-02-28 04:09:57 +00:00
|
|
|
DownloadThread::DownloadThread(QObject *parent, device_data_t *data) : QThread(parent),
|
2013-08-25 22:02:30 +00:00
|
|
|
data(data)
|
2013-05-20 19:43:33 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-09-01 14:14:54 +00:00
|
|
|
static QString str_error(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
2014-02-28 04:09:57 +00:00
|
|
|
const QString str = QString().vsprintf(fmt, args);
|
2013-09-01 14:14:54 +00:00
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2013-05-20 19:43:33 +00:00
|
|
|
void DownloadThread::run()
|
|
|
|
{
|
2014-02-12 13:51:25 +00:00
|
|
|
const char *errorText;
|
2013-12-10 07:09:42 +00:00
|
|
|
import_thread_cancelled = false;
|
2013-06-24 22:14:07 +00:00
|
|
|
if (!strcmp(data->vendor, "Uemis"))
|
2014-10-12 10:57:32 +00:00
|
|
|
errorText = do_uemis_import(data);
|
2013-06-24 22:14:07 +00:00
|
|
|
else
|
2014-02-12 13:51:25 +00:00
|
|
|
errorText = do_libdivecomputer_import(data);
|
|
|
|
if (errorText)
|
2014-02-28 04:09:57 +00:00
|
|
|
error = str_error(errorText, data->devname, data->vendor, data->product);
|
2013-05-20 19:43:33 +00:00
|
|
|
}
|
2015-01-08 12:12:11 +00:00
|
|
|
|
2015-01-08 12:13:36 +00:00
|
|
|
DiveImportedModel::DiveImportedModel(QObject *o) : QAbstractTableModel(o),
|
|
|
|
lastIndex(0),
|
2015-01-08 12:52:54 +00:00
|
|
|
firstIndex(0),
|
|
|
|
checkStates(0)
|
2015-01-08 12:12:11 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
int DiveImportedModel::columnCount(const QModelIndex &model) const
|
2015-01-08 12:12:11 +00:00
|
|
|
{
|
2015-01-08 17:31:18 +00:00
|
|
|
return 3;
|
2015-01-08 12:12:11 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
int DiveImportedModel::rowCount(const QModelIndex &model) const
|
2015-01-08 12:12:11 +00:00
|
|
|
{
|
2015-01-08 12:13:36 +00:00
|
|
|
return lastIndex - firstIndex;
|
2015-01-08 12:12:11 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 18:12:43 +00:00
|
|
|
QVariant DiveImportedModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
if (orientation == Qt::Vertical)
|
|
|
|
return QVariant();
|
|
|
|
if (role == Qt::DisplayRole) {
|
|
|
|
switch (section) {
|
|
|
|
case 0:
|
|
|
|
return QVariant(tr("Date/time"));
|
|
|
|
case 1:
|
|
|
|
return QVariant(tr("Duration"));
|
|
|
|
case 2:
|
|
|
|
return QVariant(tr("Depth"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
QVariant DiveImportedModel::data(const QModelIndex &index, int role) const
|
2015-01-08 12:12:11 +00:00
|
|
|
{
|
2015-01-08 12:25:38 +00:00
|
|
|
if (!index.isValid())
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
if (index.row() + firstIndex > lastIndex)
|
|
|
|
return QVariant();
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
struct dive *d = get_dive(index.row() + firstIndex);
|
2015-01-08 12:25:38 +00:00
|
|
|
if (role == Qt::DisplayRole) {
|
2015-01-08 18:02:28 +00:00
|
|
|
switch (index.column()) {
|
|
|
|
case 0:
|
|
|
|
return QVariant(get_short_dive_date_string(d->when));
|
|
|
|
case 1:
|
|
|
|
return QVariant(get_dive_duration_string(d->duration.seconds, tr("h:"), tr("min")));
|
|
|
|
case 2:
|
|
|
|
return QVariant(get_depth_string(d->maxdepth.mm, true, false));
|
2015-01-08 12:25:38 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-08 12:52:54 +00:00
|
|
|
if (role == Qt::CheckStateRole) {
|
2015-01-08 17:43:29 +00:00
|
|
|
if (index.column() == 0)
|
2015-01-08 19:27:47 +00:00
|
|
|
return checkStates[index.row()] ? Qt::Checked : Qt::Unchecked;
|
2015-01-08 12:52:54 +00:00
|
|
|
}
|
|
|
|
return QVariant();
|
2015-01-08 12:12:11 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 12:55:38 +00:00
|
|
|
bool DiveImportedModel::setData(const QModelIndex &index, const QVariant &value, int role)
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (index.row() + firstIndex > lastIndex)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (role != Qt::CheckStateRole)
|
|
|
|
return false;
|
|
|
|
|
2015-01-08 19:27:47 +00:00
|
|
|
checkStates[index.row()] = value.toBool();
|
2015-01-08 12:55:38 +00:00
|
|
|
dataChanged(index, index, QVector<int>() << Qt::CheckStateRole);
|
2015-01-08 17:32:36 +00:00
|
|
|
return true;
|
2015-01-08 12:55:38 +00:00
|
|
|
}
|
|
|
|
|
2015-01-09 20:17:02 +00:00
|
|
|
void DiveImportedModel::changeSelected(QModelIndex clickedIndex)
|
2015-01-08 19:39:34 +00:00
|
|
|
{
|
2015-01-09 20:17:02 +00:00
|
|
|
checkStates[clickedIndex.row()] = !checkStates[clickedIndex.row()];
|
|
|
|
dataChanged(index(0, clickedIndex.row()), index(0, clickedIndex.row()), QVector<int>() << Qt::CheckStateRole);
|
2015-01-08 19:39:34 +00:00
|
|
|
}
|
|
|
|
|
2015-01-08 20:54:53 +00:00
|
|
|
void DiveImportedModel::selectAll()
|
|
|
|
{
|
|
|
|
memset(checkStates, true, lastIndex - firstIndex);
|
|
|
|
dataChanged(index(0, 0), index(0, lastIndex - firstIndex - 1), QVector<int>() << Qt::CheckStateRole);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DiveImportedModel::selectNone()
|
|
|
|
{
|
|
|
|
memset(checkStates, false, lastIndex - firstIndex);
|
|
|
|
dataChanged(index(0, 0), index(0, lastIndex - firstIndex - 1), QVector<int>() << Qt::CheckStateRole);
|
|
|
|
}
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
Qt::ItemFlags DiveImportedModel::flags(const QModelIndex &index) const
|
|
|
|
{
|
2015-01-08 13:00:42 +00:00
|
|
|
if (index.column() != 0)
|
|
|
|
return QAbstractTableModel::flags(index);
|
|
|
|
return QAbstractTableModel::flags(index) | Qt::ItemIsUserCheckable;
|
|
|
|
}
|
|
|
|
|
2015-01-08 12:12:11 +00:00
|
|
|
void DiveImportedModel::setImportedDivesIndexes(int first, int last)
|
|
|
|
{
|
2015-01-09 20:03:29 +00:00
|
|
|
Q_ASSERT(last >= first);
|
2015-01-08 12:15:34 +00:00
|
|
|
beginRemoveRows(QModelIndex(), 0, lastIndex - firstIndex);
|
|
|
|
endRemoveRows();
|
|
|
|
beginInsertRows(QModelIndex(), 0, last - first);
|
|
|
|
lastIndex = last;
|
|
|
|
firstIndex = first;
|
2015-01-08 12:52:54 +00:00
|
|
|
delete[] checkStates;
|
2015-01-09 20:03:29 +00:00
|
|
|
checkStates = new bool[last - first + 1];
|
|
|
|
memset(checkStates, true, last - first + 1);
|
2015-01-08 12:15:34 +00:00
|
|
|
endInsertRows();
|
2015-01-08 12:12:11 +00:00
|
|
|
}
|
2015-01-08 13:52:12 +00:00
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
void DiveImportedModel::removeUnused()
|
|
|
|
{
|
|
|
|
beginRemoveRows(QModelIndex(), 0, rowCount() - 1);
|
2015-01-08 13:52:12 +00:00
|
|
|
endRemoveRows();
|
|
|
|
|
2015-01-08 18:02:28 +00:00
|
|
|
for (int i = lastIndex; i >= firstIndex; i--) {
|
2015-01-08 18:00:30 +00:00
|
|
|
if (!checkStates[i - firstIndex]) {
|
2015-01-08 13:52:12 +00:00
|
|
|
delete_single_dive(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lastIndex = 0;
|
|
|
|
firstIndex = 0;
|
|
|
|
delete[] checkStates;
|
|
|
|
}
|