【转】C#自定义控件:WinForm将其它应用程序窗体嵌入自己内部
PS:文末的附件已更新,這次我放到博客園里面了,不會彈出廣告,放心下載,O(∩_∩)O謝謝!
這是最近在做的一個(gè)項(xiàng)目中提到的需求,把一個(gè)現(xiàn)有的窗體應(yīng)用程序界面嵌入到自己開發(fā)的窗體中來,看起來就像自己開發(fā)的一樣(實(shí)際上……跟自己開發(fā)的還是有一點(diǎn)點(diǎn)區(qū)別的,就是內(nèi)嵌程序和宿主程序的窗口激活狀態(tài)問題)。
在codeproject找到了一篇相關(guān)的文章(http://www.codeproject.com/Articles/9123/Hosting-EXE-Applications-in-a-WinForm-project),雖然可用,但是很不方便,于是重新設(shè)計(jì)編寫了一個(gè)類庫,用一個(gè)控件完成內(nèi)嵌其它應(yīng)用程序的功能。
直接上圖先:
??
?
從打開Adobe Reader那張圖片可以看出來所謂的“內(nèi)嵌程序和宿主程序的窗口激活狀態(tài)問題”。當(dāng)內(nèi)嵌程序窗口激活時(shí),表面上將其包裹起來的宿主窗口卻處于非激活的狀態(tài)。想隱藏這一點(diǎn)的話,把窗口的FormBorderStyle屬性設(shè)為None吧,然后自己在窗口上畫關(guān)閉、最大化、最小化按鈕好了。
?
原作者的實(shí)現(xiàn)思路更能暴露本質(zhì),所以這里用原作者的代碼段解釋一下實(shí)現(xiàn)過程。
1、啟動要嵌入的應(yīng)用程序進(jìn)程
1 Process p = null; 2 try 3 { 4 // Start the process 5 p = System.Diagnostics.Process.Start(this.exeName); 6 7 // Wait for process to be created and enter idle condition 8 p.WaitForInputIdle(); 9 10 // Get the main handle 11 appWin = p.MainWindowHandle; 12 } 13 catch (Exception ex) 14 { 15 MessageBox.Show(this, ex.Message, "Error"); 16 }2、調(diào)用Windows API將啟動的應(yīng)用程序窗口嵌入自定義的控件(作者用的是Panel控件)
1 // Put it into this form 2 SetParent(appWin, this.Handle);//this在這里是Panel控件3 4 // Remove border and whatnot 5 SetWindowLong(appWin, GWL_STYLE, WS_VISIBLE); 6 7 // Move the window to overlay it on this window 8 MoveWindow(appWin, 0, 0, this.Width, this.Height, true);
3、設(shè)置被嵌入的窗體大小隨宿主窗體改變
1 protected override void OnResize(EventArgs e) 2 { 3 if (this.appWin != IntPtr.Zero) 4 { 5 MoveWindow(appWin, 0, 0, this.Width, this.Height, true); 6 } 7 base.OnResize (e); 8 }4、設(shè)置被嵌入的窗體應(yīng)用程序在宿主程序關(guān)閉時(shí)也關(guān)閉
1 protected override void OnHandleDestroyed(EventArgs e) 2 { 3 // Stop the application 4 if (appWin != IntPtr.Zero) 5 { 6 // Post a colse message 7 PostMessage(appWin, WM_CLOSE, 0, 0); 8 9 // Delay for it to get the message 10 System.Threading.Thread.Sleep(1000); 11 12 // Clear internal handle 13 appWin = IntPtr.Zero; 14 } 15 base.OnHandleDestroyed (e); 16 }?
原作者的代碼實(shí)際用起來是很不方便的,具體大家試試就知道,不細(xì)說了(反正我只學(xué)了學(xué)上面的步驟,也不用他的庫)。
本人開發(fā)了一個(gè)比較實(shí)用的控件,使用起來也很簡單,只需三步。
首先,在窗體應(yīng)用程序項(xiàng)目中引用類庫SmileWei.EmbeddedApp。
然后,在宿主窗體上拖一個(gè)AppContainer控件,擺放好位置。(如果工具箱里沒有AppContainer,就F6生成解決方案一下,然后再看就有了。)
最后,告訴AppContainer控件,要嵌入的應(yīng)用程序(*.exe文件)的絕對路徑(本人以使用OpenFileDialog為例),命令A(yù)ppContainer控件啟動之。
1 appContainer1.AppFilename = openEXE.FileName; 2 appContainer1.Start();這個(gè)AppContainer控件有什么好處呢?
1、原作者想到的Resize和隨宿主程序關(guān)閉而關(guān)閉的問題,AppContainer都實(shí)現(xiàn)了。
2、AppContainer指定要嵌入的應(yīng)用程序和啟動是分開的,這樣更靈活,開發(fā)過程中也不會看到如下的情況了:開發(fā)的時(shí)候原作者的控件就“情不自禁”地把內(nèi)嵌程序加載進(jìn)來了。
?
3、AppContainer防范了各種可能出錯(cuò)的情形,例如禁止自己嵌入自己(死循環(huán))、內(nèi)嵌Console程序時(shí)提示不能嵌入、參數(shù)為null或無效的檢驗(yàn)等。
4、其它。例如,AppContainer里面不會使用Thread.Sleep(1000);這樣低端的句子來保證程序正確地嵌入(而且對于類似photoshop這樣啟動很慢的程序也保證不了),而是通過Application.Ilde事件實(shí)現(xiàn)了在被嵌程序加載完畢后才將其窗體嵌入的技巧。
當(dāng)然,有些應(yīng)用程序是不能這么自動化地嵌入進(jìn)來的。因?yàn)槌绦騿哟绑w和主窗體句柄不一樣,AppContainer無法獲得主窗體句柄,所以無法自動嵌入。
為了解決這個(gè)問題,我在宿主窗體的狀態(tài)欄上設(shè)置了“句柄嵌入”標(biāo)簽,點(diǎn)擊“句柄嵌入”,你可以填入想嵌入的應(yīng)用程序主窗體句柄,然后宿主窗體就可以嵌入它了。
然后有同學(xué)就問了,我怎么知道想要嵌入的窗體句柄是多少啊?方法很多啦,我這里也提供一個(gè)自己制作的小程序,大家可以在這里下載:WindowDetective(窗口偵探)0.20.rar
界面是這個(gè)樣子的:
里面“句柄:{1903014}”那一行就給出了本人正在用的Windows Live Writer的主窗體句柄。
用法很簡單,啟動這個(gè)程序后,它會自動檢測鼠標(biāo)所在位置的窗體信息,顯示在窗口中。所以把鼠標(biāo)放在你想了解的窗體菜單欄上就OK了。QQ TM版也可以這樣嵌進(jìn)來滴。(QQ嵌不進(jìn)來,不知道騰訊在搞什么)
大家還可以試試把QQ對話框嵌進(jìn)來,很好玩哦~
?
?
我的源代碼都給出了明確的注釋,類型、變量名也都規(guī)范易懂,在此不再多做解釋了,有疑問請留言吧O(∩_∩)O
本文所有源代碼、可執(zhí)行程序均可在下面列出的鏈接中下載到。
示例宿主程序及類庫源代碼:SmileWei.EmbeddedApp.rar
示例宿主程序可執(zhí)行文件(及必需的類庫):SmileWei.EmbeddedApp.EXE.rar
窗口偵探(用于查看窗口句柄):WindowDetective(窗口偵探)0.20.rar
轉(zhuǎn)載請注明出處,http://www.cnblogs.com/bitzhuwei/archive/2012/05/24/SmileWei_EmbeddedApp.htmlO(∩_∩)O謝謝
?
轉(zhuǎn)載于:https://www.cnblogs.com/snake-hand/p/3163108.html
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎(jiǎng)勵(lì)來咯,堅(jiān)持創(chuàng)作打卡瓜分現(xiàn)金大獎(jiǎng)總結(jié)
以上是生活随笔為你收集整理的【转】C#自定义控件:WinForm将其它应用程序窗体嵌入自己内部的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: vb和vb.net事件机制
- 下一篇: 开业贺词范文模板汇集3篇