QDrag 在拖动时被破坏

Kan*_*noN 5 c++ com qt drag-and-drop

我有一个 Windows/Linux Qt 4.3 应用程序,它在 QTreeView 中使用拖放操作。我有两个非常相似的应用程序,它们使用同一组 Qt 库。拖放功能在 Linux 上都适用,但在 Windows 上仅适用于其中一种。

在不起作用的应用程序中,一旦移动鼠标,QDrag 对象就会被删除。它由事件队列中的 DeferredDelete 事件删除,该事件队列在拖动期间仍在 Qt 中处理。我不知道如何查看导致 QDrag 对象过早删除的原因。

我想不出调试这个问题的好方法。我比较了来源,找不到任何明显的东西。我尝试在另一个应用程序中使用其中一个应用程序的代码。

有什么建议么?

更新:

QDrag 操作失败的原因是 COM 未成功初始化,因此 QDrag::exec 中对 DoDragDrop 的调用立即返回。QApplication 尝试通过在 qt_init 中调用 OleInitialize 来初始化 COM,但失败并出现错误“设置后无法更改线程模式”。

有趣的是,即使 OleInitialize 是 main 中完成的第一件事,也会发生这种情况,因此线程模式最初是由某些外部依赖项设置的。在 Windows 上运行的应用程序之间的差异之一是失败的应用程序还包含 .NET 代码,因此这可能就是问题所在。

解决了:

此问题是 COM/CLR 互操作问题。CLR 在初始化时将单元状态设置为 MTA,然后当 Qt 尝试初始化 COM 时会失败。Adam NathanGotcha with STAThreadAttribute and Managed C++中讨论了这个问题和旧的解决方案。在 Visual Studio 2005 中,您可以在“配置属性”>“链接器”>“高级”中设置 /CLRTHREADATTRIBUTE:STA 编译器选项,将线程属性设置为 STA,而无需创建新的入口点。

小智 2

我不知道是什么会导致这个问题,但我会尝试通过子类化 QDrag 来找出答案,覆盖 deleteLater() (好吧,重新实现它,但因为它是一个插槽,无论如何它都会被调用),使用它而不是 QDrag 和在deleteLater() 中放置一个断点。