【问题标题】:Sending CTRL-S message to a window向窗口发送 CTRL-S 消息
【发布时间】:2011-02-28 16:46:55
【问题描述】:

我想用 C# 代码保存一个 TextPad 窗口;我可以找到窗口的句柄,但不确定如何将 CTRL - S 发送到该窗口。我想使用 P/Invoke API 来实现这一点。此外,该 TextPad 窗口将处于非活动状态,因为那时我的应用程序将处于活动状态。

[DllImport("user32.dll")]
private static extern int SendMessage(IntPtr hwnd, int msg, int wParam, int lParam);

Send Ctrl+Up to a window

我查看了这个与我的问题非常相似的讨论。我从逻辑上理解我必须像下面这样发送 4 条消息

  • KeyDown CTRL 键
  • KeyDown S 键
  • KeyUp S 键
  • KeyUp CTRL 键

我不确定如何将正确的参数发送到 SendMessage。要关闭窗口,我使用

SendMessage(hWnd, 0x0010, 0, 0);

我从 MSDN 库得到这个。

您能否指导我访问一些链接,该链接告诉我键盘中键的十六进制并解释最后两个参数代表什么?

更新 - 1
使用 spy++ 我发现这些事件是通过在记事本窗口上按 CTRL-S 生成的

1. WM_KEYDOWN nVirtKey:VK_Control, 2. WM_KEYDOWN nVirtKey:'S' .. 其他一些消息 .. 3. WM_KEYUP nVirtKey:VK_Control。 4. WM_KEYUP nVirtKey:'S'。

更新 - 2


       private IntPtr startnotepad() {
            ProcessStartInfo psi = new ProcessStartInfo();
            psi.FileName = @"notepad.exe";
            String fileName = baseDirectory + textbox1.Text;
            psi.Arguments = @"""" + fileName + @"""";
            psi.WindowStyle = ProcessWindowStyle.Minimized;
            Process p = Process.Start(psi); 
            return p.MainWindowHandle;
        }

        private void SaveNotepad(IntPtr handle)
        {
            IntPtr handle = GetWindow(handle, 5); // get the keyboard focus handle
            // verified the handle with Spy ++.
            SendMessage(handle, 0x0100, 0x11, 0); // keydown ctrl
            SendMessage(handle, 0x0100, 0x53, 0); // keydown S
            //SendMessage(handle, 0x0101, 0x11, 0); --- I tried keeping "Keyup Ctrl" here as well.
            SendMessage(handle, 0x0101, 0x53, 0); // keyup s
            SendMessage(handle, 0x0101, 0x11, 0); // keyup ctrl            
        }

即使我能够看到 WM_KEYDOWN - CTRL .. WM_KEYDOWN - 'S' .. KEYUP 'S' 和 KEYUP CTRL 被发送到 spy++ 中的右侧窗口,此代码也不起作用(保存部分代码)。谁能评论这段代码有什么问题?或者如果我在做一些非常愚蠢的事情,有什么建议。

更新 3

正如@Hans 在他的 cmets 中建议的那样,我应该使用 PostMessage 而不是 SendMessage。


 [DllImport("user32.dll")]
 public static extern IntPtr PostMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);  

        private void Save(IntPtr mainWindowHandle)
        {               
            IntPtr handle = GetWindow(mainWindowHandle, 5); // getChild window
            IntPtr CTRL_KEY = new IntPtr(0x11);
            uint KEY_DOWN = 0x0100;
            uint KEY_UP = 0x0101;
            IntPtr S_KEY = new IntPtr(0x53);

            //SetForegroundWindow(p.MainWindowHandle);
            PostMessage(handle, KEY_DOWN, CTRL_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_DOWN, S_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_UP, S_KEY, IntPtr.Zero);
            PostMessage(handle, KEY_UP, CTRL_KEY, IntPtr.Zero);                      
        } 

