首页 > 其他分享 >在winform中如何嵌入第三方软件窗体✨

在winform中如何嵌入第三方软件窗体✨

时间:2024-03-07 10:34:35浏览次数:31  
标签:IntPtr 嵌入 窗口 句柄 窗体 SetParent static winform

相关win32api的学习✨

SetParent

[DllImport("user32.dll ", EntryPoint = "SetParent")]
private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);   //将外部窗体嵌入程序

image-20240305202650242

语法:

HWND SetParent(
  [in]           HWND hWndChild,
  [in, optional] HWND hWndNewParent
);

参数:

参数名 类型 含义
hWndChild HWND 子窗口的句柄
hWndNewParent HWND 新父窗口的句柄。 如果此参数为 NULL,桌面窗口将成为新的父窗口。 如果此参数 HWND_MESSAGE,则子窗口将成为 仅消息窗口

相关解释:

什么是句柄

在计算机编程和操作系统中,句柄(Handle)是一个用于标识和引用对象或资源的抽象概念。它通常是一个整数值或指针,充当对特定资源的引用或访问标识符。句柄用于管理内存、设备、文件、窗口等各种资源。

句柄在操作系统中被广泛使用,特别是在图形用户界面(GUI)应用程序中。例如,在Windows操作系统中,窗口句柄(Window Handle)用于标识和操作窗口对象。每个窗口都有一个唯一的句柄,通过该句柄可以执行诸如移动、调整大小、关闭等操作。

另一个常见的例子是文件句柄(File Handle),它用于标识和操作打开的文件。通过文件句柄,程序可以读取、写入或关闭文件。

句柄的使用可以提高程序的安全性和效率。它们允许程序通过句柄而不是直接访问资源,从而隐藏底层实现细节并提供一种统一的接口。此外,句柄还可以用于实现资源的共享和保护,通过对句柄的权限管理来控制对资源的访问。

总的来说,句柄是一种重要的编程概念,用于标识和管理各种资源,从而使程序能够有效地操作系统资源,并提供安全和统一的访问接口。

FindWindow

 [DllImport("user32.dll")]
 public static extern IntPtr FindWindow(string lpszClass, string lpszWindow);      //按照窗体类名或窗体标题查找窗体

作用:检索其类名和窗口名称与指定字符串匹配的顶级窗口的句柄。 此函数不搜索子窗口。 此函数不执行区分大小写的搜索。

若要从指定的子窗口开始搜索子窗口,请使用 FindWindowEx 函数。

语法:

HWND FindWindowA(
  [in, optional] LPCSTR lpClassName,
  [in, optional] LPCSTR lpWindowName
);

参数:

参数名 类型 含义
lpClassName LPCTSTR 如果 lpClassName 指向字符串,则指定窗口类名。
lpWindowName LPCTSTR 窗口名称 (窗口标题) 。 如果此参数为 NULL,则所有窗口名称都匹配。

ShowWindow

image-20240306122438035

作用:设置指定窗口的显示状态。

语法:

HWND FindWindowA(
  [in, optional] LPCSTR lpClassName,
  [in, optional] LPCSTR lpWindowName
);

参数:

参数名 类型 含义
hWnd HWND 窗口的句柄。
nCmdShow int 控制窗口的显示方式。

nCmdShow不同值与含义:

0 隐藏窗口并激活另一个窗口。
1 激活并显示窗口。 如果窗口最小化、最大化或排列,系统会将其还原到其原始大小和位置。 应用程序应在首次显示窗口时指定此标志。
2 激活窗口并将其显示为最小化窗口。
3 激活窗口并显示最大化的窗口。
4 以最近的大小和位置显示窗口。
5 激活窗口并以当前大小和位置显示窗口。
6 最小化指定的窗口,并按 Z 顺序激活下一个顶级窗口。
7 将窗口显示为最小化窗口。
8 以当前大小和位置显示窗口。
9 激活并显示窗口。 如果窗口最小化、最大化或排列,系统会将其还原到其原始大小和位置。 还原最小化窗口时,应用程序应指定此标志。
10 根据启动应用程序的程序传递给 CreateProcess 函数的 STARTUPINFO 结构中指定的SW_值设置显示状态。
11 最小化窗口,即使拥有窗口的线程没有响应。 仅当最小化不同线程的窗口时,才应使用此标志。

创建一个静态类✨

