清源游民 gameogre@gmail.com
View 類
概念
在model/view架構(gòu)中,view從model中獲得數(shù)據(jù)項然后顯示給用戶。數(shù)據(jù)顯示的方式不必與model提供的表示方式相同,可以與底層存儲數(shù)據(jù)項的數(shù)據(jù)結(jié)構(gòu)完全不同。
內(nèi)容與顯式的分離是通過由QAbstractItemModel提供的標(biāo)準(zhǔn)模型接口,由QAsbstractItemview提供的標(biāo)準(zhǔn)視圖接口共同實現(xiàn)的。普遍使用model index來表示數(shù)據(jù)項。view負(fù)責(zé)管理從model中讀取的數(shù)據(jù)的外觀布局。
它們自己可以去渲染每個數(shù)據(jù)項,也可以利用delegate來既處理渲染又進(jìn)行編輯。
除了顯示數(shù)據(jù),views也處理數(shù)據(jù)項的導(dǎo)航,參與有關(guān)于數(shù)據(jù)項選擇的部分功能。view也實現(xiàn)一些基本的用戶接口特性,如上下文菜單與拖拽功能。view也為數(shù)據(jù)項提供了缺省的編程功能,也可搭配delegate實現(xiàn)更為特殊的定制編輯的需求。
一個view創(chuàng)建時必不需要model,但在它能顯示一些真正有用的信息之前,必須提供一個model。view通過使用
selections來跟蹤用戶選擇的數(shù)據(jù)項。每個view可以維護(hù)單獨使用的selections,也可以在多個views之間共享。有些views,如QTableView和QTreeView,除數(shù)據(jù)項之外也可顯示標(biāo)題(Headers),標(biāo)題部分通過一個view來實現(xiàn),QHeaderView。標(biāo)題與view一樣總是從相同的model中獲取數(shù)據(jù)。從 model中獲取數(shù)據(jù)的函數(shù)是QabstractItemModel::headerDate(),一般總是以表單的形式中顯示標(biāo)題信息。可以從QHeaderView子類化,以實現(xiàn)更為復(fù)雜的定制化需求。
使用現(xiàn)成的view
Qt提供了三個現(xiàn)成的view 類,它們能夠以用戶熟悉的方式顯示model中的數(shù)據(jù)。QListView把model中的數(shù)據(jù)項以一個簡單的列表的形式顯示,或是以經(jīng)典的圖標(biāo)視圖的形式顯示。QTreeView把model中的數(shù)據(jù)項作為具有層次結(jié)構(gòu)的列表的形式顯示,它允許以緊湊的深度嵌套的結(jié)構(gòu)進(jìn)行顯示。QTableView卻是把model中的數(shù)據(jù)項以表格的形式展現(xiàn),更像是一個電子表格應(yīng)用程序的外觀布局。

以上這些標(biāo)準(zhǔn)view的行為足以應(yīng)付大多數(shù)的應(yīng)用程序,它們也提供了一些基本的編輯功能,也可以定制特殊的需求。
使用model
以前的例子中創(chuàng)建過一個string list model,可以給它設(shè)置一些數(shù)據(jù),再創(chuàng)建一個view把model中的內(nèi)容展示出來:
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
// Unindented for quoting purposes:
QStringList numbers;
numbers << "One" << "Two" << "Three" << "Four" << "Five";
QAbstractItemModel *model = new StringListModel(numbers);
//要注意的是,這里把StringListModel作為一個QAbstractItemModel來使用。這樣我們就可以
//使用model中的抽象接口,而且如果將來我們用別的model代替了當(dāng)前這個model,這些代碼也會照樣工作。
//QListView提供的列表視圖足以滿足當(dāng)前這個model的需要了。
QListView *view = new QListView;
view->setModel(model);
view->show();
return app.exec();
}

view會渲染model中的內(nèi)容,通過model的接口來訪問它的數(shù)據(jù)。當(dāng)用戶試圖編輯數(shù)據(jù)項時,view會使用缺省的delegate來提供一個編輯構(gòu)件。
一個model,多個views
為多個views提供相同的model是非常簡單的事情,只要為每個view設(shè)置相同的model。
QTableView *firstTableView = new QTableView;
QTableView *secondTableView = new QTableView;
firstTableView->setModel(model);
secondTableView->setModel(model);
在model/view架構(gòu)中信號、槽機(jī)制的使用意味著model中發(fā)生的改變會傳遞中聯(lián)結(jié)的所有view中,這保證了
不管我們使用哪個view,訪問的都是同樣的一份數(shù)據(jù)。

上面的圖展示了一個model上的兩個不同的views,盡管在不同的view中顯示的model中的數(shù)據(jù)是一致的,每個
view都維護(hù)它們自己的內(nèi)部選擇模型,但有時候在某些情況下,共享一個選擇模型也是合理的。
處理數(shù)據(jù)項的選擇
view中數(shù)據(jù)項選擇機(jī)制由QItemSelectionModel類提供。所有標(biāo)準(zhǔn)的view缺省都構(gòu)建它們自己的選擇模型,
以標(biāo)準(zhǔn)的方式與它們交互。選擇模型可以用selectionModel()函數(shù)取得,替代的選擇模型也可以通過
setSelectionModel()來設(shè)置。當(dāng)我們想在一個model上提供多個一致的views時,這種對選擇模型的控制能力非常有用。通常來講,除非你子類化一個model或view,你不必直接操縱selections的內(nèi)容。
多個views之間共享選擇
接著上邊的例子,我們可以這樣:
secondTableView->setSelectionModel(firstTableView->selectionModel());
現(xiàn)在所有views都在同樣的選擇模型上操作,數(shù)據(jù)與選擇項都保持同步。

上面的例子中,兩個view的類型是相同的,假如這兩個view類型不同,那么所選擇的數(shù)據(jù)項在每個view
中的表現(xiàn)形式會有很大的不同。例如,在一個table view中一個連續(xù)的選擇,在一個tree view中表現(xiàn)出
來的可能會是幾個高亮的數(shù)據(jù)項片斷的組合。
posted on 2007-06-18 16:15
清源游民 閱讀(8261)
評論(3) 編輯 收藏 引用 所屬分類:
Qt