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

Ano*_*non 3 c++ qt drag-and-drop qtablewidget qt5

我使用 QListWidget 通过使用 setItemWidget 设置自定义小部件来显示它们。像这样的东西:

QListWidget* listWidget = new QListWidget;
listWidget->setAcceptDrops(true);
listWidget->setDragDropMode(QAbstractItemView::InternalMove);
listWidget->setDragEnabled(true);
listWidget->setSelectionMode(QAbstractItemView::SingleSelection);

for ( int i = 0 ; i < 50 ; ++i )
{
    ItemWidget* item = new ItemWidget;
    QListWidgetItem* listItem = new QListWidgetItem;
    listItem->setSizeHint(item->sizeHint());
    listWidget->addItem(listItem);
    listWidget->setItemWidget(listItem, item);
}
Run Code Online (Sandbox Code Playgroud)

ItemWidget 源自 QWidget,仅在布局中显示一些自定义数据,如下所示:

ItemWidget::ItemWidget()
{
    QVBoxLayout* layout = new QVBoxLayout;
    layout->setContentsMargins(0, 0, 0, 0);
    layout->setSpacing(0);

    QHBoxLayout* contentLayout = new QHBoxLayout;
    contentLayout->setSizeConstraint(QLayout::SetFixedSize);
    contentLayout->setSpacing(0);

    contentLayout->addSpacing(5);
    contentLayout->setContentsMargins(10, 20, 10, 20);

    QLabel* iconLbl = new QLabel;
    iconLbl->setPixmap(QPixmap(":/icon.png"));
    iconLbl->setMaximumWidth(20);
    contentLayout->addWidget(iconLbl, 0, Qt::AlignTop);

    contentLayout->addSpacing(14);

    QVBoxLayout* infoLayout = new QVBoxLayout;
    infoLayout->setContentsMargins(0, 0, 0, 0);
    infoLayout->setSpacing(0);

    QLabel* firstLbl = new QLabel("First line of text");
    infoLayout->addWidget(firstLbl);

    infoLayout->addSpacing(4);

    QLabel* secondLbl = new QLabel("Second line of text");
    infoLayout->addWidget(secondLbl);

    contentLayout->addLayout(infoLayout);

    layout->addLayout(contentLayout);

    setLayout(layout);
}
Run Code Online (Sandbox Code Playgroud)

我想实现拖放以便能够重新排列列表中的项目。但是,当使用 setItemWidget 时,当鼠标拖动该项目时,只会拖动背景矩形(QListWidgetItem?),而不会显示属于 ItemWidget 的任何自定义内容。我希望拖动的项目也包含 ItemWidget 内容,以便用户看到正在拖放的内容。

有人有实现这个的工作方法吗?

我已经尝试使用从 QListWidgetItem 和 QWidget 派生的自定义类,并直接在该类中设置自定义布局,因此可能不需要 ItemWidget 或使用 setItemWidget,但它没有像我希望的那样工作。

eyl*_*esc 5

要自定义QPixmapQDragof相关的方法QListWidget,我们必须重写该startDrag()方法。

主要任务是获取QPixmap为其创建的所选元素中的一个QPixmap,该元素的大小为透明的可见图像viewport(),然后我们QPixmap使用为它选择的每个项目来绘制它们QPainter

要获取QPixmap每一项的 ,请使用grab()方法,表示通过 获取的矩形visualRect()

#ifndef LISTWIDGET_H
#define LISTWIDGET_H

#include <QListWidget>
#include <QDrag>
#include <QMimeData>
#include <QPainter>

class ListWidget : public QListWidget
{
protected:
    void startDrag(Qt::DropActions supportedActions){
        QDrag *drag = new QDrag(this);
        drag->setMimeData(model()->mimeData(selectedIndexes()));
        QPixmap pixmap(viewport()->visibleRegion().boundingRect().size());
        pixmap.fill(Qt::transparent);
        QPainter painter(&pixmap);
        for(QModelIndex index: selectedIndexes()){
            painter.drawPixmap(visualRect(index), viewport()->grab(visualRect(index)));
        }
        drag->setPixmap(pixmap);
        drag->setHotSpot(viewport()->mapFromGlobal(QCursor::pos()));
        drag->exec(supportedActions, Qt::MoveAction);
    }
};

#endif // LISTWIDGET_H
Run Code Online (Sandbox Code Playgroud)

完整的示例可以在以下链接 中找到