[分享]外部exe窗体嵌入winform

失落的神庙 2013-01-11 12:00:43
加精
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Drawing;
using System.Data;
using System.Text;
using System.Windows.Forms;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.IO;
using System.Drawing.Design;

namespace HCDL
{
public class ShowForm
{
//-------------------
//Action<object, EventArgs> appIdleAction = null;
EventHandler appIdleEvent = null;

Control ParentCon = null;

string strGUID = "";

public ShowForm(Control C,string Titlestr)
{
appIdleEvent = new EventHandler(Application_Idle);
ParentCon = C;
strGUID = Titlestr;
}

/// <summary>
/// 将属性<code>AppFilename</code>指向的应用程序打开并嵌入此容器
/// </summary>
public IntPtr Start(string FileNameStr)
{
if (m_AppProcess != null)
{
Stop();
}
try
{
ProcessStartInfo info = new ProcessStartInfo(FileNameStr);
info.UseShellExecute = true;
info.WindowStyle = ProcessWindowStyle.Minimized;
m_AppProcess = System.Diagnostics.Process.Start(info);
m_AppProcess.WaitForInputIdle();
Application.Idle += appIdleEvent;
}
catch
{
if (m_AppProcess != null)
{
if (!m_AppProcess.HasExited)
m_AppProcess.Kill();
m_AppProcess = null;
}
}
return m_AppProcess.Handle;
}
/// <summary>
/// 确保应用程序嵌入此容器
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
void Application_Idle(object sender, EventArgs e)
{
if (this.m_AppProcess == null || this.m_AppProcess.HasExited)
{
this.m_AppProcess = null;
Application.Idle -= appIdleEvent;
return;
}
if (m_AppProcess.MainWindowHandle == IntPtr.Zero) return;
Application.Idle -= appIdleEvent;
EmbedProcess(m_AppProcess, ParentCon);
}
/// <summary>
/// 应用程序结束运行时要清除这里的标识
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
void m_AppProcess_Exited(object sender, EventArgs e)
{
m_AppProcess = null;
}
/// <summary>
/// 将属性<code>AppFilename</code>指向的应用程序关闭
/// </summary>
public void Stop()
{
if (m_AppProcess != null)// && m_AppProcess.MainWindowHandle != IntPtr.Zero)
{
try
{
if (!m_AppProcess.HasExited)
m_AppProcess.Kill();
}
catch (Exception)
{
}
m_AppProcess = null;
}
}


#region 属性
/// <summary>
/// application process
/// </summary>
Process m_AppProcess = null;

/// <summary>
/// 标识内嵌程序是否已经启动
/// </summary>
public bool IsStarted { get { return (this.m_AppProcess != null); } }

#endregion 属性

#region Win32 API
[DllImport("user32.dll", EntryPoint = "GetWindowThreadProcessId", SetLastError = true,
CharSet = CharSet.Unicode, ExactSpelling = true,
CallingConvention = CallingConvention.StdCall)]
private static extern long GetWindowThreadProcessId(long hWnd, long lpdwProcessId);

[DllImport("user32.dll", SetLastError = true)]
private static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

[DllImport("user32.dll", SetLastError = true)]
private static extern long SetParent(IntPtr hWndChild, IntPtr hWndNewParent);

[DllImport("user32.dll", EntryPoint = "GetWindowLongA", SetLastError = true)]
private static extern long GetWindowLong(IntPtr hwnd, int nIndex);

public static IntPtr SetWindowLong(HandleRef hWnd, int nIndex, int dwNewLong)
{
if (IntPtr.Size == 4)
{
return SetWindowLongPtr32(hWnd, nIndex, dwNewLong);
}
return SetWindowLongPtr64(hWnd, nIndex, dwNewLong);
}
[DllImport("user32.dll", EntryPoint = "SetWindowLong", CharSet = CharSet.Auto)]
public static extern IntPtr SetWindowLongPtr32(HandleRef hWnd, int nIndex, int dwNewLong);
[DllImport("user32.dll", EntryPoint = "SetWindowLongPtr", CharSet = CharSet.Auto)]
public static extern IntPtr SetWindowLongPtr64(HandleRef hWnd, int nIndex, int dwNewLong);

[DllImport("user32.dll", SetLastError = true)]
private static extern long SetWindowPos(IntPtr hwnd, long hWndInsertAfter, long x, long y, long cx, long cy, long wFlags);

[DllImport("user32.dll", SetLastError = true)]
private static extern bool MoveWindow(IntPtr hwnd, int x, int y, int cx, int cy, bool repaint);

[DllImport("user32.dll", EntryPoint = "PostMessageA", SetLastError = true)]
private static extern bool PostMessage(IntPtr hwnd, uint Msg, uint wParam, uint lParam);

[DllImport("user32.dll", SetLastError = true)]
private static extern IntPtr GetParent(IntPtr hwnd);

[DllImport("user32.dll", EntryPoint = "ShowWindow", SetLastError = true)]
static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);



private const int SWP_NOOWNERZORDER = 0x200;
private const int SWP_NOREDRAW = 0x8;
private const int SWP_NOZORDER = 0x4;
private const int SWP_SHOWWINDOW = 0x0040;
private const int WS_EX_MDICHILD = 0x40;
private const int SWP_FRAMECHANGED = 0x20;
private const int SWP_NOACTIVATE = 0x10;
private const int SWP_ASYNCWINDOWPOS = 0x4000;
private const int SWP_NOMOVE = 0x2;
private const int SWP_NOSIZE = 0x1;
private const int GWL_STYLE = (-16);
private const int WS_VISIBLE = 0x10000000;
private const int WM_CLOSE = 0x10;
private const int WS_CHILD = 0x40000000;

private const int SW_HIDE = 0; //{隐藏, 并且任务栏也没有最小化图标}
private const int SW_SHOWNORMAL = 1; //{用最近的大小和位置显示, 激活}
private const int SW_NORMAL = 1; //{同 SW_SHOWNORMAL}
private const int SW_SHOWMINIMIZED = 2; //{最小化, 激活}
private const int SW_SHOWMAXIMIZED = 3; //{最大化, 激活}
private const int SW_MAXIMIZE = 3; //{同 SW_SHOWMAXIMIZED}
private const int SW_SHOWNOACTIVATE = 4; //{用最近的大小和位置显示, 不激活}
private const int SW_SHOW = 5; //{同 SW_SHOWNORMAL}
private const int SW_MINIMIZE = 6; //{最小化, 不激活}
private const int SW_SHOWMINNOACTIVE = 7; //{同 SW_MINIMIZE}
private const int SW_SHOWNA = 8; //{同 SW_SHOWNOACTIVATE}
private const int SW_RESTORE = 9; //{同 SW_SHOWNORMAL}
private const int SW_SHOWDEFAULT = 10; //{同 SW_SHOWNORMAL}
private const int SW_MAX = 10; //{同 SW_SHOWNORMAL}

#endregion Win32 API

/// <summary>
/// 将指定的程序嵌入指定的控件
/// </summary>
private void EmbedProcess(Process app, Control control)
{
// Get the main handle
if (app == null || app.MainWindowHandle == IntPtr.Zero || control == null) return;
try
{
// Put it into this form
SetParent(app.MainWindowHandle, control.Handle);
}
catch (Exception)
{ }
try
{
// Remove border and whatnot
SetWindowLong(new HandleRef(this, app.MainWindowHandle), GWL_STYLE, WS_VISIBLE);
SendMessage(app.MainWindowHandle, WM_SETTEXT, IntPtr.Zero, strGUID);
}
catch (Exception)
{ }
try
{
// Move the window to overlay it on this window
MoveWindow(app.MainWindowHandle, -20, -20, control.Width, control.Height-10, true);
}
catch (Exception)
{ }
}

[DllImport("User32.dll", EntryPoint = "SendMessage")]
private static extern int SendMessage(IntPtr hWnd, int Msg, IntPtr wParam, string lParam);

const int WM_SETTEXT = 0x000C;
}
}







