如何捕获从此菜单发送的 Windows 消息?

Posted

技术标签:

【中文标题】如何捕获从此菜单发送的 Windows 消息?【英文标题】:How to Capture the Windows Message that is Sent from this Menu? 【发布时间】:2016-03-28 09:56:03 【问题描述】:

在大多数应用程序中,当您单击某个 MenuItem 时,会发送一个 WindowsMessage(通常为 WM_COMMAND),其中 wParam 代表所选 MenuItem 的 ID。

某个程序有一个窗口菜单(通过单击标题栏上的程序图标可访问该菜单), 我想找出当我从那个菜单中选择一个特定的 MenuItem 时发送的 WindowsMessage 是什么。

该程序是众所周知的 - Windows XP 中的命令提示符窗口: (cmd.exe)

这里是窗口菜单:

我想在那里为 MenuItem 捕获 WindowsMessage 和 wParam, 例如“粘贴”菜单项。 (但不只是它..任何其他可能也是如此)

这是我尝试过的:

方法一:

我总是尝试的第一种方法是使用 Spy++。 问题是当我尝试为这个特定程序(DOS 窗口)记录消息时,Spy++ 给了我这个消息框:

由于某种原因,Spy++ 不会为该程序捕获 WindowsMessages。

所以我继续我使用的第二种方法..

方法二:

Resource Hacker (ResHacker.exe) 也适用于查找从单击的 MenuItems 发送的 WindowsMessage,而且非常容易。

如果你运行 Resource Hacker,然后用它打开一些 EXE 文件, 您通常会看到这些树,其中之一被称为“菜单”, 它包含所有细节,包括 wParam:

问题是,当我尝试在 cmd.exe 上使用 Resource Hacker 时, 我明白了:

可以看出,那里没有“菜单”树。

我的问题:

除了我常用的2种方法外,还有其他方法吗, 可用于查找为 DOS 窗口的 Window Menu 中的“Paste”MenuItem 发送的WindowsMessage(和wParam)?

【问题讨论】:

您的目标是发送窗口消息还是在控制台窗口中粘贴文本?您似乎将很多精力集中在第一个上,但如果实际目标是第二个,请说清楚。 嗨达米安。实际上你提出了一个很好的观点。我在这里有 2 个目标:首先,以任何可能的方式生成 Paste 操作。请注意,我知道它也可以通过窗口菜单之外的另一种方式完成,即通过右键单击窗口,但这种特定方式对我不利,因为我不希望鼠标移动以实现它.第二个目标是学习拦截 WindowsMessages 的其他方法,除了我已经知道的 2 种方法(并在问题中指定)。所以这个问题有两个目标.. :) 这在 Windows XP 上会很困难,因为控制台运行在 csrss 中,这是非常高的特权。即使您知道消息编号,也无法发送。考虑改用无障碍界面。 嗨,雷蒙德。关于“即使您找出消息编号,您也无法发送它” - 这是不正确的,我设法通过模拟鼠标右键单击的 SendMessage() 进行粘贴操作。 (在该窗口上,右键单击窗口上的任意位置会导致粘贴操作)。它成功了。所以这意味着我可以在那个窗口上使用 SendMessage() 。现在唯一的问题是如何获取其他 MenuItems 的 ID.. 【参考方案1】:

0xfff1 是 wParam,所以在 C# 中(你没有指定你使用的语言,但应该很容易翻译它):

[DllImport("User32.dll")]
public static extern int SendMessage(IntPtr hWnd, int uMsg, int wParam, uint lParam);
public const int WM_KEYDOWN = 0x01000;
void PasteInCommandPrompt(IntPtr hWnd)

    SendMessage(handle, WM_COMMAND, 0xfff1, 0);

http://blogs.msdn.com/b/bill/archive/2012/06/09/programmatically-paste-clipboard-text-to-a-cmd-window-c-or-c.aspx

编辑:2019 年 9 月 22 日

在Console window - programmatic command code (wParam of WM_COMMAND) 的 cmets 中(用户使用上述链接到 Bill Lin 的博客,但无法正常工作的问题),@eryksun 给了我寻找 ConvhostV2.dll.mui 的想法找到所有可用的菜单命令。我找不到 ConvhostV2.dll.mui...

但是在我的系统上我发现了C:\Windows\System32\en-US\ConhostV1.dll.mui,当使用 Resource Hacker 查看时(正如@spaceman 尝试使用 cmd.exe),它包含所有可用于 cmd.exe 的菜单项。

cmd.exe 的完整命令列表如下:

0xfff0:复制 0xfff1:粘贴 0xfff2:马克 0xfff3:滚动 0xfff4:查找 0xfff5:全选

除了paste(允许您执行任意命令)之外,select allcopy 非常有用,因为它们可以让您获得控制台输出(尽管剥离了所有虚拟终端序列、指定文本颜色的字符)。

如果您正在深入研究操作命令窗口的路线,您可能还会对新的“Windows 伪控制台”感兴趣,它可以让您完全控制 cmd.exe 或任何基于命令行的应用程序。见https://devblogs.microsoft.com/commandline/windows-command-line-introducing-the-windows-pseudo-console-conpty/。

【讨论】:

谢谢你。我想知道他是如何获得 wParam 的。他提到了“蛮力”——这是什么意思,他尝试了所有可能的值,直到它起作用?无论如何,我很想知道如何找到 MenuItem 的值,因为它可能对命令提示符窗口的 Window Menu 中的其他 MenuItems 有用。如果有人可以在此添加信息,那就太好了。

以上是关于如何捕获从此菜单发送的 Windows 消息?的主要内容,如果未能解决你的问题,请参考以下文章

C# - 从特定应用程序捕获 Windows 消息

python screen-print不捕获所有Windows应用程序的下拉菜单

小技巧: 屏幕截图(截屏)工具如何截取(捕获)下拉菜单

如何使用 WIFI 将消息从 android 设备发送到 PC

这是不是可以挂钩子进程发送到 Windows 控制台的消息?

delphi 对右键菜单的操作