在带有 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 - 覆盖未调用的函数数据

如何在 QSqlQueryModel 中获取硬编码 QCheckBoxes 的状态?

qt数据库 qsqlquerymodel中setheaderdata怎么设置中文

在 QSqlQueryModel 中交换标题(转置表)

编辑添加的数据到 sublcased QSqlQueryModel?

QlistView setCurrentIndex()不起作用