Win32 API对文本框发送消息(多个文本Edit,动态 控件 ID)最近在群里有人说用Win32 Api不能对文本框设置内容(是别人写的一个程序,设置它的文本框的值).但是搞过win32的人都会说.这个应该不难啊,大概是搞.net的人,被微软 宠坏了.基本都不要用win32 api,这里我也不讨论用这个东西好不好,反正有人有这个需求,就要去做这个东西,我就自己建了一个工程,只要得到这个窗体的句柄,然后向他发送消息就搞定了,用到FindWindow,SendMessage搞定就可以了,我们所做的最核心的内容就是要找句柄. 要在win32 api下面使用FindWindow,SendMessage,必须这2个声明 代码
[DllImport("user32.dll", EntryPoint = "FindWindow", SetLastError =true)]
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 hwndCalc = WinAPIuser32.FindWindow(null, "Form1aa");
IntPtr hwndtext = WinAPIuser32.FindWindowEx(hwndCalc, 0,"Edit",null);
设置文本 WinAPIuser32.SendTextMessage(hwndtext, MSCODE.WM_SETTEXT, 0, "mextb1860第一个文本框");
轻松搞定了.很简单.
这个时候我想,窗体上只有一个文本框,也就是说一个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时动态的时候,我们也如何获取句柄
[DllImport("user32.dll ", EntryPoint = "GetDlgItem")]
public static extern IntPtr GetDlgItem( IntPtr hDlg, int nIDDlgItem );
当然得到了句柄还有什么做不到的,对这3个文本发送消息,设置文本内容
代码
IntPtr hwndtext = WinAPIuser32.GetDlgItem(hwndCalc, 1247226);
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方法,这个方法比较特殊,有个一个参数是一个回调函数
[DllImport("user32.dll")]
public static extern int EnumChildWindows(int hWndParent, CallBack lpfn, int lParam);
CallBack是一个委托
代码
[DllImport("user32.dll")]
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的控件类型有点不一样,不过没关系,不影响我们查找
代码
List<IntPtr> list = new List<IntPtr>();
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第三个文本框");
代码会提供下载,不明白的可以自己仔细看看. |
|