上面的代码而不是 CTRL+S 将 CTRL 和 S 发送到记事本窗口。所以,我可以看到两个“S”被扔到记事本上。查看我们按下 CTRL+S 时生成的真实消息,我发现我的 Postmessage 的最后一个参数是错误的。它应该类似于(对于 KEY UP CTRL 键“C01F0001”)

您能告诉我如何将此十六进制作为 postmessage 的最后一个参数传递吗?


// does not work
PostMessage(handle, KEY_UP, CTRL_KEY, new IntPtr(0xC01F0001);

更新 4:我认为我们不能将“热键”消息发送到最小化窗口。使用发布消息。
下面的代码适用于 SendInput: 但它会弹出窗口,这有损于目的。无论如何,只是想更新这个帖子。


 private void Save_Notepad(IntPtr mainWindowHandle) {
            //SetActiveWindow(mainWindowHandle);
            //SetFocus(GetWindow(mainWindowHandle, 5));
            ShowWindow(mainWindowHandle, 1); // show window 
            SetForegroundWindow(mainWindowHandle);
            //SetActiveWindow(mainWindowHandle);
            //SetFocus(GetWindow(mainWindowHandle, 5));
            //IntPtr returnvalue = SetFocus(mainWindowHandle);
            uint intReturn;
            INPUT structInput;
            structInput = new INPUT();
            structInput.type = 1;// keyboard input
            // key down
            structInput.ki.wScan = 0x1D;
            structInput.ki.time = 0;
            structInput.ki.dwFlags = 0;
            // key down Ctrl
            structInput.ki.wVk = 0x11;  //0x1D; //
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            // key down S
            structInput.ki.wScan = 0x1F;
            structInput.ki.wVk = 0x53; //0x41;//0x53;  //0x1F;//
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            // key up 
            structInput.ki.dwFlags = 0x0002; // key up
            // key up S
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(typeof(INPUT)));
            // key up CTRL
            structInput.ki.wVk = 0x11;  //0x1D; //
            intReturn = SendInput(1, ref structInput, Marshal.SizeOf(new INPUT()));
            //ShowWindow(mainWindowHandle, 2); // minimize it again 
        }

在 CTRL + S 之后,我可以最小化窗口,但它会创建一个 flash。 谁能建议我是否可以在没有“FLASH”的情况下(至少)实现此功能?
更新 5:使用 WM_SYSCOMMAND


IntPtr handle = GetWindow(mainWindowHandle, 5);
// send Alt F message to Notepad.
// http://msdn.microsoft.com/en-us/library/ms646360(v=VS.85).aspx
// I can see this is working.
PostMessage(handle, 0x0112, 0xF100, 'f');  // send WM_SYSCOMMAND
// how to send s on this window ??
// I have tried things which I have learned so far. 

我只需要将 S 消息发送到文件菜单,该菜单将作为 WM_SYSCOMMAND 的结果弹出。谁能指出我如何做到这一点的文档?


PostMessage(handle, KEY_DOWN, S_KEY, 0x1F0001); // POST does not work
// tried with mainwindowhandle as well
PostMessage(handle, 0x111, 'S', 0); // WM_COMMAND does not work
PostMessage(handle, 0x0112, 0xF100, 's');  // WM_SYSCOMMAND does not work (and does not make sence either)

最终更新

我无法将 ^S 消息发送到最小化窗口。当焦点在编辑器上时,我每 2-3 秒使用一次 sendkeys(^s)。

-- 卡勒菲尔

【问题讨论】:

  • 我的意思是,你的更新 4 非常好,但它不是与 .NET SendKeys 的单行吗?
  • Hans 还指出,您可以发送一个WM_COMMAND,这样可以避免在窗口最小化时显示窗口
  • @David:首先,我想要一个在窗口最小化时发送消息的解决方案。我从 SendMessage 开始,然后是 PostMessage,正如 Hans 建议的 SendInput。正如您所建议的那样,“SendKeys”确实看起来是我的更新 4 的更好解决方案。更新 4 不是我正在寻找的解决方案,但如果我必须安定下来,我将切换到 SendKeys ..
  • @David : WM_COMMAND,能否请您指出一些 API 文档或参考如何使用它来发送加速器密钥?我会很感激的。
  • 您没有发送加速键。记事本将 CTRL+S 按键转换为 WM_COMMAND 消息。您只需要发送与记事本相同的 WM_COMMAND。

