/* * models.cpp * * classes for the equipment models of Subsurface * */ #include "models.h" #include "../helpers.h" #include "../dive.h" #include "../device.h" #include "../statistics.h" #include "../qthelper.h" #include #include #include #include #include #include QFont defaultModelFont() { QFont font; font.setPointSizeF( font.pointSizeF() * 0.8); return font; } CylindersModel::CylindersModel(QObject* parent): QAbstractTableModel(parent), current(0), rows(0) { } QVariant CylindersModel::headerData(int section, Qt::Orientation orientation, int role) const { QVariant ret; if (orientation == Qt::Vertical) return ret; switch (role) { case Qt::FontRole: ret = defaultModelFont(); break; case Qt::DisplayRole: switch(section) { 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; } } return ret; } int CylindersModel::columnCount(const QModelIndex& parent) const { return COLUMNS; } QVariant CylindersModel::data(const QModelIndex& index, int role) const { QVariant ret; if (!index.isValid() || index.row() >= MAX_CYLINDERS) return ret; cylinder_t *cyl = ¤t->cylinder[index.row()]; switch (role) { case Qt::FontRole: ret = defaultModelFont(); break; case Qt::TextAlignmentRole: ret = Qt::AlignHCenter; break; case Qt::DisplayRole: case Qt::EditRole: switch(index.column()) { case TYPE: ret = QString(cyl->type.description); break; case SIZE: // we can't use get_volume_string because the idiotic imperial tank // sizes take working pressure into account... if (cyl->type.size.mliter) { if (prefs.units.volume == prefs.units.CUFT) { int cuft = 0.5 + ml_to_cuft(gas_volume(cyl, cyl->type.workingpressure)); ret = QString("%1cuft").arg(cuft); } else { ret = QString("%1l").arg(cyl->type.size.mliter / 1000.0, 0, 'f', 1); } } break; case WORKINGPRESS: if (cyl->type.workingpressure.mbar) ret = get_pressure_string(cyl->type.workingpressure, TRUE); break; case START: if (cyl->start.mbar) ret = get_pressure_string(cyl->start, TRUE); break; case END: if (cyl->end.mbar) ret = get_pressure_string(cyl->end, TRUE ); break; case O2: ret = QString("%1%").arg((cyl->gasmix.o2.permille + 5) / 10); break; case HE: ret = QString("%1%").arg((cyl->gasmix.he.permille + 5) / 10); break; } break; case Qt::DecorationRole: if (index.column() == REMOVE) ret = QIcon(":trash"); break; } return ret; } // 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); } break; case WORKINGPRESS: if (cyl->type.workingpressure.mbar != value.toInt()) { cyl->type.workingpressure.mbar = value.toInt(); mark_divelist_changed(TRUE); } break; } } #define CHANGED(_t,_u1,_u2) value._t() != data(index, role).toString().replace(_u1,"").replace(_u2,"")._t() bool CylindersModel::setData(const QModelIndex& index, const QVariant& value, int role) { cylinder_t *cyl = ¤t->cylinder[index.row()]; switch(index.column()) { case TYPE: if (!value.isNull()) { QByteArray ba = value.toByteArray(); const char *text = ba.constData(); if (!cyl->type.description || strcmp(cyl->type.description, text)) { cyl->type.description = strdup(text); mark_divelist_changed(TRUE); } } 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) { TankInfoModel *tanks = TankInfoModel::instance(); QModelIndexList matches = tanks->match(tanks->index(0,0), Qt::DisplayRole, cyl->type.description); if (prefs.units.volume == prefs.units.CUFT) { if (cyl->type.workingpressure.mbar == 0) { // this is a hack as we can't store a wet size // without working pressure in cuft mode // so we assume it's an aluminum tank at 3000psi cyl->type.workingpressure.mbar = psi_to_mbar(3000); if (!matches.isEmpty()) tanks->setData(tanks->index(matches.first().row(), TankInfoModel::BAR), cyl->type.workingpressure.mbar / 1000.0); } if (cyl->type.size.mliter != wet_volume(value.toDouble(), cyl->type.workingpressure)) { mark_divelist_changed(TRUE); cyl->type.size.mliter = wet_volume(value.toDouble(), cyl->type.workingpressure); if (!matches.isEmpty()) tanks->setData(tanks->index(matches.first().row(), TankInfoModel::ML), cyl->type.size.mliter); } } else { if (cyl->type.size.mliter != value.toDouble() * 1000.0) { mark_divelist_changed(TRUE); cyl->type.size.mliter = value.toDouble() * 1000.0; if (!matches.isEmpty()) tanks->setData(tanks->index(matches.first().row(), TankInfoModel::ML), cyl->type.size.mliter); } } } } break; case WORKINGPRESS: if (CHANGED(toDouble, "psi", "bar")) { if (value.toDouble() != 0.0) { TankInfoModel *tanks = TankInfoModel::instance(); QModelIndexList matches = tanks->match(tanks->index(0,0), Qt::DisplayRole, cyl->type.description); if (prefs.units.pressure == prefs.units.PSI) cyl->type.workingpressure.mbar = psi_to_mbar(value.toDouble()); else cyl->type.workingpressure.mbar = value.toDouble() * 1000; if (!matches.isEmpty()) tanks->setData(tanks->index(matches.first().row(), TankInfoModel::BAR), cyl->type.workingpressure.mbar / 1000.0); 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: if (CHANGED(toInt, "%", "%")) { cyl->gasmix.o2.permille = value.toInt() * 10 - 5; mark_divelist_changed(TRUE); } break; case HE: if (CHANGED(toInt, "%", "%")) { cyl->gasmix.he.permille = value.toInt() * 10 - 5; mark_divelist_changed(TRUE); } break; } return QAbstractItemModel::setData(index, value, role); } int CylindersModel::rowCount(const QModelIndex& parent) const { return rows; } void CylindersModel::add() { if (rows >= MAX_CYLINDERS) { return; } int row = rows; beginInsertRows(QModelIndex(), row, row); rows++; endInsertRows(); } void CylindersModel::update() { setDive(current); } void CylindersModel::clear() { if (rows > 0) { beginRemoveRows(QModelIndex(), 0, rows-1); endRemoveRows(); } } void CylindersModel::setDive(dive* d) { if (current) clear(); int amount = MAX_CYLINDERS; for(int i = 0; i < MAX_CYLINDERS; i++) { cylinder_t *cylinder = &d->cylinder[i]; if (cylinder_none(cylinder)) { amount = i; break; } } beginInsertRows(QModelIndex(), 0, amount-1); rows = amount; current = d; endInsertRows(); } Qt::ItemFlags CylindersModel::flags(const QModelIndex& index) const { if (index.column() == REMOVE) return Qt::ItemIsEnabled; return QAbstractItemModel::flags(index) | Qt::ItemIsEditable; } void CylindersModel::remove(const QModelIndex& index) { if (index.column() != REMOVE) { return; } beginRemoveRows(QModelIndex(), index.row(), index.row()); // yah, know, ugly. rows--; remove_cylinder(current, index.row()); mark_divelist_changed(TRUE); endRemoveRows(); } WeightModel::WeightModel(QObject* parent): QAbstractTableModel(parent), current(0), rows(0) { } void WeightModel::remove(const QModelIndex& index) { if (index.column() != REMOVE) { return; } beginRemoveRows(QModelIndex(), index.row(), index.row()); // yah, know, ugly. rows--; remove_weightsystem(current, index.row()); mark_divelist_changed(TRUE); endRemoveRows(); } void WeightModel::clear() { if (rows > 0) { beginRemoveRows(QModelIndex(), 0, rows-1); endRemoveRows(); } } int WeightModel::columnCount(const QModelIndex& parent) const { return COLUMNS; } QVariant WeightModel::data(const QModelIndex& index, int role) const { QVariant ret; if (!index.isValid() || index.row() >= MAX_WEIGHTSYSTEMS) return ret; weightsystem_t *ws = ¤t->weightsystem[index.row()]; switch (role) { case Qt::FontRole: ret = defaultModelFont(); break; case Qt::TextAlignmentRole: ret = Qt::AlignRight; break; case Qt::DisplayRole: case Qt::EditRole: switch(index.column()) { case TYPE: ret = QString(ws->description); break; case WEIGHT: ret = get_weight_string(ws->weight, TRUE); break; } break; case Qt::DecorationRole: if (index.column() == REMOVE) ret = QIcon(":trash"); break; } return ret; } // 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); } } } bool WeightModel::setData(const QModelIndex& index, const QVariant& value, int role) { weightsystem_t *ws = ¤t->weightsystem[index.row()]; switch(index.column()) { case TYPE: if (!value.isNull()) { QByteArray ba = value.toString().toUtf8(); const char *text = ba.constData(); if (!ws->description || strcmp(ws->description, text)) { ws->description = strdup(text); mark_divelist_changed(TRUE); } } break; case WEIGHT: if (CHANGED(toDouble, "kg", "lbs")) { if (prefs.units.weight == prefs.units.LBS) ws->weight.grams = lbs_to_grams(value.toDouble()); else ws->weight.grams = value.toDouble() * 1000.0; // 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); } break; } return QAbstractItemModel::setData(index, value, role); } Qt::ItemFlags WeightModel::flags(const QModelIndex& index) const { if (index.column() == REMOVE) return Qt::ItemIsEnabled; return QAbstractItemModel::flags(index) | Qt::ItemIsEditable; } int WeightModel::rowCount(const QModelIndex& parent) const { return rows; } QVariant WeightModel::headerData(int section, Qt::Orientation orientation, int role) const { QVariant ret; if (orientation == Qt::Vertical) return ret; switch (role) { case Qt::FontRole: ret = defaultModelFont(); break; case Qt::DisplayRole: switch(section) { case TYPE: ret = tr("Type"); break; case WEIGHT: ret = tr("Weight"); break; } break; } return ret; } void WeightModel::add() { if (rows >= MAX_WEIGHTSYSTEMS) return; int row = rows; beginInsertRows(QModelIndex(), row, row); rows++; endInsertRows(); } void WeightModel::update() { setDive(current); } void WeightModel::setDive(dive* d) { if (current) clear(); int amount = MAX_WEIGHTSYSTEMS; for(int i = 0; i < MAX_WEIGHTSYSTEMS; i++) { weightsystem_t *weightsystem = &d->weightsystem[i]; if (weightsystem_none(weightsystem)) { amount = i; break; } } beginInsertRows(QModelIndex(), 0, amount-1); rows = amount; current = d; endInsertRows(); } 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()]; switch(index.column()) { case DESCRIPTION: info->name = strdup(value.toByteArray().data()); break; case GR: info->grams = value.toInt(); break; } 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; 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; } return ret; } QVariant WSInfoModel::headerData(int section, Qt::Orientation orientation, int role) const { QVariant ret; if (orientation != Qt::Horizontal) return ret; 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; } return ret; } int WSInfoModel::rowCount(const QModelIndex& parent) const { return rows+1; } const QString& WSInfoModel::biggerString() const { return biggerEntry; } WSInfoModel::WSInfoModel() : QAbstractTableModel(), rows(-1) { struct ws_info *info = ws_info; for (info = ws_info; info->name; info++, rows++){ QString wsInfoName(info->name); if( wsInfoName.count() > biggerEntry.count()){ biggerEntry = wsInfoName; } } if (rows > -1) { beginInsertRows(QModelIndex(), 0, rows); endInsertRows(); } } void WSInfoModel::update() { if (rows > -1) { beginRemoveRows(QModelIndex(), 0, rows); endRemoveRows(); rows = -1; } struct ws_info *info = ws_info; for (info = ws_info; info->name; info++, rows++); if (rows > -1) { beginInsertRows(QModelIndex(), 0, rows); endInsertRows(); } } TankInfoModel* TankInfoModel::instance() { static TankInfoModel *self = new TankInfoModel(); return self; } const QString& TankInfoModel::biggerString() const { return biggerEntry; } 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()]; 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; } return TRUE; } 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; } if (role == Qt::FontRole){ return defaultModelFont(); } struct tank_info *info = &tank_info[index.row()]; int ml = info->ml; int bar = ((info->psi) ? psi_to_bar(info->psi) : info->bar) * 1000 + 0.5; if (info->cuft && info->psi) { pressure_t p; p.mbar = psi_to_mbar(info->psi); ml = wet_volume(info->cuft, p); } if (role == Qt::DisplayRole || role == Qt::EditRole) { switch(index.column()) { case BAR: ret = bar; break; case ML: ret = ml; 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; switch(role){ case Qt::FontRole: ret = defaultModelFont(); break; case Qt::DisplayRole: switch(section) { case BAR: ret = tr("Bar"); break; case ML: ret = tr("Ml"); break; case DESCRIPTION: ret = tr("Description"); break; } break; } return ret; } int TankInfoModel::rowCount(const QModelIndex& parent) const { return rows+1; } TankInfoModel::TankInfoModel() : QAbstractTableModel(), rows(-1) { struct tank_info *info = tank_info; for (info = tank_info; info->name; info++, rows++){ QString infoName(info->name); if (infoName.count() > biggerEntry.count()){ biggerEntry = infoName; } } if (rows > -1) { beginInsertRows(QModelIndex(), 0, rows); endInsertRows(); } } void TankInfoModel::update() { if (rows > -1) { beginRemoveRows(QModelIndex(), 0, rows); endRemoveRows(); rows = -1; } struct tank_info *info = tank_info; for (info = tank_info; info->name; info++, rows++); if (rows > -1) { beginInsertRows(QModelIndex(), 0, rows); endInsertRows(); } } //################################################################################################# //# //# Tree Model - a Basic Tree Model so I don't need to kill myself repeating this for every model. //# //################################################################################################# /*! 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. * */ TreeItem::~TreeItem() { qDeleteAll(children); } int TreeItem::row() const { if (parent) return parent->children.indexOf(const_cast(this)); return 0; } QVariant TreeItem::data(int column, int role) const { 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(); if (role == Qt::FontRole) { return defaultModelFont(); } TreeItem* item = static_cast(index.internalPointer()); return item->data(index.column(), role); } 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(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(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(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 { virtual QVariant data(int column, int role) const; dive_trip_t* trip; }; QVariant TripItem::data(int column, int role) const { QVariant ret; if (role == DiveTripModel::SORT_ROLE) return (qulonglong)trip->when; if (role == Qt::DisplayRole) { switch (column) { case DiveTripModel::NR: ret = QString(trip->location) + ", " + QString(get_trip_date_string(trip->when, trip->nrdives)); break; } } return ret; } struct DiveItem : public TreeItem { enum Column {NR, DATE, RATING, DEPTH, DURATION, TEMPERATURE, TOTALWEIGHT, SUIT, CYLINDER, NITROX, SAC, OTU, MAXCNS, LOCATION, COLUMNS }; virtual QVariant data(int column, int role) const; struct dive* dive; QString displayDuration() const; QString displayDepth() const; QString displayTemperature() const; QString displayWeight() const; QString displaySac() const; int weight() const; }; static int nitrox_sort_value(struct dive *dive) { int o2, he, o2low; get_dive_gas(dive, &o2, &he, &o2low); return he*1000 + o2; } QVariant DiveItem::data(int column, int role) const { QVariant retVal; switch (role) { case Qt::TextAlignmentRole: switch (column) { case DATE: /* fall through */ case SUIT: /* fall through */ case LOCATION: retVal = Qt::AlignLeft; break; default: retVal = Qt::AlignRight; break; } break; case DiveTripModel::SORT_ROLE: switch (column) { case NR: retVal = (qulonglong) dive->when; break; case DATE: retVal = (qulonglong) dive->when; break; case RATING: retVal = dive->rating; break; 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; } break; case Qt::DisplayRole: switch (column) { case NR: retVal = dive->number; break; case DATE: retVal = QString(get_dive_date_string(dive->when)); break; 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; } break; } if (role == DiveTripModel::STAR_ROLE) retVal = dive->rating; if (role == DiveTripModel::DIVE_ROLE) retVal = QVariant::fromValue(dive); return retVal; } QString DiveItem::displayDepth() const { const int scale = 1000; QString fract, str; if (get_units()->length == units::METERS) { fract = QString::number((unsigned)(dive->maxdepth.mm % scale) / 10); str = QString("%1.%2").arg((unsigned)(dive->maxdepth.mm / scale)).arg(fract, 2, QChar('0')); } if (get_units()->length == units::FEET) { str = QString::number(mm_to_feet(dive->maxdepth.mm),'f',0); } return str; } QString DiveItem::displayDuration() const { int hrs, mins, secs; secs = dive->duration.seconds % 60; mins = dive->duration.seconds / 60; hrs = mins / 60; mins -= hrs * 60; QString displayTime; if (hrs) displayTime = QString("%1:%2:").arg(hrs).arg(mins, 2, 10, QChar('0')); else displayTime = QString("%1:").arg(mins); displayTime += QString("%1").arg(secs, 2, 10, QChar('0')); return displayTime; } QString DiveItem::displayTemperature() const { QString str; if (!dive->watertemp.mkelvin) return str; if (get_units()->temperature == units::CELSIUS) str = QString::number(mkelvin_to_C(dive->watertemp.mkelvin), 'f', 1); else str = QString::number(mkelvin_to_F(dive->watertemp.mkelvin), 'f', 1); return str; } QString DiveItem::displaySac() const { QString str; if (get_units()->volume == units::LITER) str = QString::number(dive->sac / 1000.0, 'f', 1); else str = QString::number(ml_to_cuft(dive->sac), 'f', 2); return str; } QString DiveItem::displayWeight() const { QString str; if (get_units()->weight == units::KG) { int gr = weight() % 1000; int kg = weight() / 1000; str = QString("%1.%2").arg(kg).arg((unsigned)(gr) / 100); } else { str = QString("%1").arg((unsigned)(grams_to_lbs(weight()))); } return str; } int DiveItem::weight() const { weight_t tw = { total_weight(dive) }; return tw.grams; } DiveTripModel::DiveTripModel(QObject* parent): TreeModel(parent) { columns = COLUMNS; } Qt::ItemFlags DiveTripModel::flags(const QModelIndex& index) const { if (!index.isValid()) return 0; return Qt::ItemIsEnabled | Qt::ItemIsSelectable; } QVariant DiveTripModel::headerData(int section, Qt::Orientation orientation, int role) const { QVariant ret; if (orientation == Qt::Vertical) return ret; switch(role){ case Qt::FontRole : 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; } return ret; } void DiveTripModel::setupModelData() { int i = dive_table.nr; if (rowCount()){ beginRemoveRows(QModelIndex(), 0, rowCount()-1); endRemoveRows(); } if (autogroup) autogroup_dives(); dive_table.preexisting = dive_table.nr; while (--i >= 0) { struct dive* dive = get_dive(i); update_cylinder_related_info(dive); dive_trip_t* trip = dive->divetrip; DiveItem* diveItem = new DiveItem(); diveItem->dive = dive; if (!trip || currentLayout == LIST) { diveItem->parent = rootItem; rootItem->children.push_back(diveItem); continue; } if (currentLayout == LIST) continue; if (!trips.keys().contains(trip)) { TripItem* tripItem = new TripItem(); tripItem->trip = trip; tripItem->parent = rootItem; tripItem->children.push_back(diveItem); trips[trip] = tripItem; rootItem->children.push_back(tripItem); continue; } TripItem* tripItem = trips[trip]; tripItem->children.push_back(diveItem); } if (rowCount()){ beginInsertRows(QModelIndex(), 0, rowCount() - 1); endInsertRows(); } } DiveTripModel::Layout DiveTripModel::layout() const { return currentLayout; } void DiveTripModel::setLayout(DiveTripModel::Layout layout) { currentLayout = layout; setupModelData(); } /*#################################################################### * * Dive Computer Model * *#################################################################### */ DiveComputerModel::DiveComputerModel(QMultiMap &dcMap, QObject* parent): QAbstractTableModel(parent) { dcWorkingMap = dcMap; numRows = 0; } 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){ case ID: ret = tr("Device ID"); break; case MODEL: ret = tr("Model"); break; case NICKNAME: ret = tr("Nickname"); break; } return ret; } QVariant DiveComputerModel::data(const QModelIndex& index, int role) const { QList values = dcWorkingMap.values(); DiveComputerNode node = values.at(index.row()); QVariant ret; if (role == Qt::DisplayRole || role == Qt::EditRole){ switch(index.column()){ case ID: ret = QString("0x").append(QString::number(node.deviceId, 16)); break; case MODEL: ret = node.model; break; case NICKNAME: ret = node.nickName; break; } } if (role == Qt::DecorationRole && index.column() == REMOVE){ ret = QIcon(":trash"); } return ret; } int DiveComputerModel::rowCount(const QModelIndex& parent) const { return numRows; } void DiveComputerModel::update() { QList values = dcWorkingMap.values(); int count = values.count(); if(numRows){ beginRemoveRows(QModelIndex(), 0, numRows-1); numRows = 0; endRemoveRows(); } if (count){ beginInsertRows(QModelIndex(), 0, count-1); numRows = count; endInsertRows(); } } 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) { QList values = dcWorkingMap.values(); DiveComputerNode node = values.at(index.row()); dcWorkingMap.remove(node.model, node); node.nickName = value.toString(); dcWorkingMap.insert(node.model, node); return true; } void DiveComputerModel::remove(const QModelIndex& index) { QList values = dcWorkingMap.values(); DiveComputerNode node = values.at(index.row()); dcWorkingMap.remove(node.model, node); update(); } /*################################################################# * # * # Yearly Statistics Model * # * ################################################################ */ class YearStatisticsItem : public TreeItem{ public: 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}; QVariant data(int column, int role) const; YearStatisticsItem(stats_t interval); private: stats_t stats_interval; }; YearStatisticsItem::YearStatisticsItem(stats_t interval) : stats_interval(interval) { } QVariant YearStatisticsItem::data(int column, int role) const { const char *unit; QVariant ret; if (role != Qt::DisplayRole){ return ret; } 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; case AVG_DEPTH: ret = stats_interval.avg_depth.mm;break; case MIN_DEPTH: ret = stats_interval.min_depth.mm;break; case MAX_DEPTH: ret = stats_interval.max_depth.mm;break; case AVG_SAC: ret = stats_interval.avg_sac.mliter;break; case MIN_SAC: ret = stats_interval.min_sac.mliter;break; case MAX_SAC: ret = stats_interval.max_sac.mliter;break; case AVG_TEMP:{ get_temp_units(stats_interval.min_temp, &unit); if (stats_interval.combined_temp && stats_interval.combined_count) { ret = QString("%1 %2").arg(stats_interval.combined_temp / stats_interval.combined_count).arg(unit); } }break; case MIN_TEMP:{ double value = get_temp_units(stats_interval.min_temp, &unit); if (value > -100.0) { ret = QString("%1 %2").arg(value).arg(unit); } }break; case MAX_TEMP:{ double value = get_temp_units(stats_interval.max_temp, &unit); if (value > -100.0) { ret = QString("%1 %2").arg(value).arg(unit); } }break; } return ret; } YearlyStatisticsModel::YearlyStatisticsModel(QObject* parent) { columns = COLUMNS; update_yearly_stats(); } QVariant YearlyStatisticsModel::headerData(int section, Qt::Orientation orientation, int role) const { QVariant val; if (role == Qt::FontRole){ val = defaultModelFont(); } 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; case AVERAGE_TIME: val = tr("Average"); break; case SHORTEST_TIME: val = tr("Shortest"); break; case LONGEST_TIME: val = tr("Longest"); break; case AVG_DEPTH: val = tr("Depth \n Average"); break; case MIN_DEPTH: val = tr("Minimum"); break; case MAX_DEPTH: val = tr("Maximum"); break; case AVG_SAC: val = tr("SAC \n Average"); break; case MIN_SAC: val = tr("Minimum"); break; case MAX_SAC: val = tr("Maximum"); break; case AVG_TEMP: val = tr("Temperature \n Average"); break; case MIN_TEMP: val = tr("Minimum"); break; case MAX_TEMP: val = tr("Maximum"); break; } } return val; } void YearlyStatisticsModel::update_yearly_stats() { int i, month = 0; unsigned int j, combined_months; for (i = 0; stats_yearly != NULL && stats_yearly[i].period; ++i){ 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); month++; } rootItem->children.append(item); } } void DiveComputerModel::dropWorkingList() { // how do I prevent the memory leak ? } void DiveComputerModel::keepWorkingList() { if (dcList.dcMap != dcWorkingMap) mark_divelist_changed(TRUE); dcList.dcMap = dcWorkingMap; }