为了便于进行相关的操作,创建一个静态类:

  public static class WindowManager
  {
      public static IntPtr intPtr;         //第三方应用窗口的句柄

      /// <summary>
      /// 调整第三方应用窗体大小
      /// </summary>
      public static void ResizeWindow()
      {
          ShowWindow(intPtr, 0);  //先将窗口隐藏
          ShowWindow(intPtr, 3);  //再将窗口最大化,可以让第三方窗口自适应容器的大小
      }

      /// <summary>
      /// 循环查找第三方窗体
      /// </summary>
      /// <returns></returns>
      public static bool FindWindow(string formName)
      {
          for (int i = 0; i < 100; i++)
          {
              //按照窗口标题查找Python窗口
              IntPtr vHandle = FindWindow(null, formName);
              if (vHandle == IntPtr.Zero)
              {
                  Thread.Sleep(100);  //每100ms查找一次,直到找到,最多查找10s
                  continue;
              }
              else      //找到返回True
              {
                  intPtr = vHandle;
                  return true;
              }
          }
          intPtr = IntPtr.Zero;
          return false;
      }


      /// <summary>
      /// 将第三方窗体嵌入到容器内
      /// </summary>
      /// <param name="hWndNewParent">父容器句柄</param>
      /// <param name="windowName">窗体名</param>
      public static void SetParent(IntPtr hWndNewParent, string windowName)
      {
          ShowWindow(intPtr, 0);                 //先将窗体隐藏,防止出现闪烁
          SetParent(intPtr, hWndNewParent);      //将第三方窗体嵌入父容器                    
          Thread.Sleep(100);                      //略加延时
          ShowWindow(intPtr, 3);                 //让第三方窗体在容器中最大化显示
          RemoveWindowTitle(intPtr);             // 去除窗体标题
      }


      /// <summary>
      /// 去除窗体标题
      /// </summary>
      /// <param name="vHandle">窗口句柄</param>
      public static void RemoveWindowTitle(IntPtr vHandle)
      {
          long style = GetWindowLong(vHandle, -16);
          style &= ~0x00C00000;
          SetWindowLong(vHandle, -16, style);
      }


      #region API 需要using System.Runtime.InteropServices;

      [DllImport("user32.dll ", EntryPoint = "SetParent")]
      private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);   //将外部窗体嵌入程序

      [DllImport("user32.dll")]
      public static extern IntPtr FindWindow(string lpszClass, string lpszWindow);      //按照窗体类名或窗体标题查找窗体

      [DllImport("user32.dll", EntryPoint = "ShowWindow", CharSet = CharSet.Auto)]
      private static extern int ShowWindow(IntPtr hwnd, int nCmdShow);                  //设置窗体属性

      [DllImport("user32.dll", EntryPoint = "SetWindowLong", CharSet = CharSet.Auto)]
      public static extern IntPtr SetWindowLong(IntPtr hWnd, int nIndex, long dwNewLong);

      [DllImport("user32.dll", EntryPoint = "GetWindowLong", CharSet = CharSet.Auto)]
      public static extern long GetWindowLong(IntPtr hWnd, int nIndex);

      #endregion
  }

首先查看最下方的内容,以

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

为例进行说明。

这段代码是在C#中使用平台调用(Platform Invoke,或P/Invoke)来调用Windows的user32.dll中的一个函数,名为SetParent。这是一种在.NET中调用本地方法(通常是C或C++编写的)的技术。

[DllImport("user32.dll ", EntryPoint = "SetParent")]:这是一个属性,它告诉.NET运行时你要调用的DLL的名称(在这里是"user32.dll")和函数的入口点(在这里是"SetParent")。

private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent):这是函数的声明。它告诉.NET运行时函数的签名。在这个例子中,函数名为SetParent,它接受两个IntPtr类型的参数(hWndChild和hWndNewParent),并返回一个IntPtr类型的值。

在C#中,extern关键字用于声明一个方法,该方法在外部实现,通常是在一个DLL中。

在该静态类中定义了一个类型为IntPtr的静态成员intPtr表示第三方应用窗口的句柄。

IntPtr类型介绍

在C#中,IntPtr是一个特殊的数据类型,用于表示指针或句柄。它的大小会根据当前操作系统的位数而变化,32位系统下为4字节,64位系统下为8字节。IntPtr主要用于在托管代码和非托管代码之间传递指针或句柄,以及处理不确定性大小的内存操作。它通常用于与操作系统API进行交互、处理内存分配和操作句柄等场景。

IntPtr类型提供了一种安全的方式来处理指针,因为它是托管代码中的数据类型,受到.NET运行时的管理和保护。通过IntPtr,可以在托管代码中安全地表示非托管资源的地址或句柄,而无需担心内存泄漏或其他不安全的操作。

使用IntPtr类型时,需要谨慎处理,并遵循.NET平台的内存管理规则,以确保代码的稳定性和安全性。通常情况下,IntPtr主要用于与非托管代码进行交互,处理平台特定的资源或操作系统API,同时尽量避免直接使用指针操作,以减少内存管理和安全性方面的问题。