标签: c# pinvoke


【解决方案1】:

您不能使用 SendMessage(或 PostMessage,正确的那个)来模拟修饰键的状态,例如 CTRL。您必须使用 SendInput()。

像 Ctrl+S 这样的击键通常不会被翻译成 WM_COMMAND 消息。使用 Spy++ 工具查看手动键入 Ctrl+S 时会发生什么。如果您看到 WM_COMMAND,那么您就是金子,您可以使用 SendMessage() 发送该消息。这当然只适用于特定流程。

【讨论】:

  • 我可以在 spy++ 中按顺序看到以下消息:1. WM_KEYDOWN nVirtKey:VK_Control, 2. WM_KEYDOWN nVirtKey:'S' .. 3. WM_KEYUP nVirtKey:VK_Control。 4. WM_KEYUP nVirtKey:'S'。
    序列与我最初的想法略有不同,但它在那些线上。谢谢指点
  • 将此放在您的问题中,而不是在评论中。我没有看到 WM_COMMAND,请使用 SendInput()。
  • 我认为 WM_KEYDOWN 是 WM_COMMAND 的一种。我将研究 SendInput() - 这看起来像是我只能在活动窗口上做的事情。我想向 In Active 窗口发送一个 CTRL S .. 这样用户就不会意识到他的记事本数据正在被保存。
  • 记事本确实使用WM_COMMAND,看看框架窗口。否则,这是使用户意外保存文件并且永远无法再次找到它的好方法。 Ctrl+S 第一次调出另存为对话框,当用户忙于输入时非常糟糕。
  • 我们试图只控制几个记事本窗口,这些窗口将为用户创建和打开[这将阻止另存为对话框],Ctrl-S 不会关闭窗口,所以我们很好也有。那么,您认为 sendmessage() API 应该可以工作吗?如果是,那么我只需要找出正确的窗口句柄来将这些消息发送到[键盘焦点句柄]
【解决方案2】:

我需要做你需要做的事情,在某个背景窗口中调用 Ctrl+S,经过大约一天的研究,似乎有很多方法可以实现这一点。

例如,按 Alt-F 然后按“S”可以调用:

PostMessage(handle, 0x0112, 0xF100, 0x0046);
PostMessage(handle, 0x0102, 0x0053, 0);

Alt-F 部分适用于背景窗口,而后面的“S”则不适用。

另一种方法是使用 WM_COMMAND 和 WM_MENUSELECT、MF_MOUSESELECT:

IntPtr menu = GetMenu(handle);
IntPtr subMenu = GetSubMenu(menu, 0);//0 = first menu item
uint menuItemID = GetMenuItemID(subMenu, 2);//2 = second item in submenu
SendMessage(handle, 0x0111, 0x20000000 + menuItemID, menu);

最后,有点讽刺的是,最简单的解决方案是使用 menuItemID 调用 WM_COMMAND:

PostMessage(handle, 0x0111, 0x0003, 0x0);

0x0003(保存在记事本中)由应用程序及其菜单结构确定,您可以在使用组合键组合或在菜单上按下鼠标按钮时,通过在窗口中侦听 spy++ 中的 WM_COMMAND 来获取此代码命令。

似乎也可以使用 WM_SYSKEYUP/DOWN 和 WM_MENUCOMMAND,还有像 Ctrl-C、Ctrl-V 这样的键具有常量定义的值,可以作为一个字符传递,但仅限于侦听这些硬编码字符的应用程序。 .

感谢您的起点。

【讨论】:

猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-09-28
  • 1970-01-01
  • 1970-01-01
  • 2020-09-14
相关资源
最近更新 更多