#include "diveimportedmodel.h" #include "core/dive.h" #include "core/qthelper.h" #include "core/divelist.h" #include "commands/command.h" DiveImportedModel::DiveImportedModel(QObject *o) : QAbstractTableModel(o) { connect(&thread, &QThread::finished, this, &DiveImportedModel::downloadThreadFinished); } int DiveImportedModel::columnCount(const QModelIndex&) const { return 3; } int DiveImportedModel::rowCount(const QModelIndex&) const { return static_cast(log.dives.size()); } QVariant DiveImportedModel::headerData(int section, Qt::Orientation orientation, int role) const { if (orientation == Qt::Vertical) return QVariant(); // widgets access the model via index.column(), qml via role. int column = section; if (role == DateTime || role == Duration || role == Depth) { column = role - DateTime; role = Qt::DisplayRole; } if (role == Qt::DisplayRole) { switch (column) { case 0: return QVariant(tr("Date/time")); case 1: return QVariant(tr("Duration")); case 2: return QVariant(tr("Depth")); } } return QVariant(); } QVariant DiveImportedModel::data(const QModelIndex &index, int role) const { if (index.row() < 0 || static_cast(index.row()) >= log.dives.size()) return QVariant(); const struct dive &d = *log.dives[index.row()]; // widgets access the model via index.column(), qml via role. int column = index.column(); if (role >= DateTime) { column = role - DateTime; role = Qt::DisplayRole; } if (role == Qt::DisplayRole) { switch (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)); case 3: return checkStates[index.row()]; } } if (role == Qt::CheckStateRole) { if (index.column() == 0) return checkStates[index.row()] ? Qt::Checked : Qt::Unchecked; } return QVariant(); } void DiveImportedModel::changeSelected(QModelIndex clickedIndex) { checkStates[clickedIndex.row()] = !checkStates[clickedIndex.row()]; dataChanged(index(clickedIndex.row(), 0), index(clickedIndex.row(), 0), QVector() << Qt::CheckStateRole << Selected); } void DiveImportedModel::selectAll() { if (log.dives.empty()) return; std::fill(checkStates.begin(), checkStates.end(), true); dataChanged(index(0, 0), index(log.dives.size() - 1, 0), QVector() << Qt::CheckStateRole << Selected); } void DiveImportedModel::selectRow(int row) { checkStates[row] = !checkStates[row]; dataChanged(index(row, 0), index(row, 0), QVector() << Qt::CheckStateRole << Selected); } void DiveImportedModel::selectNone() { if (log.dives.empty()) return; std::fill(checkStates.begin(), checkStates.end(), false); dataChanged(index(0, 0), index(log.dives.size() - 1, 0 ), QVector() << Qt::CheckStateRole << Selected); } Qt::ItemFlags DiveImportedModel::flags(const QModelIndex &index) const { if (index.column() != 0) return QAbstractTableModel::flags(index); return QAbstractTableModel::flags(index) | Qt::ItemIsUserCheckable; } void DiveImportedModel::clearTable() { beginResetModel(); log.clear(); endResetModel(); } void DiveImportedModel::downloadThreadFinished() { beginResetModel(); // Move the table data from thread to model. Replace the downloads thread's log // with an empty log, because it may reuse it. log.clear(); std::swap(log, thread.log); checkStates.resize(log.dives.size()); std::fill(checkStates.begin(), checkStates.end(), true); endResetModel(); emit downloadFinished(); } void DiveImportedModel::startDownload() { thread.start(); } void DiveImportedModel::waitForDownload() { thread.wait(); downloadThreadFinished(); } struct divelog DiveImportedModel::consumeTables() { beginResetModel(); // Move tables to result and reset local tables (oldschool pre-C++11 flair). struct divelog res; std::swap(res, log); // Reset indices checkStates.clear(); endResetModel(); return res; } int DiveImportedModel::numDives() const { return static_cast(log.dives.size()); } // Delete non-selected dives void DiveImportedModel::deleteDeselected() { size_t total = log.dives.size(); size_t j = 0; for (size_t i = 0; i < total; i++) { if (checkStates[i]) { j++; } else { beginRemoveRows(QModelIndex(), j, j); log.dives.erase(log.dives.begin() + j); endRemoveRows(); } } checkStates.resize(log.dives.size()); std::fill(checkStates.begin(), checkStates.end(), true); } // Note: this function is only used from mobile - perhaps move it there or unify. void DiveImportedModel::recordDives(int flags) { // delete non-selected dives deleteDeselected(); struct divelog log = consumeTables(); if (!log.dives.empty()) { auto data = thread.data(); Command::importDives(&log, flags, data->devName()); } } QHash DiveImportedModel::roleNames() const { static QHash roles = { { DateTime, "datetime"}, { Depth, "depth"}, { Duration, "duration"}, { Selected, "selected"} }; return roles; }