43

我们正在 WPF 中开发一个布局管理器,它具有可以由用户移动/调整大小/等的视口。视口通常通过我们在布局管理器中控制的提供程序填充数据(图片/电影/等)。我的工作是检查它是否也可以在视口中托管任何外部 Windows 应用程序(即记事本、calc、adobe 阅读器等)。我遇到了很多问题。

大多数资源都指向使用 HwndHost 类。我正在试验微软本身的这个演练:http: //msdn.microsoft.com/en-us/library/ms752055.aspx

我已经对此进行了调整,因此列表框将替换为来自外部应用程序的 Windows 句柄。谁能帮我解决这些问题:

  1. 演练添加了一个额外的静态子窗口,其中ListBox放置了 。我认为外部应用程序不需要它。如果我忽略它,我必须将外部应用程序设为子窗口(使用 user32.dll 中的 Get/SetWindowLong 设置GWL_STYLEWS_CHILD)。但如果我这样做,应用程序的菜单栏就会消失(因为WS_CHILD样式)并且它不再接收输入。
  2. 如果我确实使用了子窗口,并使外部应用程序成为该事物的子窗口,则可以合理地工作,但有时外部应用程序无法正常绘制。
  3. 此外,我需要将子窗口调整为视口。这可能吗?
  4. 当外部应用程序产生一个子窗口(即记事本->帮助->关于)时,此窗口不由该窗口托管HwndHost(因此可以移动到视口之外)。有什么办法可以防止这种情况发生吗?
  5. 由于我不需要外部应用程序和布局管理器之间的进一步交互,我是否正确地假设我不需要捕获和转发消息?(演练将 HwndSourceHook 添加到子窗口以捕获列表框中的选择更改)。
  6. 当您运行(未修改的)示例 VS2010 并关闭窗口时,VS2010 不会看到程序结束。如果你打破一切,你最终会在没有源代码的情况下组装。有异味发生,但我找不到。
  7. 演练本身的编码似乎很草率,但我没有找到关于这个主题的任何更好的文档。还有其他例子吗?
  8. 另一种方法是不使用HwndHost,但正如这里WindowsFormHost所讨论的那样。它可以工作(而且更简单!)但我无法控制应用程序的大小?另外,WinFormHost 不是真的为此而生吗?

感谢您提供正确方向的任何指示。

4

6 回答 6

32

嗯......如果这个问题是在 20 年前提出的,那么有人会回答,“当然,看看 'OLE'!”,这里是“对象链接和嵌入”的链接:

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

如果你读了这篇文章,你会看到这个规范定义的接口数量,不是因为它的作者认为它很有趣,而是因为在一般情况下它在技术上很难实现

它实际上仍然受到一些应用程序的支持(主要是微软的,因为微软几乎是 OLE 的唯一赞助商......)

您可以使用称为 DSOFramer 的东西嵌入这些应用程序(请参阅此处的链接:MS KB311765 和 DsoFramer are missing from MS site),该组件允许您在应用程序内直观地托管 OLE 服务器(即:作为另一个进程运行的外部应用程序) . 这是微软几年前发布的某种大黑客攻击,不再受支持,以至于二进制文件很难找到!

它(可能)仍然适用于简单的 OLE 服务器,但我想我在某处读到它甚至不适用于新的 Microsoft 应用程序,例如 Word 2010。因此,您可以将 DSOFramer 用于支持它的应用程序。你可以试试。

对于其他应用程序,今天,在我们生活的现代世界中,您不托管应用程序,在外部进程中运行,您托管组件,并且它们通常应该在进程内运行。这就是为什么你通常很难做你想做的。您将面临的一个问题(尤其是最新版本的 Windows)是安全性:我不信任的进程如何能够合法地处理由的进程创建的窗口和菜单 :-) ?

尽管如此,您仍然可以使用各种 Windows hack 逐个应用程序做很多事情。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 (如何从表单的控制菜单框中删除菜单项)。

您也可以用“winword.exe”替换“notepad.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 作为我的主机类的基类,你可以在这里找到它:http ://microsoftdwayneneed.codeplex.com/SourceControl/changeset/view/69631#1034035

示例代码:

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 回答
2

