将CTRL-S消息发送到窗口

时间:2011-02-28 16:46:55

标签: c# pinvoke

我想使用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 key
  • KeyUp CTRL键

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

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

我是从MSDN库获得的。

请您指点一些链接,告诉我键盘上的键的十六进制,并说明最后两个参数代表什么?

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

1. WM_KEYDOWN nVirtKey:VK_Control, 
2. WM_KEYDOWN nVirtKey:'S' .. some other messates ..
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

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


 [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”(至少)的情况下实现此功能?
更新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)。

- Karephul

2 个答案:

答案 0 :(得分:5)

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

像Ctrl + S这样的击键不会非典型地转换为WM_COMMAND消息。使用Spy ++工具查看手动键入Ctrl + S时会发生什么。如果你看到WM_COMMAND然后你是黄金,你可以使用SendMessage()来发送该消息。这当然只适用于特定的过程。

答案 1 :(得分:3)

我需要做你需要做的事情,在某个背景窗口中调用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这样的键具有常量定义的值,并且可以作为一个字符传递,但仅限于听取这些硬编码字符的应用程序。

感谢您的起点。