QT案例 记录解决在管理员权限下QFrame控件获取拖拽到控件上的文件路径

news2025/1/16 15:44:55

参考知乎问答 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 i=0;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-dragqueryfilew
                    UINT 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获取文件路径信息

这有点难以理解,注册后再取消掉,关键还生效了,实现了窗体的鼠标显示拖拽。
后面我改成先RevokeDragDropDragAcceptFiles也实现了鼠标显示拖拽。


  • 在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 i=0;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-dragqueryfilew
                    UINT 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 <QMimeData>

DragDrop_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();
}

值得注意的是:

  1. qDebug() << winId() << this->window()->winId() << this->window()->effectiveWinId();
    先打印一次WId,只要注释这一行代码,后面的DragAcceptFiles,ChangeWindowMessageFilterEx函数都会失败,完全不理解为啥,先记下来。
  2. 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::function<void(const std::vector<std::string>&)> callback);
    ~DropHandler();

    std::function<void(const std::vector<std::string>&)> GetCallback() const { return m_callback; };

private:
    LONG m_cRef;
    std::function<void(const std::vector<std::string>&)> m_callback;
};

drophandler.cpp

#include "drophandler.h"
#include <QDebug>

DropHandler::DropHandler(std::function<void(const std::vector<std::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_cast<IDropTarget *>(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类中的DragEnter,DragLeave等函数…
但是,在没有管理员权限的情况下显然用自带的dragEnterEvent等函数更合适…
所以,继承 IDropTarget 类实现拖拽 没多大用,仅供参考…


测试源码

完整测试源码已上传Github :
https://github.com/MliesMoT/Qt_Administrator_Drop_Text
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1790882.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[SaaS] AI+数据,tiktok选品,找达人,看广告数据

TK观察专访丨前阿里“鲁班”创始人用AIGC赋能TikTok获千万融资用AI数据做TikTokhttps://mp.weixin.qq.com/s/xp5UM3ROo48DK4jS9UBMuQ主要还是爬虫做数据的。 商家做内容&#xff1a;1.找达人拍内容&#xff0c;2.商家自己做原生自制内容&#xff0c;3.广告内容。 短视频&…

俞敏洪一句“乱七八糟”,让东方甄选跌了40亿

“虽然直播的特点就是能说会道&#xff0c;但是网上那种买买买的嚎叫&#xff0c;我是完全看不起的”&#xff0c;俞敏洪在2023亚布力论坛上颇为自豪地说。 在他看来&#xff0c;“直播带货”本质上也是教育的一种&#xff0c;对产品进行知识性讲解才是最核心的一环。而知识传…

Centos 7部署NTP

介绍 NTP是Network Time Protocol&#xff08;网络时间协议&#xff09;的简称&#xff0c;它是用来通过互联网或局域网将计算机时钟同步到世界协调时间&#xff08;UTC&#xff09;的协议。 安装 # yum安装 yum install -y ntp# 离线安装 #下载地址&#xff1a;https://mir…

Meta Llama 3 残差结构

Meta Llama 3 残差结构 flyfish 在Transformer架构中&#xff0c;残差结构&#xff08;Residual Connections&#xff09;是一个关键组件&#xff0c;它在模型的性能和训练稳定性上起到了重要作用。残差结构最早由He et al.在ResNet中提出&#xff0c;并被广泛应用于各种深度…

【python】爬虫记录每小时金价

数据来源&#xff1a; https://www.cngold.org/img_date/ 因为这个网站是数据随时变动的&#xff0c;用requests、BeautifulSoup的方式解析html的话&#xff0c;数据的位置显示的是“--”&#xff0c;并不能取到数据。 所以采用webdriver访问网站&#xff0c;然后从界面上获取…

3389远程连接器,3389远程连接器如何进行远程连接

3389远程连接器是一款专业的远程桌面连接工具&#xff0c;它允许用户通过网络远程访问和控制另一台计算机&#xff0c;实现远程办公、技术支持、文件传输等多种功能。下面将详细介绍如何使用3389远程连接器进行远程连接。 首先&#xff0c;确保被连接的计算机已经开启了远程桌面…

TCP的核心属性

TCP的核心属性 一: TCP的核心属性1.1: 确认应答:1.2 : 超时重传1.3 : 连接管理1.3.1 三次握手1.3.2 四次挥手 1.4 滑动窗口1.5: 流量控制:1.6 拥塞控制1.7 延时应答1.8 :捎带应答1.9: 面向字节流1.10 : 异常情况 一: TCP的核心属性 1.1: 确认应答: 保证可靠性最核心的机制 1…

二刷算法训练营Day22 | 二叉树(8/9)

目录 详细布置&#xff1a; 1. 235. 二叉搜索树的最近公共祖先 2. 701. 二叉搜索树中的插入操作 3. 450. 删除二叉搜索树中的节点 详细布置&#xff1a; 1. 235. 二叉搜索树的最近公共祖先 给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共…

二叉树的算法题目

二叉树的遍历题目 二叉树遍历一般包含三种分别为&#xff1a;根左右、左根右、左右根&#xff08;又称为前序遍历、中序遍历、后序遍历&#xff09; 方法一&#xff1a;使用递归遍历 方法二&#xff1a;使用迭代使用栈 我们以左根右&#xff08;中序遍历&…

修复Windows上“发生意外错误”问题的5种方法,总有一种适合你

在尝试启动网络适配器的设置菜单时,是否收到“发生意外错误”消息?不用担心,因为在大多数情况下解决这个问题很容易。我们将向你展示在Windows 11或Windows 10计算机上解决此问题的多种方法。 为什么我收到“发生意外错误”的消息 当网络适配器出现问题时,Windows会显示一…

MariaDB数据导入与导出操作演示

文章目录 整个数据库导出导入先删除库然后再导入 参考这里&#xff1a; MariaDB数据库导出导入. 整个数据库 该部分演示&#xff1a;导出数据库&#xff0c;然后重建数据库&#xff0c;并导入数据的整个过程。 导出 Win R &#xff0c;打开运行输入cmd并回车&#xff0c;然…

【docker】docker的安装

如果之前安装了旧版本的docker我们需要进行卸载&#xff1a; 卸载之前的旧版本 卸载 # 卸载旧版本 sudo apt-get remove docker docker-engine docker.io containerd runc # 卸载历史版本 apt-get purge docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker…

如何在Weblogic环境中启动认证方式对接Zabbix监控

在WebLogic Server中&#xff0c;启动认证可用于确保只有经过授权的用户和系统能够访问WebLogic Server及其应用程序&#xff0c;通过合理配置认证提供者和安全领域&#xff0c;管理员可以有效管理和控制用户访问。 本文将详细介绍如何在Weblogic环境中配置启动认证并对接Zabb…

植物大战僵尸杂交版2.0.88最新版+防闪退工具V2+修改工具+高清工具

植物大战僵尸杂交版&#xff0c;不仅继承原作的经典玩法&#xff0c;而且引入了全新的植物融合玩法&#xff0c;将各式各样的植物进行巧妙的杂交&#xff0c;孕育出前所未有、功能各异的全新植物。 创新的杂交合成系统 游戏引入了创新的杂交合成系统&#xff0c;让玩家可以将不…

Swift 中的Getter 和 Setter

目录 前言 1. 什么是Getter和Setter 1.定义 2.作用 2.属性 1.存储属性 2.计算属性 3.属性观察者 3. 使用 Getter 和 Setter 的场景 1.数据转换 2.懒加载 3.数据验证和限制 4.触发相关操作 4.自定义Getter 和 Setter 5. 参考资料 前言 属性是 Swift 编程中的基本…

Ubuntu中PDF阅读器和编辑器

1. 福昕PDF编辑器 1.1. 下载地址 PDF阅读器下载_PDF编辑器下载_PDF软件官方下载_福昕软件官网 1.2. 安装 sudo dpkg -i signed_com.foxit.foxitpdfeditor_xxx_amd64_UOS.deb 2. WPS DPF 2.1. 下载地址 WPS Office 2019 for Linux-支持多版本下载_WPS官方网站 2.2. 使用 …

NSS题目练习7

[MoeCTF 2022]baby_file 打开看见一串源代码&#xff0c;需要get传参传入file 题目提示php伪协议 用dirsearch扫描发现flag.php 用php伪协议查看&#xff0c;回显一串base64编码 解码后得到flag [鹤城杯 2021]Middle magic 读取这两个文件 一个php正则表达式 补充&#xff1a…

背包问题(01背包及其优化(滚动数组和逆序枚举))

终于是完结了AC自动机&#xff0c;接下来开个新坑——背包问题&#xff0c;背包的种类还是很多的&#xff0c;之前有学过&#xff0c;但都是这里看一点&#xff0c;那里看一点&#xff0c;导致现在都搞混了&#xff0c;所以重新系统看看这方面的内容。 先从简单的入手——01背包…

如何在 Java 中使用 JOptionPane 显示消息对话框

在 Java 开发中&#xff0c;JOptionPane 是一个非常实用的类&#xff0c;可以用来显示各种类型的对话框&#xff0c;例如信息对话框、警告对话框、错误对话框等。今天&#xff0c;我们将深入探讨如何使用 JOptionPane.showMessageDialog 方法来显示消息对话框&#xff0c;以及如…

面试被问准备多久要孩子?这样回答

听说有人面试被问到多久要孩子的问题&#xff0c;当时觉得很尴尬&#xff0c;不知如何回答&#xff0c;怕回答的不好不被录用&#xff0c;其实你可以这样回答&#xff0c;让面试官心满意足。 A 面试官&#xff1a;结婚了吗&#xff1f; 我&#xff1a;结婚了 面试官&#xff1…