C/C++ Windows API——获取鼠标位置的窗口句柄_根据坐标获取右击窗口句柄 c++-程序员宅基地

技术标签: # Windows API  

// SpyDemo.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <Windows.h>

#define MAX_TEXT_LEN 255

BOOL CALLBACK EnumChildProcess(HWND hwnd, LPARAM lParam)
{
    if (hwnd == NULL) {
        return FALSE;
    }
    BOOL ret;
    RECT rect;
    ret = GetWindowRect(hwnd, &rect);
    if (!ret) {
        printf("GetWindowRect hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
    }
    else {
        //printf("GetWindowRect hwnd = %p -> rect=(left=%ld, top=%ld, right=%ld, bottom=%ld)\n", hwnd, rect.left, rect.top, rect.right, rect.bottom);
        ret = PtInRect(&rect, *(POINT *)lParam);
        if (ret) {
            printf("GetWindowRect hwnd = %p -> rect=(left=%ld, top=%ld, right=%ld, bottom=%ld)\n", hwnd, rect.left, rect.top, rect.right, rect.bottom);
            //printf("PtInRect\n");

            /*
            WINUSERAPI int WINAPI GetWindowText(
            _In_ HWND hWnd,
            _Out_writes_(nMaxCount) LPTSTR lpString,    //可能是标题名或者file:///打头的文件完整路径
            _In_ int nMaxCount
            );
            如果函数成功,返回值是拷贝的字符串的字符个数,不包括中断的空字符;如果窗口无标题栏或文本,或标题栏为空,或窗口或控制的句柄无效,则返回值为零。若想获得更多错误信息,请调用GetLastError函数。
            */
            TCHAR windowText[MAX_TEXT_LEN];
            int lenRet = GetWindowText(hwnd, windowText, MAX_TEXT_LEN);
            if (lenRet == 0 && GetLastError() != 0) {
                //GetLastError()〖0〗-操作成功完成
                printf("GetWindowText hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
            }
            else {
                _tprintf(_T("GetWindowText hwnd=%p -> windowText=%s, lenRet=%d\n"), hwnd, windowText, lenRet);
            }

            /*
            WINUSERAPI int WINAPI GetClassNameW(
                _In_ HWND hWnd,
                _Out_writes_to_(nMaxCount, return) LPWSTR lpClassName,
                _In_ int nMaxCount
            );

            如果函数成功,返回值为拷贝到指定缓冲区的字符个数:如果函数失败,返回值为0。若想获得更多错误信息,请调用GetLastError函数。
            */
            TCHAR className[MAX_TEXT_LEN];
            lenRet = GetClassName(hwnd, className, MAX_TEXT_LEN);
            if (lenRet == 0) {
                printf("GetClassName hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
            }
            else {
                _tprintf(_T("GetClassName hwnd=%p -> className=%s, lenRet=%d\n"), hwnd, className, lenRet);
            }

            /*
            找出某个窗口的创建者(线程或进程),返回创建者的标志符
            哪个线程创建了这个窗口,返回的就是这个线程的id号 (进程只有一个线程的话,那么线程标志符与进程标志符就是指同一个标志符)
            WINUSERAPI DWORD WINAPI GetWindowThreadProcessId(
                _In_ HWND hWnd,
                _Out_opt_ LPDWORD lpdwProcessId //进程号的存放地址(变量地址)
            );
            返回线程号
            */
            DWORD dwProcessId;
            DWORD dwThreadId = GetWindowThreadProcessId(hwnd, &dwProcessId);
            printf("GetWindowThreadProcessId hwnd=%p -> processId=%ld, threadId=%ld\n", hwnd, dwProcessId, dwThreadId);

            /*
            WINUSERAPI UINT WINAPI GetWindowModuleFileName(
                _In_ HWND hwnd,
                _Out_writes_to_(cchFileNameMax, return) LPTSTR pszFileName, //模块完整路径
                _In_ UINT cchFileNameMax
            );
            返回值是复制到缓冲区的字符总数。
            */
            TCHAR fileName[MAX_PATH];
            lenRet = GetWindowModuleFileName(hwnd, fileName, MAX_PATH);
            if (lenRet == 0) {
                //错误码〖126〗-找不到指定的模块。
                printf("GetWindowModuleFileName hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
            } else {
                _tprintf(_T("GetWindowModuleFileName hwnd=%p -> fileName=%s\n"), hwnd, fileName);
            }

            /*
            WINUSERAPI BOOL WINAPI GetWindowInfo(
                _In_ HWND hwnd,
                _Inout_ PWINDOWINFO pwi
            );

            typedef struct tagWINDOWINFO
            {
                DWORD cbSize;
                RECT rcWindow;
                RECT rcClient;
                DWORD dwStyle;
                DWORD dwExStyle;
                DWORD dwWindowStatus;
                UINT cxWindowBorders;
                UINT cyWindowBorders;
                ATOM atomWindowType;
                WORD wCreatorVersion;
            } WINDOWINFO, *PWINDOWINFO, *LPWINDOWINFO;
            */
            WINDOWINFO windowInfo;
            windowInfo.cbSize = sizeof(WINDOWINFO);
            ret = GetWindowInfo(hwnd, &windowInfo);
            if (!ret) {
                printf("GetWindowInfo hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
            }
            else {
                printf("GetWindowInfo hwnd=%p -> dwStyle=%ld, dwExStyle=%ld, dwWindowStatus=%ld, cxWindowBorders=%d, cyWindowBorders=%d, wCreatorVersion=%d\n", hwnd, windowInfo.dwStyle, windowInfo.dwExStyle, windowInfo.dwWindowStatus, windowInfo.cxWindowBorders, windowInfo.cyWindowBorders, windowInfo.wCreatorVersion);
            }
            printf("\n");
        }
    }

    return TRUE;
}

int main()
{
    BOOL ret;
    while (true) {
        /*
        typedef struct tagPOINT
        {
            LONG  x;
            LONG  y;
        } POINT, *PPOINT, NEAR *NPPOINT, FAR *LPPOINT;
        */
        POINT point;
        ret = GetCursorPos(&point);
        if (!ret) {
            printf("GetCursorPos -> fail(%ld)\n", GetLastError());
        }
        else {
            printf("GetCursorPos -> (%ld, %ld)\n", point.x, point.y);

            //获取桌面句柄
            HWND desktopHwnd = GetDesktopWindow();

            /*
            BOOL EnumChildWindows(
                HWND hWndParent,         // handle to parent window // 父窗口句柄
                WNDENUMPROC lpEnumFunc,  // callback function // 回调函数的地址
                LPARAM lParam            // application-defined value // 你自已定义的参数
            );
            直到调用到最个一个子窗口被枚举或回调函数返回一个false,否则将一直自动枚举下去。
            */
            ret = EnumChildWindows(desktopHwnd, EnumChildProcess, (LPARAM)&point);
        }

        /*
        WINBASEAPI VOID WINAPI Sleep(
            _In_ DWORD dwMilliseconds
        );
        Sleep会将线程挂起,把CPU让给其它线程,单位是毫秒
        */
        Sleep(20000);
    }

    system("pause");
    return 0;
}

这里写图片描述

在评论里有人说道,可以用“WindowFromPoint”来实现这个功能,于是就有了下列代码。

// SpyDemo.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include <Windows.h>

int main()
{
    BOOL ret;
    UINT count;
    /*
    typedef struct tagPOINT
    {
        LONG  x;
        LONG  y;
    } POINT, *PPOINT, NEAR *NPPOINT, FAR *LPPOINT;
    */
    POINT point;
    TCHAR pszFileName[MAX_PATH];
    TCHAR lpClassName[MAX_PATH];
    TCHAR windowText[MAX_PATH];
    while(true) {
        ret = GetCursorPos(&point);
        if (!ret) {
            printf("GetCursorPos -> fail(%ld)\n", GetLastError());
        } else {
            printf("GetCursorPos -> (%ld, %ld)\n", point.x, point.y);
            HWND hwnd = WindowFromPoint(point);
            if (hwnd == NULL || hwnd == INVALID_HANDLE_VALUE) {
                printf("WindowFromPoint point=(%ld, %ld) -> hwnd=%p -> fail(%ld)\n", point.x, point.y, hwnd, GetLastError());
            }
            else {
                printf("WindowFromPoint -> hwnd=%p\n", hwnd);

                /*
                检索与指定的窗口句柄关联的模块的完整路径和文件名称
                WINUSERAPI UINT WINAPI GetWindowModuleFileNameW(
                    _In_ HWND hwnd,
                    _Out_writes_to_(cchFileNameMax, return) LPWSTR pszFileName,
                    _In_ UINT cchFileNameMax
                );
                返回值是复制到缓冲区的字符总数。
                */
                count = GetWindowModuleFileName(hwnd, pszFileName, MAX_PATH);
                _tprintf(_T("GetWindowModuleFileName hwnd=%p -> count=%d, fileName=%s\n"), hwnd, count, pszFileName);

                /*
                WINUSERAPI int WINAPI GetClassName(
                    _In_ HWND hWnd,
                    _Out_writes_to_(nMaxCount, return) LPTSTR lpClassName,
                    _In_ int nMaxCount
                );
                */
                count = GetClassName(hwnd, lpClassName, MAX_PATH);
                _tprintf(_T("GetClassName hwnd=%p -> count=%d, lpClassName=%s\n"), hwnd, count, lpClassName);


                count = GetWindowText(hwnd, windowText, MAX_PATH);
                _tprintf(_T("GetWindowText hwnd=%p -> count=%d, lpClassName=%s\n"), hwnd, count, windowText);
            }

            //Screen(屏幕坐标)到Client(客户区坐标)的转换。
            ret = ScreenToClient(hwnd, &point);
            if (!ret) {
                printf("ScreenToClient hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
            }
            else {
                printf("ScreenToClient hwnd=%p -> point=(%ld, %ld)\n", hwnd, point.x, point.y);

                //返回父窗口中包含了指定点的第一个子窗口的句柄
                HWND childHwnd = ChildWindowFromPoint(hwnd, point);
                printf("ChildWindowFromPoint hwnd=%p, point=(%ld, %ld) -> childHwnd=%p\n", hwnd, point.x, point.y, childHwnd);

                count = GetClassName(hwnd, lpClassName, MAX_PATH);
                _tprintf(_T("GetClassName hwnd=%p -> count=%d, lpClassName=%s\n"), hwnd, count, lpClassName);

                count = GetWindowText(hwnd, windowText, MAX_PATH);
                _tprintf(_T("GetWindowText hwnd=%p -> count=%d, lpClassName=%s\n"), hwnd, count, windowText);
            }
        }
        Sleep(5000);
    }
    return 0;
}

这里写图片描述


但是有一个问题:
  为什么 GetWindowModuleFileName 虽然句柄每次都不一样,但是获取到的模块文件路径每次都是一样的当前的路径。
  希望有大神能帮我解答一下,感激不尽。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/chy555chy/article/details/52900785

智能推荐

pythontcp服务器框架_GitHub - xiaowang359/ChatServer: 基于python-tornado 与 sqlalchemy建立的1个TCP聊天服务器框架示例。...-程序员宅基地

文章浏览阅读140次。version 0.1 版本还存在一些BUG,采用sqlite数据库做为测试关于推送部分大家可以在pypi搜索anps 下载安装apnsclient 测试###通用部分提交length = json整体包长action = 协议关键字部分提交部分提交 uid ,为了使协议通用语web环境返回status = 状态成功失败errcode = 错误代码,需要具体定义common1000010001 ..._python tornado tcp 聊天

TensorFlow安装过程问题汇总_loading channels: failed-程序员宅基地

文章浏览阅读1.6k次。1. 问题: conda search numpy 以及 conda search --full-name python 失败。失败的现象:Loading channels: failedCondaHTTPError: HTTP 404 NOT FOUND for url <http://pypi.douban.com/simple/noarch/repodata.json>..._loading channels: failed

pip install sklearn安装成功后,提示ModuleNotFoundError: No module named ‘sklearn‘错误解决办法_pip sklearn库安装成功但是报错-程序员宅基地

文章浏览阅读6.5k次,点赞10次,收藏16次。pip install sklearn安装成功后,提示ModuleNotFoundError: No module named 'sklearn'错误解决办法_pip sklearn库安装成功但是报错

玩转传感器——DHT11温湿度传感器(STM32版)_温湿度传感器接线图-程序员宅基地

文章浏览阅读7.4w次,点赞327次,收藏2.3k次。玩转传感器——DHT11温湿度传感器(STM32版)文章目录玩转传感器——DHT11温湿度传感器(STM32版)前言一、接口说明1 接线图2 电源引脚3 串行接口(单线双向)二、通信过程三、测量分辨率与电气特性四、使用注意事项1 工作与贮存条件2 暴露在化学物质中3 恢复处理4 温度影响5 光线6 配线注意事项五、DHT11驱动程序1 DHT11.c1.1 配置输入输出GPIO1.2 复位DHT111.3 检查DHT11是否正常1.4 DHT11初始化1.5 读取一位数据(返回值0/1)1.6 读取一个_温湿度传感器接线图

如何把自己的驱动编译进内核或模块(Kconfig和Makefile)_nvp6324 驱动-程序员宅基地

文章浏览阅读905次。本说明以NVP6324为例。1、首先在drivers\media\i2c中修改Kconfig和Makefile,如下: 在Kconfig中添加如下:config VIDEO_NVP6324 tristate "NVP6324 AHD sensor support" depends on I2C ---help--- This is a V4L2 sensor-le..._nvp6324 驱动

自适应直方图均衡(CLAHE) 代码及详细注释【OpenCV】_自适应双平台直方图均衡算法代码-程序员宅基地

文章浏览阅读2.7w次,点赞9次,收藏80次。理论请参考博客OpenCV源码的本地路径: %OPENCV%\opencv\sources\modules\imgproc\src\clahe.cppclahe.cpp// ----------------------------------------------------------------------// CLAHEnamespace{ class C_自适应双平台直方图均衡算法代码

随便推点

中科数安 | 防泄密软件-程序员宅基地

文章浏览阅读419次,点赞16次,收藏3次。此外,中科数安防泄密软件还具有智能加密功能,可以识别散落在企业不同位置的机密文件,并对其强制加密,非核心数据不被过分加密,防止敏感内容泄漏。同时,它还支持离网办公,针对出差人员或网络故障等原因引起的客户端离网,用户可以发起离网审批,确保终端密文在出差过程中保持可用状态,不影响正常办公。它采用了多种加密机制和技术手段,确保企业数据的安全性、完整性和机密性。总之,中科数安防泄密软件是一种功能强大、技术先进的企业数据保护软件,可以有效地防止敏感数据的泄露和非法访问,保障企业的信息安全和业务连续性。

JavaScript——leetcode剑指 Offer 36. 二叉搜索树与双向链表_leetcode 链表的head是什么js-程序员宅基地

文章浏览阅读157次。题目描述输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的循环双向链表。要求不能创建任何新的节点,只能调整树中节点指针的指向。为了让您更好地理解问题,以下面的二叉搜索树为例:我们希望将这个二叉搜索树转化为双向循环链表。链表中的每个节点都有一个前驱和后继指针。对于双向循环链表,第一个节点的前驱是最后一个节点,最后一个节点的后继是第一个节点。下图展示了上面的二叉搜索树转化成的链表。“head” 表示指向链表中有最小元素的节点。特别地,我们希望可以就地完成转换操作。当转化完成以后,树中节点的左指针需_leetcode 链表的head是什么js

一篇文章让你全面了解TDengine-程序员宅基地

文章浏览阅读1.5w次,点赞3次,收藏46次。一篇文章让你全面了解TDengine本文将从以下几个方面全面介绍TDengine。TDengine的基本介绍TDengine的发展历程TDengine的优势TDengine的适用场景TDengine的写入存储策略TDengine的特点TDengine的基本介绍一句话了解TDengineTDengine是一个高效的存储、查询、分析时序大数据的平台,专为物联网、车联网、工业互联网、运维监测等优化而设计。你可以像使用关系型数据库MySQL一样来使用它,简单又方便。为什么会有TDengin_tdengine

成为JavaGC专家Part II:如何监控Java垃圾回收机制-程序员宅基地

文章浏览阅读215次。 成为JavaGC专家Part II :如何监控Java垃圾回收机制 本文是成为Java GC专家系列文章的第二篇。在第一篇《深入浅出Java垃圾回收机制》中我们学习了不同GC算法的执行过程,GC是如何工作的,什么是新生代和老年代,你应该了解的JDK7中的5种GC类型,以及这5种类型对于应用性能的影响。 在本文中,我将解释JVM到底是如何执行垃圾回收处理..._成为javagc专家part ii — 如何监控java垃圾回收机制。

python学习导航线_python点线导航-程序员宅基地

文章浏览阅读122次。文章目录python学习导航线一、seleniumpython-selenium二、python基础知识python的聊天室python学习导航线一、seleniumpython-selenium二、python基础知识python的聊天室_python点线导航

静态成员-静态成员变量-程序员宅基地

文章浏览阅读3.4k次,点赞4次,收藏22次。静态成员静态成员都是用static修饰,它的特点是不论创建多少个对象,程序都只创建一个静态成员。最主要的特点:共享什么是共享呢?例如:统计超市中所有商品数量的总和,商品数量的总和是随着每一个数量的变化而变化的,这是我们就可以用静态成员处理。(代码下面有写)静态成员又分为静态成员变量和静态成员函数。(一)静态成员变量特点:1、所有对象共享一份数据。 2、在编译阶段分配内存。 3、类内声明,类外初始化。#include<io..._静态成员变量

推荐文章

热门文章

相关标签