1、利用句柄操作窗口C#写个类操作窗口(句柄操作)实现过程:过程一:找到当前鼠标位置的句柄您的使用2个WinAPI(俺喜欢自己封装下来用):View Code DllImport(user32.dll, EntryPoint = GetCursorPos) public static extern bool GetCursorPos(out Point pt); DllImport(user32.dll, EntryPoint = WindowFromPoint) public static extern IntPtr WindowFromPoint(Point pt); /鼠标位置的坐标 pub
2、lic static Point GetCursorPosPoint() Point p = new Point(); if (GetCursorPos(out p) return p; return default(Point); / / 找到句柄 / / 坐标 / public static IntPtr GetHandle(Point p) return WindowFromPoint(p); 过程二:改变窗口的Text您的使用1个WinAPI:View Code DllImport(user32.dll, EntryPoint = SendMessage) private static
3、 extern int SendMessage(IntPtr hWnd, int Msg, IntPtr wParam, string lParam); / / 给窗口发送内容 / / 句柄 / 要发送的内容 public static void SetText(IntPtr hWnd, string lParam) SendMessage(hWnd, WM_SETTEXT, IntPtr.Zero, lParam); private const int WM_SETTEXT = 0x000C;通过这个方法就能改变Text的值了。思考:如果俺把这个窗口的句柄记录下来,下次不用鼠标获取,直接就能
4、改变值不蛮好的嘛。例如:我有个桌面系统老叫我输入用户名,密码。我记录用户名和密码的窗口句柄,然后改变他们的输入值,这样多省事。(只是举例,不考虑安全性)问题:你会告诉我,窗口句柄的每次重建会变的呀,咋办。回答:类名不变呀。过程三:您的准备一些工具吧,例如:句柄找类名呀,类名找句柄什么的等等,下面会用到一些WinAPIView Code DllImport(user32.dll, EntryPoint = FindWindow) private static extern IntPtr FindWindow(string IpClassName, string IpWindowName); Dl
5、lImport(user32.dll, EntryPoint = FindWindowEx) private static extern IntPtr FindWindowEx(IntPtr hwndParent, IntPtr hwndChildAfter, string lpszClass, string lpszWindow); DllImport(user32.dll, EntryPoint = GetParent) public static extern IntPtr GetParent(IntPtr hWnd); DllImport(user32.dll, CharSet = C
6、harSet.Auto) public static extern int GetClassName(IntPtr hWnd, StringBuilder lpClassName, int nMaxCount); / / 找到句柄 / / 类名 / public static IntPtr GetHandle(string IpClassName) return FindWindow(IpClassName, null); / / 子窗口句柄 / / 父窗口句柄 / 前一个同目录级同名窗口句柄 / 类名 / public static IntPtr GetChildHandle(IntPtr
7、hwndParent, IntPtr hwndChildAfter, string lpszClass) return FindWindowEx(hwndParent, hwndChildAfter, lpszClass, null); / / 全部子窗口句柄 / / 父窗口句柄 / 类名 / public static List GetChildHandles(IntPtr hwndParent, string className) List resultList = new List(); for (IntPtr hwndClient = GetChildHandle(hwndParent
8、, IntPtr.Zero, className); hwndClient != IntPtr.Zero; hwndClient = GetChildHandle(hwndParent, hwndClient, className) resultList.Add(hwndClient); return resultList; / / 找类名 / / 句柄 / public static string GetClassName(IntPtr hWnd) StringBuilder lpClassName = new StringBuilder(128); if (GetClassName(hWn
9、d, lpClassName, lpClassName.Capacity) = 0) throw new Exception(not found IntPtr!); return lpClassName.ToString(); 思考:遍历桌面上所有的窗口,然后找类名,然后改他的Text,扯淡嘛,相同的类名太多了,找毛呀。实现:不仅记录类名,而且记录类名在父窗口出现的位置,然后通过桌面一层层找下来,最后找到这个句柄。(虽然不是太准,但是一般的还是能解决了,如果你有什么好方式一起研究)。过程四:实现一个WinHWND的类,可以把他的规则,他的父窗口类名,以及在父窗口中同类名出现的顺序记录下来,然后
10、通过这些记录的信息还原句柄。View Code public class WinHWND public IntPtr HWND get; set; public string ClassName get; set; public WinHWND Parent get; set; public int InParentSequence get; set; private WinHWND() public WinHWND(IntPtr hWnd) this.HWND = hWnd; this.ClassName = GetClassName(); this.Parent = GetParent()
11、; this.InParentSequence = GetInParentSequence(); private string GetClassName() return WinAPI.GetClassName(this.HWND); private WinHWND GetParent() if (WinAPI.GetParent(this.HWND) = null) throw new Exception(not found IntPtr!); if (WinAPI.GetParent(this.HWND) = IntPtr.Zero) return null; return new Win
12、HWND(WinAPI.GetParent(this.HWND); private int GetInParentSequence() IntPtr IntprtParent = this.Parent = null ? IntPtr.Zero : this.Parent.HWND; return WinAPI.GetChildHandles(IntprtParent, this.ClassName).IndexOf(this.HWND); public override string ToString() StringBuilder result = new StringBuilder();
13、 for (WinHWND winHandle = this; winHandle != null; winHandle = winHandle.Parent) result.Append(string.Format(0:1; if (winHandle.InParentSequence = -1) break; return result.ToString().TrimEnd(;); private static string GetBaseMark(string sMark) string sMarks = sMark.Split(;); return sMarkssMarks.Lengt
14、h - 1.Split(:)0; private static string GetChildMarks(string sMark) string sMarks = sMark.Split(;); string sChildMarks = new stringsMarks.Length - 1; for (int i = 0; i sChildMarks.Length; i + ) sChildMarksi = sMarksi ; return sChildMarks; /我一直觉得这段写很丑陋,谁能帮帮我改改 public static WinHWND GetWinHWND(string s
15、Mark) List baseHwnds = WinAPI.GetChildHandles(IntPtr.Zero, GetBaseMark(sMark); string sChildMarks = GetChildMarks(sMark); /由于主窗口在桌面出现所以很可能同名,所以要看看他的儿子和孙子.是不是都匹配 foreach (IntPtr baseHwnd in baseHwnds) IntPtr handle = baseHwnd; for (int i = sChildMarks.Length - 1; i = 0; i-) string sChildMark = sChild
16、Marksi.Split(:); try handle = WinAPI.GetChildHandles(handle, UnEscape(sChildMark0)int.Parse(sChildMark1); catch break; if (i = 0) return new WinHWND(handle); continue; return null; #region 转义 private static string Escape(string arg) return arg.Replace(:, :).Replace(;,;); private static string UnEsca
17、pe(string arg) return arg.Replace(:, :).Replace(;, ;); #endregion public static WinHWND GetWinHWND() return new WinHWND(WinAPI.GetHandle(WinAPI.GetCursorPosPoint(); 上全部代码,里面加了窗口的部分属性,扩展其他的属性,自己发挥吧,就是搞WinAPIView Code using System;usingusing System.Linq;using System.Text;usingusing System.Drawing;usin
18、g System.Collections;namespace InformationCollectionDataFill public class WinAPI #region WinodwsAPI DllImport(user32.dll, EntryPoint = FindWindow) private static extern IntPtr FindWindow(string IpClassName, string IpWindowName); DllImport(user32.dll, EntryPoint = FindWindowEx) private static extern
19、IntPtr FindWindowEx(IntPtr hwndParent, IntPtr hwndChildAfter, string lpszClass, string lpszWindow); DllImport(user32.dll, EntryPoint = SendMessage) private static extern int SendMessage(IntPtr hWnd, int Msg, IntPtr wParam, string lParam); DllImport(user32.dll, EntryPoint = GetParent) public static e
20、xtern IntPtr GetParent(IntPtr hWnd); DllImport(user32.dll, EntryPoint = GetCursorPos) public static extern bool GetCursorPos(out Point pt); DllImport(user32.dll, EntryPoint = WindowFromPoint, CharSet = CharSet.Auto, ExactSpelling = true) public static extern IntPtr WindowFromPoint(Point pt); DllImpo
21、rt(user32.dll, CharSet = CharSet.Auto) public static extern int GetClassName(IntPtr hWnd, StringBuilder lpClassName, int nMaxCount); DllImport(user32.dll, CharSet = CharSet.Auto) public static extern int GetWindowText(IntPtr hWnd, Out, MarshalAs(UnmanagedType.LPTStr) StringBuilder lpString, int nMax
22、Count); DllImport(user32.dll, CharSet = CharSet.Auto) public static extern int GetWindowRect(IntPtr hwnd, ref Rectangle rc); DllImport(user32.dll, CharSet = CharSet.Auto) public static extern int GetClientRect(IntPtr hwnd, ref Rectangle rc); DllImport(user32.dll, CharSet = CharSet.Auto) public stati
23、c extern int MoveWindow(IntPtr hwnd, int x, int y, int nWidth, int nHeight, bool bRepaint); DllImport(user32.dll, CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true) public static extern int ScreenToClient(IntPtr hWnd, ref Rectangle rect); #endregion #region 封装API方法 / / 找到句柄 / / 类名 /
24、public static IntPtr GetHandle(string IpClassName) return FindWindow(IpClassName, null); / / 找到句柄 / / 坐标 / public static IntPtr GetHandle(Point p) return WindowFromPoint(p); /鼠标位置的坐标 public static Point GetCursorPosPoint() Point p = new Point(); if (GetCursorPos(out p) return p; return default(Point
25、); / / 子窗口句柄 / / 父窗口句柄 / 前一个同目录级同名窗口句柄 / 类名 / public static IntPtr GetChildHandle(IntPtr hwndParent, IntPtr hwndChildAfter, string lpszClass) return FindWindowEx(hwndParent, hwndChildAfter, lpszClass, null); / / 全部子窗口句柄 / / 父窗口句柄 / 类名 / public static List GetChildHandles(IntPtr hwndParent, string cl
26、assName) List resultList = new List(); for (IntPtr hwndClient = GetChildHandle(hwndParent, IntPtr.Zero, className); hwndClient != IntPtr.Zero; hwndClient = GetChildHandle(hwndParent, hwndClient, className) resultList.Add(hwndClient); return resultList; / / 给窗口发送内容 / / 句柄 / 要发送的内容 public static void SetT
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1