類QabstractItemModel,QabstractListModel,QAbstractTableModel不保存數據,用戶需要從這些類派生出子類,並在子類中定義某種數據結構來保存數據。與此不同,類QStandardItemModel負責保存數據,每個數據項被表示為類QStandardItem的對象。我們首先闡述如何使用類QStandardItem保存一個數據項,再闡述如何使用類QStandardItemModel將這些數據項組織起來,形成列表、表格或者樹,以供其他視圖類顯示。
理論參考:http://book.51cto.com/art/201207/347880.htm
關鍵是QAbstractItemView.setItemDelegateForColumn函數,可以設置進度條代理。
其中QAbstractItemView的函數特別多,QStandardItem的函數也特別多。
int main( int argc, char **argv ) { QApplication app( argc, argv ); QTableView table; QStandardItemModel model( 10, 2 ); for( int r=0; r<10; ++r ) { QStandardItem *item = new QStandardItem( QString("Row %1").arg(r+1) ); model.setItem( r, 0, item ); model.setItem( r, 1, new QStandardItem( QString::number((r*30)%100 )) ); } table.setModel( &model ); // 正常設置模型,沒有任何特殊之處 BarDelegate delegate; table.setItemDelegateForColumn( 1, &delegate ); // 設置第一列為代理 table.show(); return app.exec(); }
除了調用父類構造函數,還要覆蓋兩個純虛函數paint和sizeHint
class BarDelegate : public QAbstractItemDelegate { public: BarDelegate( QObject *parent = 0 ); // 覆蓋兩個函數就可以顯示進度條 void paint( QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index ) const; QSize sizeHint( const QStyleOptionViewItem &option, const QModelIndex &index ) const; };
具體每一步的解釋:
BarDelegate::BarDelegate( QObject *parent ) : QAbstractItemDelegate( parent ) { } void BarDelegate::paint( QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index ) const { if( option.state & QStyle::State_Selected ) painter->fillRect( option.rect, option.palette.highlight() ); // 數據是存儲在QStandardItemModel的QStandardItem,會自動根據當前行進行匹配(我認為) int value = index.model()->data( index, Qt::DisplayRole ).toInt(); // 這句,取得當前行的數據 qDebug() << value; double factor = double(value)/100.0; // 計算比例因子 painter->save(); // 保存舊畫板(我認為) // 進度條背景色 if( factor > 0.8 ) { painter->setBrush( Qt::red ); // 超過0.8畫純紅色 factor = 1; } else painter->setBrush( QColor( 0, int(factor*255), 255-int(factor*255) ) ); // 否則顏色依次變淡 painter->setPen( Qt::black ); // 畫筆顏色(這里沒用到,我認為) // 前面都是准備工作,這里才是真正畫進度條 painter->drawRect( option.rect.x()+2, option.rect.y()+2, int(factor*(option.rect.width()-5)), option.rect.height()-5 ); painter->restore(); // 恢復新畫板(我認為) } QSize BarDelegate::sizeHint( const QStyleOptionViewItem &option, const QModelIndex &index ) const { return QSize( 45, 15 ); // 隨便改,不影響(我認為) }