【发布时间】: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);
我查看了这个与我的问题非常相似的讨论。我从逻辑上理解我必须像下面这样发送 4 条消息
- KeyDown CTRL 键
- KeyDown S 键
- KeyUp S 键
- KeyUp CTRL 键
我不确定如何将正确的参数发送到 SendMessage。要关闭窗口,我使用
SendMessage(hWnd, 0x0010, 0, 0);
我从 MSDN 库得到这个。
您能否指导我访问一些链接,该链接告诉我键盘中键的十六进制并解释最后两个参数代表什么?
更新 - 1
使用 spy++ 我发现这些事件是通过在记事本窗口上按 CTRL-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。