相關win32api的學習
SetParent
[DllImport("user32.dll ", EntryPoint = "SetParent")]
private static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent); //將外部窗體嵌入程序
語法:
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
作用:設置指定窗口的顯示狀態。
語法:
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,同時盡量避免直接使用指針操作,以減少內存管理和安全性方面的問題。
這個靜態類還有ResizeWindow
、FindWindow
、SetParent
、RemoveWindowTitle
方法,等后面用到了再做解釋。
創建一個winform
winform的設計如下所示:
啟動軟件按鈕點擊事件處理程序:
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++工具。
什么是Spy++
Spy++(Spy++)是Microsoft Visual Studio套件中的一個實用工具,用于Windows平臺的應用程序開發和調試。它允許開發人員查看和分析正在運行的Windows應用程序的窗口層次結構、消息流和屬性。
Spy++的主要功能包括:
- 窗口層次結構:Spy++可以顯示當前系統上所有可見和隱藏的窗口,并以層次結構的形式展示它們之間的父子關系。這使得開發人員可以快速了解應用程序的界面組織和窗口之間的相互作用。
- 消息監視:Spy++可以捕獲和顯示應用程序之間發送和接收的Windows消息。這對于調試和分析應用程序的行為非常有用,特別是在處理用戶輸入、事件處理和消息傳遞方面。
- 屬性查看:Spy++允許開發人員查看和修改窗口的屬性,如標題、類名、位置、大小、樣式等。這對于調試和修改窗口屬性以及理解窗口如何與應用程序交互非常有幫助。
- 窗口捕獲:Spy++可以捕獲特定窗口的消息,并將其導出為日志文件,以供進一步分析和調試使用。
總之,Spy++是一個強大的工具,可用于Windows平臺的應用程序開發和調試,它提供了豐富的功能來幫助開發人員理解和調試復雜的窗口應用程序。
打開之后,如下所示:
可以通過這樣查看窗體名:
得到了關于這個窗體的一些信息,其中紅框部分就是窗體標題,如下所示:
找到窗體標題之后,看看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); // 去除窗體標題
}
現在查看一下效果:
但是我們發現嵌入的效果不是很好,而且無法隨著窗體的變化而變化,需要再做下修改:
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();
}
});
}
現在再來看一下效果:
總結
以上就是在winform中嵌入第三方窗體的一次實踐,希望對你有所幫助。
參考
1、C#完美將第三方窗體嵌入Panel容器(WPF、Winform)_c#嵌入另一個exe文件到panel控件中,exe打開的子窗口也識別進來-CSDN博客
2、技術文檔 | Microsoft Learn
該文章在 2024/3/8 15:29:03 編輯過