43

ユーザーが移動/サイズ変更/などできるビューポートを持つWPFでレイアウトマネージャーを開発しています。通常、ビューポートには、レイアウト マネージャーで制御されているプロバイダーを介してデータ (写真/動画など) が入力されます。私の仕事は、ビューポートで外部 Windows アプリ (つまり、メモ帳、電卓、Adobe Reader など) をホストすることも可能かどうかを調べることです。私は多くの問題に遭遇します。

ほとんどのリソースは、HwndHost クラスの使用を指しています。私はマイクロソフト自身からこのチュートリアルを試しています: http://msdn.microsoft.com/en-us/library/ms752055.aspx

リスト ボックスが外部アプリケーションからの Windows ハンドルに置き換えられるように、これを調整しました。誰でもこれらの質問で私を助けることができます:

  1. このチュートリアルでは、 が配置される追加の静的サブ ウィンドウを追加しますListBox。外部アプリには必要ないと思います。省略すると、外部アプリを子ウィンドウにする必要があります (user32.dll から Get/SetWindowLong を使用して as に設定GWL_STYLEしますWS_CHILD)。しかし、それを行うと、アプリのメニュー バーが (WS_CHILDスタイルのために) 消えてしまい、入力を受け取ることができなくなります。
  2. サブウィンドウを使用し、外部アプリをその子にすると、合理的に機能しますが、外部アプリが正常に描画されない場合があります。
  3. また、ビューポートに合わせて子ウィンドウのサイズを変更する必要があります。これは可能ですか?
  4. 外部アプリが子ウィンドウを生成する場合 (つまり、メモ帳 -> ヘルプ -> バージョン情報)、このウィンドウは によってホストされませんHwndHost(したがって、ビューポートの外に移動できます)。それを防ぐ方法はありますか?
  5. 外部アプリケーションとレイアウト マネージャーの間でそれ以上の対話は必要ないので、メッセージをキャッチして転送する必要がないと想定してよいでしょうか? (チュートリアルでは、HwndSourceHook をサブ ウィンドウに追加して、リスト ボックスでの選択の変更をキャッチします)。
  6. (変更されていない) サンプル VS2010 を実行してウィンドウを閉じると、VS2010 はプログラムが終了したことを認識しません。すべてを壊すと、ソースなしでアセンブリになってしまいます。何か異臭がするのですが、見つかりません。
  7. ウォークスルー自体は非常にずさんなコーディングのようですが、この件に関するより良いドキュメントは見つかりませんでした。他の例はありますか?
  8. もう 1 つのアプローチは、使用しないことですHwndHostが、ここでWindowsFormHost説明したとおりです。動作します (そして、はるかに簡単です!) が、アプリケーションのサイズを制御できませんか? また、WinFormHost は実際にはこのためのものではありませんか?

正しい方向への指針をありがとう。

4

6 に答える 6

32

ええと... 20 年前のように質問が提起されていた場合、「もちろん、'OLE' を見てください!」と答えるでしょう。「オブジェクトのリンクと埋め込み」とは何かへのリンクを次に示します。

http://en.wikipedia.org/wiki/Object_Linking_and_Embedding

この記事を読むと、この仕様で定義されたインターフェースの数がわかります。これは、作成者が面白いと思ったからではなく、一般的なケースで実現するのが技術的に難しいためです。

実際には、一部のアプリではまだサポートされています (Microsoft はほとんど OLE の唯一のスポンサーだったため、ほとんどが Microsoft のアプリです)。

DSOFramer と呼ばれるものを使用してこれらのアプリを埋め込むことができます (SO のリンクを参照してください: MS KB311765 と DsoFramer が MS サイトにありません)。これは、アプリケーション内で視覚的に OLE サーバー (つまり、別のプロセスとして実行される外部アプリ) をホストできるようにするコンポーネントです。 . これは、Microsoft が数年前に公開したある種の大きなハックであり、バイナリを見つけるのが非常に困難なほどサポートされていません。

