我们正在 WPF 中开发一个布局管理器,它具有可以由用户移动/调整大小/等的视口(viewport)。视口(viewport)通常通过我们在布局管理器中控制的提供程序填充数据(图片/电影/等)。我的工作是检查它是否也可以在视口(viewport)中托管任何外部 Windows 应用程序(即记事本、calc、adobe reader 等)。我遇到了很多问题。
大多数资源都指向使用 HwndHost 类。我正在试验 Microsoft 本身的演练:http://msdn.microsoft.com/en-us/library/ms752055.aspx
我对此进行了调整,因此列表框已替换为来自外部应用程序的窗口句柄。谁能帮我解决这些问题:
- 演练添加了一个额外的静态子窗口,其中放置了
ListBox
。我认为我不需要外部应用程序。如果我忽略它,我必须使外部应用程序成为子窗口(使用 user32.dll 中的 Get/SetWindowLong 将GWL_STYLE
设置为WS_CHILD
)。但如果我这样做,应用程序的菜单栏就会消失(因为WS_CHILD
样式)并且它不再接收输入。 - 如果我确实使用子窗口,并使外部应用程序成为其子窗口,那么一切正常,但有时外部应用程序无法正常绘制。
- 另外,我需要根据视口(viewport)调整子窗口的大小。这可能吗?
- 当外部应用程序生成子窗口(即记事本->帮助->关于)时,该窗口不由
HwndHost
托管(因此可以移到视口(viewport)之外)。有什么办法可以防止这种情况发生吗? - 由于我不需要外部应用程序和布局管理器之间的进一步交互,我假设我不需要捕获和转发消息是否正确? (本演练将 HwndSourceHook 添加到子窗口以捕获列表框中的选择更改)。
- 当您运行(未修改的)示例 VS2010 并关闭窗口时,VS2010 看不到程序已结束。如果你打破一切,你最终会在没有源代码的情况下进行组装。有什么臭东西正在发生,但我找不到它。
- 演练本身的编码似乎非常草率,但我还没有找到关于这个主题的任何更好的文档。还有其他例子吗?
- 另一种方法是不使用
HwndHost
,而是使用WindowsFormHost
,如讨论的 here .它有效(而且更简单!)但我无法控制应用程序的大小?另外,WinFormHost 不是真的用于此目的吗?
感谢任何指向正确方向的指示。
最佳答案
嗯...如果这个问题是在 20 年前提出的,人们会回答“当然,看看‘OLE’!”,这里是“对象链接和嵌入”的链接:
http://en.wikipedia.org/wiki/Object_Linking_and_Embedding
如果你读过这篇文章,你会看到这个规范定义的接口(interface)数量,不是因为它的作者认为它很有趣,而是因为它在技术上很难实现在一般情况下
它实际上仍然受到一些应用程序的支持(主要是微软的,因为微软几乎是 OLE 的唯一赞助商......)
您可以使用称为 DSOFramer 的东西嵌入这些应用程序(参见 SO 上的链接:MS KB311765 and DsoFramer are missing from MS site),一个允许您托管 OLE 服务器的组件(即:作为另一个进程运行的外部应用程序) 在应用程序中可视化。这是 Microsoft 几年前发布的某种大 hack,不再支持到二进制文件很难找到的程度!
它(可能)仍然适用于简单的 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”,它似乎可以工作。但这有一些限制(键盘、鼠标、焦点等)。
祝你好运!
关于c# - 在 WPF 窗口中托管外部应用程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5028598/