mvc是經典的三層結構,將數據,視圖和邏輯分離。Qt中的Model/View框架,實現了這個模式。在Qt中這個模式設計到三個類,model類,view類和delegate類。model類保存數據,view複製顯示,而delegate負責協調model和view之間的數據edit(編輯)和render(渲染)。mvc
這些在model子類中須要實現的方法能夠分爲三組。框架
項數據綁定:全部的model須要實現方法使視圖和代理可以查詢model...函數
Models可以提供各類程度的數據訪問限制:read-only,resizing,editedui
Read-Only access 只讀訪問this
若是隻讀訪問,只須要實現下面幾個函數在繼承的子類中spa
Flags,其餘的組件能夠經過這個得知每一個Item的信息,在大多數的models中,包含Qt::ItemIsEnable,Qt::ItemIsSelectable.net
data,被用來提供數據給視圖和代理,通常的,models只要提供Qt::DisplayRole和任何程序特殊的角色,也有一些特殊的Qt::ToolTipRole等,詳細能夠看Qt::ItemDataRole。設計
headerData,爲視圖的頭部提供信息數據。代理
rowCount提供這個model有多少行數據。code
上述的四個函數在任何類型的model中都要實現,無論是QAbstractListModel仍是QAbstractTableModel。另外,下面的函數必須被實現,在QAbstractTableModel和QAbstractItemModel中,columnCount。
編輯項目
可編輯的模型容許數據項被修改,和能夠提供函數來插入數據在行和列。
Flags,必須包含Qt::ItemDataRole。
setData,被用來修改和特殊的模型索引相關的項目。修改的數據必須是Qt::EditRole,發送一個dataChanged信號。
setHeaderData,用來修改水平和垂直的頭信息,發出一個headerDataChanged信號。
改變models的size
全部類型的model可以提供插入和移除行。Table Model和分級的model也支持列的插入和刪除操做。
下面的例子是基於QAbstractListModel實現的一個QStringListModel
/************************************************ * *author:周翔 *e-mail:604487178@qq.com *blog:http://blog.csdn.net/zhx6044 * * *************************************************/ #ifndef STRINGLISTMODEL_HPP #define STRINGLISTMODEL_HPP #include <QAbstractListModel> #include <QStringList> class StringListModel : public QAbstractListModel { Q_OBJECT public: explicit StringListModel( const QStringList &stringList, QObject *parent = 0); //從新實現的函數 int rowCount(const QModelIndex &parent) const; QVariant data(const QModelIndex &index, int role) const; QVariant headerData(int section, Qt::Orientation orientation, int role) const; Qt::ItemFlags flags(const QModelIndex &index) const; bool setData(const QModelIndex &index, const QVariant &value, int role); signals: public slots: private: QStringList m_slist;//存放數據的容器 }; #endif // STRINGLISTMODEL_HPP /************************************************ * *author:周翔 *e-mail:604487178@qq.com *blog:http://blog.csdn.net/zhx6044 * * *************************************************/ #include "stringlistmodel.hpp" #include <QDebug> StringListModel::StringListModel(const QStringList &stringList, QObject *parent) : QAbstractListModel(parent), m_slist(stringList) { } /** * @brief StringListModel::rowCount model數據的行數 * @return */ int StringListModel::rowCount(const QModelIndex &/*parent*/) const { return m_slist.length();//就是鏈表的長度 } /** * @brief StringListModel::data 得到對應index項的數據 * @param index * @param role 數據的角色 * @return */ QVariant StringListModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) { return QVariant(); } //row從0開始,有效的範圍爲0~鏈表長度減1 if (index.row() >= m_slist.length()) { return QVariant(); } if (role == Qt::DisplayRole) { return m_slist.at(index.row()); } else { return QVariant(); } } QVariant StringListModel::headerData(int section, Qt::Orientation orientation, int role) const { if (role != Qt::DisplayRole) { return QVariant(); } if (orientation == Qt::Horizontal) { return QString("col %1").arg(section); } else { return QString("row %1").arg(section); } } /** * @brief StringListModel::flags 被其餘組件訪問時得到每一個Item的信息 * @param index * @return */ Qt::ItemFlags StringListModel::flags(const QModelIndex &index) const { if (!index.isValid()) { return Qt::ItemIsEnabled; } return QAbstractItemModel::flags(index) | Qt::ItemIsEditable;//可編輯的 } /** * @brief StringListModel::setData 當視圖的顯示的數據被改變的時候,model也相應的改變 * @param index * @param value * @param role * @return */ bool StringListModel::setData(const QModelIndex &index, const QVariant &value, int role) { //這個index必須是有效的,必須仍是可編輯的 if (index.isValid() && role == Qt::EditRole) { m_slist.replace(index.row(),value.toString()); emit dataChanged(index,index);//發出這個信號,外部使用這個信號沒用 return true; } return false; }
StringListModel *model = new StringListModel(QStringList() << "chenchen" << "love" << "zhou xiang",this); ui->listView->setModel(model); //這邊的listView是一個QListView對象
修改數據項