在WPF中托管外部exe

时间:2013-12-04 21:31:40

标签: wpf winforms wpf-4.0 windowsformshost

您好我正在尝试在WPF应用中托管一个exe(记事本)。 请在此处找到代码:

public partial class MainWindow : Window
{
    private Process _process;
    private System.Windows.Forms.Panel _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;
    const string patran = "patran";
    public MainWindow()
    {
        InitializeComponent();
        _process = Process.Start("notepad.exe");

        _panel = new System.Windows.Forms.Panel();
        wfHost.Child = _panel;
        var patranPanelHandle = _panel.Handle;
        SetParent(_process.MainWindowHandle, patranPanelHandle);

        // 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();
    }

    private void ResizeEmbeddedApp()
    {
        if (_process == null)
            return;
        SetWindowPos(_process.MainWindowHandle, IntPtr.Zero, 0, 0, (int)_panel.Width, (int)_panel.Height, SWP_NOZORDER | SWP_NOACTIVATE);
    }

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

相同的代码适用于Winforms,因此我使用了WinformsHost,然后添加了一个winforms面板,如上所述。但结果并不如预期。记事本不在WPF应用程序之外且父级子项无法正常工作

this也未找到答案

1 个答案:

答案 0 :(得分:1)

我想,你过早地调用SetParent。父窗口和子窗口都准备就绪后,您需要SetParent。因此,您可以做的是等到父窗口Loaded,然后启动子进程。但是,在调用WaitForInputIdle之前,请在子进程上调用SetParent

下面是我尝试过的示例代码,它可以运行:

public partial class MainWindow : Window
{
    private Process _process;

    [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;
    const string patran = "patran";
    public MainWindow()
    {
        InitializeComponent();

        Loaded += (s, e) => LaunchChildProcess();
    }

    private void LaunchChildProcess()
    {
        _process = Process.Start("notepad.exe");
        _process.WaitForInputIdle();

        var helper = new WindowInteropHelper(this);

        SetParent(_process.MainWindowHandle, helper.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();
    }

    private void ResizeEmbeddedApp()
    {
        if (_process == null)
            return;
        SetWindowPos(_process.MainWindowHandle, IntPtr.Zero, 0, 0, (int)ActualWidth, (int)ActualHeight, SWP_NOZORDER | SWP_NOACTIVATE);
    }

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

有关WaitForInputIdle的详细信息,请参阅此MSDN页。

<强>更新

附加结果图像:

enter image description here