Qt Model/View 學習筆記 (一)
Qt 4推出了一組新的item view類,它們使用model/view結構來管理數據與表示層的關係。這種結構帶來的
功能上的分離給了開發人員更大的彈性來定製數據項的表示,它也提供一個標準的model接口,使得更多的
數據源可以被這些item view使用。這裏對model/view的結構進行了描述,結構中的每個組件都進行了解釋,
給出了一些例子說明了提供的這些類如何使用。
Model/View 結構
Model-View-Controller(MVC), 是從Smalltalk發展而來的一種設計模式,常被用於構建用戶界面。經典設計模式的著作中有這樣的描述:
MVC 由三種對象組成。Model是應用程序對象,View是它的屏幕表示,Controller定義了用戶界面如何對用戶輸入進行響應。在MVC之前,用戶界面設計傾向於三者揉合在一起,MVC對它們進行了解耦,提高了靈活性與重用性。
假如把view與controller結合在一起,結果就是model/view結構。這個結構依然是把數據存儲與數據表示進行了分離,它與MVC都基於同樣的思想,但它更簡單一些。這種分離使得在幾個不同的view上顯示同一個數據成爲可能,也可以重新實現新的view,而不必改變底層的數據結構。爲了更靈活的對用戶輸入進行處理,引入了delegate這個概念。它的好處是,數據項的渲染與編程可以進行定製。
如上圖所示,model與數據源通訊,並提供接口給結構中的別的組件使用。通訊的性質依賴於數據源的種類
與model實現的方式。view從model獲取model indexes,後者是數據項的引用。通過把model indexes提供給model,view可以從數據源中獲取數據。
在標準的views中,delegate會對數據項進行渲染,當某個數據項被選中時,delegate通過model indexes與model直接進行交流。總的來說,model/view 相關類可以被分成上面所提到的三組:models,views,delegates。這些組件通過抽象類來定義,它們提供了共同的接口,在某些情況下,還提供了缺省的實現。抽象類意味着需要子類化以提供完整的其他組件希望的功能。這也允許實現定製的組件。models,views,delegates之間通過信號,槽機制來進行通訊:
從model發出的信號通知view數據源中的數據發生了改變。
從view發出的信號提供了有關被顯示的數據項與用戶交互的信息。
從delegate發生的信號被用於在編輯時通知model和view關於當前編輯器的狀態信息。
Models
所有的item models都基於QAbstractItemModel類,這個類定義了用於views和delegates訪問數據的接口。
數據本身不必存儲在model,數據可被置於一個數據結構或另外的類,文件,數據庫,或別的程序組件中。
關於model的基本概念在Model Classes部分中描述。
QAbstractItemModel提供給數據一個接口,它非常靈活,基本滿足views的需要,無論數據用以下任何樣的形式
表現,如tables,lists,trees。然而,當你重新實現一個model時,如果它基於table或list形式的數據結構,最好從QAbstractListModel,QAbstractTableModel開始做起,因爲它們提供了適當的常規功能的缺省實現。這些類可以被子類化以支持特殊的定製需求。子類化model的過程在Create New Model部分討論
QT提供了一些現成的models用於處理數據項:
QStringListModel 用於存儲簡單的QString列表。
QStandardItemModel 管理複雜的樹型結構數據項,每項都可以包含任意數據。
QDirModel 提供本地文件系統中的文件與目錄信息。
QSqlQueryModel, QSqlTableModel,QSqlRelationTableModel用來訪問數據庫。
假如這些標準Model不滿足你的需要,你應該子類化QAbstractItemModel,QAbstractListModel或是
QAbstractTableModel來定製。
Views
不同的view都完整實現了各自的功能:QListView把數據顯示爲一個列表,QTableView把Model 中的數據以table的形式表現,QTreeView 用具有層次結構的列表來顯示model中的數據。這些類都基於QAbstractItemView抽象基類,儘管這些類都是現成的,完整的進行了實現,但它們都可以用於子類化以便滿足定製需求。
Delegates
QAbstractItemDelegate 是model/view架構中的用於delegate的抽象基類。缺省的delegate實現在QItemDelegate類中提供。它可以用於Qt標準views的缺省 delegate.
排序
在model/view架構中,有兩種方法進行排序,選擇哪種方法依賴於你的底層Model。
假如你的model是可排序的,也就是它重新實現了QAbstractItemModel::sort()函數,QTableView與QTreeView都提供了API,允許你以編程的方式對Model數據進行排序。另外,你也可以進行交互方式下的排序(例如,允許用戶通過點擊view表頭的方式對數據進行排序),可以這樣做:把QHeaderView::sectionClicked()信號與QTableView::sortByColum()槽或QTreeView::sortByColumn()槽進行聯結就好了。
另一種方法是,假如你的model沒有提供需要的接口或是你想用list view表示數據,可以用一個代理
model在用view表示數據之前對你的model數據結構進行轉換。
便利類
許多便利類都源於標準的view類,它們方便了那些使用Qt中基於項的view與table類,它們不應該被子類化,
它們只是爲Qt 3的等價類提供一個熟悉的接口。這些類有QListWidget,QTreeWidget,QTableWidget,它們提供瞭如Qt 3中的QListBox, QlistView,QTable相似的行爲。這些類比View類缺少靈活性,不能用於任意的models,推介使用model/view的方法處理數據。
Qt Model/View 學習筆記 (二)
介紹
Qt提供了兩個標準的models:QStandardItemModel和QDirModel。QStandardItemModel是一個多用途的
model,可用於表示list,table,tree views所需要的各種不同的數據結構。這個model也持有數據。QDirModel
維護相關的目錄內容的信息,它本身不持有數據,僅是對本地文件系統中的文件與目錄的描述。
QDirModel是一個現成的model,很容易進行配置以用於現存的數據,使用這個model,可以很好地展示如何
給一個現成的view設定model,研究如何用model indexes來操縱數據。
model與views的搭配使用
QListView與QTreeView很適合與QDirModel搭配。下面的例子在tree view與list view顯示了相同的信息,QDirModel提供了目錄內容數據。這兩個Views共享用戶選擇,因此每個被選擇的項在每個view中都會被高亮。
先裝配出一個QDirModel以供使用,再創建views去顯示目錄的內容。這給我展示了使用model的最簡單的方式。
model的創建與使用都在main()函數中完成:
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
QSplitter *splitter = new QSplitter;
QDirModel *model = new QDirModel;
//從缺省目錄創建數據
QTreeView *tree = new QTreeView(splitter);
tree->setModel(model);
tree->setRootIndex(model->index(QDir::currentPath()));
QListView *list = new QListView(splitter);
list->setModel(model);
list->setRootIndex(model->index(QDir::currentPath()));
//配置一個view去顯示model中的數據,只需要簡單地調用setModel(),並把目錄model作爲參數傳遞
//setRootIndex()告訴views顯示哪個目錄的信息,這需要提供一個model index,然後用這個
//model index去model中去獲取數據
//index()這個函數是QDirModel特有的,通過把一個目錄做爲參數,得到了需要的model index
//其他的代碼只是窗口show出來,進入程序的事件循環就好了
splitter->setWindowTitle("Two views onto the same directory model");
splitter->show();
return app.exec();
}