这个静态类还有ResizeWindowFindWindowSetParentRemoveWindowTitle方法,等后面用到了再做解释。

创建一个winform✨

winform的设计如下所示:

image-20240306150128808

启动软件按钮点击事件处理程序:

 private void button2_Click(object sender, EventArgs e)
 {
     Process.Start("程序路径");
 }

嵌入窗体按钮点击事件处理程序:

  private void button1_Click(object sender, EventArgs e)
  {
      Task.Run(() =>
      {               
          if (WindowManager.FindWindow("Sysplorer [演示版]"))
          {
              this.Invoke(new Action(() =>
              {
                  WindowManager.SetParent(panel1.Handle, "Sysplorer [演示版]");  
              }));
          }
          else
          {
              MessageBox.Show("未能查找到窗体");
          }
      });
  }

在这里就会遇到一个问题就是如何确定窗体的标题是什么?

可以使用VS中的Spy++工具。

image-20240306151051044

什么是Spy++

Spy++(Spy++)是Microsoft Visual Studio套件中的一个实用工具,用于Windows平台的应用程序开发和调试。它允许开发人员查看和分析正在运行的Windows应用程序的窗口层次结构、消息流和属性。

Spy++的主要功能包括:

  1. 窗口层次结构:Spy++可以显示当前系统上所有可见和隐藏的窗口,并以层次结构的形式展示它们之间的父子关系。这使得开发人员可以快速了解应用程序的界面组织和窗口之间的相互作用。
  2. 消息监视:Spy++可以捕获和显示应用程序之间发送和接收的Windows消息。这对于调试和分析应用程序的行为非常有用,特别是在处理用户输入、事件处理和消息传递方面。
  3. 属性查看:Spy++允许开发人员查看和修改窗口的属性,如标题、类名、位置、大小、样式等。这对于调试和修改窗口属性以及理解窗口如何与应用程序交互非常有帮助。
  4. 窗口捕获:Spy++可以捕获特定窗口的消息,并将其导出为日志文件,以供进一步分析和调试使用。

总之,Spy++是一个强大的工具,可用于Windows平台的应用程序开发和调试,它提供了丰富的功能来帮助开发人员理解和调试复杂的窗口应用程序。

打开之后,如下所示:

image-20240306151615839

可以通过这样查看窗体名:

查找窗体名

得到了关于这个窗体的一些信息,其中红框部分就是窗体标题,如下所示:

image-20240306152034487

找到窗体标题之后,看看WindowManager.FindWindow方法:

 public static bool FindWindow(string formName)
 {
     for (int i = 0; i < 100; i++)
     {
         //按照窗体标题查找窗体
         IntPtr vHandle = FindWindow(null, formName);
         if (vHandle == IntPtr.Zero)
         {
             Thread.Sleep(100);  //每100ms查找一次,直到找到,最多查找10s
             continue;
         }
         else      //找到返回True
         {
             intPtr = vHandle;
             return true;
         }
     }
     intPtr = IntPtr.Zero;
     return false;
 }

再看看 WindowManager.SetParent方法:

  public static void SetParent(IntPtr hWndNewParent, string windowName)
  {
      ShowWindow(intPtr, 0);                 //先将窗体隐藏,防止出现闪烁
      SetParent(intPtr, hWndNewParent);      //将第三方窗体嵌入父容器                    
      Thread.Sleep(100);                      //略加延时
      ShowWindow(intPtr, 3);                 //让第三方窗体在容器中最大化显示
      RemoveWindowTitle(intPtr);             // 去除窗体标题
  }

现在查看一下效果:

查看效果1

但是我们发现嵌入的效果不是很好,而且无法随着窗体的变化而变化,需要再做下修改:

 public Form1()
 {
     InitializeComponent();
     this.Resize += new EventHandler(Form1_Resize);
 }

注册窗体的Resize事件。

事件处理程序:

  private void Form1_Resize(object sender, EventArgs e)
  {
      Task.Run(() =>
      {
          //第三方窗体句柄不为空
          if (WindowManager.intPtr != IntPtr.Zero)
          {
              WindowManager.ResizeWindow();
          }
      });
    
  }

现在再来看一下效果:

查看效果2

总结✨

以上就是在winform中嵌入第三方窗体的一次实践,希望对你有所帮助。

参考✨

1、C#完美将第三方窗体嵌入Panel容器(WPF、Winform)_c#嵌入另一个exe文件到panel控件中,exe打开的子窗口也识别进来-CSDN博客

2、技术文档 | Microsoft Learn

