首页 > 解决方案 > 检测所有子 QWidget 上的拖放拖放

问题描述

我正在尝试在整个应用程序中实现拖放行为,我想在发生拖放时执行一些操作,无论在 MainWindow 中的哪个位置。我面临的问题是,在我的 MainWindow 中,我有一个包含 QGraphicsView 的小部件,它又包含一个 QGraphicsScene,我可以通过 EventFilter 检测应用程序中的任何位置,除了在视图和场景中。是否可以实现某种可以从 MainWindow 检测到的全局拖放行为?我试图避免将逻辑传播到所有可见的小部件中。

这是我的 MainWindow 中的拖放逻辑,如前所述,适用于不在视图/场景上发生的所有拖放:

void MainWindow::dropEvent(QDropEvent *event)
{
    auto pixmap = QPixmap(event->mimeData()->urls().first().toString().remove("file://"));
    if(!pixmap.isNull()) {
        load(pixmap);
    }

    event->acceptProposedAction();
}

void MainWindow::dragEnterEvent(QDragEnterEvent *event)
{
    if (event->mimeData()->hasUrls()) {
        event->acceptProposedAction();
    }
}

对于视图/场景上的拖放,我尝试安装以下事件过滤器:

bool MainWindow::eventFilter(QObject *obj, QEvent *event)
{
    if(event->type() == QEvent::DragEnter) {
        qDebug("Enter");
    } else if(event->type() == QEvent::Drop) {
        qDebug("Drop");
    } else if(event->type() == QEvent::GraphicsSceneDrop)   {
        qDebug("Scene drop");
    } else if (event->type() >= 159 && event->type() <= 168) {
        qDebug("Any Scene Event");
    }
    return QObject::eventFilter(obj, event);
}

并将其应用于 MainWindow ctor,我检测到的唯一内容是进入视图时的输入。

...
setAcceptDrops(true);
mChildWidget->installEventFilter(this);
mChildWidget->setAcceptDrops(true);
...

标签: qtdrag-and-dropqgraphicsviewqgraphicssceneqevent

解决方案


看起来我在这里面临两个问题。第一个在此问题中解释为Accepting drop on a QGraphicsScene。QGraphicsScene 会忽略没有在项目上发生的 DragAndDrops。基本上,您可以启用拖放,但只能单独为每个项目启用,这仍然会留下项目未覆盖的区域。这个问题的解决方案似乎是覆盖dragMoveEvent

void MyQGprahicsScene::dragMoveEvent(QGraphicsSceneDragDropEvent *event)
{
    Q_UNUSED(event)

    // Overriding default dragMoveEvent in order to accept drops without items under them
}

我面临的第二个问题是我试图将 eventFilter 应用于 Child 小部件,但看起来我必须将它应用于 qApp,之后一切似乎都正常工作,可能是因为 qApp 是所有事件的顶级 QObject不处理时会降落。

#include "DragAndDropHandler.h"

bool DragAndDropHandler::eventFilter(QObject *obj, QEvent *event)
{
    if(event->type() == QEvent::DragEnter) {
        handleDragEnter(dynamic_cast<QDragEnterEvent *>(event));
    } else if(event->type() == QEvent::Drop) {
        handleDrop(dynamic_cast<QDropEvent *>(event));
    } else if(event->type() == QEvent::GraphicsSceneDrop) {
        handleDrop(dynamic_cast<QGraphicsSceneDragDropEvent *>(event));
    } else if (event->type() ==  QEvent::GraphicsSceneDragEnter) {
        handleDragEnter(dynamic_cast<QGraphicsSceneDragDropEvent *>(event));
    }
    return QObject::eventFilter(obj, event);
}

void DragAndDropHandler::handleDragEnter(QDragEnterEvent *event)
{
    if (event->mimeData()->hasUrls()) {
        event->acceptProposedAction();
    }
}

void DragAndDropHandler::handleDragEnter(QGraphicsSceneDragDropEvent *event)
{
    if (event->mimeData()->hasUrls()) {
        event->acceptProposedAction();
    }
}

void DragAndDropHandler::handleDrop(QDropEvent *event)
{
    auto path = getUrlFromMimeData(event->mimeData());
    event->acceptProposedAction();
    emit imageDropped(path);
}

void DragAndDropHandler::handleDrop(QGraphicsSceneDragDropEvent *event)
{
    auto path = getUrlFromMimeData(event->mimeData());
    event->acceptProposedAction();
    emit imageDropped(path);
}

QString DragAndDropHandler::getUrlFromMimeData(const QMimeData *mimeData) const
{
    return mimeData->urls().first().toString().remove("file://");
}

而且,在我的 MainWindow 的构造函数中:

setAcceptDrops(true);
qApp->installEventFilter(mDragAndDropHandler);
connect(mDragAndDropHandler, &DragAndDropHandler::imageDropped, this, &MainWindow::loadImageFromFile);

推荐阅读