在带有 QSqlQueryModel 的 QListView 上使用 QStyledItemDelegate
Posted
技术标签:
【中文标题】在带有 QSqlQueryModel 的 QListView 上使用 QStyledItemDelegate【英文标题】:Using a QStyledItemDelegate on a QListView with QSqlQueryModel 【发布时间】:2011-03-17 03:34:26 【问题描述】:我有一个 QListView,它有一个 QSqlQueryModel 设置为其模型。如何使用 QStyledItemDelegate 来自定义 QListView 的行的外观(例如显示 2 个文本行)?
QSqlDatabase db = QSqlDatabase::addDatabase( "QSQLITE" );
db.setDatabaseName( "test.db" );
if( !db.open() )
qDebug() << db.lastError();
qFatal( "Failed to connect." );
qDebug( "Connected!" );
QSqlQueryModel *sqlModel = new QSqlQueryModel;
sqlModel->setQuery("SELECT * FROM entries");
mListWidget->setModel(sqlModel);
基本上,我认为我需要做的是以某种方式将角色“匹配”到 db 表的字段,以便能够使用以下方式从 QStyledItemDelegate 获取数据:
void ListViewDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
[...]
QString headerText = qvariant_cast<QString>(index.data(headerRole));
QString subText = qvariant_cast<QString>(index.data(subHeaderRole));
[...]
谢谢!
【问题讨论】:
【参考方案1】:您当然可以使用 QStyledItemDelegate 进行自定义项目绘制。 QModelIndex 具有对模型对象的引用,您可以使用它来获取“条目”记录字段。如果您需要显示更多数据而不是单个数据,您还必须重新定义模型的 sizeHint 方法以增加项目大小。除此之外,它或多或少是微不足道的。
请看下面的例子是否对你有用:
class ListViewDelegate : public QStyledItemDelegate
protected:
void paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const
QStyleOptionViewItemV4 opt = option;
initStyleOption(&opt, index);
QString line0 = index.model()->data(index.model()->index(index.row(), 1)).toString();
QString line1 = index.model()->data(index.model()->index(index.row(), 2)).toString();
// draw correct background
opt.text = "";
QStyle *style = opt.widget ? opt.widget->style() : QApplication::style();
style->drawControl(QStyle::CE_ItemViewItem, &opt, painter, opt.widget);
QRect rect = opt.rect;
QPalette::ColorGroup cg = opt.state & QStyle::State_Enabled ? QPalette::Normal : QPalette::Disabled;
if (cg == QPalette::Normal && !(opt.state & QStyle::State_Active))
cg = QPalette::Inactive;
// set pen color
if (opt.state & QStyle::State_Selected)
painter->setPen(opt.palette.color(cg, QPalette::HighlightedText));
else
painter->setPen(opt.palette.color(cg, QPalette::Text));
// draw 2 lines of text
painter->drawText(QRect(rect.left(), rect.top(), rect.width(), rect.height()/2),
opt.displayAlignment, line0);
painter->drawText(QRect(rect.left(), rect.top()+rect.height()/2, rect.width(), rect.height()/2),
opt.displayAlignment, line1);
QSize sizeHint(const QStyleOptionViewItem & option, const QModelIndex & index ) const
QSize result = QStyledItemDelegate::sizeHint(option, index);
result.setHeight(result.height()*2);
return result;
;
这里定义了测试数据库集:
QSqlError initDb()
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
db.setDatabaseName(":memory:");
if (!db.open())
return db.lastError();
QStringList tables = db.tables();
if (tables.contains("test", Qt::CaseInsensitive))
return QSqlError();
QSqlQuery q;
if (!q.exec(QLatin1String("create table entries(id integer primary key, first_line varchar, second_line varchar)")))
return q.lastError();
q.exec("insert into entries(id, first_line, second_line) values(0, 'first line 0', 'second line 0')");
q.exec("insert into entries(id, first_line, second_line) values(1, 'first line 1', 'second line 1')");
q.exec("insert into entries(id, first_line, second_line) values(2, 'first line 2', 'second line 2')");
return QSqlError();
模型和列表视图定义:
initDb();
QSqlQueryModel *sqlModel = new QSqlQueryModel();
sqlModel->setQuery("SELECT * FROM entries");
ui->listView->setModel(sqlModel);
ui->listView->setItemDelegate(new ListViewDelegate());
希望这会有所帮助,问候
【讨论】:
谢谢!您能否给我一个简短的示例,说明如何在单元格中绘制 QIcon? 为你的图标创建一个像素图:icon.pixmap(...);然后使用painter->drawPixmap(pos, pixmap) 来绘制它以上是关于在带有 QSqlQueryModel 的 QListView 上使用 QStyledItemDelegate的主要内容,如果未能解决你的问题,请参考以下文章
如何在 QSqlQueryModel 中获取硬编码 QCheckBoxes 的状态?
qt数据库 qsqlquerymodel中setheaderdata怎么设置中文