我有这个在生产中运行,到目前为止在 WPF 应用程序中运行良好。确保SetNativeWindowInWPFWindowAsChild()从拥有window.

    public static bool SetNativeWindowInWPFWindowAsChild(IntPtr hWndNative, Window window)
    {
        UInt32 dwSyleToRemove = WS_POPUP | WS_CAPTION | WS_THICKFRAME;
        UInt32 dwExStyleToRemove = WS_EX_DLGMODALFRAME | WS_EX_WINDOWEDGE | WS_EX_CLIENTEDGE | WS_EX_STATICEDGE;

        UInt32 dwStyle = GetWindowLong(hWndNative, GWL_STYLE);
        UInt32 dwExStyle = GetWindowLong(hWndNative, GWL_EXSTYLE);

        dwStyle &= ~dwSyleToRemove;
        dwExStyle &= ~dwExStyleToRemove;

        SetWindowLong(hWndNative, GWL_STYLE, dwStyle | WS_CHILD);
        SetWindowLong(hWndNative, GWL_EXSTYLE, dwExStyle);

        IntPtr hWndOld = SetParent(hWndNative, new WindowInteropHelper(window).Handle);
        if (hWndOld == IntPtr.Zero)
        {
            System.Diagnostics.Debug.WriteLine("SetParent() Failed -> LAST ERROR: " + Marshal.GetLastWin32Error() + "\n");
        }
        return hWndOld != IntPtr.Zero;
    }

这是我使用的本机 Win32 API。(这里有额外内容,因为我在设置窗口后调整/聚焦窗口)

        [StructLayout(LayoutKind.Sequential)]
        private struct RECT
        {
            public Int32 left;
            public Int32 top;
            public Int32 right;
            public Int32 bottom;
        }
        [DllImport("user32.dll", SetLastError = true)]
        private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);
        [DllImport("user32.dll")]
        private static extern UInt32 SetWindowLong(IntPtr hWnd, int nIndex, UInt32 dwNewLong);
        [DllImport("user32.dll")]
        private static extern UInt32 GetWindowLong(IntPtr hWnd, int nIndex);
        [DllImport("user32.dll")]
        private static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect);
        [DllImport("user32.dll")]
        private static extern IntPtr SetFocus(IntPtr hWnd);
        [DllImport("user32.dll")]
        private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, SetWindowPosFlags uFlags);

        private static int GWL_STYLE = -16;
        private static int GWL_EXSTYLE = -20;

        private static UInt32 WS_CHILD = 0x40000000;
        private static UInt32 WS_POPUP = 0x80000000;
        private static UInt32 WS_CAPTION = 0x00C00000;
        private static UInt32 WS_THICKFRAME = 0x00040000;

        private static UInt32 WS_EX_DLGMODALFRAME = 0x00000001;
        private static UInt32 WS_EX_WINDOWEDGE = 0x00000100;
        private static UInt32 WS_EX_CLIENTEDGE = 0x00000200;
        private static UInt32 WS_EX_STATICEDGE = 0x00020000;

        [Flags]
        private enum SetWindowPosFlags : uint
        {
            SWP_ASYNCWINDOWPOS = 0x4000,
            SWP_DEFERERASE = 0x2000,
            SWP_DRAWFRAME = 0x0020,
            SWP_FRAMECHANGED = 0x0020,
            SWP_HIDEWINDOW = 0x0080,
            SWP_NOACTIVATE = 0x0010,
            SWP_NOCOPYBITS = 0x0100,
            SWP_NOMOVE = 0x0002,
            SWP_NOOWNERZORDER = 0x0200,
            SWP_NOREDRAW = 0x0008,
            SWP_NOREPOSITION = 0x0200,
            SWP_NOSENDCHANGING = 0x0400,
            SWP_NOSIZE = 0x0001,
            SWP_NOZORDER = 0x0004,
            SWP_SHOWWINDOW = 0x0040
        }
        private static readonly IntPtr HWND_TOPMOST = new IntPtr(-1);
        private static readonly IntPtr HWND_NOTOPMOST = new IntPtr(-2);
        private static readonly IntPtr HWND_TOP = new IntPtr(0);
        private static readonly IntPtr HWND_BOTTOM = new IntPtr(1);
于 2017-02-13T14:59:59.883 回答
1

解决方案非常复杂。很多代码。这里有一些提示。

首先,你在正确的轨道上。

你必须使用 HwndHost 和 HwndSource 的东西。如果不这样做,您将获得视觉伪像。像闪烁一样。一个警告,如果你不使用主机和源,它看起来会起作用,但最终不会——它会有随机的小愚蠢错误。

看看这个以获得一些提示。它并不完整,但它会帮助你朝着正确的方向前进。 http://microsoftdwayneneed.codeplex.com/SourceControl/changeset/view/50925#1029346

您必须进入 Win32 才能控制很多您所询问的内容。您确实需要捕获和转发消息。您确实需要控制哪些窗口“拥有”子窗口。

大量使用 Spy++。

于 2011-02-25T18:10:16.147 回答
0

查看我的回答:如何在 wpf 应用程序中运行应用程序?

我设法让记事本示例在没有 DwayneNeed jiggery 的情况下工作。我刚刚添加了 SetParent() 和繁荣……她就像 DwayneNeed 示例一样工作。

于 2012-12-05T01:09:26.260 回答