可以,但是很乱。我实际上只是在做一些非常相似的事情。Pinvoke.net 非常适合这些东西,但我将向您展示我如何找到控件的一些示例。如果 AmiBroker 有任何关于控件名称或 AccessibleNames 的文档或任何可以让您找到所需的确切控件的文档,那将是杀手锏。因为如果它们的名称含糊不清,那么您将很难找到您要特别寻找的那些。但基本上,您想要做的是 EnumChildWindows 在您拥有的句柄上,遍历它们并寻找一个独特的属性来让您找到您想要的控件。然后,您需要执行特定的 SendMessage 以从控件中获取文本(GetWindowText 或它所称的任何内容仅适用于标签)。代码如下,
[DllImport("user32")]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool EnumChildWindows(IntPtr window, EnumWindowProc callback, IntPtr i);
public static extern uint GetClassName(IntPtr handle, StringBuilder name, int maxLength);
public delegate bool EnumWindowProc(IntPtr hWnd, IntPtr parameter);
private static List<IntPtr> GetChildWindows(IntPtr parent)
{
List<IntPtr> result = new List<IntPtr>();
GCHandle listHandle = GCHandle.Alloc(result);
try
{
EnumWindowProc childProc = new EnumWindowProc(EnumWindow);
EnumChildWindows(parent, childProc, GCHandle.ToIntPtr(listHandle));
}
finally
{
if (listHandle.IsAllocated)
listHandle.Free();
}
return result;
}
private static bool EnumWindow(IntPtr handle, IntPtr pointer)
{
GCHandle gch = GCHandle.FromIntPtr(pointer);
List<IntPtr> list = gch.Target as List<IntPtr>;
if (list == null)
{
throw new InvalidCastException("GCHandle Target could not be cast as List<IntPtr>");
}
list.Add(handle);
// You can modify this to check to see if you want to cancel the operation, then return a null here
return true;
}
//THIS IS THE ONE YOU'LL CALL!
public static IntPtr GetWindowByClass(IntPtr mainWindow, string name)
{
List<IntPtr> windows = GetChildWindows(mainWindow);
foreach (IntPtr window in windows)
{
StringBuilder response = new StringBuilder();
response.Capacity = 500;
if (GetClassName(window, response, response.Capacity) > 0)
if (response.ToString() == name)
return window;
}
return IntPtr.Zero;
}
因此,基本上它会遍历您在应用程序上拥有的句柄的一整套子窗口,查看类名是否与您正在寻找的控件匹配,然后返回它。有成千上万种方法可以改进它(一次搜索所有你想要的方法,FindWindow 可以按类名工作,等等)但我想向你展示更多它是如何完成的,而不是声明它应该是这样的完毕。最后,从窗口/控件获取文本的调用如下(也改编自 pinvoke.net:在 User32.dll 下查找所有这些内容):
public static string GetText(IntPtr control)
{
StringBuilder builder = new StringBuilder(40);
IntPtr result = IntPtr.Zero;
uint response = SendMessageTimeoutText(control, 0xd, 40, builder, APITypes.SendMessageTimeoutFlags.SMTO_NORMAL, 2000, out result);
return builder.ToString();
}
[DllImport("user32.dll", EntryPoint = "SendMessageTimeout", SetLastError = true, CharSet = CharSet.Auto)]
public static extern uint SendMessageTimeoutText(
IntPtr hWnd,
int Msg, // Use WM_GETTEXT
int countOfChars,
StringBuilder text,
APITypes.SendMessageTimeoutFlags flags,
uint uTImeoutj,
out IntPtr result);
[Flags]
public enum SendMessageTimeoutFlags : uint
{
SMTO_NORMAL = 0x0,
SMTO_BLOCK = 0x1,
SMTO_ABORTIFHUNG = 0x2,
SMTO_NOTIMEOUTIFNOTHUNG = 0x8
}
编辑:附录:我用来访问像这样的另一个表单的应用程序实际上没有唯一的控件名称,所以我最终使用 Spy++ 来确定它在窗口层次结构中的位置并拉动子项并依次选择每个子项。如果你必须走那条路,上帝会帮助你,特别是因为它可能根本不一致,特别是如果你需要的是一个没有创建的表格,或者它隐藏在另一个在 Z-Order 中跳过它的表格后面(破坏您正在搜索的分层列表)。也就是说,您应该知道 EnumChildWindows 将始终枚举给定窗口的 ALL CHILD WINDOWS,无论它们在层次结构中的哪个位置。如果您真的必须通过其父级及其父级的父级深入搜索每个控件,则需要使用 FindWindowEx,
[DllImport("user32.dll", SetLastError = true)]
public static extern IntPtr FindWindowEx(IntPtr parentHandle, IntPtr childAfter, string className, string windowTitle);