0

C# で AmiBroker という製品のプラグインを実装しています。

AmiBroker は、サードパーティ ベンダーが株式データをソリューションに渡すために使用できるいくつかの機能を公開した取引ソフトウェアです。したがって、AmiBroker によって認識されるプラグインを C# で作成できます。

私のシナリオでは、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;
    }

したがって、基本的には、アプリにあるハンドルの子ウィンドウのセット全体を反復処理し、探しているコントロールとクラス名が一致するかどうかを確認してから返します。それを改善する方法は何千もあります (1 回のショットで必要なものをすべて検索する、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 を使用する必要があります。

    [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 に答える