1

意图

使用以下代码,我设法在我的 Windows 窗体中加载了一些应用程序。

代码

这个函数的作用是...

  • 陈述一个过程
  • 将流程嵌入到我的表单面板中
  • 最大化嵌入式过程
  • 向面板添加调整大小事件处理程序以更新面板调整大小时嵌入进程的大小
  • 向表单添加关闭的事件处理程序以在表单关闭时终止嵌入式进程

用途

using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Windows.Forms;

常数

const int   GWL_STYLE   = -16;
const long  WS_VISIBLE  = 0x10000000,
            WS_MAXIMIZE = 0x01000000,
            WS_BORDER   = 0x00800000,
            WS_CHILD    = 0x40000000;

功能

IntPtr LoadExtern(Control Panel, string Path)
{
    try
    {
        Process Process = Process.Start(Path);
        Process.WaitForInputIdle();
        IntPtr Handle = Process.MainWindowHandle;
        SetParent(Handle, Panel.Handle);
        SetWindowLong(Handle, GWL_STYLE, (int)(WS_VISIBLE+(WS_MAXIMIZE|WS_BORDER)));

        MoveWindow(Handle, 0, 0, Panel.Width, Panel.Height, true);

        Panel.Resize += new EventHandler(
             delegate(object sender, EventArgs e)
             {
                  MoveWindow(Handle, 0, 0, Panel.Width, Panel.Height, true);
             }
        );

        this.FormClosed += new FormClosedEventHandler(
             delegate(object sender, FormClosedEventArgs e) {
                  SendMessage(Handle, 83, 0, 0);
                  Thread.Sleep(1000);
                  Handle = IntPtr.Zero;
             }
        );

        return Handle;
    }
    catch (Exception e) { MessageBox.Show(this, e.Message, "Error"); }
    return new IntPtr();
}

DLL 导入

[DllImport("user32.dll")]
static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);

[DllImport("user32.dll")]
static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);

[DllImport("user32.dll")]
static extern bool MoveWindow(IntPtr Handle, int x, int y, int w, int h, bool repaint);

[DllImport("user32.dll")]
static extern IntPtr SendMessage(IntPtr Handle, int Msg, int wParam, int lParam);

结果

此代码适用于某些应用程序,例如 Windows 记事本。记事本已启动并包含在我的表单面板中。没有标题,也没有边界,因为它应该是。

LoadExtern(panel1, "notepad.exe");

关闭表单后,嵌入式进程会按预期终止。

问题

不幸的是,我的代码不适用于其他一些(更大的)应用程序,如 firefox 或 sublimetext。

LoadExtern(panel2, @"C:\Program Files (x86)\Mozilla Firefox\firefox.exe");

发生的事情是我的表单启动并且 firefox 启动,但是在它自己的窗口中。你能帮我在我的应用程序中包含 sublimetext 或 firefox 吗?

部分解决方案

感谢盛江的回答,我让它适用于更多的应用程序。我所做的是等待主窗口句柄。

Process.WaitForInputIdle();
IntPtr Handle = new IntPtr();
for (int i = 0; Handle == IntPtr.Zero && i < 300; i++)
{
     Handle = Process.MainWindowHandle;
     Thread.Sleep(10);
}

但我仍然无法嵌入 Windows 资源管理器之类的应用程序。

4

2 回答 2

3

巧合的是,您的代码运行良好。

  • WaitForInputIdle 不需要等待 UI 线程。例如,由其他程序创建的输入法或挂钩可能会创建一个简单的线程,该线程在 UI 线程仍在忙于初始化时变得空闲。
  • MainWindowHandle 搜索第一个可见的顶级窗口。它不会返回逻辑主窗口
    • 主窗口不是创建的第一个可见窗口(例如,首先创建登录对话框)
    • 主窗口不是以可见样式创建的(考虑一个在系统托盘的通知区域中只有一个图标的程序)
    • 根本没有创建主窗口(例如,某些应用程序在现有实例中打开新文档/url,如浏览器和 Windows 资源管理器)
    • 没有一个主窗口,而是多个具有相同状态的顶级窗口。想想 IE6/Outlook/Word。

即使主窗口是可见创建的,实际上是新进程中的第一个可见窗口,您仍然可能遇到问题。

从 SetParent 的文档中:

应用程序可以使用 SetParent 函数来设置弹出窗口、重叠窗口或子窗口的父窗口。

它并没有说您可以重新设置顶级窗口。事实上,顶层窗口提供了许多程序可能依赖的服务,例如

  • 作为衡量工具来确定全屏请求是否完整(与您要求新程序需要出现在面板中的要求相冲突)
  • 当新的 DDE 对话开始时、活动窗口/程序更改时、新硬件到达时、系统设置更改时、用户注销时、Windows 资源管理器启动时、用户在嵌套上按 Enter 键时收到通知对话框等。仅发送到顶级窗口的窗口消息列表太长,无法在此处列出,
  • 如果程序选择,则充当模态对话框的默认所有者窗口(并且如果您也在程序中显示模态对话框,请注意崩溃
于 2012-09-29T16:06:27.213 回答
0

此代码适用于大多数应用程序。我仅在表单上使用 webbrowser 控件嵌入了文件资源管理器,并将其 url 设置为文件位置。然后,Internet Explorer 控件神奇地变成了文件浏览器。

这是我的最终代码,请随意将其用于您自己的项目。

IntPtr EmbedProcess(Control Panel, string Path)
{
    string Name = NameFromPath(Path);

    foreach (Process Task in Process.GetProcesses())
    {
        if (NameFromPath(Task.ProcessName).Contains(Name))
        {
            try { Task.Kill(); }
            catch (Exception e) {  }
        }
    }

    try
    {
        Process Task = Process.Start(Path);
        Task.WaitForInputIdle();
        IntPtr Handle = new IntPtr();
        for (int i = 0; Handle == IntPtr.Zero && i < 10; i++) { Handle = Task.MainWindowHandle; Thread.Sleep(100); }
        SetParent(Handle, Panel.Handle);
        SetWindowLong(Handle, GWL_STYLE, (int)(WS_VISIBLE + (WS_MAXIMIZE | WS_BORDER)));

        MoveWindow(Handle, 0, 0, Panel.Width, Panel.Height, true);

        Panel.Resize += new EventHandler(delegate(object sender, EventArgs e) { MoveWindow(Handle, 0, 0, Panel.Width, Panel.Height, true); });

        this.FormClosed += new FormClosedEventHandler(delegate(object sender, FormClosedEventArgs e)
        {
            SendMessage(Handle, 83, 0, 0);
            Thread.Sleep(100);
            Handle = IntPtr.Zero;
        });

        return Handle;
    }
    catch (Exception e) { MessageBox.Show(this, e.Message, "Error"); }
    return new IntPtr();
}

我有人对用于将窗口进程和控制台进程嵌入到表单中的洞 C# 类感兴趣,请查看此 github 存储库

于 2012-09-29T20:45:01.123 回答