- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我们正在 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
样式)并且它不再接收输入。 HwndHost
托管(因此可以移到视口(viewport)之外)。有什么办法可以防止这种情况发生吗? 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/
https://github.com/mattdiamond/Recorderjs/blob/master/recorder.js中的代码 我不明白 JavaScript 语法,比如 (functio
在 iOS 7 及更早版本中,如果我们想在应用程序中找到 topMostWindow,我们通常使用以下代码行 [[[UIApplication sharedApplication] windows]
我已经尝试解决这个问题很长一段时间了:我无法访问窗口的 url,因为它位于另一个域上..有一些解决方案吗? function login() { var cb = window.ope
是否可以将 FFMPEG 视频流传递到 C# 窗口?现在它在新窗口中作为新进程打开,我只是想将它传递给我自己的 SessionWindow。 此时我像这样执行ffplay: public void E
我有一个名为 x 的矩阵看起来像这样: pTime Close 1 1275087600 1.2268 2 1275264000 1.2264 3 1275264300 1.2
在编译时,发生搜索,grep搜索等,Emacs会在单独的窗口中创建一个新的缓冲区来显示结果,有没有自动跳转到那个窗口的方法?这很有用,因为我可以使用 n 和 p 而不是 M-g n 和 M-g p 移
我有一个启动 PowerShell 脚本的批处理文件。 批处理文件: START Powershell -executionpolicy RemoteSigned -noexit -file "MyS
我有一个基于菜单栏的应用程序,单击图标时会显示一个窗口。在 Mac OS X Lion 上一切正常,但由于某种原因,在 Snow Leopard 和早期版本的 Mac OS X 上会出现错误。任何时候
在 macOS 中,如何在 Xcode 和/或 Interface Builder 中创建带有“集成标题栏和工具栏”的窗口? 这是“宽标题栏”类型的窗口,已添加到 OS X 10.10 Yosemit
在浏览器 (Chrome) 中 JavaScript: var DataModler = { Data: { Something: 'value' }, Process: functi
我有 3 个 html 页面。第 1 页链接到第 2 页,第 2 页链接到第 3 页(为了简单起见)。 我希望页面 2 中的链接打开页面 3 并关闭页面 1(选项卡 1)。 据我了解,您无法使用 Ja
当点击“创建节点”按钮时,如何打开一个新的框架或窗口?我希望新框架包含一个文本字段和下拉菜单,以便用户可以选择一个选项。 Create node Search node
我有一个用户控件,用于编辑应用程序中的某些对象。 我最近遇到一个实例,我想弹出一个新的对话框(窗口)来托管此用户控件。 如何实例化新窗口并将需要设置的任何属性从窗口传递到用户控件? 感谢您的宝贵时间。
我有一个Observable,它发出许多对象,我想使用window或buffer操作对这些对象进行分组。但是,我不想指定count参数来确定窗口中应包含多少个对象,而是希望能够使用自定义条件。 例如,
我有以下代码,它打开一个新的 JavaFX 阶段(我们称之为窗口)。 openAlertBox.setOnAction(e -> { AlertBox alert = AlertBox
我要添加一个“在新窗口中打开”上下文菜单项,该菜单项将以新的UIScene打开我的应用程序文档之一。当然,我只想在实际上支持多个场景的设备上显示该菜单项。 目前,我只是在检查设备是否是使用旧设备的iP
我正在尝试创建一个 AIR 应用程序来记录应用程序的使用情况,使用 AIR 从系统获取信息的唯一简单方法是使用命令行工具和抓取 标准输出 . 我知道像 这样的工具顶部 和 ps 对于 OS X,但它们
所以我有这个简单的 turtle 螺旋制作器,我想知道是否有一种方法可以打印出由该程序创建的我的设计副本。 代码: import turtle x= float(input("Angle: ")) y
我正在编写一个 C# WPF 程序,它将文本消息发送到另一个程序的窗口。我有一个宏程序作为我的键盘驱动程序 (Logitech g15) 的一部分,它已经这样做了,尽管它不会将击键直接发送到进程,而是
我尝试使用以下代码通过 UDP 发送,但得到了奇怪的结果。 if((sendto(newSocket, sendBuf, totalLength, 0, (SOCKADDR *)&sendAd
我是一名优秀的程序员,十分优秀!