単純な OLE サーバーではまだ動作する可能性がありますが、Word 2010 などの新しい Microsoft アプリケーションでは動作しないとどこかで読んだと思います。したがって、DSOFramer をサポートするアプリケーションには DSOFramer を使用できます。あなたはそれを試すことができます。

他のアプリケーションについては、今日、私たちが住んでいる現代の世界では、アプリケーションをホストせず、外部プロセスで実行し、コンポーネントをホストし、一般にインプロセスで実行することになっていますそのため、一般的にやりたいことを行うのは非常に困難になります。あなたが直面する問題の 1 つ (最近のバージョンの Windows では特にそうではありません) はセキュリティです:私が信頼していないあなたのプロセスが、私のプロセスによって作成されウィンドウとメニューを合法的に処理するにはどうすればよいでしょうか :-) ?

それでも、さまざまな Windows ハックを使用して、アプリケーションごとにかなり多くのことができます。SetParent は基本的にすべてのハックの母です :-)

これは、指定したサンプルを拡張し、自動サイズ変更を追加し、キャプション ボックスを削除するコードです。例として、システム メニューであるコントロール ボックスを暗黙的に削除する方法を示します。

public partial class Window1 : Window
{
    private System.Windows.Forms.Panel _panel;
    private Process _process;

    public Window1()
    {
        InitializeComponent();
        _panel = new System.Windows.Forms.Panel();
        windowsFormsHost1.Child = _panel;
    }

    [DllImport("user32.dll")]
    private static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);

    [DllImport("user32.dll", SetLastError = true)]
    private static extern int GetWindowLong(IntPtr hWnd, int nIndex);

    [DllImport("user32")]
    private static extern IntPtr SetParent(IntPtr hWnd, IntPtr hWndParent);

    [DllImport("user32")]
    private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, int uFlags);

    private const int SWP_NOZORDER = 0x0004;
    private const int SWP_NOACTIVATE = 0x0010;
    private const int GWL_STYLE = -16;
    private const int WS_CAPTION = 0x00C00000;
    private const int WS_THICKFRAME = 0x00040000;

    private void button1_Click(object sender, RoutedEventArgs e)
    {
        button1.Visibility = Visibility.Hidden;
        ProcessStartInfo psi = new ProcessStartInfo("notepad.exe");
        _process = Process.Start(psi);
        _process.WaitForInputIdle();
        SetParent(_process.MainWindowHandle, _panel.Handle);

        // remove control box
        int style = GetWindowLong(_process.MainWindowHandle, GWL_STYLE);
        style = style & ~WS_CAPTION & ~WS_THICKFRAME;
        SetWindowLong(_process.MainWindowHandle, GWL_STYLE, style);

        // resize embedded application & refresh
        ResizeEmbeddedApp();
    }

    protected override void OnClosing(System.ComponentModel.CancelEventArgs e)
    {
        base.OnClosing(e);
        if (_process != null)
        {
            _process.Refresh();
            _process.Close();
        }
    }

    private void ResizeEmbeddedApp()
    {
        if (_process == null)
            return;

        SetWindowPos(_process.MainWindowHandle, IntPtr.Zero, 0, 0, (int)_panel.ClientSize.Width, (int)_panel.ClientSize.Height, SWP_NOZORDER | SWP_NOACTIVATE);
    }

    protected override Size MeasureOverride(Size availableSize)
    {
        Size size = base.MeasureOverride(availableSize);
        ResizeEmbeddedApp();
        return size;
    }
}

これは基本的にすべて Windows の「従来の」ハックです。ここで説明されているように、気に入らない項目メニューを削除することもできます: http://support.microsoft.com/kb/110393/en-us (フォームのコントロール メニュー ボックスからメニュー項目を削除する方法)。

「notepad.exe」を「winword.exe」に置き換えることもでき、動作するようです。ただし、これには制限があります (キーボード、マウス、フォーカスなど)。

幸運を!

