gpt4 book ai didi

c# - 在 c# 中向应用程序发送击键(sendkeys、postmessage、sendmessage 都不起作用)

转载 作者:可可西里 更新时间:2023-11-01 13:54:48 27 4
gpt4 key购买 nike

我正在尝试执行以下操作之一 1.打开所需的程序并以编程方式按下一个键 2.找到打开的程序窗口并以编程方式按下一个键 (两者都可以)

我已经尝试过多次 SendKeys.SendWait()、PostMessage() 和 SendMessage() 的实现,但均未成功。以下是我的代码片段

    //included all these for attempts 
[DllImport("User32.dll")]
static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

[DllImport("User32.dll")]
static extern int SetForegroundWindow(IntPtr hWnd);

[DllImport("User32.Dll", EntryPoint = "PostMessageA")]
static extern bool PostMessage(IntPtr hWnd, uint msg, int wParam, int lParam);

[DllImport("user32.dll")]
static extern byte VkKeyScan(char ch);

[DllImport("user32.dll", CharSet = CharSet.Auto)]
static extern IntPtr SendMessage(IntPtr hWnd, UInt32 Msg, IntPtr wParam, IntPtr lParam);

获取窗口句柄,sendmessage/postmessage/sendkeys使用的变量

IntPtr ptrOBS = proc.Handle;//this works properly, proc is instantiated properly
//IntPtr ptrOBS = FindWindow(null, "Open Broadcaster Software v0.472b");
SetForegroundWindow(ptrOBS);
const UInt32 WM_CHAR = 0x0102;
const uint WM_KEYDOWN = 0x100;
const int VK_R = 0x52; // taken from http://msdn.microsoft.com/en-us/library/dd375731(v=vs.85).aspx
const int VK_S = 0x53;

发送消息尝试:

SendMessage(ptrOBS, WM_KEYDOWN, (IntPtr)VK_R, (IntPtr)1);//tried both WM_CHAR and WM_KEYDOWN

PostMessage 尝试:

string message = "rs";
bool sent = PostMessage(ptrOBS, WM_KEYDOWN, VkKeyScan(message[0]), 0);

发送 key 尝试:

SendKeys.SendWait("{r}");

尝试在父窗口(应用程序)和子窗口(由我尝试发送的按键触发的按钮)上设置焦点:

static void SetFocus(IntPtr hwndTarget, string childClassName)
{
// hwndTarget is the other app's main window
// ...
IntPtr targetThreadID = GetWindowThreadProcessId(hwndTarget, IntPtr.Zero); //target thread id
IntPtr myThreadID = GetCurrentThread(); // calling thread id, our thread id
try
{
bool lRet = AttachThreadInput(myThreadID, targetThreadID, -1); // attach current thread id to target window

// if it's not already in the foreground...
lRet = BringWindowToTop(hwndTarget);
SetForegroundWindow(hwndTarget);

// if you know the child win class name do something like this (enumerate windows using Win API again)...
IntPtr hwndChild = (IntPtr)1183492;//(IntPtr)EnumAllWindows(hwndTarget, childClassName).FirstOrDefault();

if (hwndChild == IntPtr.Zero)
{
// or use keyboard etc. to focus, i.e. send keys/input...
// SendInput (...);
return;
}

// you can use also the edit control's hwnd or some child window (of target) here
SetFocus(hwndChild); // hwndTarget);
SendKeys.SendWait("{r}");
}
finally
{
SendKeys.SendWait("{r}");
bool lRet = AttachThreadInput(myThreadID, targetThreadID, 0); //detach from foreground window
SendKeys.SendWait("{r}");
}
}

对于 NSGaga:

    string windowName = "Open Broadcaster Software v0.472b";
IntPtr outerPtr = FindWindow(null, windowName);
IntPtr ptrOBS = (IntPtr)527814;//button that im trying to trigger keypress on
SetForegroundWindow(outerPtr);
SetForegroundWindow(ptrOBS);
SetFocus(outerPtr, "OBSWindowClass");//SetFocus(ptrOBS, "Button");
const UInt32 WM_CHAR = 0x0102;
const int VK_R = 0x52; // taken from http://msdn.microsoft.com/en-us/library/dd375731(v=vs.85).aspx
const int VK_S = 0x53;

//SetForegroundWindow(ptrOBS);
System.Threading.Thread.Sleep(3000);
SendKeys.SendWait("{r}");
SendMessage(outerPtr, WM_KEYDOWN, (IntPtr)VK_R, (IntPtr)1);
PostMessage(outerPtr, WM_KEYDOWN, VkKeyScan('r'), 0);

最佳答案

You cannot reliably use SendMessage and PostMessage for synthesizing keyboard input .它们不是为此而设计的。这些消息(WM_CHARWM_KEYDOWN 等)是在接收、处理和转发键盘输入时由较低级别的子系统发出的通知到适当的收件人。自己发送或发布这些消息就像给某人打电话恶作剧。

SendKeys(与所有其他输入合成器方法一样,包括 SendInput 函数,该函数明确设计用于合成键盘输入并且至少在某些实现中是 SendKeys 实际上在引擎盖下使用)仅当您希望接收键盘输入的窗口具有焦点时才有效。在 Windows 中,只有焦点(事件)窗口接收输入事件。

所以 SendKeys 可能是您要让它工作的方法(或者是 P/Invoking SendInput 及其所有相关结构),但您确实需要遵守收件人窗口必须具有焦点的警告。否则,它不会得到任何东西。

从您的示例代码看来,您正在尝试使用 SetForegroundWindow满足这个前提的函数。不幸的是,您向它传递了一个无效值,并且没有进行任何可能会提醒您注意此错误的错误检查。具体来说,这段代码是错误的:

IntPtr ptrOBS = proc.Handle;//this works properly, proc is instantiated properly
SetForegroundWindow(ptrOBS); // WRONG, ptrOBS is not a window handle

即使我相信您认为 ptrOBS 被正确初始化,这也使它成为一个有效的进程句柄,这与一个有效的进程句柄截然不同窗口。除了明显的名义上的差异外,进程可以有多个窗口并且只有一个窗口可以有焦点(即“在前台”)。

在调用 SetForegroundWindow 之前,您需要获取特定窗口的句柄,并且鉴于我们知道一个进程可以有多个窗口,这可能会很棘手。您需要一些可靠的方法来确定您想要的窗口。许多人通过将窗口的名称硬编码为字符串来实现这一点,这种方法在重新编译目标应用程序并且此实现细节发生更改之前效果很好。我能想到的唯一可靠方法是让用户单击目标窗口和您的代码以检索当前位于鼠标指针下的窗口的句柄。

当然,所有这些都假设您已经遵守了 SetForegroundWindow 的使用限制,这些限制在链接的 SDK 文档的“备注”部分中列举。

关于c# - 在 c# 中向应用程序发送击键(sendkeys、postmessage、sendmessage 都不起作用),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15536980/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com