这是最近在做的一个项目中提到的需求,把一个现有的窗体应用程序界面嵌入到自己开发的窗体中来,看起来就像自己开发的一样(实际上……跟自己开发的还是有一点点区别的,就是内嵌程序和宿主程序的窗口激活状态问题)。

在codeproject找到了一篇相关的文章(http://www.codeproject.com/Articles/9123/Hosting-EXE-Applications-in-a-WinForm-project),虽然可用,但是很不方便,于是重新设计编写了一个类库,用一个控件完成内嵌其它应用程序的功能。

直接上图先:

从打开Adobe Reader那张图片可以看出来所谓的“内嵌程序和宿主程序的窗口激活状态问题”。当内嵌程序窗口激活时,表面上将其包裹起来的宿主窗口却处于非激活的状 态。想隐藏这一点的话,把窗口的FormBorderStyle属性设为None吧,然后自己在窗口上画关闭、最大化、最小化按钮好了。

原作者的实现思路更能暴露本质,所以这里用原作者的代码段解释一下实现过程。

1、启动要嵌入的应用程序进程

 Process p = null;
try
{
// Start the process
p = System.Diagnostics.Process.Start(this.exeName); // Wait for process to be created and enter idle condition
p.WaitForInputIdle(); // Get the main handle
appWin = p.MainWindowHandle;
}
catch (Exception ex)
{
MessageBox.Show(this, ex.Message, "Error");
}

2、调用Windows API将启动的应用程序窗口嵌入自定义的控件(作者用的是Panel控件)

 // Put it into this form
SetParent(appWin, this.Handle);//this在这里是Panel控件 // Remove border and whatnot
SetWindowLong(appWin, GWL_STYLE, WS_VISIBLE); // Move the window to overlay it on this window
MoveWindow(appWin, , , this.Width, this.Height, true);

3、设置被嵌入的窗体大小随宿主窗体改变

 protected override void OnResize(EventArgs e)
{
if (this.appWin != IntPtr.Zero)
{
MoveWindow(appWin, , , this.Width, this.Height, true);
}
base.OnResize (e);
}

4、设置被嵌入的窗体应用程序在宿主程序关闭时也关闭

 protected override void OnHandleDestroyed(EventArgs e)
{
// Stop the application
if (appWin != IntPtr.Zero)
{
// Post a colse message
PostMessage(appWin, WM_CLOSE, , ); // Delay for it to get the message
System.Threading.Thread.Sleep(); // Clear internal handle
appWin = IntPtr.Zero;
}
base.OnHandleDestroyed (e);
}

原作者的代码实际用起来是很不方便的,具体大家试试就知道,不细说了(反正我只学了学上面的步骤,也不用他的库)。

本人开发了一个比较实用的控件,使用起来也很简单,只需三步。

首先,在窗体应用程序项目中引用类库SmileWei.EmbeddedApp。

然后,在宿主窗体上拖一个AppContainer控件,摆放好位置。(如果工具箱里没有AppContainer,就F6生成解决方案一下,然后再看就有了。)

最后,告诉AppContainer控件,要嵌入的应用程序(*.exe文件)的绝对路径(本人以使用OpenFileDialog为例),命令AppContainer控件启动之。

  appContainer1.AppFilename = openEXE.FileName;
appContainer1.Start();

这个AppContainer控件有什么好处呢?

1、原作者想到的Resize和随宿主程序关闭而关闭的问题,AppContainer都实现了。

2、AppContainer指定要嵌入的应用程序和启动是分开的,这样更灵活,开发过程中也不会看到如下的情况了:开发的时候原作者的控件就“情不自禁”地把内嵌程序加载进来了。

3、AppContainer防范了各种可能出错的情形,例如禁止自己嵌入自己(死循环)、内嵌Console程序时提示不能嵌入、参数为null或无效的检验等。

4、其它。例如,AppContainer里面不会使用Thread.Sleep(1000);这样低端的句子来保证程序正确地嵌入(而且对于类似 photoshop这样启动很慢的程序也保证不了),而是通过Application.Ilde事件实现了在被嵌程序加载完毕后才将其窗体嵌入的技巧。

当然,有些应用程序是不能这么自动化地嵌入进来的。因为程序启动窗体和主窗体句柄不一样,AppContainer无法获得主窗体句柄,所以无法自动嵌入。

为了解决这个问题,我在宿主窗体的状态栏上设置了“句柄嵌入”标签,点击“句柄嵌入”,你可以填入想嵌入的应用程序主窗体句柄,然后宿主窗体就可以嵌入它了。

然后有同学就问了,我怎么知道想要嵌入的窗体句柄是多少啊?方法很多啦,我这里也提供一个自己制作的小程序,大家可以在这里下载:WindowDetective(窗口侦探)0.20.rar

界面是这个样子的:

里面“句柄:{1903014}”那一行就给出了本人正在用的Windows Live Writer的主窗体句柄。

用法很简单,启动这个程序后,它会自动检测鼠标所在位置的窗体信息,显示在窗口中。所以把鼠标放在你想了解的窗体菜单栏上就OK了。QQ TM版也可以这样嵌进来滴。(QQ嵌不进来,不知道腾讯在搞什么)

大家还可以试试把QQ对话框嵌进来,很好玩哦~

我的源代码都给出了明确的注释,类型、变量名也都规范易懂,在此不再多做解释了,有疑问请留言吧O(∩_∩)O

本文所有源代码、可执行程序均可在下面列出的链接中下载到。

示例宿主程序及类库源代码:SmileWei.EmbeddedApp.rar

示例宿主程序可执行文件(及必需的类库):SmileWei.EmbeddedApp.EXE.rar

窗口侦探(用于查看窗口句柄):WindowDetective(窗口侦探)0.20.rar

原文地址:http://www.cnblogs.com/bitzhuwei/archive/2012/05/24/SmileWei_EmbeddedApp.html

C#自定义控件:WinForm将其它应用程序窗体嵌入自己内部【转载】的更多相关文章

  1. 【转】C#自定义控件:WinForm将其它应用程序窗体嵌入自己内部

    PS:文末的附件已更新,这次我放到博客园里面了,不会弹出广告,放心下载,O(∩_∩)O谢谢! 这是最近在做的一个项目中提到的需求,把一个现有的窗体应用程序界面嵌入到自己开发的窗体中来,看起来就像自己开 ...

  2. C#WinForm窗体内Panel容器中嵌入子窗体、程序主窗体设计例子

    C#WinForm父级窗体内Panel容器中嵌入子窗体.程序主窗体设计例子 在项目开发中经常遇到父级窗体嵌入子窗体所以写了一个例子程序,顺便大概划分了下界面模块和配色,不足之处还望指点 主窗体窗体采用 ...

  3. 外部exe窗体嵌入winform

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Drawing; u ...

  4. C#WinForm应用程序中嵌入ECharts图表

    C#WinForm应用程序中嵌入ECharts图表 程序运行效果: 下载ECharts: 官网下载ECharts :http://echarts.baidu.com/download.html 或者直 ...

  5. 在WinForm应用程序中嵌入WPF控件

    我们知道,在WPF界面上添加WinForm的控件需要使用WindowsFormHost类.而在WinForm界面上添加WPF控件该如何做呢?有没有类似的类呢?明显是有的,ElementHost就是为了 ...

  6. Delphi实现窗体内嵌其他应用程序窗体

    实现原理是启动一个应用程序,通过ProcessID得到窗体句柄,然后对其设定父窗体句柄为本程序某控件句柄(本例是窗体内一个Panel的句柄),这样就达成了内嵌的效果. 本文实现的是内嵌一个记事本程序, ...

  7. SNF开发平台WinForm之八-自动升级程序部署使用说明-SNF快速开发平台3.3-Spring.Net.Framework

    9.1运行效果: 9.2开发实现: 1.首先配置服务器端,把“SNFAutoUpdate2.0\服务器端部署“目录按网站程序进行发布到IIS服务器上. 2.粘贴语句,生成程序 需要调用的应用程序的Lo ...

  8. Winform中如何实现父窗体传递数据到子窗体并刷新子窗体

    原理:利用委托和事件,本文将以图文并茂的例子讲述,告诉我们So Easy --------------------------------------------------------------- ...

  9. 通过 WIN32 API 实现嵌入程序窗体

    写了一个不使用 COM, 而是通过 WIN32 API 实现的示例, 它把写字板程序嵌在了自己的一个面板中. 这么做可能没有实际意义, 因为两个程序之前没有进行有价值的交互, 这里仅仅是为了演示这么做 ...

随机推荐

  1. sizeof(结构体)的计算

    摘要: 经常被计算结构体的sizeof给搞晕,于是找了个时间,静下心来,搞定它. 一.为什么结构体计算这么乱? 答案是字节对齐,计算机存储系统中以Byte为单位存储数据,不同数据类型所占的空间不同,如 ...

  2. Grunt 初体验

    对于没有接触过类似自动化工具的朋友,对 grunt 也许只是停留在听过阶段,而并没有真正的使用过.今天就从最初级的教程说起.在开始教程之前,需要先确保你已经安装了 node. 下面就开始来讲解 gru ...

  3. Glimpse

    给自己程序配好Glimpse. Glimpse.Mvc 有问题 遇到稍微复杂点的内套多个PartialView,内存就爆了彪1.7g,不开Glimpse一点问题都没.另外Glimpse.Nlog也有问 ...

  4. BZOJ3509: [CodeChef] COUNTARI

    3509: [CodeChef] COUNTARI Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 339  Solved: 85[Submit][St ...

  5. I.MX6 Android iperf3 porting failed

    /***************************************************************************** * I.MX6 Android iperf ...

  6. (七)学习MVC之CodeFirst迁移更新数据库

    1.首先在程序包管理控制台输入:enable-migrations -force ,然后回车: 问题1: The EntityFramework package is not installed on ...

  7. Entity Framework4.0 (七) EF4的存储过程

    前面了解了EF4的CRUD的操作,你会发现EF4使用起来比较简单的.呵呵,之前我们使用数据库的时候,有时会使用存储过程代替在代码中直接使用SQL语句. 使用存储过程的好处: 提高效率:因为存储过程是经 ...

  8. 我的WCF之旅(1):创建一个简单的WCF程序

    为了使读者对基于WCF的编程模型有一个直观的映像,我将带领读者一步一步地创建一个完整的WCF应用.本应用功能虽然简单,但它涵盖了一个完整WCF应用的基本结构.对那些对WCF不是很了解的读者来说,这个例 ...

  9. 黑盒测试用例设计方法&理论结合实际 -> 正交试验法

    一. 概念 依据Galois理论,从大量的(实验)数据(测试例)中挑选适量的,有代表性的点(例),从而合理地安排实验(测试)的一种科学实验设计方法.类似的方法有:聚类分析方法,因子方法方法等. 二. ...

  10. DNS(一)简介

    最近学习相关DNS知识,顺便总结下相关内容. 1.什么是DNS DNS(Domain Name System)服务,可以使用域名代替复杂的IP地址来访问网络服务器,使得网络服务的访问更加简单,而且可以 ...