/* * models.cpp * * classes for the equipment models of Subsurface * */ #include "models.h" #include "diveplanner.h" #include "mainwindow.h" #include "helpers.h" #include "dive.h" #include "device.h" #include "statistics.h" #include "qthelper.h" #include "gettextfromc.h" #include "display.h" #include "color.h" #include "divetripmodel.h" #include "cleanertablemodel.h" #include "weigthsysteminfomodel.h" #include #include #include #include #include #include #include #include #include #include // initialize the trash icon if necessary const QPixmap &trashIcon() { static QPixmap trash = QPixmap(":trash").scaledToHeight(defaultIconMetrics().sz_small); return trash; } /*################################################################ * * Implementation of the Dive List. * * ############################################################### */ /*################################################################# * # * # Yearly Statistics Model * # * ################################################################ */ /*################################################################# * # * # Table Print Model * # * ################################################################ */ TablePrintModel::TablePrintModel() { columns = 7; rows = 0; } TablePrintModel::~TablePrintModel() { for (int i = 0; i < list.size(); i++) delete list.at(i); } void TablePrintModel::insertRow(int index) { struct TablePrintItem *item = new struct TablePrintItem(); item->colorBackground = 0xffffffff; if (index == -1) { beginInsertRows(QModelIndex(), rows, rows); list.append(item); } else { beginInsertRows(QModelIndex(), index, index); list.insert(index, item); } endInsertRows(); rows++; } void TablePrintModel::callReset() { beginResetModel(); endResetModel(); } QVariant TablePrintModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) return QVariant(); if (role == Qt::BackgroundRole) return QColor(list.at(index.row())->colorBackground); if (role == Qt::DisplayRole) switch (index.column()) { case 0: return list.at(index.row())->number; case 1: return list.at(index.row())->date; case 2: return list.at(index.row())->depth; case 3: return list.at(index.row())->duration; case 4: return list.at(index.row())->divemaster; case 5: return list.at(index.row())->buddy; case 6: return list.at(index.row())->location; } if (role == Qt::FontRole) { QFont font; font.setPointSizeF(7.5); if (index.row() == 0 && index.column() == 0) { font.setBold(true); } return QVariant::fromValue(font); } return QVariant(); } bool TablePrintModel::setData(const QModelIndex &index, const QVariant &value, int role) { if (index.isValid()) { if (role == Qt::DisplayRole) { switch (index.column()) { case 0: list.at(index.row())->number = value.toString(); case 1: list.at(index.row())->date = value.toString(); case 2: list.at(index.row())->depth = value.toString(); case 3: list.at(index.row())->duration = value.toString(); case 4: list.at(index.row())->divemaster = value.toString(); case 5: list.at(index.row())->buddy = value.toString(); case 6: { /* truncate if there are more than N lines of text, * we don't want a row to be larger that a single page! */ QString s = value.toString(); const int maxLines = 15; int count = 0; for (int i = 0; i < s.length(); i++) { if (s.at(i) != QChar('\n')) continue; count++; if (count > maxLines) { s = s.left(i - 1); break; } } list.at(index.row())->location = s; } } return true; } if (role == Qt::BackgroundRole) { list.at(index.row())->colorBackground = value.value(); return true; } } return false; } int TablePrintModel::rowCount(const QModelIndex &parent) const { Q_UNUSED(parent); return rows; } int TablePrintModel::columnCount(const QModelIndex &parent) const { Q_UNUSED(parent); return columns; } /*################################################################# * # * # Profile Print Model * # * ################################################################ */ ProfilePrintModel::ProfilePrintModel(QObject *parent) { } void ProfilePrintModel::setDive(struct dive *divePtr) { diveId = divePtr->id; // reset(); } void ProfilePrintModel::setFontsize(double size) { fontSize = size; } int ProfilePrintModel::rowCount(const QModelIndex &parent) const { return 12; } int ProfilePrintModel::columnCount(const QModelIndex &parent) const { return 5; } QVariant ProfilePrintModel::data(const QModelIndex &index, int role) const { const int row = index.row(); const int col = index.column(); switch (role) { case Qt::DisplayRole: { struct dive *dive = get_dive_by_uniq_id(diveId); struct DiveItem di; di.diveId = diveId; const QString unknown = tr("unknown"); // dive# + date, depth, location, duration if (row == 0) { if (col == 0) return tr("Dive #%1 - %2").arg(dive->number).arg(di.displayDate()); if (col == 3) { QString unit = (get_units()->length == units::METERS) ? "m" : "ft"; return tr("Max depth: %1 %2").arg(di.displayDepth()).arg(unit); } } if (row == 1) { if (col == 0) return QString(get_dive_location(dive)); if (col == 3) return QString(tr("Duration: %1 min")).arg(di.displayDuration()); } // headings if (row == 2) { if (col == 0) return tr("Gas used:"); if (col == 2) return tr("Tags:"); if (col == 3) return tr("SAC:"); if (col == 4) return tr("Weights:"); } // notes if (col == 0) { if (row == 6) return tr("Notes:"); if (row == 7) return QString(dive->notes); } // more headings if (row == 4) { if (col == 0) return tr("Divemaster:"); if (col == 1) return tr("Buddy:"); if (col == 2) return tr("Suit:"); if (col == 3) return tr("Viz:"); if (col == 4) return tr("Rating:"); } // values for gas, sac, etc... if (row == 3) { if (col == 0) { int added = 0; QString gas, gases; for (int i = 0; i < MAX_CYLINDERS; i++) { if (!is_cylinder_used(dive, i)) continue; gas = dive->cylinder[i].type.description; gas += QString(!gas.isEmpty() ? " " : "") + gasname(&dive->cylinder[i].gasmix); // if has a description and if such gas is not already present if (!gas.isEmpty() && gases.indexOf(gas) == -1) { if (added > 0) gases += QString(" / "); gases += gas; added++; } } return gases; } if (col == 2) { char buffer[256]; taglist_get_tagstring(dive->tag_list, buffer, 256); return QString(buffer); } if (col == 3) return di.displaySac(); if (col == 4) { weight_t tw = { total_weight(dive) }; return get_weight_string(tw, true); } } // values for DM, buddy, suit, etc... if (row == 5) { if (col == 0) return QString(dive->divemaster); if (col == 1) return QString(dive->buddy); if (col == 2) return QString(dive->suit); if (col == 3) return (dive->visibility) ? QString::number(dive->visibility).append(" / 5") : QString(); if (col == 4) return (dive->rating) ? QString::number(dive->rating).append(" / 5") : QString(); } return QString(); } case Qt::FontRole: { QFont font; font.setPointSizeF(fontSize); if (row == 0 && col == 0) { font.setBold(true); } return QVariant::fromValue(font); } case Qt::TextAlignmentRole: { // everything is aligned to the left unsigned int align = Qt::AlignLeft; // align depth and duration right if (row < 2 && col == 4) align = Qt::AlignRight | Qt::AlignVCenter; return QVariant::fromValue(align); } } // switch (role) return QVariant(); } Qt::ItemFlags GasSelectionModel::flags(const QModelIndex &index) const { return Qt::ItemIsEnabled | Qt::ItemIsSelectable; } GasSelectionModel *GasSelectionModel::instance() { static QScopedPointer self(new GasSelectionModel()); return self.data(); } //TODO: Remove this #include here when the issue below is fixed. #include "diveplannermodel.h" void GasSelectionModel::repopulate() { /* TODO: * getGasList shouldn't be a member of DivePlannerPointsModel, * it has nothing to do with the current plain being calculated: * it's internal to the current_dive. */ setStringList(DivePlannerPointsModel::instance()->getGasList()); } QVariant GasSelectionModel::data(const QModelIndex &index, int role) const { if (role == Qt::FontRole) { return defaultModelFont(); } return QStringListModel::data(index, role); } // Language Model, The Model to populate the list of possible Languages. LanguageModel *LanguageModel::instance() { static LanguageModel *self = new LanguageModel(); QLocale l; return self; } LanguageModel::LanguageModel(QObject *parent) : QAbstractListModel(parent) { QSettings s; QDir d(getSubsurfaceDataPath("translations")); Q_FOREACH (const QString &s, d.entryList()) { if (s.startsWith("subsurface_") && s.endsWith(".qm")) { languages.push_back((s == "subsurface_source.qm") ? "English" : s); } } } QVariant LanguageModel::data(const QModelIndex &index, int role) const { QLocale loc; QString currentString = languages.at(index.row()); if (!index.isValid()) return QVariant(); switch (role) { case Qt::DisplayRole: { QLocale l(currentString.remove("subsurface_")); return currentString == "English" ? currentString : QString("%1 (%2)").arg(l.languageToString(l.language())).arg(l.countryToString(l.country())); } case Qt::UserRole: return currentString == "English" ? "en_US" : currentString.remove("subsurface_"); } return QVariant(); } int LanguageModel::rowCount(const QModelIndex &parent) const { return languages.count(); }