分享

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

 quasiceo 2012-11-29

[代码]Delphi实现窗体内嵌其他应用程序窗体

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

本文实现的是内嵌一个记事本程序,如下图:

内嵌程序

在实现细节上需要注意几点

  1. 为了美化程序的嵌入效果,需要隐藏其标题栏
  2. 在外部窗体大小变化时,需要内嵌的窗体也随之变化大小
  3. 外部程序退出时,内嵌的程序也要退出

下面是例子程序。新建窗体,上面放置一个Panel控件,名为pnlApp,然后按下面代码编写:

unit frmTestEmbedApp;
 
interface
 
uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, ExtCtrls;
 
type
 
  TForm1 = class(TForm)
    pnlApp: TPanel;
    procedure FormCreate(Sender: TObject);
    procedure FormClose(Sender: TObject; var Action: TCloseAction);
    procedure FormResize(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
  end;
 
var
  Form1: TForm1;
  hWin: HWND = 0;
 
implementation
 
{$R *.dfm}
 
type
  // 存储窗体信息
  PProcessWindow = ^TProcessWindow;
  TProcessWindow = record
    ProcessID: Cardinal;
    FoundWindow: hWnd;
  end;
 
// 窗体枚举函数
 
function EnumWindowsProc(Wnd: HWND; ProcWndInfo: PProcessWindow): BOOL; stdcall;
var
  WndProcessID: Cardinal;
begin
  GetWindowThreadProcessId(Wnd, @WndProcessID);
  if WndProcessID = ProcWndInfo^.ProcessID then begin
    ProcWndInfo^.FoundWindow := Wnd;
    Result := False;                                    // 已找到,故停止 EnumWindows
  end
  else
    Result := True;                                     // 继续查找
end;
 
// 由 ProcessID 查找窗体 Handle
 
function GetProcessWindow(ProcessID: Cardinal): HWND;
var
  ProcWndInfo: TProcessWindow;
begin
  ProcWndInfo.ProcessID := ProcessID;
  ProcWndInfo.FoundWindow := 0;
  EnumWindows(@EnumWindowsProc, Integer(@ProcWndInfo)); // 查找窗体
  Result := ProcWndInfo.FoundWindow;
end;
 
// 在 Panel 上内嵌运行程序
 
function RunAppInPanel(const AppFileName: string; ParentHandle: HWND; var WinHandle: HWND): Boolean;
var
  si: STARTUPINFO;
  pi: TProcessInformation;
begin
  Result := False;
 
  // 启动进程
  FillChar(si, SizeOf(si), 0);
  si.cb := SizeOf(si);
  si.wShowWindow := SW_SHOW;
  if not CreateProcess(nil, PChar(AppFileName), nil, nil, true,
    CREATE_NEW_CONSOLE or NORMAL_PRIORITY_CLASS, nil, nil, si, pi) then Exit;
 
  // 等待进程启动
  WaitForInputIdle(pi.hProcess, 10000);
 
  // 取得进程的 Handle
  WinHandle := GetProcessWindow(pi.dwProcessID);
  if WinHandle > 0 then begin
    // 设定父窗体
    Windows.SetParent(WinHandle, ParentHandle);
 
    // 设定窗体位置
    SetWindowPos(WinHandle, 0, 0, 0, 0, 0, SWP_NOSIZE or SWP_NOZORDER);
 
    // 去掉标题栏
    SetWindowLong(WinHandle, GWL_STYLE, GetWindowLong(WinHandle, GWL_STYLE)
      and (not WS_CAPTION) and (not WS_BORDER) and (not WS_THICKFRAME));
 
    Result := True;
  end;
 
  // 释放 Handle
  CloseHandle(pi.hProcess);
  CloseHandle(pi.hThread);
end;
 
procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
  // 退出时向内嵌程序发关闭消息
  if hWin > 0 then PostMessage(hWin, WM_CLOSE, 0, 0);
end;
 
procedure TForm1.FormCreate(Sender: TObject);
const
  App = 'C:\Windows\Notepad.exe';
begin
  pnlApp.Align := alClient;
 
  // 启动内嵌程序
  if not RunAppInPanel(App, pnlApp.Handle, hWin) then ShowMessage('App not found');
end;
 
procedure TForm1.FormResize(Sender: TObject);
begin
  // 保持内嵌程序充满 pnlApp
  if hWin <> 0 then MoveWindow(hWin, 0, 0, pnlApp.ClientWidth, pnlApp.ClientHeight, True);
end;
 
end.

这种方式也存在几个问题:

问题1:如果程序有Splash窗体先显示,则实际窗体无法内嵌,因为仅将Splash窗体的父窗体设定为本程序的控件句柄,后续窗体无法设定。

解决方法:可以通过轮询方式查询后续窗体,并设定其父窗体为本程序的控件句柄。

问题2:点击内嵌程序的窗体,则本程序的标题栏失去焦点

解决方法:不详。

问题3:点击内嵌程序的窗体,按下ALT+F4,则内嵌程序退出,仅留下本程序

解决方法:可以通过Hook方式拦截ALT+F4。

爱生活,爱拉风
分类: Delphi

博主后一篇:[笔记]如何屏蔽视频网站的片头广告——优酷
posted @ 2011-07-21 23:48 ET民工[源自火星] 阅读(1169) 评论(12) 编辑 收藏
#1楼 2011-08-15 13:39 vcc  

请教 ET民工
我若改执行其他exe
1.
'C:\Windows\system32\mspaint.exe';
'C:\Windows\system32\calc.exe';
加sleep(500) 还可以include 进来

但'C:\Program Files\Internet Explorer\iexplore.exe';
就有点难

2.我另外自己写的exe 被include後,在下面的windows的工作列,会像一般exe一样有ICON 出现,这能将他 隐藏吗?像记事本就不会出现
#2楼[楼主] 2011-08-15 16:06 ET民工[源自火星]  
@vcc
1.
'C:\Windows\system32\mspaint.exe';
'C:\Windows\system32\calc.exe';
加sleep(500) 还可以include 进来

但'C:\Program Files\Internet Explorer\iexplore.exe';
就有点难

不建议嵌入IE,建议使用WebBrowser控件

2.我另外自己写的exe 被include後,在下面的windows的工作列,会像一般exe一样有ICON 出现,这能将他 隐藏吗?像记事本就不会出现

可以参考这篇文章,不过我没有测试,仅供参考吧:
利用ITaskbarList接口隐藏任务栏图标
http://blog.csdn.net/3150379/article/details/3169335

#3楼 2011-08-15 17:15 vcc  
感谢指导
2个方法都未能成功

利用ITaskbarList接口隐藏任务栏图标 只能隐藏 主程序application.handle


而是要内嵌delphi 写的exe,,
不要让内嵌的exe ICON 出现在下方任务档
(利用ITaskbarList接口隐藏任务栏图标)
未能成功

#6楼[楼主] 2011-08-17 11:45 ET民工[源自火星] 
@vcc
如果内嵌的程序是你自己写的,可以将其主窗体直接当作外壳程序的子窗体,这样就没问题了。我的一个程序就这么做的,运行良好。
或者将内嵌的程序修改为运行时不显示在任务栏按钮里,这样可以达到目的,当然,前提是你可以修改内嵌程序的代码。

#7楼 2011-08-17 12:20 vcc  
1.内嵌的程序是自己写的,但是我要他成为单一exe,而不要为MDI 子视窗方式,,我要用外壳程序 一次遥控多个同样的exe,
exe可以同时同步运行,,
MDI子视窗无法实现同时同步运行

2.最好是 A.像记事本一样 ,一般exe 内嵌後,不显示在任务栏按钮里
(包括任何delphi exe[我内嵌他人写的delphi TILED.exe 也一样会显示,所以我怀疑delphi exe 比较特殊])

B.要如何 内嵌的程序修改为运行时不显示在任务栏按钮里

#8楼[楼主] 2011-08-17 12:32 ET民工[源自火星]  

我的做法不是使用MDI(MDI已经过时了),如下:
1、外壳程序只做外壳的界面,但是里面有个PageControl控件,每一页是一个子窗体的容器。
2、初始化外壳程序时,读取配置文件,初始化每个子窗体程序
3、外壳程序可以通过Notify模式和子窗体交互,说白了就是双向广播
4、子窗体因为是Form级的被调用,所以和主窗体是完整的一个程序,不存在多余进程

以上的基础是,外壳和子窗体都是Delphi程序,都是源码级别
#9楼 2011-08-17 13:13 vcc 
外壳将多个子窗体定义为窗体数组。
同时控制,可以通过遍历数组每一项来发送指令和获取回复。
外壳程序 用PageControl 将子视窗 分页管理?
外壳程序 若是sendmessage 给所有子视窗(子视窗内的程序是独立执行绪吗?)
我的子程式是控制USB,,我利用外壳程序 同时 同步控制多个子程序(多个USB)

我试过MDI 将控制USB作成子Form,,但发现 只能有单一子视窗运作,,
所以往Thread 跟 外壳程序 2方面找资料,,
未尝试 Thread我的[控制USB](担心debug...)
刚好找到了 楼主这篇,,用了几天 大致都能得到我的预期需求
现在就是 任务栏按钮 会显示出来 处理不来,,



    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多