2013-04-14 03:44:02 +00:00
|
|
|
/*
|
|
|
|
* models.cpp
|
|
|
|
*
|
|
|
|
* classes for the equipment models of Subsurface
|
|
|
|
*
|
|
|
|
*/
|
2013-04-13 13:17:59 +00:00
|
|
|
#include "models.h"
|
2013-05-21 19:03:05 +00:00
|
|
|
#include "../helpers.h"
|
2013-05-22 17:02:28 +00:00
|
|
|
#include "../dive.h"
|
2013-06-07 14:43:45 +00:00
|
|
|
#include "../device.h"
|
2013-06-18 07:33:03 +00:00
|
|
|
#include "../statistics.h"
|
2013-06-17 22:58:26 +00:00
|
|
|
#include "../qthelper.h"
|
2013-06-07 18:25:29 +00:00
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QDebug>
|
2013-05-02 21:19:15 +00:00
|
|
|
#include <QColor>
|
|
|
|
#include <QBrush>
|
2013-05-19 13:46:49 +00:00
|
|
|
#include <QFont>
|
2013-05-22 14:00:20 +00:00
|
|
|
#include <QIcon>
|
2013-04-24 15:57:30 +00:00
|
|
|
|
2013-06-16 14:13:32 +00:00
|
|
|
QFont defaultModelFont()
|
|
|
|
{
|
|
|
|
QFont font;
|
|
|
|
font.setPointSizeF( font.pointSizeF() * 0.8);
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
|
2013-05-21 12:33:55 +00:00
|
|
|
CylindersModel::CylindersModel(QObject* parent): QAbstractTableModel(parent), current(0), rows(0)
|
2013-04-13 13:17:59 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CylindersModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
2013-05-26 03:04:31 +00:00
|
|
|
|
2013-05-02 05:00:08 +00:00
|
|
|
if (orientation == Qt::Vertical)
|
2013-04-13 13:17:59 +00:00
|
|
|
return ret;
|
|
|
|
|
2013-05-26 03:04:31 +00:00
|
|
|
switch (role) {
|
|
|
|
case Qt::FontRole:
|
2013-06-16 14:13:32 +00:00
|
|
|
ret = defaultModelFont();
|
|
|
|
break;
|
2013-05-26 03:04:31 +00:00
|
|
|
case Qt::DisplayRole:
|
2013-04-13 13:17:59 +00:00
|
|
|
switch(section) {
|
2013-05-29 20:43:14 +00:00
|
|
|
case TYPE: ret = tr("Type"); break;
|
|
|
|
case SIZE: ret = tr("Size"); break;
|
|
|
|
case WORKINGPRESS: ret = tr("WorkPress"); break;
|
|
|
|
case START: ret = tr("StartPress"); break;
|
|
|
|
case END: ret = tr("EndPress "); break;
|
|
|
|
case O2: ret = tr("O2% "); break;
|
|
|
|
case HE: ret = tr("He% "); break;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CylindersModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
2013-05-22 14:00:20 +00:00
|
|
|
return COLUMNS;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2013-06-20 04:22:09 +00:00
|
|
|
static QVariant percent_string(fraction_t fraction)
|
|
|
|
{
|
|
|
|
int permille = fraction.permille;
|
|
|
|
|
|
|
|
if (!permille)
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
return QString("%1%").arg(permille / 10.0, 0, 'f', 1);
|
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
QVariant CylindersModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
2013-05-26 03:04:31 +00:00
|
|
|
|
2013-05-02 05:00:08 +00:00
|
|
|
if (!index.isValid() || index.row() >= MAX_CYLINDERS)
|
2013-04-13 13:17:59 +00:00
|
|
|
return ret;
|
2013-05-02 05:00:08 +00:00
|
|
|
|
2013-05-21 19:03:05 +00:00
|
|
|
cylinder_t *cyl = ¤t->cylinder[index.row()];
|
2013-05-26 03:04:31 +00:00
|
|
|
switch (role) {
|
2013-06-20 04:42:48 +00:00
|
|
|
case Qt::FontRole: {
|
|
|
|
QFont font = defaultModelFont();
|
|
|
|
switch (index.column()) {
|
|
|
|
case START:
|
|
|
|
if (!cyl->start.mbar)
|
|
|
|
font.setItalic(true);
|
|
|
|
break;
|
|
|
|
case END:
|
|
|
|
if (!cyl->end.mbar)
|
|
|
|
font.setItalic(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ret = font;
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
2013-06-20 04:42:48 +00:00
|
|
|
}
|
2013-05-26 03:04:31 +00:00
|
|
|
case Qt::TextAlignmentRole:
|
2013-06-16 15:33:27 +00:00
|
|
|
ret = Qt::AlignHCenter;
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
case Qt::EditRole:
|
2013-04-13 13:17:59 +00:00
|
|
|
switch(index.column()) {
|
|
|
|
case TYPE:
|
2013-05-21 19:03:05 +00:00
|
|
|
ret = QString(cyl->type.description);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
case SIZE:
|
2013-05-21 19:03:05 +00:00
|
|
|
// we can't use get_volume_string because the idiotic imperial tank
|
|
|
|
// sizes take working pressure into account...
|
|
|
|
if (cyl->type.size.mliter) {
|
Fix the imperial cylinder size calculations in equipment handling
This makes us use the same linear calculations as we did in the Gtk
branch. We don't take compressibility into account, since tank
manufacturers don't seem to either. A Luxfer AL80 is 11.1 liters, and
with the standard (non-compressibility) calculations, 80 cuft of air at
3000 psi is 11.094 liter, so that is the right model to use.
Also, stop with the horrible "units in edited numbers" stuff. It uses
up precious space, and doesn't look any better. If the user asked for
cuft, give him cuft without making a big deal about it.
Oh, and if the working pressure doesn't exist, sizes are always in
liters. That's what we did in the Gtk branch, that's what we do here.
Again, no reason to even bother stating units, it's not helping.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-20 02:33:49 +00:00
|
|
|
double volume;
|
|
|
|
int mbar = cyl->type.workingpressure.mbar;
|
|
|
|
|
|
|
|
if (mbar && prefs.units.volume == prefs.units.CUFT) {
|
|
|
|
volume = ml_to_cuft(cyl->type.size.mliter);
|
|
|
|
volume *= bar_to_atm(mbar / 1000.0);
|
2013-05-21 19:03:05 +00:00
|
|
|
} else {
|
Fix the imperial cylinder size calculations in equipment handling
This makes us use the same linear calculations as we did in the Gtk
branch. We don't take compressibility into account, since tank
manufacturers don't seem to either. A Luxfer AL80 is 11.1 liters, and
with the standard (non-compressibility) calculations, 80 cuft of air at
3000 psi is 11.094 liter, so that is the right model to use.
Also, stop with the horrible "units in edited numbers" stuff. It uses
up precious space, and doesn't look any better. If the user asked for
cuft, give him cuft without making a big deal about it.
Oh, and if the working pressure doesn't exist, sizes are always in
liters. That's what we did in the Gtk branch, that's what we do here.
Again, no reason to even bother stating units, it's not helping.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-20 02:33:49 +00:00
|
|
|
volume = cyl->type.size.mliter / 1000.0;
|
2013-05-21 19:03:05 +00:00
|
|
|
}
|
Fix the imperial cylinder size calculations in equipment handling
This makes us use the same linear calculations as we did in the Gtk
branch. We don't take compressibility into account, since tank
manufacturers don't seem to either. A Luxfer AL80 is 11.1 liters, and
with the standard (non-compressibility) calculations, 80 cuft of air at
3000 psi is 11.094 liter, so that is the right model to use.
Also, stop with the horrible "units in edited numbers" stuff. It uses
up precious space, and doesn't look any better. If the user asked for
cuft, give him cuft without making a big deal about it.
Oh, and if the working pressure doesn't exist, sizes are always in
liters. That's what we did in the Gtk branch, that's what we do here.
Again, no reason to even bother stating units, it's not helping.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-20 02:33:49 +00:00
|
|
|
ret = QString("%1").arg(volume, 0, 'f', 1);
|
2013-05-21 19:03:05 +00:00
|
|
|
}
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
2013-05-22 20:29:17 +00:00
|
|
|
case WORKINGPRESS:
|
2013-05-21 19:03:05 +00:00
|
|
|
if (cyl->type.workingpressure.mbar)
|
|
|
|
ret = get_pressure_string(cyl->type.workingpressure, TRUE);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
case START:
|
2013-05-21 19:03:05 +00:00
|
|
|
if (cyl->start.mbar)
|
2013-06-20 04:42:48 +00:00
|
|
|
ret = get_pressure_string(cyl->start, FALSE);
|
|
|
|
else if (cyl->sample_start.mbar)
|
|
|
|
ret = get_pressure_string(cyl->sample_start, FALSE);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
case END:
|
2013-05-21 19:03:05 +00:00
|
|
|
if (cyl->end.mbar)
|
2013-06-20 04:42:48 +00:00
|
|
|
ret = get_pressure_string(cyl->end, FALSE);
|
|
|
|
else if (cyl->sample_end.mbar)
|
|
|
|
ret = get_pressure_string(cyl->sample_end, FALSE);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
case O2:
|
2013-06-20 04:22:09 +00:00
|
|
|
ret = percent_string(cyl->gasmix.o2);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
case HE:
|
2013-06-20 04:22:09 +00:00
|
|
|
ret = percent_string(cyl->gasmix.he);
|
2013-04-13 13:17:59 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
|
|
|
case Qt::DecorationRole:
|
2013-05-23 04:25:05 +00:00
|
|
|
if (index.column() == REMOVE)
|
2013-05-22 14:00:20 +00:00
|
|
|
ret = QIcon(":trash");
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
2013-07-18 13:24:02 +00:00
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-07-18 13:24:02 +00:00
|
|
|
cylinder_t* CylindersModel::cylinderAt(const QModelIndex& index)
|
|
|
|
{
|
|
|
|
return ¤t->cylinder[index.row()];
|
|
|
|
}
|
|
|
|
|
2013-05-23 20:31:46 +00:00
|
|
|
// this is our magic 'pass data in' function that allows the delegate to get
|
|
|
|
// the data here without silly unit conversions;
|
|
|
|
// so we only implement the two columns we care about
|
|
|
|
void CylindersModel::passInData(const QModelIndex& index, const QVariant& value)
|
|
|
|
{
|
|
|
|
cylinder_t *cyl = ¤t->cylinder[index.row()];
|
|
|
|
switch(index.column()) {
|
|
|
|
case SIZE:
|
|
|
|
if (cyl->type.size.mliter != value.toInt()) {
|
|
|
|
cyl->type.size.mliter = value.toInt();
|
|
|
|
mark_divelist_changed(TRUE);
|
2013-07-16 22:13:58 +00:00
|
|
|
dataChanged(index, index);
|
2013-05-23 20:31:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case WORKINGPRESS:
|
|
|
|
if (cyl->type.workingpressure.mbar != value.toInt()) {
|
|
|
|
cyl->type.workingpressure.mbar = value.toInt();
|
|
|
|
mark_divelist_changed(TRUE);
|
2013-07-16 22:13:58 +00:00
|
|
|
dataChanged(index, index);
|
2013-05-23 20:31:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-18 15:28:28 +00:00
|
|
|
#define CHANGED(_t,_u1,_u2) value._t() != data(index, role).toString().remove(_u1).remove(_u2)._t()
|
2013-05-22 20:29:17 +00:00
|
|
|
|
2013-05-22 15:20:00 +00:00
|
|
|
bool CylindersModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
|
|
{
|
|
|
|
cylinder_t *cyl = ¤t->cylinder[index.row()];
|
2013-05-22 20:29:17 +00:00
|
|
|
switch(index.column()) {
|
|
|
|
case TYPE:
|
|
|
|
if (!value.isNull()) {
|
Fix usage of temporary QByteArrays
This commit fixes three different things:
- a memory leak in WeightModel::setData()
- getSetting() calling strdup() on a QByteArray
- a possible usage of memory after deallocation
Here's an explanation of the last issue (taken from the mailing list, slightly
adapted):
toByteArray(), as well as others "toSomething()" methods, returns
a new object which the compiler allocates on the stack. The compiler
will consider it a temporary data, and destroy it on the next line. So,
when one does
char *text= value.toByteArray().data(); // line 1
if (strcmp(description, text)) { // line 2
the compiler creates a QByteArray on line 1, calls ::data() on it, which
returns a valid char *, and assigns its value to "text". So far, so
good. But before jumping to line 2, the compiler destroys the temporary
QByteArray, and this will in turn invoke the QByteArray destructor,
which will destroy the internal data. The result is that on line 2,
"text" will point to some memory which has already been freed.
One solution is to store a copy of the temporary QByteArray into a local
variable: the compiler will still destroy the temporary QByteArray it created,
but (thanks to the reference-counted data sharing built in QByteArray) now the
destructor will see that the data is referenced by another instance of
QByteArray (the local variable "ba") and will not free the internal data.
In this way, the internal data will be available until the local variable is
destroyed, which will happen at the end of the {} block where it is defined.
Please note that when one uses the data in the same line, one doesn't need to
worry about this issue. In fact,
text = strdup(value.toString().toUtf8().data());
works just fine.
Signed-off-by: Alberto Mardegan <mardy@users.sourceforge.net>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-05-24 11:39:37 +00:00
|
|
|
QByteArray ba = value.toByteArray();
|
|
|
|
const char *text = ba.constData();
|
2013-05-22 23:22:21 +00:00
|
|
|
if (!cyl->type.description || strcmp(cyl->type.description, text)) {
|
2013-05-22 20:29:17 +00:00
|
|
|
cyl->type.description = strdup(text);
|
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
2013-05-22 15:20:00 +00:00
|
|
|
}
|
2013-05-22 20:29:17 +00:00
|
|
|
break;
|
|
|
|
case SIZE:
|
|
|
|
if (CHANGED(toDouble, "cuft", "l")) {
|
|
|
|
// if units are CUFT then this value is meaningless until we have working pressure
|
|
|
|
if (value.toDouble() != 0.0) {
|
2013-05-24 05:56:12 +00:00
|
|
|
TankInfoModel *tanks = TankInfoModel::instance();
|
|
|
|
QModelIndexList matches = tanks->match(tanks->index(0,0), Qt::DisplayRole, cyl->type.description);
|
Fix the imperial cylinder size calculations in equipment handling
This makes us use the same linear calculations as we did in the Gtk
branch. We don't take compressibility into account, since tank
manufacturers don't seem to either. A Luxfer AL80 is 11.1 liters, and
with the standard (non-compressibility) calculations, 80 cuft of air at
3000 psi is 11.094 liter, so that is the right model to use.
Also, stop with the horrible "units in edited numbers" stuff. It uses
up precious space, and doesn't look any better. If the user asked for
cuft, give him cuft without making a big deal about it.
Oh, and if the working pressure doesn't exist, sizes are always in
liters. That's what we did in the Gtk branch, that's what we do here.
Again, no reason to even bother stating units, it's not helping.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-20 02:33:49 +00:00
|
|
|
int mbar = cyl->type.workingpressure.mbar;
|
|
|
|
int mliter;
|
|
|
|
|
|
|
|
if (mbar && prefs.units.volume == prefs.units.CUFT) {
|
|
|
|
double liters = cuft_to_l(value.toDouble());
|
|
|
|
liters /= bar_to_atm(mbar / 1000.0);
|
|
|
|
mliter = rint(liters * 1000);
|
2013-05-22 15:20:00 +00:00
|
|
|
} else {
|
Fix the imperial cylinder size calculations in equipment handling
This makes us use the same linear calculations as we did in the Gtk
branch. We don't take compressibility into account, since tank
manufacturers don't seem to either. A Luxfer AL80 is 11.1 liters, and
with the standard (non-compressibility) calculations, 80 cuft of air at
3000 psi is 11.094 liter, so that is the right model to use.
Also, stop with the horrible "units in edited numbers" stuff. It uses
up precious space, and doesn't look any better. If the user asked for
cuft, give him cuft without making a big deal about it.
Oh, and if the working pressure doesn't exist, sizes are always in
liters. That's what we did in the Gtk branch, that's what we do here.
Again, no reason to even bother stating units, it's not helping.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-20 02:33:49 +00:00
|
|
|
mliter = rint(value.toDouble() * 1000);
|
|
|
|
}
|
|
|
|
if (cyl->type.size.mliter != mliter) {
|
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
cyl->type.size.mliter = mliter;
|
|
|
|
if (!matches.isEmpty())
|
|
|
|
tanks->setData(tanks->index(matches.first().row(), TankInfoModel::ML), cyl->type.size.mliter);
|
2013-05-22 15:20:00 +00:00
|
|
|
}
|
|
|
|
}
|
2013-05-22 20:29:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case WORKINGPRESS:
|
|
|
|
if (CHANGED(toDouble, "psi", "bar")) {
|
2013-07-18 15:28:28 +00:00
|
|
|
QString vString = value.toString();
|
|
|
|
vString.remove("psi").remove("bar");
|
|
|
|
if (vString.toDouble() != 0.0) {
|
2013-05-24 05:56:12 +00:00
|
|
|
TankInfoModel *tanks = TankInfoModel::instance();
|
|
|
|
QModelIndexList matches = tanks->match(tanks->index(0,0), Qt::DisplayRole, cyl->type.description);
|
2013-05-22 20:29:17 +00:00
|
|
|
if (prefs.units.pressure == prefs.units.PSI)
|
2013-07-18 15:28:28 +00:00
|
|
|
cyl->type.workingpressure.mbar = psi_to_mbar(vString.toDouble());
|
2013-05-22 20:29:17 +00:00
|
|
|
else
|
2013-07-18 15:28:28 +00:00
|
|
|
cyl->type.workingpressure.mbar = vString.toDouble() * 1000;
|
2013-05-24 05:56:12 +00:00
|
|
|
if (!matches.isEmpty())
|
|
|
|
tanks->setData(tanks->index(matches.first().row(), TankInfoModel::BAR), cyl->type.workingpressure.mbar / 1000.0);
|
2013-05-22 20:29:17 +00:00
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case START:
|
|
|
|
if (CHANGED(toDouble, "psi", "bar")) {
|
|
|
|
if (value.toDouble() != 0.0) {
|
|
|
|
if (prefs.units.pressure == prefs.units.PSI)
|
|
|
|
cyl->start.mbar = psi_to_mbar(value.toDouble());
|
|
|
|
else
|
|
|
|
cyl->start.mbar = value.toDouble() * 1000;
|
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case END:
|
|
|
|
if (CHANGED(toDouble, "psi", "bar")) {
|
|
|
|
if (value.toDouble() != 0.0) {
|
|
|
|
if (prefs.units.pressure == prefs.units.PSI)
|
|
|
|
cyl->end.mbar = psi_to_mbar(value.toDouble());
|
|
|
|
else
|
|
|
|
cyl->end.mbar = value.toDouble() * 1000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case O2:
|
2013-06-20 01:40:15 +00:00
|
|
|
if (CHANGED(toDouble, "%", "%")) {
|
2013-07-18 15:28:28 +00:00
|
|
|
cyl->gasmix.o2.permille = value.toString().remove('%').toDouble() * 10 + 0.5;
|
2013-05-22 20:29:17 +00:00
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case HE:
|
2013-06-20 01:40:15 +00:00
|
|
|
if (CHANGED(toDouble, "%", "%")) {
|
2013-07-18 15:28:28 +00:00
|
|
|
cyl->gasmix.he.permille = value.toString().remove('%').toDouble() * 10 + 0.5;
|
2013-05-22 20:29:17 +00:00
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
|
|
|
break;
|
2013-05-22 15:20:00 +00:00
|
|
|
}
|
2013-07-16 22:13:58 +00:00
|
|
|
dataChanged(index, index);
|
|
|
|
return true;
|
2013-05-22 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
int CylindersModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
2013-05-21 12:33:55 +00:00
|
|
|
return rows;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2013-05-22 17:52:38 +00:00
|
|
|
void CylindersModel::add()
|
2013-04-13 13:17:59 +00:00
|
|
|
{
|
2013-05-21 12:33:55 +00:00
|
|
|
if (rows >= MAX_CYLINDERS) {
|
2013-05-01 21:49:17 +00:00
|
|
|
return;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2013-05-21 12:33:55 +00:00
|
|
|
int row = rows;
|
2013-04-13 13:17:59 +00:00
|
|
|
|
|
|
|
beginInsertRows(QModelIndex(), row, row);
|
2013-05-21 12:33:55 +00:00
|
|
|
rows++;
|
2013-04-13 13:17:59 +00:00
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CylindersModel::update()
|
|
|
|
{
|
2013-05-21 12:33:55 +00:00
|
|
|
setDive(current);
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CylindersModel::clear()
|
|
|
|
{
|
2013-05-21 12:33:55 +00:00
|
|
|
if (rows > 0) {
|
|
|
|
beginRemoveRows(QModelIndex(), 0, rows-1);
|
2013-04-13 13:17:59 +00:00
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-21 12:33:55 +00:00
|
|
|
void CylindersModel::setDive(dive* d)
|
|
|
|
{
|
|
|
|
if (current)
|
|
|
|
clear();
|
|
|
|
|
2013-05-21 18:29:21 +00:00
|
|
|
int amount = MAX_CYLINDERS;
|
2013-05-23 04:25:05 +00:00
|
|
|
for(int i = 0; i < MAX_CYLINDERS; i++) {
|
2013-05-21 18:29:21 +00:00
|
|
|
cylinder_t *cylinder = &d->cylinder[i];
|
|
|
|
if (cylinder_none(cylinder)) {
|
2013-05-21 12:33:55 +00:00
|
|
|
amount = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
beginInsertRows(QModelIndex(), 0, amount-1);
|
|
|
|
rows = amount;
|
|
|
|
current = d;
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
2013-05-22 14:00:20 +00:00
|
|
|
Qt::ItemFlags CylindersModel::flags(const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
if (index.column() == REMOVE)
|
|
|
|
return Qt::ItemIsEnabled;
|
2013-05-22 15:20:00 +00:00
|
|
|
return QAbstractItemModel::flags(index) | Qt::ItemIsEditable;
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CylindersModel::remove(const QModelIndex& index)
|
|
|
|
{
|
2013-05-23 04:25:05 +00:00
|
|
|
if (index.column() != REMOVE) {
|
2013-05-22 14:00:20 +00:00
|
|
|
return;
|
|
|
|
}
|
2013-05-22 14:40:57 +00:00
|
|
|
beginRemoveRows(QModelIndex(), index.row(), index.row()); // yah, know, ugly.
|
2013-05-22 17:02:28 +00:00
|
|
|
rows--;
|
|
|
|
remove_cylinder(current, index.row());
|
|
|
|
mark_divelist_changed(TRUE);
|
2013-05-22 14:40:57 +00:00
|
|
|
endRemoveRows();
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
|
|
|
|
2013-05-24 01:40:16 +00:00
|
|
|
WeightModel::WeightModel(QObject* parent): QAbstractTableModel(parent), current(0), rows(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-07-18 14:53:47 +00:00
|
|
|
weightsystem_t* WeightModel::weightSystemAt(const QModelIndex& index)
|
|
|
|
{
|
|
|
|
return ¤t->weightsystem[index.row()];
|
|
|
|
}
|
|
|
|
|
2013-05-22 14:00:20 +00:00
|
|
|
void WeightModel::remove(const QModelIndex& index)
|
|
|
|
{
|
2013-05-23 04:25:05 +00:00
|
|
|
if (index.column() != REMOVE) {
|
2013-05-22 14:00:20 +00:00
|
|
|
return;
|
|
|
|
}
|
2013-05-22 14:40:57 +00:00
|
|
|
beginRemoveRows(QModelIndex(), index.row(), index.row()); // yah, know, ugly.
|
2013-05-22 17:02:28 +00:00
|
|
|
rows--;
|
|
|
|
remove_weightsystem(current, index.row());
|
|
|
|
mark_divelist_changed(TRUE);
|
2013-05-22 14:40:57 +00:00
|
|
|
endRemoveRows();
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
void WeightModel::clear()
|
|
|
|
{
|
2013-05-21 12:59:41 +00:00
|
|
|
if (rows > 0) {
|
|
|
|
beginRemoveRows(QModelIndex(), 0, rows-1);
|
2013-05-01 16:04:14 +00:00
|
|
|
endRemoveRows();
|
|
|
|
}
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int WeightModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
2013-05-22 14:00:20 +00:00
|
|
|
return COLUMNS;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant WeightModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
2013-05-01 17:11:46 +00:00
|
|
|
QVariant ret;
|
2013-05-02 05:00:08 +00:00
|
|
|
if (!index.isValid() || index.row() >= MAX_WEIGHTSYSTEMS)
|
2013-05-01 17:11:46 +00:00
|
|
|
return ret;
|
2013-05-02 05:00:08 +00:00
|
|
|
|
2013-05-24 01:40:16 +00:00
|
|
|
weightsystem_t *ws = ¤t->weightsystem[index.row()];
|
2013-05-01 17:11:46 +00:00
|
|
|
|
2013-05-26 03:04:31 +00:00
|
|
|
switch (role) {
|
|
|
|
case Qt::FontRole:
|
2013-06-16 14:13:32 +00:00
|
|
|
ret = defaultModelFont();
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
|
|
|
case Qt::TextAlignmentRole:
|
|
|
|
ret = Qt::AlignRight;
|
|
|
|
break;
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
case Qt::EditRole:
|
2013-05-01 17:11:46 +00:00
|
|
|
switch(index.column()) {
|
|
|
|
case TYPE:
|
|
|
|
ret = QString(ws->description);
|
|
|
|
break;
|
|
|
|
case WEIGHT:
|
2013-05-21 19:03:05 +00:00
|
|
|
ret = get_weight_string(ws->weight, TRUE);
|
2013-05-01 17:11:46 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
|
|
|
case Qt::DecorationRole:
|
2013-05-23 04:25:05 +00:00
|
|
|
if (index.column() == REMOVE)
|
2013-05-22 14:00:20 +00:00
|
|
|
ret = QIcon(":trash");
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
2013-05-01 17:11:46 +00:00
|
|
|
return ret;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
2013-05-24 01:40:16 +00:00
|
|
|
// this is our magic 'pass data in' function that allows the delegate to get
|
|
|
|
// the data here without silly unit conversions;
|
|
|
|
// so we only implement the two columns we care about
|
|
|
|
void WeightModel::passInData(const QModelIndex& index, const QVariant& value)
|
|
|
|
{
|
|
|
|
weightsystem_t *ws = ¤t->weightsystem[index.row()];
|
|
|
|
if (index.column() == WEIGHT) {
|
|
|
|
if (ws->weight.grams != value.toInt()) {
|
|
|
|
ws->weight.grams = value.toInt();
|
|
|
|
mark_divelist_changed(TRUE);
|
2013-07-16 22:13:58 +00:00
|
|
|
dataChanged(index, index);
|
2013-05-24 01:40:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-22 15:20:00 +00:00
|
|
|
bool WeightModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
|
|
{
|
2013-05-24 01:40:16 +00:00
|
|
|
weightsystem_t *ws = ¤t->weightsystem[index.row()];
|
2013-05-22 15:20:00 +00:00
|
|
|
switch(index.column()) {
|
2013-05-24 01:40:16 +00:00
|
|
|
case TYPE:
|
|
|
|
if (!value.isNull()) {
|
Fix usage of temporary QByteArrays
This commit fixes three different things:
- a memory leak in WeightModel::setData()
- getSetting() calling strdup() on a QByteArray
- a possible usage of memory after deallocation
Here's an explanation of the last issue (taken from the mailing list, slightly
adapted):
toByteArray(), as well as others "toSomething()" methods, returns
a new object which the compiler allocates on the stack. The compiler
will consider it a temporary data, and destroy it on the next line. So,
when one does
char *text= value.toByteArray().data(); // line 1
if (strcmp(description, text)) { // line 2
the compiler creates a QByteArray on line 1, calls ::data() on it, which
returns a valid char *, and assigns its value to "text". So far, so
good. But before jumping to line 2, the compiler destroys the temporary
QByteArray, and this will in turn invoke the QByteArray destructor,
which will destroy the internal data. The result is that on line 2,
"text" will point to some memory which has already been freed.
One solution is to store a copy of the temporary QByteArray into a local
variable: the compiler will still destroy the temporary QByteArray it created,
but (thanks to the reference-counted data sharing built in QByteArray) now the
destructor will see that the data is referenced by another instance of
QByteArray (the local variable "ba") and will not free the internal data.
In this way, the internal data will be available until the local variable is
destroyed, which will happen at the end of the {} block where it is defined.
Please note that when one uses the data in the same line, one doesn't need to
worry about this issue. In fact,
text = strdup(value.toString().toUtf8().data());
works just fine.
Signed-off-by: Alberto Mardegan <mardy@users.sourceforge.net>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-05-24 11:39:37 +00:00
|
|
|
QByteArray ba = value.toString().toUtf8();
|
|
|
|
const char *text = ba.constData();
|
2013-05-24 01:40:16 +00:00
|
|
|
if (!ws->description || strcmp(ws->description, text)) {
|
|
|
|
ws->description = strdup(text);
|
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
}
|
|
|
|
}
|
2013-05-22 15:20:00 +00:00
|
|
|
break;
|
|
|
|
case WEIGHT:
|
2013-05-24 01:40:16 +00:00
|
|
|
if (CHANGED(toDouble, "kg", "lbs")) {
|
|
|
|
if (prefs.units.weight == prefs.units.LBS)
|
|
|
|
ws->weight.grams = lbs_to_grams(value.toDouble());
|
|
|
|
else
|
2013-06-20 01:40:15 +00:00
|
|
|
ws->weight.grams = value.toDouble() * 1000.0 + 0.5;
|
2013-05-24 05:56:12 +00:00
|
|
|
// now update the ws_info
|
|
|
|
WSInfoModel *wsim = WSInfoModel::instance();
|
|
|
|
QModelIndexList matches = wsim->match(wsim->index(0,0), Qt::DisplayRole, ws->description);
|
|
|
|
if (!matches.isEmpty())
|
|
|
|
wsim->setData(wsim->index(matches.first().row(), WSInfoModel::GR), ws->weight.grams);
|
2013-05-24 01:40:16 +00:00
|
|
|
}
|
2013-05-22 15:20:00 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-07-16 22:13:58 +00:00
|
|
|
dataChanged(index, index);
|
|
|
|
return true;
|
2013-05-22 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2013-05-22 14:00:20 +00:00
|
|
|
Qt::ItemFlags WeightModel::flags(const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
if (index.column() == REMOVE)
|
|
|
|
return Qt::ItemIsEnabled;
|
2013-05-22 15:20:00 +00:00
|
|
|
return QAbstractItemModel::flags(index) | Qt::ItemIsEditable;
|
2013-05-22 14:00:20 +00:00
|
|
|
}
|
|
|
|
|
2013-04-13 13:17:59 +00:00
|
|
|
int WeightModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
2013-05-21 12:59:41 +00:00
|
|
|
return rows;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant WeightModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
2013-05-02 05:00:08 +00:00
|
|
|
if (orientation == Qt::Vertical)
|
2013-04-13 13:17:59 +00:00
|
|
|
return ret;
|
|
|
|
|
2013-05-26 03:04:31 +00:00
|
|
|
switch (role) {
|
|
|
|
case Qt::FontRole:
|
2013-06-16 14:13:32 +00:00
|
|
|
ret = defaultModelFont();
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
|
|
|
case Qt::DisplayRole:
|
2013-05-01 21:30:34 +00:00
|
|
|
switch(section) {
|
|
|
|
case TYPE:
|
|
|
|
ret = tr("Type");
|
|
|
|
break;
|
|
|
|
case WEIGHT:
|
|
|
|
ret = tr("Weight");
|
|
|
|
break;
|
|
|
|
}
|
2013-05-26 03:04:31 +00:00
|
|
|
break;
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-05-22 17:52:38 +00:00
|
|
|
void WeightModel::add()
|
2013-04-13 13:17:59 +00:00
|
|
|
{
|
2013-05-21 18:29:21 +00:00
|
|
|
if (rows >= MAX_WEIGHTSYSTEMS)
|
2013-05-01 21:49:17 +00:00
|
|
|
return;
|
2013-05-01 17:11:46 +00:00
|
|
|
|
2013-05-21 12:59:41 +00:00
|
|
|
int row = rows;
|
2013-05-01 17:11:46 +00:00
|
|
|
beginInsertRows(QModelIndex(), row, row);
|
2013-05-21 12:59:41 +00:00
|
|
|
rows++;
|
2013-05-01 17:11:46 +00:00
|
|
|
endInsertRows();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void WeightModel::update()
|
|
|
|
{
|
2013-05-21 12:59:41 +00:00
|
|
|
setDive(current);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WeightModel::setDive(dive* d)
|
|
|
|
{
|
|
|
|
if (current)
|
|
|
|
clear();
|
|
|
|
|
2013-05-21 18:29:21 +00:00
|
|
|
int amount = MAX_WEIGHTSYSTEMS;
|
|
|
|
for(int i = 0; i < MAX_WEIGHTSYSTEMS; i++) {
|
|
|
|
weightsystem_t *weightsystem = &d->weightsystem[i];
|
|
|
|
if (weightsystem_none(weightsystem)) {
|
2013-05-21 12:59:41 +00:00
|
|
|
amount = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
beginInsertRows(QModelIndex(), 0, amount-1);
|
|
|
|
rows = amount;
|
|
|
|
current = d;
|
|
|
|
endInsertRows();
|
2013-04-13 13:17:59 +00:00
|
|
|
}
|
2013-04-15 18:04:35 +00:00
|
|
|
|
2013-05-24 01:40:16 +00:00
|
|
|
WSInfoModel* WSInfoModel::instance()
|
|
|
|
{
|
|
|
|
static WSInfoModel *self = new WSInfoModel();
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WSInfoModel::insertRows(int row, int count, const QModelIndex& parent)
|
|
|
|
{
|
|
|
|
beginInsertRows(parent, rowCount(), rowCount());
|
|
|
|
rows += count;
|
|
|
|
endInsertRows();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WSInfoModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
|
|
{
|
|
|
|
struct ws_info *info = &ws_info[index.row()];
|
2013-05-24 05:56:12 +00:00
|
|
|
switch(index.column()) {
|
|
|
|
case DESCRIPTION:
|
|
|
|
info->name = strdup(value.toByteArray().data());
|
|
|
|
break;
|
|
|
|
case GR:
|
|
|
|
info->grams = value.toInt();
|
|
|
|
break;
|
|
|
|
}
|
2013-07-16 22:13:58 +00:00
|
|
|
emit dataChanged(index, index);
|
2013-05-24 01:40:16 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WSInfoModel::clear()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int WSInfoModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant WSInfoModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
|
|
|
if (!index.isValid()) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
struct ws_info *info = &ws_info[index.row()];
|
|
|
|
|
|
|
|
int gr = info->grams;
|
2013-06-16 14:13:32 +00:00
|
|
|
switch(role){
|
|
|
|
case Qt::FontRole :
|
|
|
|
ret = defaultModelFont();
|
|
|
|
break;
|
|
|
|
case Qt::DisplayRole :
|
|
|
|
case Qt::EditRole :
|
|
|
|
switch(index.column()) {
|
|
|
|
case GR:
|
|
|
|
ret = gr;
|
|
|
|
break;
|
|
|
|
case DESCRIPTION:
|
|
|
|
ret = QString(info->name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2013-05-24 01:40:16 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant WSInfoModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
|
|
|
|
|
|
|
if (orientation != Qt::Horizontal)
|
|
|
|
return ret;
|
|
|
|
|
2013-06-16 14:13:32 +00:00
|
|
|
switch(role){
|
|
|
|
case Qt::FontRole :
|
|
|
|
ret = defaultModelFont();
|
|
|
|
break;
|
|
|
|
case Qt::DisplayRole :
|
|
|
|
switch(section) {
|
|
|
|
case GR:
|
|
|
|
ret = tr("kg");
|
|
|
|
break;
|
|
|
|
case DESCRIPTION:
|
|
|
|
ret = tr("Description");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2013-05-24 01:40:16 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int WSInfoModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return rows+1;
|
|
|
|
}
|
|
|
|
|
2013-06-16 16:28:59 +00:00
|
|
|
const QString& WSInfoModel::biggerString() const
|
|
|
|
{
|
|
|
|
return biggerEntry;
|
|
|
|
}
|
|
|
|
|
2013-05-24 01:40:16 +00:00
|
|
|
WSInfoModel::WSInfoModel() : QAbstractTableModel(), rows(-1)
|
|
|
|
{
|
|
|
|
struct ws_info *info = ws_info;
|
2013-06-16 16:28:59 +00:00
|
|
|
for (info = ws_info; info->name; info++, rows++){
|
|
|
|
QString wsInfoName(info->name);
|
|
|
|
if( wsInfoName.count() > biggerEntry.count()){
|
|
|
|
biggerEntry = wsInfoName;
|
|
|
|
}
|
|
|
|
}
|
2013-05-24 01:40:16 +00:00
|
|
|
|
|
|
|
if (rows > -1) {
|
|
|
|
beginInsertRows(QModelIndex(), 0, rows);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WSInfoModel::update()
|
|
|
|
{
|
|
|
|
if (rows > -1) {
|
|
|
|
beginRemoveRows(QModelIndex(), 0, rows);
|
|
|
|
endRemoveRows();
|
2013-05-24 05:56:12 +00:00
|
|
|
rows = -1;
|
2013-05-24 01:40:16 +00:00
|
|
|
}
|
|
|
|
struct ws_info *info = ws_info;
|
|
|
|
for (info = ws_info; info->name; info++, rows++);
|
|
|
|
|
|
|
|
if (rows > -1) {
|
|
|
|
beginInsertRows(QModelIndex(), 0, rows);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-23 18:33:20 +00:00
|
|
|
TankInfoModel* TankInfoModel::instance()
|
2013-04-15 18:04:35 +00:00
|
|
|
{
|
2013-05-23 18:33:20 +00:00
|
|
|
static TankInfoModel *self = new TankInfoModel();
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2013-06-16 16:28:59 +00:00
|
|
|
const QString& TankInfoModel::biggerString() const
|
|
|
|
{
|
|
|
|
return biggerEntry;
|
|
|
|
}
|
|
|
|
|
2013-05-23 18:33:20 +00:00
|
|
|
bool TankInfoModel::insertRows(int row, int count, const QModelIndex& parent)
|
|
|
|
{
|
|
|
|
beginInsertRows(parent, rowCount(), rowCount());
|
|
|
|
rows += count;
|
|
|
|
endInsertRows();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TankInfoModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
|
|
{
|
|
|
|
struct tank_info *info = &tank_info[index.row()];
|
2013-05-24 05:56:12 +00:00
|
|
|
switch(index.column()) {
|
|
|
|
case DESCRIPTION:
|
|
|
|
info->name = strdup(value.toByteArray().data());
|
|
|
|
break;
|
|
|
|
case ML:
|
|
|
|
info->ml = value.toInt();
|
|
|
|
break;
|
|
|
|
case BAR:
|
|
|
|
info->bar = value.toInt();
|
|
|
|
break;
|
|
|
|
}
|
2013-07-16 22:13:58 +00:00
|
|
|
emit dataChanged(index, index);
|
2013-05-24 01:40:16 +00:00
|
|
|
return TRUE;
|
2013-04-15 18:04:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TankInfoModel::clear()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int TankInfoModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant TankInfoModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
|
|
|
if (!index.isValid()) {
|
|
|
|
return ret;
|
|
|
|
}
|
2013-06-16 15:33:27 +00:00
|
|
|
if (role == Qt::FontRole){
|
|
|
|
return defaultModelFont();
|
|
|
|
}
|
2013-06-19 21:36:48 +00:00
|
|
|
if (role == Qt::DisplayRole || role == Qt::EditRole) {
|
|
|
|
struct tank_info *info = &tank_info[index.row()];
|
|
|
|
int ml = info->ml;
|
|
|
|
double bar = (info->psi) ? psi_to_bar(info->psi) : info->bar;
|
2013-06-16 15:33:27 +00:00
|
|
|
|
2013-06-19 21:36:48 +00:00
|
|
|
if (info->cuft && info->psi)
|
|
|
|
ml = cuft_to_l(info->cuft) * 1000 / bar_to_atm(bar);
|
2013-04-15 18:04:35 +00:00
|
|
|
|
|
|
|
switch(index.column()) {
|
2013-05-02 05:00:08 +00:00
|
|
|
case BAR:
|
2013-06-19 21:36:48 +00:00
|
|
|
ret = bar * 1000;
|
2013-04-15 18:04:35 +00:00
|
|
|
break;
|
2013-05-02 05:00:08 +00:00
|
|
|
case ML:
|
|
|
|
ret = ml;
|
2013-04-15 18:04:35 +00:00
|
|
|
break;
|
|
|
|
case DESCRIPTION:
|
|
|
|
ret = QString(info->name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant TankInfoModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
|
|
|
|
|
|
|
if (orientation != Qt::Horizontal)
|
|
|
|
return ret;
|
|
|
|
|
2013-06-16 14:13:32 +00:00
|
|
|
switch(role){
|
|
|
|
case Qt::FontRole:
|
|
|
|
ret = defaultModelFont();
|
|
|
|
break;
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
switch(section) {
|
2013-04-15 18:04:35 +00:00
|
|
|
case BAR:
|
|
|
|
ret = tr("Bar");
|
|
|
|
break;
|
|
|
|
case ML:
|
|
|
|
ret = tr("Ml");
|
|
|
|
break;
|
|
|
|
case DESCRIPTION:
|
|
|
|
ret = tr("Description");
|
|
|
|
break;
|
2013-06-16 14:13:32 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-04-15 18:04:35 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int TankInfoModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return rows+1;
|
|
|
|
}
|
|
|
|
|
|
|
|
TankInfoModel::TankInfoModel() : QAbstractTableModel(), rows(-1)
|
|
|
|
{
|
|
|
|
struct tank_info *info = tank_info;
|
2013-06-16 16:28:59 +00:00
|
|
|
for (info = tank_info; info->name; info++, rows++){
|
|
|
|
QString infoName(info->name);
|
|
|
|
if (infoName.count() > biggerEntry.count()){
|
|
|
|
biggerEntry = infoName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-15 18:04:35 +00:00
|
|
|
if (rows > -1) {
|
|
|
|
beginInsertRows(QModelIndex(), 0, rows);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TankInfoModel::update()
|
|
|
|
{
|
2013-05-23 04:25:05 +00:00
|
|
|
if (rows > -1) {
|
2013-04-15 18:04:35 +00:00
|
|
|
beginRemoveRows(QModelIndex(), 0, rows);
|
|
|
|
endRemoveRows();
|
2013-05-24 05:56:12 +00:00
|
|
|
rows = -1;
|
2013-04-15 18:04:35 +00:00
|
|
|
}
|
|
|
|
struct tank_info *info = tank_info;
|
2013-05-02 05:00:08 +00:00
|
|
|
for (info = tank_info; info->name; info++, rows++);
|
2013-04-15 18:04:35 +00:00
|
|
|
|
|
|
|
if (rows > -1) {
|
|
|
|
beginInsertRows(QModelIndex(), 0, rows);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
2013-04-22 01:12:36 +00:00
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
//#################################################################################################
|
|
|
|
//#
|
|
|
|
//# Tree Model - a Basic Tree Model so I don't need to kill myself repeating this for every model.
|
|
|
|
//#
|
|
|
|
//#################################################################################################
|
2013-04-27 15:27:27 +00:00
|
|
|
|
2013-04-22 01:12:36 +00:00
|
|
|
/*! A DiveItem for use with a DiveTripModel
|
|
|
|
*
|
|
|
|
* A simple class which wraps basic stats for a dive (e.g. duration, depth) and
|
|
|
|
* tidies up after it's children. This is done manually as we don't inherit from
|
|
|
|
* QObject.
|
|
|
|
*
|
|
|
|
*/
|
2013-05-02 02:51:34 +00:00
|
|
|
|
2013-06-18 20:29:37 +00:00
|
|
|
TreeItem::TreeItem()
|
|
|
|
{
|
|
|
|
parent = NULL;
|
|
|
|
}
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
TreeItem::~TreeItem()
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-05-14 07:28:30 +00:00
|
|
|
qDeleteAll(children);
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
2013-04-25 07:50:01 +00:00
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
int TreeItem::row() const
|
2013-05-02 02:51:34 +00:00
|
|
|
{
|
|
|
|
if (parent)
|
2013-06-17 21:59:50 +00:00
|
|
|
return parent->children.indexOf(const_cast<TreeItem*>(this));
|
2013-05-02 02:51:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2013-04-22 01:12:36 +00:00
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
QVariant TreeItem::data(int column, int role) const
|
2013-05-02 02:51:34 +00:00
|
|
|
{
|
2013-06-17 21:59:50 +00:00
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
TreeModel::TreeModel(QObject* parent): QAbstractItemModel(parent)
|
|
|
|
{
|
|
|
|
rootItem = new TreeItem();
|
|
|
|
}
|
|
|
|
|
|
|
|
TreeModel::~TreeModel()
|
|
|
|
{
|
|
|
|
delete rootItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant TreeModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
TreeItem* item = static_cast<TreeItem*>(index.internalPointer());
|
2013-06-19 17:30:36 +00:00
|
|
|
QVariant val = item->data(index.column(), role);
|
2013-06-17 21:59:50 +00:00
|
|
|
|
2013-06-19 17:30:36 +00:00
|
|
|
if (role == Qt::FontRole && !val.isValid())
|
|
|
|
return defaultModelFont();
|
|
|
|
else
|
|
|
|
return val;
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
QModelIndex TreeModel::index(int row, int column, const QModelIndex& parent)
|
|
|
|
const
|
|
|
|
{
|
|
|
|
if (!hasIndex(row, column, parent))
|
|
|
|
return QModelIndex();
|
|
|
|
|
|
|
|
TreeItem* parentItem = (!parent.isValid()) ? rootItem : static_cast<TreeItem*>(parent.internalPointer());
|
|
|
|
|
|
|
|
TreeItem* childItem = parentItem->children[row];
|
|
|
|
|
|
|
|
return (childItem) ? createIndex(row, column, childItem) : QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex TreeModel::parent(const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return QModelIndex();
|
|
|
|
|
|
|
|
TreeItem* childItem = static_cast<TreeItem*>(index.internalPointer());
|
|
|
|
TreeItem* parentItem = childItem->parent;
|
|
|
|
|
|
|
|
if (parentItem == rootItem || !parentItem)
|
|
|
|
return QModelIndex();
|
|
|
|
|
|
|
|
return createIndex(parentItem->row(), 0, parentItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
int TreeModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
TreeItem* parentItem;
|
|
|
|
|
|
|
|
if (!parent.isValid())
|
|
|
|
parentItem = rootItem;
|
|
|
|
else
|
|
|
|
parentItem = static_cast<TreeItem*>(parent.internalPointer());
|
|
|
|
|
|
|
|
int amount = parentItem->children.count();
|
|
|
|
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
|
|
|
int TreeModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return columns;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*################################################################
|
|
|
|
*
|
|
|
|
* Implementation of the Dive List.
|
|
|
|
*
|
|
|
|
* ############################################################### */
|
|
|
|
struct TripItem : public TreeItem {
|
2013-05-02 02:51:34 +00:00
|
|
|
virtual QVariant data(int column, int role) const;
|
|
|
|
dive_trip_t* trip;
|
|
|
|
};
|
|
|
|
|
|
|
|
QVariant TripItem::data(int column, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
2013-04-25 14:04:41 +00:00
|
|
|
|
2013-06-27 05:16:40 +00:00
|
|
|
if (role == DiveTripModel::TRIP_ROLE)
|
|
|
|
return QVariant::fromValue<void*>(trip);
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
if (role == DiveTripModel::SORT_ROLE)
|
2013-05-29 05:54:39 +00:00
|
|
|
return (qulonglong)trip->when;
|
|
|
|
|
2013-05-02 16:33:51 +00:00
|
|
|
if (role == Qt::DisplayRole) {
|
|
|
|
switch (column) {
|
2013-06-17 21:59:50 +00:00
|
|
|
case DiveTripModel::NR:
|
2013-06-13 15:19:23 +00:00
|
|
|
ret = QString(trip->location) + ", " + QString(get_trip_date_string(trip->when, trip->nrdives));
|
2013-05-02 16:33:51 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-05-29 05:54:39 +00:00
|
|
|
static int nitrox_sort_value(struct dive *dive)
|
|
|
|
{
|
|
|
|
int o2, he, o2low;
|
|
|
|
get_dive_gas(dive, &o2, &he, &o2low);
|
|
|
|
return he*1000 + o2;
|
|
|
|
}
|
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
QVariant DiveItem::data(int column, int role) const
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-05-02 02:51:34 +00:00
|
|
|
QVariant retVal;
|
|
|
|
|
2013-05-02 21:05:53 +00:00
|
|
|
switch (role) {
|
|
|
|
case Qt::TextAlignmentRole:
|
2013-05-02 02:51:34 +00:00
|
|
|
switch (column) {
|
|
|
|
case DATE: /* fall through */
|
|
|
|
case SUIT: /* fall through */
|
|
|
|
case LOCATION:
|
|
|
|
retVal = Qt::AlignLeft;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
retVal = Qt::AlignRight;
|
|
|
|
break;
|
|
|
|
}
|
2013-05-02 21:05:53 +00:00
|
|
|
break;
|
2013-06-17 21:59:50 +00:00
|
|
|
case DiveTripModel::SORT_ROLE:
|
2013-05-29 05:54:39 +00:00
|
|
|
switch (column) {
|
Sort 'dive number' column by date, not number
Ok, so this sounds insane, but it fixes our currently broken sorting
of dive trips vs plain dives not in trips.
The reason for that is simple: the dive trips are sorted by date, but
that's column #0, and for plain dives is the dive number.
So currently the trip-vs-dive sorting looks at the date of the trip,
and compares that to the number of the dive. Since the date of the
trip is expressed as seconds-since-1970-in-UTC, unsurprisingly the
dive number is generally much smaller (even for some very avid divers
;), and so the plain dives end up sorting way at the bottom (or at the
top, if you do "oldest trips first"
Since the dive number *should* sort as the date, this stupid attached
patch just makes us return the dive date instead.
Now, there are other possible solutions to this:
- make the date of the dive be column 0, and make the dive number be column 1.
Quite frankly, while more logical for this particular problem, it
probably sucks as a solution. We do want to have a column we can sort
dives by that is date-based, but doesn't include trips. And while the
dive number *should* sort identically to the date one, the fact is
that you can have dives without any numbering, so it doesn't.
In contrast, all dives have dates, and sorting numbered dives by
date should still result in sane behavior (and if it doesn't, then the
insanity comes from the dive numbering, and odd sorting is the fault
of the user and indicative of a problem)
- We could possibly do something magical like sorting dives by number
when they are inside trips, or when no trips exist at all. But then
we'd sort by date when there are trips and the dive is outside the
trip. But quite frankly, that just sounds insane.
So on the whole, I don't love this patch, but it seems to be the least
confusing of the possibilities.
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
2013-06-09 00:58:16 +00:00
|
|
|
case NR: retVal = (qulonglong) dive->when; break;
|
2013-05-29 20:43:14 +00:00
|
|
|
case DATE: retVal = (qulonglong) dive->when; break;
|
2013-05-29 20:53:50 +00:00
|
|
|
case RATING: retVal = dive->rating; break;
|
2013-05-29 20:43:14 +00:00
|
|
|
case DEPTH: retVal = dive->maxdepth.mm; break;
|
|
|
|
case DURATION: retVal = dive->duration.seconds; break;
|
|
|
|
case TEMPERATURE: retVal = dive->watertemp.mkelvin; break;
|
|
|
|
case TOTALWEIGHT: retVal = total_weight(dive); break;
|
|
|
|
case SUIT: retVal = QString(dive->suit); break;
|
|
|
|
case CYLINDER: retVal = QString(dive->cylinder[0].type.description); break;
|
|
|
|
case NITROX: retVal = nitrox_sort_value(dive); break;
|
|
|
|
case SAC: retVal = dive->sac; break;
|
|
|
|
case OTU: retVal = dive->otu; break;
|
|
|
|
case MAXCNS: retVal = dive->maxcns; break;
|
|
|
|
case LOCATION: retVal = QString(dive->location); break;
|
2013-05-29 05:54:39 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-05-02 21:05:53 +00:00
|
|
|
case Qt::DisplayRole:
|
2013-05-02 02:51:34 +00:00
|
|
|
switch (column) {
|
2013-05-29 20:43:14 +00:00
|
|
|
case NR: retVal = dive->number; break;
|
2013-07-11 09:41:50 +00:00
|
|
|
case DATE: retVal = displayDate(); break;
|
2013-05-29 20:43:14 +00:00
|
|
|
case DEPTH: retVal = displayDepth(); break;
|
|
|
|
case DURATION: retVal = displayDuration(); break;
|
|
|
|
case TEMPERATURE: retVal = displayTemperature(); break;
|
|
|
|
case TOTALWEIGHT: retVal = displayWeight(); break;
|
|
|
|
case SUIT: retVal = QString(dive->suit); break;
|
|
|
|
case CYLINDER: retVal = QString(dive->cylinder[0].type.description); break;
|
|
|
|
case NITROX: retVal = QString(get_nitrox_string(dive)); break;
|
|
|
|
case SAC: retVal = displaySac(); break;
|
|
|
|
case OTU: retVal = dive->otu; break;
|
|
|
|
case MAXCNS: retVal = dive->maxcns; break;
|
|
|
|
case LOCATION: retVal = QString(dive->location); break;
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
2013-05-02 21:05:53 +00:00
|
|
|
break;
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
2013-05-02 23:32:57 +00:00
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
if (role == DiveTripModel::STAR_ROLE)
|
2013-05-02 23:32:57 +00:00
|
|
|
retVal = dive->rating;
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
if (role == DiveTripModel::DIVE_ROLE)
|
2013-05-02 23:32:57 +00:00
|
|
|
retVal = QVariant::fromValue<void*>(dive);
|
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
return retVal;
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2013-07-11 09:41:50 +00:00
|
|
|
QString DiveItem::displayDate() const
|
|
|
|
{
|
|
|
|
char *buf = get_dive_date_string(dive->when);
|
|
|
|
QString date(buf);
|
|
|
|
free(buf);
|
|
|
|
return date;
|
|
|
|
}
|
|
|
|
|
2013-04-24 15:57:30 +00:00
|
|
|
QString DiveItem::displayDepth() const
|
|
|
|
{
|
|
|
|
const int scale = 1000;
|
|
|
|
QString fract, str;
|
|
|
|
if (get_units()->length == units::METERS) {
|
2013-04-25 14:04:41 +00:00
|
|
|
fract = QString::number((unsigned)(dive->maxdepth.mm % scale) / 10);
|
|
|
|
str = QString("%1.%2").arg((unsigned)(dive->maxdepth.mm / scale)).arg(fract, 2, QChar('0'));
|
2013-04-24 15:57:30 +00:00
|
|
|
}
|
|
|
|
if (get_units()->length == units::FEET) {
|
2013-04-25 14:04:41 +00:00
|
|
|
str = QString::number(mm_to_feet(dive->maxdepth.mm),'f',0);
|
2013-04-24 15:57:30 +00:00
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString DiveItem::displayDuration() const
|
|
|
|
{
|
2013-04-24 23:03:14 +00:00
|
|
|
int hrs, mins, secs;
|
2013-04-25 14:04:41 +00:00
|
|
|
secs = dive->duration.seconds % 60;
|
|
|
|
mins = dive->duration.seconds / 60;
|
2013-04-24 23:03:14 +00:00
|
|
|
hrs = mins / 60;
|
|
|
|
mins -= hrs * 60;
|
2013-04-24 15:57:30 +00:00
|
|
|
|
|
|
|
QString displayTime;
|
2013-04-24 23:03:14 +00:00
|
|
|
if (hrs)
|
|
|
|
displayTime = QString("%1:%2:").arg(hrs).arg(mins, 2, 10, QChar('0'));
|
2013-04-24 15:57:30 +00:00
|
|
|
else
|
2013-04-24 23:03:14 +00:00
|
|
|
displayTime = QString("%1:").arg(mins);
|
|
|
|
displayTime += QString("%1").arg(secs, 2, 10, QChar('0'));
|
2013-04-24 15:57:30 +00:00
|
|
|
return displayTime;
|
|
|
|
}
|
|
|
|
|
2013-04-25 06:21:57 +00:00
|
|
|
QString DiveItem::displayTemperature() const
|
|
|
|
{
|
|
|
|
QString str;
|
|
|
|
|
2013-05-14 07:45:01 +00:00
|
|
|
if (!dive->watertemp.mkelvin)
|
|
|
|
return str;
|
|
|
|
|
2013-05-02 05:00:08 +00:00
|
|
|
if (get_units()->temperature == units::CELSIUS)
|
2013-04-25 14:04:41 +00:00
|
|
|
str = QString::number(mkelvin_to_C(dive->watertemp.mkelvin), 'f', 1);
|
2013-05-02 05:00:08 +00:00
|
|
|
else
|
2013-04-25 14:04:41 +00:00
|
|
|
str = QString::number(mkelvin_to_F(dive->watertemp.mkelvin), 'f', 1);
|
2013-05-02 05:00:08 +00:00
|
|
|
|
2013-04-25 06:21:57 +00:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString DiveItem::displaySac() const
|
|
|
|
{
|
|
|
|
QString str;
|
|
|
|
|
2013-05-02 05:00:08 +00:00
|
|
|
if (get_units()->volume == units::LITER)
|
2013-06-12 15:58:57 +00:00
|
|
|
str = QString::number(dive->sac / 1000.0, 'f', 1);
|
2013-05-02 05:00:08 +00:00
|
|
|
else
|
2013-04-25 14:04:41 +00:00
|
|
|
str = QString::number(ml_to_cuft(dive->sac), 'f', 2);
|
2013-05-02 05:00:08 +00:00
|
|
|
|
2013-04-25 06:21:57 +00:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString DiveItem::displayWeight() const
|
|
|
|
{
|
|
|
|
QString str;
|
|
|
|
|
|
|
|
if (get_units()->weight == units::KG) {
|
2013-05-02 02:51:34 +00:00
|
|
|
int gr = weight() % 1000;
|
|
|
|
int kg = weight() / 1000;
|
2013-05-14 07:45:01 +00:00
|
|
|
str = QString("%1.%2").arg(kg).arg((unsigned)(gr) / 100);
|
2013-04-25 06:21:57 +00:00
|
|
|
} else {
|
2013-05-14 07:45:01 +00:00
|
|
|
str = QString("%1").arg((unsigned)(grams_to_lbs(weight())));
|
2013-04-25 06:21:57 +00:00
|
|
|
}
|
2013-05-02 02:51:34 +00:00
|
|
|
|
2013-04-25 06:21:57 +00:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
int DiveItem::weight() const
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-05-02 02:51:34 +00:00
|
|
|
weight_t tw = { total_weight(dive) };
|
|
|
|
return tw.grams;
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
DiveTripModel::DiveTripModel(QObject* parent): TreeModel(parent)
|
2013-05-02 02:51:34 +00:00
|
|
|
{
|
2013-06-17 21:59:50 +00:00
|
|
|
columns = COLUMNS;
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
Qt::ItemFlags DiveTripModel::flags(const QModelIndex& index) const
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-05-02 02:51:34 +00:00
|
|
|
if (!index.isValid())
|
2013-04-22 01:12:36 +00:00
|
|
|
return 0;
|
2013-05-02 02:51:34 +00:00
|
|
|
|
|
|
|
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
QVariant DiveTripModel::headerData(int section, Qt::Orientation orientation, int role) const
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-06-17 21:59:50 +00:00
|
|
|
QVariant ret;
|
|
|
|
if (orientation == Qt::Vertical)
|
|
|
|
return ret;
|
2013-04-22 01:12:36 +00:00
|
|
|
|
2013-06-16 14:13:32 +00:00
|
|
|
switch(role){
|
|
|
|
case Qt::FontRole :
|
2013-06-17 21:59:50 +00:00
|
|
|
ret = defaultModelFont(); break;
|
|
|
|
case Qt::DisplayRole :
|
|
|
|
switch (section) {
|
|
|
|
case NR: ret = tr("#"); break;
|
|
|
|
case DATE: ret = tr("Date"); break;
|
|
|
|
case RATING: ret = UTF8_BLACKSTAR; break;
|
|
|
|
case DEPTH: ret = (get_units()->length == units::METERS) ? tr("m") : tr("ft"); break;
|
|
|
|
case DURATION: ret = tr("min"); break;
|
|
|
|
case TEMPERATURE:ret = QString("%1%2").arg(UTF8_DEGREE).arg((get_units()->temperature == units::CELSIUS) ? "C" : "F"); break;
|
|
|
|
case TOTALWEIGHT:ret = (get_units()->weight == units::KG) ? tr("kg") : tr("lbs"); break;
|
|
|
|
case SUIT: ret = tr("Suit"); break;
|
|
|
|
case CYLINDER: ret = tr("Cyl"); break;
|
|
|
|
case NITROX: ret = QString("O%1%").arg(UTF8_SUBSCRIPT_2); break;
|
|
|
|
case SAC: ret = tr("SAC"); break;
|
|
|
|
case OTU: ret = tr("OTU"); break;
|
|
|
|
case MAXCNS: ret = tr("maxCNS"); break;
|
|
|
|
case LOCATION: ret = tr("Location"); break;
|
|
|
|
}break;
|
2013-06-16 14:13:32 +00:00
|
|
|
}
|
|
|
|
|
2013-06-17 21:59:50 +00:00
|
|
|
return ret;
|
2013-05-02 02:51:34 +00:00
|
|
|
}
|
2013-04-22 01:12:36 +00:00
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
void DiveTripModel::setupModelData()
|
2013-04-22 01:12:36 +00:00
|
|
|
{
|
2013-05-02 02:51:34 +00:00
|
|
|
int i = dive_table.nr;
|
|
|
|
|
2013-05-28 19:56:58 +00:00
|
|
|
if (rowCount()){
|
|
|
|
beginRemoveRows(QModelIndex(), 0, rowCount()-1);
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
|
2013-06-15 23:45:04 +00:00
|
|
|
if (autogroup)
|
|
|
|
autogroup_dives();
|
|
|
|
dive_table.preexisting = dive_table.nr;
|
2013-05-02 02:51:34 +00:00
|
|
|
while (--i >= 0) {
|
|
|
|
struct dive* dive = get_dive(i);
|
2013-05-07 03:36:37 +00:00
|
|
|
update_cylinder_related_info(dive);
|
2013-05-02 02:51:34 +00:00
|
|
|
dive_trip_t* trip = dive->divetrip;
|
|
|
|
|
|
|
|
DiveItem* diveItem = new DiveItem();
|
|
|
|
diveItem->dive = dive;
|
|
|
|
|
2013-05-28 19:56:58 +00:00
|
|
|
if (!trip || currentLayout == LIST) {
|
2013-05-02 02:51:34 +00:00
|
|
|
diveItem->parent = rootItem;
|
2013-05-14 07:28:30 +00:00
|
|
|
rootItem->children.push_back(diveItem);
|
2013-05-02 02:51:34 +00:00
|
|
|
continue;
|
|
|
|
}
|
2013-05-28 19:56:58 +00:00
|
|
|
if (currentLayout == LIST)
|
|
|
|
continue;
|
|
|
|
|
2013-05-02 02:51:34 +00:00
|
|
|
if (!trips.keys().contains(trip)) {
|
|
|
|
TripItem* tripItem = new TripItem();
|
|
|
|
tripItem->trip = trip;
|
|
|
|
tripItem->parent = rootItem;
|
2013-05-14 07:28:30 +00:00
|
|
|
tripItem->children.push_back(diveItem);
|
2013-05-02 02:51:34 +00:00
|
|
|
trips[trip] = tripItem;
|
2013-05-14 07:28:30 +00:00
|
|
|
rootItem->children.push_back(tripItem);
|
2013-05-02 02:51:34 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
TripItem* tripItem = trips[trip];
|
2013-05-14 07:28:30 +00:00
|
|
|
tripItem->children.push_back(diveItem);
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
2013-05-28 19:56:58 +00:00
|
|
|
|
|
|
|
if (rowCount()){
|
2013-05-29 20:43:14 +00:00
|
|
|
beginInsertRows(QModelIndex(), 0, rowCount() - 1);
|
2013-05-28 19:56:58 +00:00
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DiveTripModel::Layout DiveTripModel::layout() const
|
|
|
|
{
|
|
|
|
return currentLayout;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DiveTripModel::setLayout(DiveTripModel::Layout layout)
|
|
|
|
{
|
|
|
|
currentLayout = layout;
|
|
|
|
setupModelData();
|
2013-04-22 01:12:36 +00:00
|
|
|
}
|
2013-06-07 14:43:45 +00:00
|
|
|
|
2013-06-17 19:46:35 +00:00
|
|
|
/*####################################################################
|
2013-06-07 14:43:45 +00:00
|
|
|
*
|
2013-06-17 19:46:35 +00:00
|
|
|
* Dive Computer Model
|
2013-06-07 14:43:45 +00:00
|
|
|
*
|
|
|
|
*####################################################################
|
|
|
|
*/
|
|
|
|
|
2013-06-17 22:58:26 +00:00
|
|
|
DiveComputerModel::DiveComputerModel(QMultiMap<QString, DiveComputerNode> &dcMap, QObject* parent): QAbstractTableModel(parent)
|
2013-06-07 14:43:45 +00:00
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
dcWorkingMap = dcMap;
|
|
|
|
numRows = 0;
|
2013-06-07 14:43:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int DiveComputerModel::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return COLUMNS;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DiveComputerModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant ret;
|
|
|
|
if (role != Qt::DisplayRole || orientation != Qt::Horizontal){
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
switch(section){
|
2013-06-17 19:46:35 +00:00
|
|
|
case ID: ret = tr("Device ID"); break;
|
|
|
|
case MODEL: ret = tr("Model"); break;
|
|
|
|
case NICKNAME: ret = tr("Nickname"); break;
|
2013-06-07 14:43:45 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DiveComputerModel::data(const QModelIndex& index, int role) const
|
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
QList<DiveComputerNode> values = dcWorkingMap.values();
|
|
|
|
DiveComputerNode node = values.at(index.row());
|
2013-06-07 14:43:45 +00:00
|
|
|
|
|
|
|
QVariant ret;
|
|
|
|
if (role == Qt::DisplayRole || role == Qt::EditRole){
|
|
|
|
switch(index.column()){
|
2013-06-17 22:58:26 +00:00
|
|
|
case ID: ret = QString("0x").append(QString::number(node.deviceId, 16)); break;
|
|
|
|
case MODEL: ret = node.model; break;
|
|
|
|
case NICKNAME: ret = node.nickName; break;
|
2013-06-07 14:43:45 +00:00
|
|
|
}
|
|
|
|
}
|
2013-06-17 19:46:35 +00:00
|
|
|
|
2013-06-07 14:43:45 +00:00
|
|
|
if (role == Qt::DecorationRole && index.column() == REMOVE){
|
|
|
|
ret = QIcon(":trash");
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DiveComputerModel::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
return numRows;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DiveComputerModel::update()
|
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
QList<DiveComputerNode> values = dcWorkingMap.values();
|
|
|
|
int count = values.count();
|
2013-06-07 15:57:35 +00:00
|
|
|
|
2013-06-07 14:43:45 +00:00
|
|
|
if(numRows){
|
|
|
|
beginRemoveRows(QModelIndex(), 0, numRows-1);
|
|
|
|
numRows = 0;
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
2013-06-07 15:57:35 +00:00
|
|
|
|
2013-06-07 14:43:45 +00:00
|
|
|
if (count){
|
|
|
|
beginInsertRows(QModelIndex(), 0, count-1);
|
|
|
|
numRows = count;
|
|
|
|
endInsertRows();
|
|
|
|
}
|
2013-06-07 15:57:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Qt::ItemFlags DiveComputerModel::flags(const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
Qt::ItemFlags flags = QAbstractItemModel::flags(index);
|
|
|
|
if (index.column() == NICKNAME)
|
|
|
|
flags |= Qt::ItemIsEditable;
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DiveComputerModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
QList<DiveComputerNode> values = dcWorkingMap.values();
|
|
|
|
DiveComputerNode node = values.at(index.row());
|
|
|
|
dcWorkingMap.remove(node.model, node);
|
|
|
|
node.nickName = value.toString();
|
|
|
|
dcWorkingMap.insert(node.model, node);
|
2013-07-16 22:13:58 +00:00
|
|
|
emit dataChanged(index, index);
|
2013-06-07 18:25:29 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-17 22:58:26 +00:00
|
|
|
void DiveComputerModel::remove(const QModelIndex& index)
|
2013-06-07 18:25:29 +00:00
|
|
|
{
|
2013-06-17 22:58:26 +00:00
|
|
|
QList<DiveComputerNode> values = dcWorkingMap.values();
|
|
|
|
DiveComputerNode node = values.at(index.row());
|
|
|
|
dcWorkingMap.remove(node.model, node);
|
2013-06-07 18:25:29 +00:00
|
|
|
update();
|
2013-06-07 14:43:45 +00:00
|
|
|
}
|
2013-06-17 22:41:05 +00:00
|
|
|
|
2013-06-18 16:25:24 +00:00
|
|
|
void DiveComputerModel::dropWorkingList()
|
|
|
|
{
|
|
|
|
// how do I prevent the memory leak ?
|
|
|
|
}
|
|
|
|
|
|
|
|
void DiveComputerModel::keepWorkingList()
|
|
|
|
{
|
|
|
|
if (dcList.dcMap != dcWorkingMap)
|
|
|
|
mark_divelist_changed(TRUE);
|
|
|
|
dcList.dcMap = dcWorkingMap;
|
|
|
|
}
|
|
|
|
|
2013-06-17 22:41:05 +00:00
|
|
|
/*#################################################################
|
|
|
|
* #
|
|
|
|
* # Yearly Statistics Model
|
|
|
|
* #
|
|
|
|
* ################################################################
|
|
|
|
*/
|
|
|
|
|
2013-06-18 00:05:17 +00:00
|
|
|
class YearStatisticsItem : public TreeItem{
|
|
|
|
public:
|
2013-06-18 16:26:23 +00:00
|
|
|
enum {YEAR, DIVES, TOTAL_TIME, AVERAGE_TIME, SHORTEST_TIME, LONGEST_TIME, AVG_DEPTH, MIN_DEPTH,
|
|
|
|
MAX_DEPTH, AVG_SAC, MIN_SAC, MAX_SAC, AVG_TEMP, MIN_TEMP, MAX_TEMP, COLUMNS};
|
2013-06-18 00:05:17 +00:00
|
|
|
|
|
|
|
QVariant data(int column, int role) const;
|
|
|
|
YearStatisticsItem(stats_t interval);
|
|
|
|
private:
|
|
|
|
stats_t stats_interval;
|
|
|
|
};
|
|
|
|
|
2013-06-18 16:26:23 +00:00
|
|
|
YearStatisticsItem::YearStatisticsItem(stats_t interval) : stats_interval(interval)
|
2013-06-18 00:05:17 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant YearStatisticsItem::data(int column, int role) const
|
|
|
|
{
|
2013-06-18 16:26:23 +00:00
|
|
|
double value;
|
2013-06-18 00:05:17 +00:00
|
|
|
QVariant ret;
|
|
|
|
|
2013-06-19 17:30:36 +00:00
|
|
|
if (role == Qt::FontRole) {
|
|
|
|
QFont font = defaultModelFont();
|
|
|
|
font.setBold(stats_interval.is_year);
|
|
|
|
return font;
|
|
|
|
} else if (role != Qt::DisplayRole) {
|
|
|
|
return ret;
|
|
|
|
}
|
2013-06-18 16:26:23 +00:00
|
|
|
switch(column) {
|
|
|
|
case YEAR: ret = stats_interval.period; break;
|
|
|
|
case DIVES: ret = stats_interval.selection_size; break;
|
|
|
|
case TOTAL_TIME: ret = get_time_string(stats_interval.total_time.seconds, 0); break;
|
|
|
|
case AVERAGE_TIME: ret = get_minutes(stats_interval.total_time.seconds / stats_interval.selection_size); break;
|
|
|
|
case SHORTEST_TIME: ret = get_minutes(stats_interval.shortest_time.seconds); break;
|
|
|
|
case LONGEST_TIME: ret = get_minutes(stats_interval.longest_time.seconds); break;
|
2013-06-18 17:48:46 +00:00
|
|
|
case AVG_DEPTH: ret = get_depth_string(stats_interval.avg_depth); break;
|
|
|
|
case MIN_DEPTH: ret = get_depth_string(stats_interval.min_depth); break;
|
|
|
|
case MAX_DEPTH: ret = get_depth_string(stats_interval.max_depth); break;
|
|
|
|
case AVG_SAC: ret = get_volume_string(stats_interval.avg_sac); break;
|
|
|
|
case MIN_SAC: ret = get_volume_string(stats_interval.min_sac); break;
|
|
|
|
case MAX_SAC: ret = get_volume_string(stats_interval.max_sac); break;
|
2013-06-18 16:26:23 +00:00
|
|
|
case AVG_TEMP:
|
|
|
|
if (stats_interval.combined_temp && stats_interval.combined_count) {
|
2013-06-18 17:48:46 +00:00
|
|
|
ret = QString::number(stats_interval.combined_temp / stats_interval.combined_count, 'f', 1);
|
2013-06-18 16:26:23 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MIN_TEMP:
|
2013-06-18 17:48:46 +00:00
|
|
|
value = get_temp_units(stats_interval.min_temp, NULL);
|
|
|
|
if (value > -100.0)
|
|
|
|
ret = QString::number(value, 'f', 1);
|
2013-06-18 16:26:23 +00:00
|
|
|
break;
|
|
|
|
case MAX_TEMP:
|
2013-06-18 17:48:46 +00:00
|
|
|
value = get_temp_units(stats_interval.max_temp, NULL);
|
|
|
|
if (value > -100.0)
|
|
|
|
ret = QString::number(value, 'f', 1);
|
2013-06-18 16:26:23 +00:00
|
|
|
break;
|
2013-06-18 00:05:17 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-06-17 22:41:05 +00:00
|
|
|
YearlyStatisticsModel::YearlyStatisticsModel(QObject* parent)
|
|
|
|
{
|
2013-06-17 23:02:30 +00:00
|
|
|
columns = COLUMNS;
|
2013-06-18 00:05:17 +00:00
|
|
|
update_yearly_stats();
|
2013-06-17 22:41:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant YearlyStatisticsModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
QVariant val;
|
2013-06-18 16:26:23 +00:00
|
|
|
if (role == Qt::FontRole)
|
2013-06-17 23:02:30 +00:00
|
|
|
val = defaultModelFont();
|
2013-06-18 00:05:17 +00:00
|
|
|
|
2013-06-18 16:26:23 +00:00
|
|
|
if (role == Qt::DisplayRole && orientation == Qt::Horizontal) {
|
|
|
|
switch(section) {
|
|
|
|
case YEAR: val = tr("Year \n > Month"); break;
|
|
|
|
case DIVES: val = tr("#"); break;
|
|
|
|
case TOTAL_TIME: val = tr("Duration \n Total"); break;
|
2013-06-18 17:48:46 +00:00
|
|
|
case AVERAGE_TIME: val = tr("\nAverage"); break;
|
|
|
|
case SHORTEST_TIME: val = tr("\nShortest"); break;
|
|
|
|
case LONGEST_TIME: val = tr("\nLongest"); break;
|
|
|
|
case AVG_DEPTH: val = QString(tr("Depth (%1)\n Average")).arg(get_depth_unit()); break;
|
|
|
|
case MIN_DEPTH: val = tr("\nMinimum"); break;
|
|
|
|
case MAX_DEPTH: val = tr("\nMaximum"); break;
|
|
|
|
case AVG_SAC: val = QString(tr("SAC (%1)\n Average")).arg(get_volume_unit()); break;
|
|
|
|
case MIN_SAC: val = tr("\nMinimum"); break;
|
|
|
|
case MAX_SAC: val = tr("\nMaximum"); break;
|
|
|
|
case AVG_TEMP: val = QString(tr("Temp. (%1)\n Average").arg(get_temp_unit())); break;
|
|
|
|
case MIN_TEMP: val = tr("\nMinimum"); break;
|
|
|
|
case MAX_TEMP: val = tr("\nMaximum"); break;
|
2013-06-17 23:02:30 +00:00
|
|
|
}
|
2013-06-17 22:41:05 +00:00
|
|
|
}
|
2013-06-17 23:02:30 +00:00
|
|
|
return val;
|
2013-06-17 22:41:05 +00:00
|
|
|
}
|
2013-06-18 00:05:17 +00:00
|
|
|
|
|
|
|
void YearlyStatisticsModel::update_yearly_stats()
|
|
|
|
{
|
|
|
|
int i, month = 0;
|
|
|
|
unsigned int j, combined_months;
|
|
|
|
|
2013-06-18 16:26:23 +00:00
|
|
|
for (i = 0; stats_yearly != NULL && stats_yearly[i].period; ++i) {
|
2013-06-18 00:05:17 +00:00
|
|
|
YearStatisticsItem *item = new YearStatisticsItem(stats_yearly[i]);
|
|
|
|
combined_months = 0;
|
|
|
|
for (j = 0; combined_months < stats_yearly[i].selection_size; ++j) {
|
|
|
|
combined_months += stats_monthly[month].selection_size;
|
|
|
|
YearStatisticsItem *iChild = new YearStatisticsItem(stats_monthly[month]);
|
|
|
|
item->children.append(iChild);
|
2013-06-18 20:29:37 +00:00
|
|
|
iChild->parent = item;
|
2013-06-18 00:05:17 +00:00
|
|
|
month++;
|
|
|
|
}
|
|
|
|
rootItem->children.append(item);
|
2013-06-18 20:29:37 +00:00
|
|
|
item->parent = rootItem;
|
2013-06-18 00:05:17 +00:00
|
|
|
}
|
|
|
|
}
|