Win32 API对文本框发送消息(多个文本Edit,动态 控件 ID)
最近在群里有人说用Win32 Api不能对文本框设置内容(是别人写的一个程序,设置它的文本框的值).但是搞过win32的人都会说.这个应该不难啊,大概是搞.net的人,被微软 宠坏了.基本都不要用win32 api,这里我也不讨论用这个东西好不好,反正有人有这个需求,就要去做这个东西,我就自己建了一个工程,只要得到这个窗体的句柄,然后向他发送消息就搞定了,用到FindWindow,SendMessage搞定就可以了,我们所做的最核心的内容就是要找句柄.
要在win32 api下面使用FindWindow,SendMessage,必须这2个声明
public static extern IntPtr FindWindow(string lpClassName, string lpWindowName);
[DllImport("User32.dll", EntryPoint = "SendMessage", CharSet = CharSet.Auto)]
public static extern int SendTextMessage(
IntPtr hWnd,
int Msg,
int wParam,
string lParam
);
记得这里要加上CharSet,否则发送中文可能是乱码,调用的时候
查找窗体的句柄,然后再在这个窗体下面查找这个文本框的句柄,窗口我们是根据窗体的标题文本来查找,文本控件时根据控件的类型来找.
IntPtr hwndtext = WinAPIuser32.FindWindowEx(hwndCalc, 0,"Edit",null);
设置文本
轻松搞定了.很简单.
这个时候我想,窗体上只有一个文本框,也就是说一个Edit,查找起来是很方便,可是往往,我们在实际情况中,一个窗体上有很多文本框,我们要找到其中一个文本框设置它的值,这个时候你在用WinAPIuser32.FindWindowEx(hwndCalc, 0,"Edit",null);就做不到了.因为他得到的,始终是最后一个 文本框的句柄,也就是说如果页面上有
3个文本,你使用这个时候只会获取到最后一个文本框的 句柄,如果你要设置第2个文本框你是做不到的,我们有2个办法,一个是EnumChildWindows方法来遍历下面的所有文本框,对这些文本框进行赋值,第2中方法就是 根据控件ID来查找句柄,在一个程序编译完成以后,也就是发布给客户用的时候,窗体上的控件ID就是固定的了,不可改变,我这里说的控件ID不是指.net的立面一个TextBox控件的ID,而是在windows下面,显现出来的ID,这样我们就可以通过固定的ID来查找,注意ID是固定的,不会再改变,这样我们就可以用GetDlgItem的方法来通过ID号来获取句柄,这里我们先讲第2中方法,第一种方法,比较复杂,而且后面我会用第一种方法来做一个非常特殊的演示,那就是如果控件ID时动态的时候,我们也如何获取句柄
public static extern IntPtr GetDlgItem(
IntPtr hDlg,
int nIDDlgItem
);
当然得到了句柄还有什么做不到的,对这3个文本发送消息,设置文本内容
WinAPIuser32.SendTextMessage(hwndtext, MSCODE.WM_SETTEXT, 0, "mextb1860第一个文本框");
hwndtext = WinAPIuser32.GetDlgItem(hwndCalc, 1181678);
WinAPIuser32.SendTextMessage(hwndtext, MSCODE.WM_SETTEXT, 0, "mextb1860第二个文本框");
hwndtext = WinAPIuser32.GetDlgItem(hwndCalc, 919180);
WinAPIuser32.SendTextMessage(hwndtext, MSCODE.WM_SETTEXT, 0, "mextb1860第三个文本框");
一切很顺利,就像我们想的一样,3个文本框的内容都改变了,太好了,不过不要太高兴了,因为我们这里的ID是固定所以都硬编码进去了,在一般情况下是没有问题,因为大部分的都是固定的,这个时候我发现.net的程序的控件ID时随时改变的,而且每次运行一次ID都不一样,这个ID是跟着句柄改变,句柄是多少ID就是多少,老火啊.这回要根据ID来获取句柄是获取是行不通了,现在的情况是 一个页面多个Edit类控件的ID是动态的,程序每次运行都不一样,不固定.
关闭程序,再重新打开,在看下ID
那么我现在用第二种办法来解决,请出EnumChildWindows方法,这个方法比较特殊,有个一个参数是一个回调函数
public static extern int EnumChildWindows(int hWndParent, CallBack lpfn, int lParam);
CallBack是一个委托
public static extern int EnumChildWindows(int hWndParent, CallBack lpfn, int lParam);
/// <summary>
/// 回调业务
/// </summary>
public delegate void CallBusiness(IntPtr hwnd);
public delegate bool CallBack(IntPtr hwnd, int lParam);
/// <summary>
/// 遍历子窗体的父窗体句柄
/// </summary>
public static CallBack callBackEnumChildWindows = new CallBack(ChildWindowProcess);
/// <summary>
/// 委托业务,需要客户端添加
/// </summary>
public static CallBusiness CallFuntion;
/// <summary>
/// 遍历子窗体或控件
/// </summary>
/// <param name="hWnd"></param>
/// <param name="lParam"></param>
/// <returns></returns>
public static bool EnumChildWindows(IntPtr hWnd, int lParam)
{
EnumChildWindows(hWnd.ToInt32(), callBackEnumChildWindows, 0);
return true;
}
/// <summary>
/// 获取类名字
/// </summary>
/// <param name="hwnd">需要获取类名的句柄</param>
/// <param name="lpClassName">类名(执行完成以后查看)</param>
/// <param name="nMaxCount">缓冲区</param>
/// <returns></returns>
[DllImport("user32.dll", EntryPoint = "GetClassName")]
public static extern int GetClassName(
IntPtr hwnd,
StringBuilder lpClassName,
int nMaxCount
);
/// <summary>
/// 遍历子控件
/// </summary>
/// <param name="hwnd"></param>
/// <param name="lParam"></param>
/// <returns></returns>
public static bool ChildWindowProcess(IntPtr hwnd, int lParam)
{
if (CallFuntion != null)
{
CallFuntion(hwnd);
}
return true;
}
EnumChildWindows用来遍历所有的子控件的句柄,有一个回调函数,CallBusiness也是一个代理,是提供给客户端调用的时候来编写逻辑的.代码很简单,应该很容易理解,客户端调用的代码,因为是.net开发的程序 所以 Edit的控件类型有点不一样,不过没关系,不影响我们查找
WinAPIuser32.CallFuntion = delegate(IntPtr enumIntPtr)
{
StringBuilder s = new StringBuilder(2000);
WinAPIuser32.GetClassName(enumIntPtr, s, 255);
if (s.ToString() == "WindowsForms10.EDIT.app.0.378734a")
{
list.Add(enumIntPtr);
}
};
WinAPIuser32.EnumChildWindows(hwndCalc, 0);
WinAPIuser32.CallFuntion = null;
//第1个文本框
WinAPIuser32.SendTextMessage(list[2], MSCODE.WM_SETTEXT, 0, "mextb1860第一个文本框");
//第2个文本框
WinAPIuser32.SendTextMessage(list[1], MSCODE.WM_SETTEXT, 0, "mextb1860第二个文本框");
//第3个文本框
WinAPIuser32.SendTextMessage(list[0], MSCODE.WM_SETTEXT, 0, "mextb1860第三个文本框");
代码会提供下载,不明白的可以自己仔细看看.