我想使用C#代码保存TextPad窗口; 我可以找到窗口的句柄但不知道如何将CTRL-S发送到该窗口.我想使用P/Invoke API来实现这一点.此外,该TextPad窗口将处于非活动状态,因为此时我的应用程序将处于活动状态.
[DllImport("user32.dll")]
private static extern int SendMessage(IntPtr hwnd, int msg, int wParam, int lParam);
Run Code Online (Sandbox Code Playgroud)
我看了一下这个与我的问题非常相似的讨论.我逻辑上理解我必须做4发送消息,如下所示
我不知道如何将正确的参数发送到SendMessage.关闭我使用的窗口
SendMessage(hWnd, 0x0010, 0, 0);
Run Code Online (Sandbox Code Playgroud)
我从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
}
Run Code Online (Sandbox Code Playgroud)
这段代码不起作用(保存部分内容),即使我能够看到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);
}
Run Code Online (Sandbox Code Playgroud)
上面的代码而不是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);
Run Code Online (Sandbox Code Playgroud)
更新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
}
Run Code Online (Sandbox Code Playgroud)
在CTRL + SI可以最小化窗口之后,它会创建一个闪存.任何人都可以建议我是否可以在没有"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.
Run Code Online (Sandbox Code Playgroud)
我只需要将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)
Run Code Online (Sandbox Code Playgroud)
最后更新
我无法将^ S消息发送到最小化窗口.当焦点在编辑器上时,我每2-3秒使用一次sendkeys(^ s).
- Karephul
您不能使用SendMessage(或PostMessage,正确的)来模拟修饰符键的状态,如CTRL.您必须使用SendInput().
像Ctrl + S这样的击键不会非典型地转换为WM_COMMAND消息.使用Spy ++工具查看手动键入Ctrl + S时会发生什么.如果你看到WM_COMMAND然后你是黄金,你可以使用SendMessage()发送该消息.这当然只适用于特定的过程.