You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
prodejna/core/autotablemodel.h

288 lines
7.6 KiB
C

#ifndef ODBTABLEMODEL_H
#define ODBTABLEMODEL_H
#include <QAbstractTableModel>
#include <QSharedPointer>
#include <QMetaProperty>
#include <QModelIndex>
#include <QDebug>
#include "../qdecimal/src/QDecDouble.hh"
#include "define.h"
#include "core_global.h"
#include "exprevaluator.h"
#include "itablemodel.h"
#include "data/comboitem.h"
template<class T>
class AutoTableModel : public ITableModel
{
public:
explicit AutoTableModel(QObject *parent = NULL)
:ITableModel(parent)
{
filtered = false;
m_checkboxSelect = false;
}
virtual ~AutoTableModel() {}
// QAbstractItemModel interface
public:
int rowCount(const QModelIndex &parent = QModelIndex()) const
{
Q_UNUSED(parent)
return m_list.size();
}
int columnCount(const QModelIndex &parent = QModelIndex()) const
{
Q_UNUSED(parent)
QObject *entity = (QObject*)new T();
int colCount = entity->metaObject()->propertyCount() - 1;
delete entity;
return colCount;
}
QVariant data(const QModelIndex &index, int role) const
{
if (index.column() == 0 && m_checkboxSelect)
{
if (role == Qt::CheckStateRole)
{
if (m_selectedRows.contains(index.row()))
{
return Qt::Checked;
}
else
{
return Qt::Unchecked;
}
}
}
QSharedPointer<T> entity = m_list.at(index.row());
QObject *rawEntity = (QObject*)entity.data();
QVariant dispData = rawEntity->property(rawEntity->metaObject()->property(index.column() + 1).name());
if (role == Qt::DisplayRole)
{
if (dispData.canConvert<QObject*>() && qobject_cast<ComboItem*>(dispData.value<QObject*>()))
{
return qobject_cast<ComboItem*>(dispData.value<QObject*>())->toString();
}
if (dispData.canConvert<QDecDouble>())
{
return dispData.value<QDecDouble>().toDouble();
}
return dispData;
}
if (role == Qt::TextAlignmentRole)
{
if (dispData.canConvert<QDecDouble>()
|| dispData.type() == QVariant::Date
|| dispData.type() == QVariant::Time
|| dispData.type() == QVariant::DateTime
|| dispData.type() == QVariant::Int
|| dispData.type() == QVariant::Double)
{
return Qt::AlignRight;
}
return Qt::AlignLeft;
}
if (role == Qt::EditRole)
{
return dispData;
}
return QVariant::Invalid;
}
QList<QSharedPointer<T> > list()
{
return m_list;
}
QVariant headerData(int section, Qt::Orientation orientation, int role) const
{
if (role != Qt::DisplayRole) {
return QVariant();
}
if (orientation == Qt::Horizontal) {
QObject *entity = (QObject*)new T();
for (int i = 0; i < entity->metaObject()->propertyCount(); i++) {
if (i == section) {
QString colName(entity->metaObject()->property(i + 1).name());
delete entity;
if (m_translations.contains(colName))
{
return m_translations[colName];
}
return colName;
}
}
delete entity;
}
return QVariant(section + 1);
}
virtual void sort(int column, Qt::SortOrder order) {
if (m_list.isEmpty()) {
return;
}
beginResetModel();
QObject *rawEntity = (QObject*)m_list.at(0).data();
const char *prop = rawEntity->metaObject()->property(column + 1).name();
std::sort(ALL(m_list), [prop, order](QSharedPointer<T> entA, QSharedPointer<T> entB) -> bool {
if (order == Qt::AscendingOrder) {
return ((QObject*)entA.data())->property(prop) < ((QObject*)entB.data())->property(prop);
} else {
return ((QObject*)entB.data())->property(prop) < ((QObject*)entA.data())->property(prop);
}
});
endResetModel();
}
/*///////////////////////*/
QSharedPointer<T> itemFromIndex(const QModelIndex &index) const
{
return m_list.at(index.row());
}
void setItemToIndex(const QModelIndex &index, QSharedPointer<T> data)
{
m_list.removeAt(index.row());
m_list.insert(index.row(), data);
emit dataChanged(index, index);
}
void addRow(QSharedPointer<T> data)
{
int index = rowCount() == 0 ? 0 : rowCount() - 1;
beginInsertRows(QModelIndex(), index, index);
insertRow(rowCount());
m_list.append(data);
endInsertRows();
}
void removeRowAt(const QModelIndex &index)
{
beginRemoveRows(QModelIndex(), index.row(), index.row());
removeRow(index.row());
m_list.removeAt(index.row());
endRemoveRows();
}
void setData(QList<QSharedPointer<T> > list)
{
beginResetModel();
m_list = list;
endResetModel();
}
void setTranslations(const QMap<QString, QString> &translations)
{
m_translations = translations;
}
QList<int> selectedRows() const
{
return m_selectedRows;
}
QList<QSharedPointer<T> > selectedItems()
{
QList<QSharedPointer<T> > ret;
foreach (int row, m_selectedRows) {
ret.append(m_list[row]);
}
return ret;
}
protected:
void handleFilter(const QString &filter) override
{
beginResetModel();
if (!filtered)
{
m_fullList = m_list;
filtered = true;
}
ExprEvaluator ev;
auto it = std::copy_if(ALL(m_fullList), m_list.begin(), [&filter, &ev](QSharedPointer<T> obj){ return ev.evaluate((QObject*)obj.data(), filter); });
m_list.erase(it, m_list.end());
endResetModel();
}
void handleRestore() override
{
if (filtered)
{
beginResetModel();
m_list = m_fullList;
endResetModel();
filtered = false;
}
}
private:
QList<QSharedPointer<T> > m_list;
QList<QSharedPointer<T> > m_fullList;
QList<int> m_selectedRows;
QMap<QString, QString> m_translations;
bool filtered;
// QAbstractItemModel interface
public:
virtual bool setData(const QModelIndex &index, const QVariant &value, int role) override
{
if (role == Qt::EditRole)
{
QSharedPointer<T> entity = m_list.at(index.row());
QObject *rawEntity = (QObject*)entity.data();
rawEntity->setProperty(rawEntity->metaObject()->property(index.column() + 1).name(), value);
}
if (role == Qt::CheckStateRole)
{
if (m_selectedRows.contains(index.row()))
{
m_selectedRows.removeOne(index.row());
}
else
{
m_selectedRows.append(index.row());
}
}
emit editCompleted();
return true;
}
};
#endif // ODBTABLEMODEL_H