mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-19 14:25:27 +00:00
Create DivesiteImportDialog to select sites to import
Creates the dialog box to select which sites to import from the file selected in mainwindow.cpp. The DivesiteImportModel is created as a table to display and select which sites are to be imported. Once the sites are selected, the Command::importDiveSites command is called to add the sites to the core dive site table with undo/redo functions. Signed-off-by: Doug Junkins <junkins@foghead.com>
This commit is contained in:
parent
98b3a326bd
commit
00ec824129
7 changed files with 421 additions and 0 deletions
|
@ -22,6 +22,7 @@ set (SUBSURFACE_UI
|
||||||
divecomputermanagementdialog.ui
|
divecomputermanagementdialog.ui
|
||||||
divelogexportdialog.ui
|
divelogexportdialog.ui
|
||||||
divelogimportdialog.ui
|
divelogimportdialog.ui
|
||||||
|
divesiteimportdialog.ui
|
||||||
diveplanner.ui
|
diveplanner.ui
|
||||||
diveshareexportdialog.ui
|
diveshareexportdialog.ui
|
||||||
downloadfromdivecomputer.ui
|
downloadfromdivecomputer.ui
|
||||||
|
@ -87,6 +88,8 @@ set(SUBSURFACE_INTERFACE
|
||||||
diveplanner.h
|
diveplanner.h
|
||||||
diveshareexportdialog.cpp
|
diveshareexportdialog.cpp
|
||||||
diveshareexportdialog.h
|
diveshareexportdialog.h
|
||||||
|
divesiteimportdialog.cpp
|
||||||
|
divesiteimportdialog.h
|
||||||
downloadfromdivecomputer.cpp
|
downloadfromdivecomputer.cpp
|
||||||
downloadfromdivecomputer.h
|
downloadfromdivecomputer.h
|
||||||
filterwidget2.cpp
|
filterwidget2.cpp
|
||||||
|
|
80
desktop-widgets/divesiteimportdialog.cpp
Normal file
80
desktop-widgets/divesiteimportdialog.cpp
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
#include "desktop-widgets/divesiteimportdialog.h"
|
||||||
|
#include "desktop-widgets/command.h"
|
||||||
|
#include "core/display.h"
|
||||||
|
#include "core/qthelper.h"
|
||||||
|
#include "core/metrics.h"
|
||||||
|
#include "core/subsurface-string.h"
|
||||||
|
#include "desktop-widgets/mainwindow.h"
|
||||||
|
#include "qt-models/divesiteimportmodel.h"
|
||||||
|
#include "qt-models/models.h"
|
||||||
|
|
||||||
|
#include <QShortcut>
|
||||||
|
|
||||||
|
// Caller keeps ownership of "imported". The contents of "imported" will be consumed on execution of the dialog.
|
||||||
|
// On return, it will be empty.
|
||||||
|
DivesiteImportDialog::DivesiteImportDialog(struct dive_site_table &imported, QString source, QWidget *parent) : QDialog(parent),
|
||||||
|
importedSource(source)
|
||||||
|
{
|
||||||
|
QShortcut *close = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_W), this);
|
||||||
|
QShortcut *quit = new QShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q), this);
|
||||||
|
|
||||||
|
divesiteImportedModel = new DivesiteImportedModel(this);
|
||||||
|
|
||||||
|
int startingWidth = defaultModelFont().pointSize();
|
||||||
|
|
||||||
|
ui.setupUi(this);
|
||||||
|
ui.importedDivesitesView->setModel(divesiteImportedModel);
|
||||||
|
ui.importedDivesitesView->setSelectionBehavior(QAbstractItemView::SelectRows);
|
||||||
|
ui.importedDivesitesView->setSelectionMode(QAbstractItemView::SingleSelection);
|
||||||
|
ui.importedDivesitesView->horizontalHeader()->setStretchLastSection(true);
|
||||||
|
ui.importedDivesitesView->verticalHeader()->setVisible(false);
|
||||||
|
ui.importedDivesitesView->setColumnWidth(0, startingWidth * 14);
|
||||||
|
ui.importedDivesitesView->setColumnWidth(1, startingWidth * 12);
|
||||||
|
ui.importedDivesitesView->setColumnWidth(2, startingWidth * 8);
|
||||||
|
ui.importedDivesitesView->setColumnWidth(3, startingWidth * 14);
|
||||||
|
ui.selectAllButton->setEnabled(true);
|
||||||
|
ui.unselectAllButton->setEnabled(true);
|
||||||
|
|
||||||
|
connect(ui.importedDivesitesView, &QTableView::clicked, divesiteImportedModel, &DivesiteImportedModel::changeSelected);
|
||||||
|
connect(ui.selectAllButton, &QPushButton::clicked, divesiteImportedModel, &DivesiteImportedModel::selectAll);
|
||||||
|
connect(ui.unselectAllButton, &QPushButton::clicked, divesiteImportedModel, &DivesiteImportedModel::selectNone);
|
||||||
|
connect(close, SIGNAL(activated()), this, SLOT(close()));
|
||||||
|
connect(quit, SIGNAL(activated()), parent, SLOT(close()));
|
||||||
|
|
||||||
|
ui.ok->setEnabled(true);
|
||||||
|
|
||||||
|
importedSites = imported;
|
||||||
|
imported.nr = imported.allocated = 0;
|
||||||
|
imported.dive_sites = nullptr;
|
||||||
|
|
||||||
|
divesiteImportedModel->repopulate(&importedSites);
|
||||||
|
}
|
||||||
|
|
||||||
|
DivesiteImportDialog::~DivesiteImportDialog()
|
||||||
|
{
|
||||||
|
clear_dive_site_table(&importedSites);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportDialog::on_cancel_clicked()
|
||||||
|
{
|
||||||
|
clear_dive_site_table(&importedSites);
|
||||||
|
done(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportDialog::on_ok_clicked()
|
||||||
|
{
|
||||||
|
// delete non-selected dive sites
|
||||||
|
struct dive_site_table selectedSites = { 0 };
|
||||||
|
for (int i = 0; i < importedSites.nr; i++)
|
||||||
|
if (divesiteImportedModel->data(divesiteImportedModel->index(i, 0), Qt::CheckStateRole) == Qt::Checked) {
|
||||||
|
struct dive_site *newSite = alloc_dive_site();
|
||||||
|
copy_dive_site(importedSites.dive_sites[i], newSite);
|
||||||
|
add_dive_site_to_table(newSite, &selectedSites);
|
||||||
|
}
|
||||||
|
|
||||||
|
Command::importDiveSites(&selectedSites, importedSource);
|
||||||
|
clear_dive_site_table(&selectedSites);
|
||||||
|
clear_dive_site_table(&importedSites);
|
||||||
|
accept();
|
||||||
|
}
|
40
desktop-widgets/divesiteimportdialog.h
Normal file
40
desktop-widgets/divesiteimportdialog.h
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
// SPDX-License-Identifier: GPL-2.0
|
||||||
|
#ifndef DIVESITEIMPORTDIALOG_H
|
||||||
|
#define DIVESITEIMPORTDIALOG_H
|
||||||
|
|
||||||
|
#include <QDialog>
|
||||||
|
#include <QThread>
|
||||||
|
#include <QHash>
|
||||||
|
#include <QMap>
|
||||||
|
#include <QAbstractTableModel>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "ui_divesiteimportdialog.h"
|
||||||
|
#include "core/divesite.h"
|
||||||
|
|
||||||
|
namespace Ui {
|
||||||
|
class DivesiteImportDialog;
|
||||||
|
}
|
||||||
|
|
||||||
|
class DivesiteImportedModel;
|
||||||
|
|
||||||
|
class DivesiteImportDialog : public QDialog {
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
DivesiteImportDialog(struct dive_site_table &imported, QString source, QWidget *parent = 0 );
|
||||||
|
~DivesiteImportDialog();
|
||||||
|
|
||||||
|
public
|
||||||
|
slots:
|
||||||
|
void on_ok_clicked();
|
||||||
|
void on_cancel_clicked();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Ui::DivesiteImportDialog ui;
|
||||||
|
struct dive_site_table importedSites;
|
||||||
|
QString importedSource;
|
||||||
|
|
||||||
|
DivesiteImportedModel *divesiteImportedModel;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // DIVESITEIMPORTDIALOG_H
|
119
desktop-widgets/divesiteimportdialog.ui
Normal file
119
desktop-widgets/divesiteimportdialog.ui
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ui version="4.0">
|
||||||
|
<class>DivesiteImportDialog</class>
|
||||||
|
<widget class="QDialog" name="DivesiteImportDialog">
|
||||||
|
<property name="geometry">
|
||||||
|
<rect>
|
||||||
|
<x>0</x>
|
||||||
|
<y>0</y>
|
||||||
|
<width>747</width>
|
||||||
|
<height>535</height>
|
||||||
|
</rect>
|
||||||
|
</property>
|
||||||
|
<property name="windowTitle">
|
||||||
|
<string>Select dive sites to import</string>
|
||||||
|
</property>
|
||||||
|
<property name="windowIcon">
|
||||||
|
<iconset>
|
||||||
|
<normalon>:subsurface-icon</normalon>
|
||||||
|
</iconset>
|
||||||
|
</property>
|
||||||
|
<layout class="QVBoxLayout" name="verticalLayout_3">
|
||||||
|
<property name="spacing">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<property name="leftMargin">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<property name="topMargin">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<property name="rightMargin">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<property name="bottomMargin">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="labelSelectAllNoneLayout">
|
||||||
|
<property name="spacing">
|
||||||
|
<number>5</number>
|
||||||
|
</property>
|
||||||
|
<item>
|
||||||
|
<spacer name="horizontalSpacer_2">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Horizontal</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>40</width>
|
||||||
|
<height>20</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="selectAllButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>Select all</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="unselectAllButton">
|
||||||
|
<property name="text">
|
||||||
|
<string>Unselect all</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QTableView" name="importedDivesitesView">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="MinimumExpanding" vsizetype="MinimumExpanding">
|
||||||
|
<horstretch>1</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="buttonBoxLayout">
|
||||||
|
<property name="spacing">
|
||||||
|
<number>0</number>
|
||||||
|
</property>
|
||||||
|
<item>
|
||||||
|
<spacer name="horizontalSpacer">
|
||||||
|
<property name="orientation">
|
||||||
|
<enum>Qt::Horizontal</enum>
|
||||||
|
</property>
|
||||||
|
<property name="sizeHint" stdset="0">
|
||||||
|
<size>
|
||||||
|
<width>40</width>
|
||||||
|
<height>20</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
</spacer>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="ok">
|
||||||
|
<property name="text">
|
||||||
|
<string>OK</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPushButton" name="cancel">
|
||||||
|
<property name="text">
|
||||||
|
<string>Cancel</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</widget>
|
||||||
|
<resources/>
|
||||||
|
<connections/>
|
||||||
|
</ui>
|
|
@ -29,6 +29,8 @@ set(SUBSURFACE_DESKTOP_MODELS_LIB_SRCS
|
||||||
divepicturemodel.h
|
divepicturemodel.h
|
||||||
diveplannermodel.cpp
|
diveplannermodel.cpp
|
||||||
diveplannermodel.h
|
diveplannermodel.h
|
||||||
|
divesiteimportmodel.cpp
|
||||||
|
divesiteimportmodel.h
|
||||||
divetripmodel.cpp
|
divetripmodel.cpp
|
||||||
divetripmodel.h
|
divetripmodel.h
|
||||||
filtermodels.cpp
|
filtermodels.cpp
|
||||||
|
|
142
qt-models/divesiteimportmodel.cpp
Normal file
142
qt-models/divesiteimportmodel.cpp
Normal file
|
@ -0,0 +1,142 @@
|
||||||
|
#include "divesiteimportmodel.h"
|
||||||
|
#include "core/qthelper.h"
|
||||||
|
#include "core/taxonomy.h"
|
||||||
|
|
||||||
|
DivesiteImportedModel::DivesiteImportedModel(QObject *o) : QAbstractTableModel(o),
|
||||||
|
firstIndex(0),
|
||||||
|
lastIndex(-1),
|
||||||
|
importedSitesTable(nullptr)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int DivesiteImportedModel::columnCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
int DivesiteImportedModel::rowCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return lastIndex - firstIndex + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant DivesiteImportedModel::headerData(int section, Qt::Orientation orientation, int role) const
|
||||||
|
{
|
||||||
|
if (orientation == Qt::Vertical)
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
if (role == Qt::DisplayRole) {
|
||||||
|
switch (section) {
|
||||||
|
case NAME:
|
||||||
|
return tr("Name");
|
||||||
|
case LOCATION:
|
||||||
|
return tr("Location");
|
||||||
|
case COUNTRY:
|
||||||
|
return tr("Country");
|
||||||
|
case NEAREST:
|
||||||
|
return tr("Nearest\nExisting Site");
|
||||||
|
case DISTANCE:
|
||||||
|
return tr("Distance");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant DivesiteImportedModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid())
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
if (index.row() + firstIndex > lastIndex)
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
struct dive_site *ds = get_dive_site(index.row() + firstIndex, importedSitesTable);
|
||||||
|
if (!ds)
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
// widgets access the model via index.column()
|
||||||
|
// Not supporting QML access via roles
|
||||||
|
if (role == Qt::DisplayRole) {
|
||||||
|
switch (index.column()) {
|
||||||
|
case NAME:
|
||||||
|
return ds->name;
|
||||||
|
case LOCATION:
|
||||||
|
return printGPSCoords(&ds->location);
|
||||||
|
case COUNTRY:
|
||||||
|
return taxonomy_get_country(&ds->taxonomy);
|
||||||
|
case NEAREST: {
|
||||||
|
// 40075000 is circumference of the earth in meters
|
||||||
|
struct dive_site *nearest_ds =
|
||||||
|
get_dive_site_by_gps_proximity(&ds->location,
|
||||||
|
40075000, &dive_site_table);
|
||||||
|
if (nearest_ds)
|
||||||
|
return nearest_ds->name;
|
||||||
|
else
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
case DISTANCE: {
|
||||||
|
unsigned int distance = 0;
|
||||||
|
struct dive_site *nearest_ds =
|
||||||
|
get_dive_site_by_gps_proximity(&ds->location,
|
||||||
|
40075000, &dive_site_table);
|
||||||
|
if (nearest_ds)
|
||||||
|
distance = get_distance(&ds->location,
|
||||||
|
&nearest_ds->location);
|
||||||
|
return distance_string(distance);
|
||||||
|
}
|
||||||
|
case SELECTED:
|
||||||
|
return checkStates[index.row()];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (role == Qt::CheckStateRole) {
|
||||||
|
if (index.column() == 0)
|
||||||
|
return checkStates[index.row()] ? Qt::Checked : Qt::Unchecked;
|
||||||
|
}
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportedModel::changeSelected(QModelIndex clickedIndex)
|
||||||
|
{
|
||||||
|
checkStates[clickedIndex.row()] = !checkStates[clickedIndex.row()];
|
||||||
|
dataChanged(index(clickedIndex.row(), 0), index(clickedIndex.row(), 0), QVector<int>() << Qt::CheckStateRole << SELECTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportedModel::selectAll()
|
||||||
|
{
|
||||||
|
std::fill(checkStates.begin(), checkStates.end(), true);
|
||||||
|
dataChanged(index(0, 0), index(lastIndex - firstIndex, 0), QVector<int>() << Qt::CheckStateRole << SELECTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportedModel::selectRow(int row)
|
||||||
|
{
|
||||||
|
checkStates[row] = !checkStates[row];
|
||||||
|
dataChanged(index(row, 0), index(row, 0), QVector<int>() << Qt::CheckStateRole << SELECTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportedModel::selectNone()
|
||||||
|
{
|
||||||
|
std::fill(checkStates.begin(), checkStates.end(), false);
|
||||||
|
dataChanged(index(0, 0), index(lastIndex - firstIndex,0 ), QVector<int>() << Qt::CheckStateRole << SELECTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
Qt::ItemFlags DivesiteImportedModel::flags(const QModelIndex &index) const
|
||||||
|
{
|
||||||
|
if (index.column() != 0)
|
||||||
|
return QAbstractTableModel::flags(index);
|
||||||
|
return QAbstractTableModel::flags(index) | Qt::ItemIsUserCheckable;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DivesiteImportedModel::repopulate(struct dive_site_table *sites)
|
||||||
|
{
|
||||||
|
beginResetModel();
|
||||||
|
|
||||||
|
importedSitesTable = sites;
|
||||||
|
firstIndex = 0;
|
||||||
|
lastIndex = importedSitesTable->nr - 1;
|
||||||
|
checkStates.resize(importedSitesTable->nr);
|
||||||
|
for (int row = 0; row < importedSitesTable->nr; row++)
|
||||||
|
if (get_dive_site_by_gps(&importedSitesTable->dive_sites[row]->location, &dive_site_table))
|
||||||
|
checkStates[row] = false;
|
||||||
|
else
|
||||||
|
checkStates[row] = true;
|
||||||
|
endResetModel();
|
||||||
|
}
|
35
qt-models/divesiteimportmodel.h
Normal file
35
qt-models/divesiteimportmodel.h
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
#ifndef DIVESITEIMPORTEDMODEL_H
|
||||||
|
#define DIVESITEIMPORTEDMODEL_H
|
||||||
|
|
||||||
|
#include <QAbstractTableModel>
|
||||||
|
#include <vector>
|
||||||
|
#include "core/divesite.h"
|
||||||
|
|
||||||
|
class DivesiteImportedModel : public QAbstractTableModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
enum columnNames { NAME, LOCATION, COUNTRY, NEAREST, DISTANCE, SELECTED };
|
||||||
|
|
||||||
|
DivesiteImportedModel(QObject *parent = 0);
|
||||||
|
int columnCount(const QModelIndex& index = QModelIndex()) const;
|
||||||
|
int rowCount(const QModelIndex& index = QModelIndex()) const;
|
||||||
|
QVariant data(const QModelIndex& index, int role) const;
|
||||||
|
QVariant headerData(int section, Qt::Orientation orientation, int role) const;
|
||||||
|
Qt::ItemFlags flags(const QModelIndex &index) const;
|
||||||
|
void repopulate(dive_site_table_t *sites);
|
||||||
|
public
|
||||||
|
slots:
|
||||||
|
void changeSelected(QModelIndex clickedIndex);
|
||||||
|
void selectRow(int row);
|
||||||
|
void selectAll();
|
||||||
|
void selectNone();
|
||||||
|
|
||||||
|
private:
|
||||||
|
int firstIndex;
|
||||||
|
int lastIndex;
|
||||||
|
std::vector<char> checkStates; // char instead of bool to avoid silly pessimization of std::vector.
|
||||||
|
struct dive_site_table *importedSitesTable;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
Loading…
Add table
Reference in a new issue