C#的winform如何嵌套另一個(gè)exe程序
C#winform嵌套另一個(gè)exe程序
一共有二種方法,也不知道作者從哪里復(fù)制來(lái)的,先感謝原作者。
首先建立一個(gè)程序,加2個(gè)按鈕,為了區(qū)分,界面修改成紅色。
第一種
1.建立一個(gè)主程序,加一個(gè)panel1,為了區(qū)分背景是綠色
2.代碼調(diào)用
3.所有代碼
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Diagnostics; using System.Drawing; using System.Linq; using System.Runtime.InteropServices; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; namespace WindowsFormsApp1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } /// <summary> /// 嵌入外部exe /// </summary> public class EmbeddedExeTool { [DllImport("User32.dll", EntryPoint = "SetParent")] private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent); [DllImport("user32.dll", EntryPoint = "ShowWindow")] private static extern int ShowWindow(IntPtr hwnd, int nCmdShow); [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")] private static extern int GetWindowLong(IntPtr hWnd, int nIndex); [DllImport("user32.dll")] private static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong); [DllImport("user32.dll", SetLastError = true)] private static extern IntPtr FindWindow(string lpClassName, string lpWindowName); IntPtr WindowHandle = IntPtr.Zero; private const int WS_THICKFRAME = 262144; private const int WS_BORDER = 8388608; private const int GWL_STYLE = -16; private const int WS_CAPTION = 0xC00000; private Process proApp = null; private Control ContainerControl = null; private const int WS_VISIBLE = 0x10000000; [DllImport("user32.dll", EntryPoint = "SetWindowLong", CharSet = CharSet.Auto)] private static extern IntPtr SetWindowLongPtr32(HandleRef hWnd, int nIndex, int dwNewLong); [DllImport("user32.dll", EntryPoint = "SetWindowLongPtr", CharSet = CharSet.Auto)] private static extern IntPtr SetWindowLongPtr64(HandleRef hWnd, int nIndex, int dwNewLong); private IntPtr SetWindowLong(HandleRef hWnd, int nIndex, int dwNewLong) { if (IntPtr.Size == 4) { return SetWindowLongPtr32(hWnd, nIndex, dwNewLong); } return SetWindowLongPtr64(hWnd, nIndex, dwNewLong); } /// <summary> /// 加載外部exe程序到程序容器中 /// </summary> /// <param name="control">要顯示exe的容器控件</param> /// <param name="exepath">exe的完整絕對(duì)路徑</param> public void LoadEXE(Control control, string exepath) { ContainerControl = control; control.SizeChanged += Control_SizeChanged; ProcessStartInfo info = new ProcessStartInfo(exepath); info.WindowStyle = ProcessWindowStyle.Minimized; info.UseShellExecute = false; info.CreateNoWindow = false; proApp = Process.Start(info); Application.Idle += Application_Idle; EmbedProcess(proApp, control); } /// <summary> /// 加載外部exe程序到程序容器中 /// </summary> /// <param name="form">要顯示exe的窗體</param> /// <param name="exepath">exe的完整絕對(duì)路徑</param> public void LoadEXE(Form form, string exepath) { ContainerControl = form; form.SizeChanged += Control_SizeChanged; proApp = new Process(); proApp.StartInfo.UseShellExecute = false; proApp.StartInfo.CreateNoWindow = false; proApp.StartInfo.WindowStyle = ProcessWindowStyle.Minimized; proApp.StartInfo.FileName = exepath; proApp.StartInfo.Arguments = Process.GetCurrentProcess().Id.ToString(); proApp.Start(); Application.Idle += Application_Idle; EmbedProcess(proApp, form); } /// <summary> /// 確保應(yīng)用程序嵌入此容器 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void Application_Idle(object sender, EventArgs e) { if (this.proApp == null || this.proApp.HasExited) { this.proApp = null; Application.Idle -= Application_Idle; return; } if (proApp.MainWindowHandle == IntPtr.Zero) return; Application.Idle -= Application_Idle; EmbedProcess(proApp, ContainerControl); } /// <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); // Remove border and whatnot SetWindowLong(new HandleRef(this, app.MainWindowHandle), GWL_STYLE, WS_VISIBLE); ShowWindow(app.MainWindowHandle, (int)ProcessWindowStyle.Maximized); MoveWindow(app.MainWindowHandle, 0, 0, control.Width, control.Height, true); } catch (Exception ex3) { Console.WriteLine(ex3.Message); } } /// <summary> /// 嵌入容器大小改變事件 /// </summary> private void Control_SizeChanged(object sender, EventArgs e) { if (proApp == null) { return; } if (proApp.MainWindowHandle != IntPtr.Zero && ContainerControl != null) { MoveWindow(proApp.MainWindowHandle, 0, 0, ContainerControl.Width, ContainerControl.Height, true); } } } private void Form1_Load(object sender, EventArgs e) { EmbeddedExeTool exetool = new EmbeddedExeTool(); //WindowsFormsApp4.exe 為要嵌入外部exe的具體路徑 exetool.LoadEXE(panel1, AppDomain.CurrentDomain.BaseDirectory+ "WindowsFormsApp4.exe");//debug下面的文件夾 } } }
4.效果很好。
紅配綠,絕配。
第二種
和第一種方式有點(diǎn)不一樣。
1.建立一個(gè)winform主程序,為了區(qū)分背景是綠色
2.把代碼直接復(fù)制進(jìn)去,修改命名空間,再加上一個(gè)load事件即可
3.代碼
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Diagnostics; using System.Runtime.InteropServices; namespace WindowsFormsApp1 { public partial class Form1 : Form { private const int GWL_STYLE = (-16); private const int WS_VISIBLE = 0x10000000; EventHandler appIdleEvent = null; Action<object, EventArgs> appIdleAction = null; Process m_AppProcess; [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 SetParent(IntPtr hWndChild, IntPtr hWndNewParent); //SetParent(IntPtr hWndChild, IntPtr hWndNewParent);這個(gè)方法很重要,就是將hWndChild指向開啟exe的窗體嵌入到hWndNewParent窗體的某個(gè)控件上,或者是窗體本 身的容器 [DllImport("user32.dll", SetLastError = true)] private static extern bool MoveWindow(IntPtr hwnd, int x, int y, int cx, int cy, bool repaint); // MoveWindow(IntPtr hwnd, int x, int y, int cx, int cy, bool repaint);這個(gè)方法是windows的api,見(jiàn)名知意,是移動(dòng)hwnd所指的窗體到指定的位置,并且指定是否需要重繪 public static IntPtr SetWindowLong(HandleRef hWnd, int nIndex, int dwNewLong) { if (IntPtr.Size == 4) { return SetWindowLongPtr32(hWnd, nIndex, dwNewLong); } return SetWindowLongPtr64(hWnd, nIndex, dwNewLong); } public Form1() { InitializeComponent(); appIdleAction = new Action<object, EventArgs>(Application_Idle); appIdleEvent = new EventHandler(appIdleAction); } /// <summary> /// 確保應(yīng)用程序嵌入此容器,再次確認(rèn)exe嵌入,如果不調(diào)用這個(gè)方法,程序不一定能嵌入外部exe /// </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;//這一步一直不知道有什么用,但是不用這行代碼程序有時(shí)候能嵌入有時(shí)候又不行 return; } if (m_AppProcess.MainWindowHandle == IntPtr.Zero) return; Application.Idle -= appIdleEvent; EmbedProcess(m_AppProcess, this); } /// <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 ex1) { Console.WriteLine(ex1.Message); } try { // Remove border and whatnot SetWindowLong(new HandleRef(this, app.MainWindowHandle), GWL_STYLE, WS_VISIBLE); } catch (Exception ex2) { Console.WriteLine(ex2.Message); } try { MoveWindow(app.MainWindowHandle, 0, 0, control.Width, control.Height, true); } catch (Exception ex3) { Console.WriteLine(ex3.Message); } } private void Form1_Load(object sender, EventArgs e) { //以下這段代碼是通過(guò)命令行方式調(diào)起一個(gè)exe程序,獲取這個(gè)程序的句柄然后嵌入主的winform窗體中 ProcessStartInfo info = new ProcessStartInfo(AppDomain.CurrentDomain.BaseDirectory + "WindowsFormsApp4.exe");//debug下面的文件夾 info.WindowStyle = ProcessWindowStyle.Minimized; info.UseShellExecute = false; info.CreateNoWindow = false; m_AppProcess = System.Diagnostics.Process.Start(info); Application.Idle += appIdleEvent; try { EmbedProcess(m_AppProcess, this); } catch (Exception ex) { Console.WriteLine(ex.Message); } } } }
4.效果
可見(jiàn),第一種和第二種的效果有所區(qū)別的。
拓展。
WPF簽入winform
1.依然用上面的winform程序,把輸出類型改成類庫(kù)
2.建立一個(gè)WPF程序,引用System.Windows.Forms和WindowsFormsIntegration,紅色。綠色是步驟1生產(chǎn)的dll
3.在wpf中增加WindowsFormsHost控件
4.cs后臺(tái)代碼
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows; using System.Windows.Controls; using System.Windows.Data; using System.Windows.Documents; using System.Windows.Input; using System.Windows.Media; using System.Windows.Media.Imaging; using System.Windows.Navigation; using System.Windows.Shapes; using WindowsFormsApp4; namespace WpfApp1 { /// <summary> /// MainWindow.xaml 的交互邏輯 /// </summary> public partial class MainWindow : Window { public MainWindow() { InitializeComponent(); Form1 mainform = new Form1(); mainform.TopLevel = false; winform.Child = mainform; } } }
5.效果
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
淺談C#跨線程調(diào)用窗體控件(比如TextBox)引發(fā)的線程安全問(wèn)題
下面小編就為大家分享一篇淺談C#跨線程調(diào)用窗體控件(比如TextBox)引發(fā)的線程安全問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2017-11-11C#調(diào)用打印機(jī)實(shí)現(xiàn)打印
這篇文章介紹了C#調(diào)用打印機(jī)實(shí)現(xiàn)打印的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04使用C#實(shí)現(xiàn)阿拉伯?dāng)?shù)字到大寫中文的轉(zhuǎn)換
這篇文章主要介紹了C#實(shí)現(xiàn)阿拉伯?dāng)?shù)字轉(zhuǎn)為大寫中文的實(shí)現(xiàn)代碼,需要的朋友可以參考下2007-03-03C#基于Sockets類實(shí)現(xiàn)TCP通訊
這篇文章主要為大家詳細(xì)介紹了C#基于Sockets類實(shí)現(xiàn)TCP通訊,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01WPF使用Dragablz構(gòu)建可拖拽分離的Tab頁(yè)程序
這篇文章介紹了WPF使用Dragablz構(gòu)建可拖拽分離Tab頁(yè)的方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06Unity UGUI的InputField輸入框組件使用詳解
這篇文章主要為大家介紹了Unity UGUI的InputField輸入框組件使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07