0

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

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

在我的场景中,我得到了 AmiBroker 主窗口的处理程序 [注意:AmiBroker 完全用 C++ 编写] 在 C# 中,我们可以检索主窗口的处理程序,所以使用这个句柄可以读取窗口的数据,例如子窗口,显示股票列表或用户可见的东西的面板,如果是这样,我将如何去做?

4

2 回答 2

3

可以,但是很乱。我实际上只是在做一些非常相似的事情。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);
于 2013-08-07T17:03:01.010 回答
1

如果您正在为其编写插件的软件提供 API,那么您所要求的似乎与直觉相悖。你真的应该使用它。

虽然可以使用 Win32 API 来枚举给定句柄的主窗口的子窗口,然后使用更多的 Win32 API 函数来确定 UI 的状态(即“读取数据”),但这将非常繁琐和错误-易于。

这是EnumChildWindows的 MSDN 链接,它允许您枚举给定句柄的主窗口的子窗口。

如果你想深入那个兔子洞,你可能还会发现SendMessageGetWindowText很有用。如果您打算从 C# 中使用这些 Win32 API,那么您绝对应该查看pinvoke.net 。

于 2013-08-07T16:59:36.163 回答