使用要存储的额外数据自定义 QListWidgetItem,如何?

Posted

技术标签:

【中文标题】使用要存储的额外数据自定义 QListWidgetItem,如何?【英文标题】:Customize QListWidgetItem with extra data to store in, How? 【发布时间】:2011-05-03 10:49:38 【问题描述】:

QListWidgetItem 包含 2 个数据:图标和文本。我想在其中存储另一个 QString 。我能怎么做?这是我的测试代码。调用 addItem 后,ListWidget 什么也不显示。

我怎么知道点击了哪个项目? SLOT 函数是“void on_listWidget_itemClicked(QListWidgetItem* item)”。显然参数项是父类:QListWidgetItem,而不是子类:ListWidgetItem

ListWidgetItem::ListWidgetItem(const QIcon &icon, const QString &text,QString &ip, QListWidget *parent, int type)

    myip = ip;
    QListWidgetItem::QListWidgetItem(icon,text,parent,type);


ListWidgetItem::~ListWidgetItem()




QVariant ListWidgetItem::data(int role) const

    if (role==IPROLE)
    
        return myip;
    
    return QListWidgetItem::data(role);


void ListWidgetItem::setData(int role, const QVariant &value)

    if (role==IPROLE)
    
        myip = value.toString();
    
    QListWidgetItem::setData(role,value);

【问题讨论】:

【参考方案1】:

QListWidgetItem 有成员函数

void QListWidgetItem::setData ( int role, const QVariant & value )

QVariant QListWidgetItem::data ( int role ) const

用于存储任意数据(包括QString)。设置role = Qt::UserRole(或任何更高的值)。

【讨论】:

【参考方案2】:

为了在类层次结构中上下移动,您可以使用强制转换,在本例中为 dynamic_cast:

dynamic_cast<QListWidgetItem*>( your_item )

您可以将其用作信号和插槽的参数。这将告诉接收函数“嘿,这曾经是一个 QListWidgetItem 上的指针,所以这样对待它!”

为了显示更多信息,您应该考虑改用 QTableWidget。我认为这符合您的要求,您不必像以前那样在“走私”信息上绕道;-)

编辑:

作为对您评论的回应:

看看QTableWidgetItem。你会看到,一个 QTableWidgetItem 也可以有一个图标 :-) 所以你需要做的是将你的信息分成几个 QTableWidgetItem。您有一个项目作为图标,on 用于文本,另一个项目用于另一个文本和您需要的内容。

现在每个 都是您之前在 ListWidget 中的一个项目。现在只要用户点击某处,您就会使用信号cellClicked( int row, int column )row 告诉您点击了哪一行。现在您有了被点击的行并且可以得到您正在寻找的信息。

假设用户点击了第 4 行。每行都有列。第一列包含图标,第二列包含名称。现在特别之处:第三列包含 IP,但 隐藏。你可以这样做,使用hideColumn( int column )

第 4 行:[ 第 0 列:图标 |第 1 列:名称 |第 2 列:(隐藏)IP]

因此,每次用户单击一行时,您只需在第 2 列中查找 IP。

当然,您可以根据需要添加任意数量的单元格...一个包含带有名称的字符串,另一个包含描述...我们只是说,最后一列是带有 IP 的隐藏列。

您还可以使用setShowGrid( false ) 使网格不可见。现在它看起来像一个正常的信息显示,每一行都是一个项目。

:-)

编辑 2:

为了向您展示我的意思,我实现了一个小示例。请参阅下面的代码。

m_table_widget = new QTableWidget( 4, 4, this );

QTableWidgetItem* my_item_0 = new QTableWidgetItem( "icon 1" );
QTableWidgetItem* my_item_1 = new QTableWidgetItem( "server 1" );
QTableWidgetItem* my_item_2 = new QTableWidgetItem( "this is server 1" );
QTableWidgetItem* my_item_3 = new QTableWidgetItem( "192.0.0.1" );

QTableWidgetItem* my_item_4 = new QTableWidgetItem( "icon 2" );
QTableWidgetItem* my_item_5 = new QTableWidgetItem( "server 2" );
QTableWidgetItem* my_item_6 = new QTableWidgetItem( "this is server 2" );
QTableWidgetItem* my_item_7 = new QTableWidgetItem( "192.0.0.2" );

m_table_widget->setItem( 0, 0, my_item_0 );
m_table_widget->setItem( 0, 1, my_item_1 );
m_table_widget->setItem( 0, 2, my_item_2 );
m_table_widget->setItem( 0, 3, my_item_3 );

m_table_widget->setItem( 1, 0, my_item_4 );
m_table_widget->setItem( 1, 1, my_item_5 );
m_table_widget->setItem( 1, 2, my_item_6 );
m_table_widget->setItem( 1, 3, my_item_7 );

m_table_widget->hideColumn( 3 );
m_table_widget->setShowGrid( false );
m_table_widget->setSelectionBehavior( QAbstractItemView::SelectRows );
m_table_widget->verticalHeader()->hide();

QStringList list;
list << "Icon" << "Name" << "Description" << "IP";
m_table_widget->setHorizontalHeaderLabels( list );

connect( m_table_widget, SIGNAL(cellClicked(int,int)), this, SLOT(on_cell_clicked(int,int)) );

现在我们有一个没有网格的表格,带有水平标题(当然你不需要)。单击一行时,会选择整行并调用以下插槽:

void main_window::on_cell_clicked( int row, int column )

    QString ip = m_table_widget->item( row, 3 )->text();
    QMessageBox message( QMessageBox::Information, "Server Info", QString( "This is row %1 and the IP is: %2" ).arg( row ).arg( ip ) );

    message.exec();

这只是我快速编写的一个简单示例...当然有更好的方法,但也许它有助于解决您的问题。

【讨论】:

谢谢。但是 QTableWidget 不支持 IconMode。我想在下面显示带有名称的图标,单击时,应用程序可以获得项目信息,在我的情况下,IP。

以上是关于使用要存储的额外数据自定义 QListWidgetItem,如何?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用自定义项目小部件拖放 QListWidget 项目?

QListWidget 拖放与通过 setItemWidget 设置的自定义小部件

Qt - 如何从 QListWidget 中获取选定的自定义项目?

将自定义 QWidget 添加到 QListWidget

PyQt5 QListWidget自定义项目[重复]

在 QListWidget 之间拖放自定义小部件项目