mirror of
https://github.com/subsurface/subsurface.git
synced 2024-11-28 05:00:20 +00:00
a5e7f4253a
Replace the fixed-size weightsystem table by a dynamically relocated table. Reuse the table-macros used in other parts of the code. The table stores weightsystem entries, not pointers to weightsystems. Thus, ownership of the description string is taken when adding a weightsystem. An extra function adds a cloned weightsystem at the end of the table. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
172 lines
4.7 KiB
C++
172 lines
4.7 KiB
C++
// SPDX-License-Identifier: GPL-2.0
|
|
#include "qt-models/weightmodel.h"
|
|
#include "core/subsurface-string.h"
|
|
#include "core/gettextfromc.h"
|
|
#include "core/metrics.h"
|
|
#include "core/qthelper.h"
|
|
#include "core/subsurface-qt/DiveListNotifier.h"
|
|
#include "qt-models/weightsysteminfomodel.h"
|
|
|
|
WeightModel::WeightModel(QObject *parent) : CleanerTableModel(parent),
|
|
changed(false),
|
|
rows(0)
|
|
{
|
|
//enum Column {REMOVE, TYPE, WEIGHT};
|
|
setHeaderDataStrings(QStringList() << tr("") << tr("Type") << tr("Weight"));
|
|
connect(&diveListNotifier, &DiveListNotifier::weightsystemsReset, this, &WeightModel::weightsystemsReset);
|
|
}
|
|
|
|
weightsystem_t *WeightModel::weightSystemAt(const QModelIndex &index)
|
|
{
|
|
return &displayed_dive.weightsystems.weightsystems[index.row()];
|
|
}
|
|
|
|
void WeightModel::remove(const QModelIndex &index)
|
|
{
|
|
if (index.column() != REMOVE)
|
|
return;
|
|
beginRemoveRows(QModelIndex(), index.row(), index.row());
|
|
rows--;
|
|
remove_weightsystem(&displayed_dive, index.row());
|
|
changed = true;
|
|
endRemoveRows();
|
|
}
|
|
|
|
void WeightModel::clear()
|
|
{
|
|
beginResetModel();
|
|
rows = 0;
|
|
endResetModel();
|
|
}
|
|
|
|
QVariant WeightModel::data(const QModelIndex &index, int role) const
|
|
{
|
|
if (!index.isValid() || index.row() >= displayed_dive.weightsystems.nr)
|
|
return QVariant();
|
|
|
|
const weightsystem_t *ws = &displayed_dive.weightsystems.weightsystems[index.row()];
|
|
|
|
switch (role) {
|
|
case Qt::FontRole:
|
|
return defaultModelFont();
|
|
case Qt::TextAlignmentRole:
|
|
return Qt::AlignCenter;
|
|
case Qt::DisplayRole:
|
|
case Qt::EditRole:
|
|
switch (index.column()) {
|
|
case TYPE:
|
|
return gettextFromC::tr(ws->description);
|
|
case WEIGHT:
|
|
return get_weight_string(ws->weight, true);
|
|
}
|
|
break;
|
|
case Qt::DecorationRole:
|
|
if (index.column() == REMOVE)
|
|
return trashIcon();
|
|
case Qt::SizeHintRole:
|
|
if (index.column() == REMOVE)
|
|
return trashIcon().size();
|
|
case Qt::ToolTipRole:
|
|
if (index.column() == REMOVE)
|
|
return tr("Clicking here will remove this weight system.");
|
|
}
|
|
return QVariant();
|
|
}
|
|
|
|
// 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 = &displayed_dive.weightsystems.weightsystems[index.row()];
|
|
if (index.column() == WEIGHT) {
|
|
if (ws->weight.grams != value.toInt()) {
|
|
ws->weight.grams = value.toInt();
|
|
dataChanged(index, index);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool WeightModel::setData(const QModelIndex &index, const QVariant &value, int role)
|
|
{
|
|
QString vString = value.toString();
|
|
weightsystem_t *ws = &displayed_dive.weightsystems.weightsystems[index.row()];
|
|
switch (index.column()) {
|
|
case TYPE:
|
|
if (!value.isNull()) {
|
|
//TODO: C-function weight_system_set_description ?
|
|
if (!ws->description || gettextFromC::tr(ws->description) != vString) {
|
|
// loop over translations to see if one matches
|
|
int i = -1;
|
|
while (ws_info[++i].name && i < MAX_WS_INFO) {
|
|
if (gettextFromC::tr(ws_info[i].name) == vString) {
|
|
ws->description = copy_string(ws_info[i].name);
|
|
break;
|
|
}
|
|
}
|
|
if (ws_info[i].name == NULL) // didn't find a match
|
|
ws->description = copy_qstring(vString);
|
|
changed = true;
|
|
}
|
|
}
|
|
break;
|
|
case WEIGHT:
|
|
if (CHANGED()) {
|
|
ws->weight = string_to_weight(qPrintable(vString));
|
|
// now update the ws_info
|
|
changed = true;
|
|
WSInfoModel *wsim = WSInfoModel::instance();
|
|
QModelIndexList matches = wsim->match(wsim->index(0, 0), Qt::DisplayRole, gettextFromC::tr(ws->description));
|
|
if (!matches.isEmpty())
|
|
wsim->setData(wsim->index(matches.first().row(), WSInfoModel::GR), ws->weight.grams);
|
|
}
|
|
break;
|
|
}
|
|
dataChanged(index, index);
|
|
return true;
|
|
}
|
|
|
|
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&) const
|
|
{
|
|
return rows;
|
|
}
|
|
|
|
void WeightModel::add()
|
|
{
|
|
int row = rows;
|
|
weightsystem_t ws { {0}, "" };
|
|
beginInsertRows(QModelIndex(), row, row);
|
|
add_cloned_weightsystem(&displayed_dive.weightsystems, ws);
|
|
rows++;
|
|
changed = true;
|
|
endInsertRows();
|
|
}
|
|
|
|
void WeightModel::updateDive()
|
|
{
|
|
beginResetModel();
|
|
rows = displayed_dive.weightsystems.nr;
|
|
endResetModel();
|
|
}
|
|
|
|
void WeightModel::weightsystemsReset(const QVector<dive *> &dives)
|
|
{
|
|
// This model only concerns the currently displayed dive. If this is not among the
|
|
// dives that had their cylinders reset, exit.
|
|
if (!current_dive || std::find(dives.begin(), dives.end(), current_dive) == dives.end())
|
|
return;
|
|
|
|
// Copy the weights from the current dive to the displayed dive.
|
|
copy_weights(¤t_dive->weightsystems, &displayed_dive.weightsystems);
|
|
|
|
// And update the model..
|
|
updateDive();
|
|
}
|