标签:IntPtr,嵌入,窗口,句柄,窗体,SetParent,static,winform
From: https://www.cnblogs.com/mingupupu/p/18058334

相关文章

  • 大语言模型常见的文本切分之语义嵌入
    语料准备选择了一篇新闻稿,有比较明显的段落区分,每个段落由若干句子构成,可用于验证切分效果。#节气释义小寒,是二十四节气中的第二十三个节气,太阳到达黄经285度时开始。《月令七十二候集解》中说:“十二月节,月初寒尚小,故云。月半则大矣。”冷气积久而寒,小寒是天气寒冷但还没有......
  • 俄罗斯套娃 (Matryoshka) 嵌入模型概述
    在这篇博客中,我们将向你介绍俄罗斯套娃嵌入的概念,并解释为什么它们很有用。我们将讨论这些模型在理论上是如何训练的,以及你如何使用SentenceTransformers来训练它们。除此之外,我们还会告诉你怎么用这种像套娃一样的俄罗斯套娃嵌入模型,并且我们会比较一下这种模型和普通嵌入模......
  • 英码嵌入式联合昇腾推出EA200I AI智能计算模组:国产化程度高,支持产品全面定制!
    此前,华为昇腾的Atlas200IDKA2开发者套件凭借其高性能、硬件接口丰富、参考代码和算法模型丰富、工具全流程覆盖、方便用户快速验证等特点在行业内极为火热。英码嵌入式致力于嵌入式计算与人工智能硬件行业,拥有多年深厚的技术沉淀和技术服务经验,2024年,英码嵌入式将携手华为昇腾......
  • 技术干货 | 英码嵌入式IVP92x开发主板上电启动及各模块测试详细教程(附工具)
    IVP92x是一款基于英码嵌入式低照度全彩视频处理模组SOM928设计的开发主板,IVP92x主板具备多路智能视觉分析(目标识别/运动跟踪/周界防范等)能力,支持[email protected]/H.264多码流编解码,同时支持智能降噪、全景拼接以及双目深度处理;除此之外,还设计了丰富的外围接口,满足无人机、智能摄......
  • 旗舰级产品 | 英码嵌入式AI+ISP机器视觉IVP92x开发主板,支持全面定制!
    IVP92x是广州英码嵌入式设备有限公司推出的一款基于英码嵌入式SOM928/SOM927核心板(支持全国产化)设计的开发主板;搭载海思SS928/SS927处理器,板载双路千兆MAC和USB3.0,提供双目摄像头输入接口(MIPI-In-FPC接口,最大支持4路图像Sensor输入)、HDMI高清输出和立体声音频接口,支......
  • 嵌入式开发笔试题99题答案
    1.a)5。b)存在风险,因为c=c++%5;这个表达式对c有两次修改,行为未定义,c的值不确定。 2.a)a=2,b=100,c=2,d=6,e=5b)stack:d;data:a;bss:b,c,eBSS段:BSS段(bsssegment)通常是指用来存放程序中未初始化的全局变量的一块内存区域。BSS是英文BlockStartedbySymbol的简称。BSS段属于静......
  • 嵌入式开发笔试题99题
    1.intmain(intargc,char*argv[]){intc=9,d=0;c=c++%5;d=c;printf("d=%d\n",d);return0;}a)写出程序输出b)在一个可移植的系统中这种表达式是否存在风险?为什么? 2.inta=0;intb;staticcharc;intmain(intargc,char*argv[]){......
  • VSTO:WinForms如何引用Ribbon.Invalidate
    问题描述:近期项目需要在VSTO插件中设计WinForms界面,该界面需要实现一个功能:当WinForms从外部应用中获取数据后,将其传递到editbox显示栏内。项目开发中遇到以下问题:WinForms中实例化Ribbon后,再引用其中的函数或Invalidate功能,在运行时会报错:System.NullReferenceException:“未将......
  • WPF 父子窗体间传值
    WPF父子窗体间传值1.父->子1.1.父窗体.cspublicpartialclassMainWindow:Window{publicMainWindow(){InitializeComponent();}publicvoidMainWindow_OnLoaded(objectsender,RoutedEventArgse){ChildrenWindowchildren=newChildrenWind......
  • C# WinForm基于owin创建WebApi
    在实际的项目开发中,可能会有在WinForm程序中提供Web服务器的需求。通过owin可以很方便的实现,并且可提供Web静态文件访问服务。操作方法:1.在NuGet引用owinMicrosoft.AspNet.WebApi.OwinMicrosoft.AspNet.WebApi.OwinSelfHostMicrosoft.Owin.StaticFiles2.添加服务启动配置类 ......