公司网站内容如何做,网站中队人物介绍怎么做,网站制作软件免费下载,网站建设的培训心得参考知乎问答 Qt管理员权限如何支持拖放操作#xff1f; 的回答和代码示例。 解决在管理员权限运行下#xff0c;通过窗体的QFrame子控件获取到拖拽的内容。 目录标题 导读解决方案详解示例详细 【管理员权限】在QFrame控件中获取拖拽内容 【管理员权限】继承 IDropTarget 类… 参考知乎问答 Qt管理员权限如何支持拖放操作 的回答和代码示例。 解决在管理员权限运行下通过窗体的QFrame子控件获取到拖拽的内容。 目录标题 导读解决方案详解示例详细 【管理员权限】在QFrame控件中获取拖拽内容 【管理员权限】继承 IDropTarget 类实现拖拽 【管理员权限下无效】 测试源码 导读
在QT 程序中非管理员运行的软件正常的拖拽功能实现 只需要重写drag 拖拽事件并且设置 setAcceptDrops(true); 就正常实现拖拽相关功能。
void dragEnterEvent(QDragEnterEvent *event) override;
void dragLeaveEvent(QDragLeaveEvent *event) override;
void dragMoveEvent(QDragMoveEvent *event) override;
void dropEvent(QDropEvent *event) override;但是程序一旦设置了管理员权限启动就获取不到拖拽事件。 这一点在 知乎问答Qt管理员权限如何支持拖放操作 有相关说明。 而我也对 qt_uac_drag_demo 示例进行了简单的测试。 因为我直接使用的Msvc2017编译器所以直接添加头文件
#include Windows.h
#include WinUser.h
#include ole2.h
#include shellapi.h
#pragma comment(lib,user32.lib)
#pragma comment(lib,Ole32.lib)然后直接调用 ChangeWindowMessageFilterExDragAcceptFilesRevokeDragDrop等函数
解决方案详解 示例详细 【管理员权限】
通过加载 qt_uac_drag_demo 项目可以了解到示例主要 先通过DragAcceptFiles注册窗口接收拖拽事件RevokeDragDrop取消掉注册已注册事件
void* user32 LoadLibraryA(user32);
FARPROC func GetProcAddress((HMODULE)user32, ChangeWindowMessageFilter);
qDebug() (*func)();
user32 LoadLibraryA(user32);
func GetProcAddress((HMODULE)user32, ChangeWindowMessageFilter);
qDebug() (*func)();
ChangeWindowMessageFilter(WM_DROPFILES, 1);
// ChangeWindowMessageFilter(WM_COPYDATA, 1);
// ChangeWindowMessageFilter(0x0049, 1);
qDebug() w.winId() w.effectiveWinId();
qDebug() ChangeWindowMessageFilterEx((HWND)w.effectiveWinId(), WM_DROPFILES, MSGFLT_ALLOW, NULL);
qDebug() ChangeWindowMessageFilterEx((HWND)w.effectiveWinId(), WM_COPYDATA, MSGFLT_ALLOW, NULL);
qDebug() ChangeWindowMessageFilterEx((HWND)w.effectiveWinId(), 0x0049, MSGFLT_ALLOW, NULL);
DragAcceptFiles((HWND)w.effectiveWinId(), true);
qDebug() GetLastError();
HRESULT res RevokeDragDrop((HWND)w.winId());
qDebug() res: res;再重构的 bool nativeEvent(const QByteArray eventType, void *message, long *result) 函数 然后通过 DragQueryFileW 获取文件/文件夹路径, bool nativeEvent(const QByteArray eventType, void *message, long *result){if (eventType windows_generic_MSG){PMSG msg (PMSG) message;if(msg-message 563){qDebug() msg-message msg-hwnd msg-wParam msg-lParam msg-time msg-pt.x msg-pt.y;UINT file_num DragQueryFile((HDROP) msg-wParam, 0xFFFFFFFF, NULL, 0);qDebug() 文件数量: file_num;for(int i0;i(int)file_num;i){UINT file_name_size DragQueryFile((HDROP) msg-wParam, i, NULL, 0);qDebug() file_name_size;LPWSTR fn (LPWSTR)malloc(sizeof(WCHAR)*file_name_size);//! https://learn.microsoft.com/zh-cn/windows/win32/api/shellapi/nf-shellapi-dragqueryfilewUINT code DragQueryFileW((HDROP) msg-wParam, i, fn, file_name_size);QString filename QString::fromStdWString(fn);free(fn);qDebug() 第 i 个文件: filename;qDebug() get name error: code;}qDebug() eventType message *result;}}return QFrame::nativeEvent(eventType, message, result);}一开始我是没有理解为什么要先调用 DragAcceptFiles((HWND)w.effectiveWinId(), true); 再调用 HRESULT res RevokeDragDrop((HWND)w.winId()); 直到我看到 评论下方的回答 以此看来如果在Qt中简单的使用ChangeWindowMessageFilter拖入文件时还是会显示禁止的标志我参考了下其他非Qt的代码。 具体实现形式差不多是 1.是先使用RevokeDragDrop取消掉注册 2.DragAcceptFiles注册窗口接收拖拽事件 3.SetWindowLongA()设置窗口及事件回调函数 4.在事件回调中处理563消息WM_DROPFILES消息。 5.再使用DragQueryFileA获取文件路径信息 这有点难以理解注册后再取消掉关键还生效了实现了窗体的鼠标显示拖拽。 后面我改成先RevokeDragDrop 再 DragAcceptFiles也实现了鼠标显示拖拽。 在QFrame控件中获取拖拽内容 【管理员权限】
示例是在 MainWindow 窗体中获取的拖拽内容而我需要限制某一个控件获取拖拽的内容所以我在MainWindow 窗体中添加了一个QFrame 控件 像这样 这就需要重写QFrame 类 DragDrop_Frame.h nativeEvent 函数和示例中一样没有改动
class DragDrop_Frame:public QFrame
{
public:DragDrop_Frame(QWidget* parent nullptr, Qt::WindowFlags f Qt::WindowFlags());void dragEnterEvent(QDragEnterEvent *e) ;bool nativeEvent(const QByteArray eventType, void *message, long *result){if (eventType windows_generic_MSG){PMSG msg (PMSG) message;if(msg-message 563){qDebug() msg-message msg-hwnd msg-wParam msg-lParam msg-time msg-pt.x msg-pt.y;UINT file_num DragQueryFile((HDROP) msg-wParam, 0xFFFFFFFF, NULL, 0);qDebug() 文件数量: file_num;for(int i0;i(int)file_num;i){UINT file_name_size DragQueryFile((HDROP) msg-wParam, i, NULL, 0);qDebug() file_name_size;LPWSTR fn (LPWSTR)malloc(sizeof(WCHAR)*file_name_size);//! https://learn.microsoft.com/zh-cn/windows/win32/api/shellapi/nf-shellapi-dragqueryfilewUINT code DragQueryFileW((HDROP) msg-wParam, i, fn, file_name_size);QString filename QString::fromStdWString(fn);free(fn);qDebug() 第 i 个文件: filename;qDebug() get name error: code;}qDebug() eventType message *result;}}return QFrame::nativeEvent(eventType, message, result);}
};
DragDrop_Frame.cpp 为了让QFrame 能获取获取到拖拽事件必须移除父窗体的注册在重新添加父窗体和QFrame窗体的注册。再通过ChangeWindowMessageFilterEx 修改指定窗口 (UIPI) 消息筛选器的用户界面特权隔离。
#include dragdrop_frame.h
#include QDebug
#include drophandler.h
#include QDragEnterEvent
#include QMimeDataDragDrop_Frame::DragDrop_Frame(QWidget* parent, Qt::WindowFlags f ):QFrame(parent,f)
{this-setAcceptDrops(true);setStyleSheet(QFrame { border-radius: 24px;border: 2px dashed #676E89;background-image: url(:/tuozhuai.png);background-position: center;background-repeat: no-repeat;background-origin: content; });//! //1.是先使用RevokeDragDrop取消掉注册qDebug() winId() this-window()-winId() this-window()-effectiveWinId();qDebug() RevokeDragDrop((HWND)this-window()-effectiveWinId());//2.DragAcceptFiles注册窗口接收拖拽事件//--3.SetWindowLongA()设置窗口及事件回调函数-- //不需要DragAcceptFiles((HWND)this-window()-effectiveWinId(), true);DragAcceptFiles((HWND)winId(), true);//在 nativeEvent(const QByteArray eventType, void *message, long *result) 事件中实现//4.在事件回调中处理563消息WM_DROPFILES消息。//5.再使用DragQueryFileA获取文件路径信息qDebug() ChangeWindowMessageFilterEx((HWND)winId(), WM_DROPFILES, MSGFLT_ALLOW, NULL);qDebug() ChangeWindowMessageFilterEx((HWND)winId(), WM_COPYDATA, MSGFLT_ALLOW, NULL);qDebug() ChangeWindowMessageFilterEx((HWND)winId(), 0x0049, MSGFLT_ALLOW, NULL);
}void DragDrop_Frame::dragEnterEvent(QDragEnterEvent *e){qDebug()[mimeData] e-mimeData()-urls().count();
}
值得注意的是
qDebug() winId() this-window()-winId() this-window()-effectiveWinId(); 先打印一次WId只要注释这一行代码后面的DragAcceptFilesChangeWindowMessageFilterEx函数都会失败完全不理解为啥先记下来。RevokeDragDrop((HWND)this-window()-effectiveWinId()); 一定要注销窗体拖拽事件否则获取拖拽事件失败。 继承 IDropTarget 类实现拖拽 【管理员权限下无效】
在测试使用RevokeDragDrop 函数的时候我找到了 RegisterDragDrop 函数 我就想着注销掉系统默认的拖拽事件在重新添加一个拖拽事件是不是就能解决掉被屏蔽的拖拽事件 于是我从Giehub上借鉴了一份 IDropTarget 类的实现 drophandler.h #include Windows.h
#include WinUser.h
#include ole2.h
#include shellapi.h
#include functional
#include vector#pragma comment(lib,user32.lib)
#pragma comment(lib,Ole32.lib)/*!
* 从github 移植的阉割版的 IDropTarget 实现
* https://github.com/WinMerge/winmerge/blob/f62b2e5b8b3e9d415045426e276dfd4d2ed271e6/Src/DropHandler.h
*/class DropHandler: public IDropTarget
{
public:HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void **ppvObject);ULONG STDMETHODCALLTYPE AddRef();ULONG STDMETHODCALLTYPE Release();HRESULT STDMETHODCALLTYPE DragEnter(IDataObject *pDataObj, DWORD grfKeyState, POINTL pt, DWORD *pdwEffect);HRESULT STDMETHODCALLTYPE DragOver(DWORD grfKeyState, POINTL pt, DWORD *pdwEffect);HRESULT STDMETHODCALLTYPE DragLeave(void);HRESULT STDMETHODCALLTYPE Drop(IDataObject *pDataObj, DWORD grfKeyState, POINTL pt, DWORD *pdwEffect);explicit DropHandler(std::functionvoid(const std::vectorstd::string) callback);~DropHandler();std::functionvoid(const std::vectorstd::string) GetCallback() const { return m_callback; };private:LONG m_cRef;std::functionvoid(const std::vectorstd::string) m_callback;
};drophandler.cpp
#include drophandler.h
#include QDebugDropHandler::DropHandler(std::functionvoid(const std::vectorstd::string) callback): m_cRef(0), m_callback(callback)
{qDebug()[DropHandler] --;
}DropHandler::~DropHandler(){}HRESULT STDMETHODCALLTYPE DropHandler::QueryInterface(REFIID riid, void **ppvObject)
{qDebug()[QueryInterface] --;if (!IsEqualIID(riid, IID_IUnknown) !IsEqualIID(riid, IID_IDropTarget)){*ppvObject nullptr;return E_NOINTERFACE;}*ppvObject static_castIDropTarget *(this);AddRef();return S_OK;
}ULONG STDMETHODCALLTYPE DropHandler::AddRef(void)
{qDebug()[AddRef] --;return InterlockedIncrement(m_cRef);
}ULONG STDMETHODCALLTYPE DropHandler::Release(void)
{qDebug()[Release] --;ULONG cRef InterlockedDecrement(m_cRef);if (cRef 0) {delete this;return 0;}return cRef;
}HRESULT STDMETHODCALLTYPE DropHandler::DragEnter(IDataObject* pDataObj, DWORD grfKeyState, POINTL pt, DWORD* pdwEffect)
{qDebug()[DragEnter] --;return S_OK;
}HRESULT STDMETHODCALLTYPE DropHandler::DragOver(DWORD, POINTL, DWORD *)
{qDebug()[DragOver] --;return S_OK;
}HRESULT STDMETHODCALLTYPE DropHandler::DragLeave(void)
{qDebug()[DragLeave] --;return S_OK;
}HRESULT DropHandler::Drop(IDataObject* pDataObj, DWORD grfKeyState, POINTL pt, DWORD* pdwEffect)
{qDebug()[Drop] --;return S_OK;
}
在通过RegisterDragDrop 注册 qDebug() winId() this-window()-winId() this-window()-effectiveWinId();HRESULT res RevokeDragDrop((HWND)this-window()-effectiveWinId());DragAcceptFiles((HWND)this-window()-effectiveWinId(), true);DragAcceptFiles((HWND)winId(), true);qDebug() RegisterDragDrop((HWND)winId(),new DropHandler(NULL));结果发现父窗体可以拖拽QFrame控件反而被禁止拖拽了、、、、 而在没有管理员权限的情况下直接注册 RegisterDragDrop((HWND)winId(),new DropHandler(NULL)); 就能在拖拽事件中执行在DropHandler类中的DragEnterDragLeave等函数… 但是在没有管理员权限的情况下显然用自带的dragEnterEvent等函数更合适… 所以继承 IDropTarget 类实现拖拽 没多大用仅供参考… 测试源码
完整测试源码已上传Github : https://github.com/MliesMoT/Qt_Administrator_Drop_Text