PS:以前是3.5版本。现在改成2.0也能用的了

用法


ShowForm Sf = new ShowForm(this, "仅供内部使用" + System.Guid.NewGuid().ToString());
ProxHandle= Sf.Start(Application.StartupPath + "\\ProxyThorn.exe");





第三方exe



winform效果图
...全文
93972 130 打赏 收藏 转发到动态 举报
写回复
用AI写文章
130 条回复
切换为时间正序
请发表友善的回复…
发表回复
飞天凤凰601 2015-09-15
  • 打赏
  • 举报
回复
这个东西好呀,谢谢!!!
sowinwork 2015-09-08
  • 打赏
  • 举报
回复
备用。。。。。。。。。
baidu_31130099 2015-09-07
  • 打赏
  • 举报
回复
要怎么把cmd命令窗口给嵌入进去啊
Jack_calt 2015-02-01
  • 打赏
  • 举报
回复
看起来很牛逼的样子
A971319231 2014-12-12
  • 打赏
  • 举报
回复 1
那个嵌进去的exe的关闭按钮和最小化按钮都不见了。。怎么解决?
A971319231 2014-12-10
  • 打赏
  • 举报
回复
请问一下楼主的可以再窗体里面拖动另外一个窗体吗?我的嵌进去的窗体的关闭按钮都不见了
a3212b12 2014-09-26
  • 打赏
  • 举报
回复
学习下,厉害。。。
zyj_604 2014-08-30
  • 打赏
  • 举报
回复
不错,标记一下。说不定以后能用到
失落的神庙 2014-08-27
  • 打赏
  • 举报
回复
引用 104 楼 tranlynn 的回复:
强大,要是源码能下载就好了。
引用 112 楼 liaodm 的回复:
这个程序,谁有联系到楼主,求楼主联系方式, 楼主如有看到回贴,烦请联系一下本人,有尝请教。77190062@qq.com
引用 111 楼 Gi_gi 的回复:
[quote=引用 31 楼 gupiao175 的回复:] 不好意思,没有注意看。有提供实例代码的。呵呵!
实例代码在哪里,我怎么没找到??[/quote] 96楼实例
  • 打赏
  • 举报
回复
Proxhandle 是什么东东?
磨砻淬砺 2014-08-19
  • 打赏
  • 举报
回复
好东西,记录下,以后慢慢学习。
d1070193993 2014-07-05
  • 打赏
  • 举报
回复
mark以后肯定有用
Ki1381 2014-07-04
  • 打赏
  • 举报
回复
牛叉,必须马克
xbl002 2013-12-19
  • 打赏
  • 举报
回复
标记下,以后也许会用到
sanweixianshi 2013-12-19
  • 打赏
  • 举报
回复
哈哈,学习,收藏一下!
riverswan 2013-12-18
  • 打赏
  • 举报
回复
真强 ......
txian001 2013-12-16
  • 打赏
  • 举报
回复
嵌入不了wps软件。做一个试试。
vanfy1983 2013-11-27
  • 打赏
  • 举报
回复
好帖子,感谢大家的热烈讨论,受益匪浅!
hao65717016 2013-11-22
  • 打赏
  • 举报
回复
楼主,有没有WPF版本的代码
chutao 2013-11-14
  • 打赏
  • 举报
回复
Good,收藏
加载更多回复(110)

110,499

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