mvc是经典的三层结构,将数据,视图和逻辑分离。Qt中的Model/View框架,实现了这个模式。在Qt中这个模式设计到三个类,model类,view类和delegate类。model类保存数据,view负责显示,而delegate负责协调model和view之间的数据edit(编辑)和render(渲染)。

模型与视图结构:
模型与数据通信,并提供接口
视图从模型中获取数据条目索引
代理绘制数据条目
通信方式:信号&槽
qt MVC(一)_抽象基类
工作过程:
数据改变时,模型发出信号通知视图;
用户对界面操作时,视图发出信号;
代理发出信号告知模型和视图编辑器目前的状态;
模型model(表示数据)
抽象基类QAbstractItemModel
列表的抽象基类QAbstractListModel、表格的抽象基类QAbstractTableModel
QDirModel类是文件与目录的存储模型
QStandardItemModel类 : 管理更复杂的树结构件,其中每一个项目可以包含任意数据。
QStringListModel类: 用于存储简单的QString的列表项。
QFileSystemModel:提供有关本地文件系统的文件和目录信息。
QSqlQueryModel、QSqlTableModel、QSqlRelationalTableModel:使用模型/视图约定来访问数据库
如果这些标准模型不能满足要求,则可以继承化QAbstractItemModel、QAbstractListModel或QAbstractTableModel来创建自定义模型。
视图view(表示用户界面)
抽象基类QAbstractItemView
QListView—QListWidget\QUndoView
QTableView—QTableWidget
QTreeView—QTreeWidget
QColumnView
QHeaderView
实际上:QListWidget、QTableWidget、QTreeWidget已经包含数据,是模型与视图集成的类
虽然这些类是准备使用的实现,他们也可以被子类化,以提供自定义的视图。
代理delegate(自定义数据条目item的显示与编辑方式)
抽象基类QAbstractItemDelegate
QItemDelegate/QStyleItemDelegate
类QItemDelegate 由类QSqlRelationDelegate继承
排序
在模型/视图结构中有两种接近的排序方式,选择哪种方式取决于你的基础模型。
如果你的模型是可排序的,也就是说,如果重新实现了QAbstractItemModel::sort()方法,QTableView和QTreeView都提供了一个API,允许以编程方式排序来排序模型数据。此外,可以启用交互式排序(即允许用户将数据通过单击视图的标题进行排序),由QHeaderView::sortIndicatorChanged()信号分别连接到QTableView:: sortByColumn()槽或QTreeView::sortByColumn()槽。
另一种方法,如果模型没有所需的接口,或者如果想使用一个列表视图来显示数据,使用代理模型呈现数据视图之前应转换模型的结构。
项数据绑定:所有的model需要实现方法使视图和代理能够查询model
Models能够提供各种程度的数据访问限制:read-only,resizing,edited
如果只读访问,只需要实现下面几个函数在继承的子类中

Flags:其他的组件可以通过这个得知每个Item的信息,在大多数的models中,包含Qt::ItemIsEnable,Qt::ItemIsSelectable

data:被用来提供数据给视图和代理,一般的,models只要提供Qt::DisplayRole和任何程序特殊的角色,也有一些特殊的Qt::ToolTipRole等,详细可以看Qt::ItemDataRole。

headerData:为视图的头部提供信息数据。

rowCount:提供这个model有多少行数据。

上述的四个函数在任何类型的model中都要实现,不管是QAbstractListModel还是QAbstractTableModel。另外,下面的函数必须被实现,在QAbstractTableModel和QAbstractItemModel中,columnCount。

编辑项目

可编辑的模型允许数据项被修改,和可以提供函数来插入数据在行和列。

Flags,必须包含Qt::ItemDataRole。

setData,被用来修改和特殊的模型索引相关的项目。修改的数据必须是Qt::EditRole,发送一个dataChanged信号。

setHeaderData,用来修改水平和垂直的头信息,发出一个headerDataChanged信号。

改变models的size

所有类型的model能够提供插入和移除行。Table Model和分级的model也支持列的插入和删除操作。

#ifndef STRINGLISTNODEL_HPP
#define STRINGLISTNODEL_HPP
#include <QAbstractListModel>
#include <QStringList>

class StringListModel :
public QAbstractListModel
{
Q_OBJECT
public:
StringListModel(const QStringList& stringList,QObject* parent = 0);

~StringListModel();

//重现实现的函数
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);

private:
QStringList m_slist;//存放数据的容器
};

#endif
#include "StringListModel.h"
#include <QDebug>

StringListModel::StringListModel(const QStringList& stringList, QObject* parent /*= 0*/) :QAbstractListModel(parent), m_slist(stringList)
{

}

StringListModel::~StringListModel()
{
}

//数据的行数
int StringListModel::rowCount(const QModelIndex&parent) const
{
return m_slist.length();//链表的长度
}

//获得对应index项的数据
QVariant StringListModel::data(const QModelIndex& index, int role)const
{
if (!index.isValid())
{
return QVariant();
}
//row从0开始,有效的范围为0~链表长度减一
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);
}
}

//被其他组件访问时获得每个item的信息
Qt::ItemFlags StringListModel::flags(const QModelIndex &index)const
{
if (!index.isValid())
{
return Qt::ItemIsEnabled;
}
return QAbstractItemModel::flags(index) | Qt::ItemIsEditable;//可编辑的
}

//当视图的显示的数据被改变的时候,model也相应的改变
bool StringListModel::setData(const QModelIndex& index, const QVariant& value, int role)
{
//这个index必须是index的,必须还是可以编辑的
if (index.isValid()&&role==Qt::EditRole)
{
m_slist.replace(index.row(), value.toString());
emit dataChanged(index, index);//发出这个信号,外部使用这信号没用
return true;
}
return false;
}
QSplitter* mainSpliter = new QSplitter(Qt::Horizontal, 0);
QListView* view = new QListView(mainSpliter);

StringListModel * model = new StringListModel(QStringList() << "sss" << "yyyy" << "ffff", view);
view->setModel(model);
mainSpliter->show();