gpt4 book ai didi

c# - 使用处理程序在 C# 中读取 C++ 窗口

转载 作者:行者123 更新时间:2023-11-27 23:13:44 25 4
gpt4 key购买 nike

我正在用 C# 为一个名为 AmiBroker 的产品实现一个插件。

AmiBroker 是一款交易软件,它公开了一些功能,第三方供应商可以使用这些功能将股票数据传递给解决方案。因此,我们可以在 C# 中创建一个 AmiBroker 可以识别的插件。

在我的场景中,我得到了 AmiBroker 主窗口的处理程序 [注意:AmiBroker 完全用 C++ 编写] 在 C# 中,我们可以检索主窗口的处理程序,因此使用这个句柄我可以读取窗口的数据子窗口、显示库存列表的面板或用户可见的内容,如果是这样,我将如何着手执行此操作?

最佳答案

可以,但是很麻烦。我真的只是在做一些非常相似的事情。 Pinvoke.net 非常适合这些东西,但我将向您展示一些我如何找到控件的示例。如果 AmiBroker 有任何关于控件名称或 AccessibleNames 的文档或任何允许您找到您正在寻找的确切控件的文档,那将是致命的。因为如果他们的名字含糊不清,你将很难找到你特别要找的人。但基本上,您要做的是在您拥有的句柄上使用 EnumChildWindows,遍历它们并寻找一个独特的属性,以允许您找到所需的控件。然后,您需要执行特定的 SendMessage 以从控件中获取文本(GetWindowText 或它所调用的任何内容仅适用于标签)。代码如下,在某些时候(很好的起点)从 Pinvoke.net 改编或刷取:

    [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 顺序中跳过它的表格后面(打破你正在搜索的层次列表)。也就是说,您应该知道 EnumChildWindows 将始终枚举给定窗口的所有子窗口,无论它们在层次结构中的哪个位置。如果您真的必须通过其父项及其父项的父项向下钻取并搜索每个控件,则需要使用 FindWindowEx,并声明您查看的最后一个子项(如果您想要第一个子项,则为 IntPtr.Zero):

    [DllImport("user32.dll", SetLastError = true)]
public static extern IntPtr FindWindowEx(IntPtr parentHandle, IntPtr childAfter, string className, string windowTitle);

关于c# - 使用处理程序在 C# 中读取 C++ 窗口,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18109002/

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