universallistmodel.h 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. #pragma once
  2. #include <QAbstractListModel>
  3. #include <QObject>
  4. #include <QList>
  5. #include <QHash>
  6. #include <QPointer>
  7. #include <QMetaProperty>
  8. #include <QMetaObject>
  9. #include <QDebug>
  10. /*! Universal list model is QObject-base list model abstraction.
  11. * It exposes all objects properties as data-roles.
  12. *
  13. */
  14. template <typename T>
  15. class UniversalListModel : public QAbstractListModel
  16. {
  17. public:
  18. UniversalListModel(QObject* parent = 0) : QAbstractListModel(parent) {}
  19. ~UniversalListModel() {
  20. clear();
  21. }
  22. int rowCount(const QModelIndex &parent) const {
  23. Q_UNUSED(parent)
  24. return m_container.count();
  25. }
  26. QHash<int, QByteArray> roleNames() const {
  27. if(s_roleNames.isEmpty()) {
  28. int propertyCount = T::staticMetaObject.propertyCount();
  29. for(int i = 1; i < propertyCount; i++) {
  30. s_roleNames.insert(Qt::UserRole + i, T::staticMetaObject.property(i).name());
  31. }
  32. s_roleNames[propertyCount] = "modelData";
  33. }
  34. return s_roleNames;
  35. }
  36. QVariant data(const QModelIndex &index, int role) const
  37. {
  38. int row = index.row();
  39. if(row < 0 || row >= m_container.count() || m_container.at(row).isNull()) {
  40. return QVariant();
  41. }
  42. T* dataPtr = m_container.at(row).data();
  43. if(s_roleNames.value(role) == "modelData") {
  44. return QVariant::fromValue(dataPtr);
  45. }
  46. return dataPtr->property(s_roleNames.value(role));
  47. }
  48. /*!
  49. * \brief append
  50. * \param value
  51. * \return
  52. */
  53. int append(T* value) {
  54. Q_ASSERT_X(value != nullptr, fullTemplateName(), "Trying to add member of NULL");
  55. if(m_container.indexOf(value) >= 0) {
  56. #ifdef DEBUG
  57. qDebug() << fullTemplateName() << "Member already exists";
  58. #endif
  59. return -1;
  60. }
  61. beginInsertRows(QModelIndex(), m_container.count(), m_container.count());
  62. m_container.append(QPointer<T>(value));
  63. endInsertRows();
  64. return m_container.count() - 1;
  65. }
  66. /*!
  67. * \brief prepend
  68. * \param value
  69. * \return
  70. */
  71. int prepend(T* value) {
  72. Q_ASSERT_X(value != nullptr, fullTemplateName(), "Trying to add member of NULL");
  73. if(m_container.indexOf(value) >= 0) {
  74. #ifdef DEBUG
  75. qDebug() << fullTemplateName() << "Member already exists";
  76. #endif
  77. return -1;
  78. }
  79. beginInsertRows(QModelIndex(), 0, 0);
  80. m_container.prepend(QPointer<T>(value));
  81. endInsertRows();
  82. return 0;
  83. }
  84. /*!
  85. * \brief remove
  86. * \param value
  87. */
  88. void remove(T* value) {
  89. Q_ASSERT_X(value != nullptr, fullTemplateName(), ": Trying to remove member of NULL");
  90. int valueIndex = m_container.indexOf(value);
  91. if(valueIndex >= 0) {
  92. beginRemoveRows(QModelIndex(), valueIndex, valueIndex);
  93. m_container.removeAt(valueIndex);
  94. endRemoveRows();
  95. }
  96. }
  97. /*!
  98. * \brief Resets container with new container passed as parameter
  99. * \param container a data for model. Should contain QPointer's to objects.
  100. * Passing empty container makes model empty. This method should be used to cleanup model.
  101. */
  102. void reset(const QList<QPointer<T> >& container) {
  103. beginResetModel();
  104. clear();
  105. m_container = container;
  106. endResetModel();
  107. }
  108. /*!
  109. * \brief Returns the item at index position i in the list. i must be a valid index position in the list (i.e., 0 <= i < rowCount()).
  110. * This function is very fast (constant time).
  111. * \param i index of looking object
  112. * \return Object at provided index
  113. */
  114. T* at(int i) const {
  115. return m_container.at(i);
  116. }
  117. /*!
  118. * \brief Looking for index of objec
  119. * \param value
  120. * \return
  121. */
  122. int indexOf(T* value) const {
  123. return m_container.indexOf(value);
  124. }
  125. protected:
  126. void clear() {
  127. m_container.clear();
  128. }
  129. QList<QPointer<T> > m_container;
  130. static QHash<int, QByteArray> s_roleNames;
  131. private:
  132. static QByteArray fullTemplateName() { //Debug helper
  133. return QString("UniversalListModel<%1>").arg(T::staticMetaObject.className()).toLatin1();
  134. }
  135. };
  136. template<typename T>
  137. QHash<int, QByteArray> UniversalListModel<T>::s_roleNames;