于 2011-02-21T18:55:39.483 に答える
7

Simon Mourierの答えは非常によく書かれています。しかし、自作のwinformアプリで試してみると失敗してしまいました。

_process.WaitForInputIdle();

で置き換えることができます

while (_process.MainWindowHandle==IntPtr.Zero)
            {
                Thread.Sleep(1);
            }

そしてすべてが順調に進みます。

素晴らしい質問をありがとう、そしてあなたの答えをくれた皆さんに感謝します。

于 2014-08-27T21:45:31.810 に答える
7

このスレッドの回答を読み、自分で試行錯誤を行った後、かなりうまく機能するものになりましたが、もちろん、特別な場合には注意が必要なものもあります.

ホスト クラスの基本クラスとしてHwndHostExを使用しました。ここで見つけることができます。

コード例:

public class NotepadHwndHost : HwndHostEx
{
    private Process _process;

    protected override HWND BuildWindowOverride(HWND hwndParent)
    {
        ProcessStartInfo psi = new ProcessStartInfo("notepad.exe");
        _process = Process.Start(psi);
        _process.WaitForInputIdle();

        // The main window handle may be unavailable for a while, just wait for it
        while (_process.MainWindowHandle == IntPtr.Zero)
        {
            Thread.Yield();
        }

        HWND hwnd = new HWND(_process.MainWindowHandle);

        const int GWL_STYLE = -16;
        const int BORDER = 0x00800000;
        const int DLGFRAME = 0x00400000;
        const int WS_CAPTION = BORDER | DLGFRAME;
        const int WS_THICKFRAME = 0x00040000;
        const int WS_CHILD = 0x40000000;

        int style = GetWindowLong(notepadHandle, GWL_STYLE);
        style = style & ~WS_CAPTION & ~WS_THICKFRAME; // Removes Caption bar and the sizing border
        style |= WS_CHILD; // Must be a child window to be hosted

        NativeMethods.SetWindowLong(hwnd, GWL.STYLE, style);

        return hwnd;
    }

    protected override void DestroyWindowOverride(HWND hwnd)
    {
        _process.CloseMainWindow();

        _process.WaitForExit(5000);

        if (_process.HasExited == false)
        {
            _process.Kill();
        }

        _process.Close();
        _process.Dispose();
        _process = null;
        hwnd.Dispose();
        hwnd = null;
    }
}

HWND、NativeMethods、および列挙型も、DwayneNeed ライブラリ (Microsoft.DwayneNeed.User32) から取得されます。

NotepadHwndHost を子として WPF ウィンドウに追加するだけで、そこにホストされているメモ帳ウィンドウが表示されます。

于 2012-11-14T08:34:44.403 に答える
1

ソリューションは信じられないほど複雑です。たくさんのコード。ここにいくつかのヒントがあります。

まず、あなたは正しい軌道に乗っています。

HwndHost と HwndSource を使用する必要があります。そうしないと、視覚的なアーティファクトが発生します。ちらつきのように。警告: ホストとソースを使用しない場合、動作するように見えますが、最終的には動作しません。ランダムで小さなばかげたバグが発生します。

いくつかのヒントについては、これを見てください。完全ではありませんが、正しい方向に進むのに役立ちます。 http://microsoftdwayneneed.codeplex.com/SourceControl/changeset/view/50925#1029346

あなたが求めていることの多くを制御するには、Win32 に入る必要があります。メッセージをキャッチして転送する必要があります。どのウィンドウが子ウィンドウを「所有」するかを制御する必要があります。

Spy++ を多用します。

于 2011-02-25T18:10:16.147 に答える
0

私の答えをチェックしてください:wpfアプリケーション内でアプリケーションを実行する方法は?

私はなんとかメモ帳の例をDwayneNeedジゲリーなしで動作させることができました。SetParent()とブームを追加しました...彼女はDwayneNeedの例と同じように機能します。

于 2012-12-05T01:09:26.260 に答える