Qt mvc學習一

 mvc是經典的三層結構,將數據,視圖和邏輯分離。Qt中的Model/View框架,實現了這個模式。在Qt中這個模式設計到三個類,model類,view類和delegate類。model類保存數據,view複製顯示,而delegate負責協調model和view之間的數據edit(編輯)和render(渲染)。mvc

                  

這些在model子類中須要實現的方法能夠分爲三組。框架

項數據綁定:全部的model須要實現方法使視圖和代理可以查詢model...函數

Models可以提供各類程度的數據訪問限制:read-onlyresizingeditedui

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。另外,下面的函數必須被實現,在QAbstractTableModelQAbstractItemModel中,columnCount

編輯項目

可編輯的模型容許數據項被修改,和能夠提供函數來插入數據在行和列。

Flags,必須包含Qt::ItemDataRole

setData,被用來修改和特殊的模型索引相關的項目。修改的數據必須是Qt::EditRole,發送一個dataChanged信號。

setHeaderData,用來修改水平和垂直的頭信息,發出一個headerDataChanged信號。

改變modelssize

全部類型的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;

}


使用這個model類


StringListModel *model = new StringListModel(QStringList() << "chenchen"
                                                 << "love"
                                                 << "zhou xiang",this);
    ui->listView->setModel(model);
    //這邊的listView是一個QListView對象


修改數據項

相關文章
相關標籤/搜索