WinUI3 使用Win32Api 實現視窗停靠常駐桌面功能。

来源:https://www.cnblogs.com/chifan/archive/2022/09/30/16743608.html
-Advertisement-
Play Games

我們可以通過使用Win32Api來製作一些強大的功能,本文將通過示例代碼來介紹使用Win32Api來之做桌面視窗停靠功能; 效果圖: 一.通過Nuget 引入 Vanara.PInvoke.Shell32 和 PInvoke.User32 這兩個庫。 二.功能列表 1.Berth 函數,將視窗停靠在 ...


  我們可以通過使用Win32Api來製作一些強大的功能,本文將通過示例代碼來介紹使用Win32Api來之做桌面視窗停靠功能;

  效果圖:

 

 

   一.通過Nuget 引入 Vanara.PInvoke.Shell32 和 PInvoke.User32 這兩個庫。

 

 

 

 

 

  二.功能列表

    1.Berth 函數,將視窗停靠在桌面的右側;

      1).使用 Shell32.SHAppBarMessage 函數的兩次調用將桌面的指定位置設置為"AppBar"區域;

      2).使用AppWindow將視窗的模式設置為菜單模式(該模式會將視窗的標題欄移除,並且禁用了用戶更改視窗大小的功能);

      3).使用 User32.SetWindowLong函數將任務欄裡面的應用圖標隱藏

      4).使用User32.MoveWindow函數設置指定的大小,並且將視窗移動到指定的位置。

    2.Detach 函數,將視窗取消停靠;

      1).使用Shell32.SHAppBarMessage 函數移除 “AppBar” ,將桌面恢復正常;

      2).使用AppWindow 將視窗設置為普通模式(將原本隱藏的標題欄顯示出來,已經更改為可以更改視窗大小);

      3).使用 User32.SetWindowLong函數將原本被移除的圖標顯示出來;

      4).使用User32.MoveWindow函數設置指定的大小,並且將視窗移動到指定的位置。

  三.所有代碼

1 <StackPanel Background="Red" Orientation="Horizontal" HorizontalAlignment="Center" VerticalAlignment="Center">
2         <Button x:Name="edge" >停靠邊緣</Button>
3         <Button x:Name="detach">取消停靠</Button>
4         <Button x:Name="close">關閉應用</Button>
5     </StackPanel>
Xaml
 1 public sealed partial class MainWindow : Window
 2     {
 3         public MainWindow()
 4         {
 5             this.InitializeComponent();
 6             Init();
 7         }
 8 
 9         void Init()
10         {
11             edge.Click += (s, e) => Berth(this);
12             detach.Click += (s, e) => Detach(this);
13             close.Click += (s, e) =>
14             {
15                 App.Current.Exit();
16             };
17             //監聽視窗關閉事件
18             this.Closed += (s, e) =>
19             {
20                 //取消停靠
21                 Detach(this);
22             };
23         }
24 
25         /// <summary>
26         /// 將視窗停靠到邊緣
27         /// </summary>
28         void Berth(Window window, int width = 400)
29         {
30             //獲取視窗句柄
31             var hwnd = WindowNative.GetWindowHandle(window);
32             //創建應用欄信息對象,並設置其大小和位置
33             var data = new APPBARDATA();
34             data.hWnd = hwnd;
35             data.uEdge = ABE.ABE_RIGHT;//設置方向
36             data.cbSize = (uint)Marshal.SizeOf(data);
37             data.rc.Top = 0;
38             data.rc.bottom = DisplayArea.Primary.OuterBounds.Height;
39             data.rc.left = DisplayArea.Primary.OuterBounds.Width - width;
40             data.rc.right = DisplayArea.Primary.OuterBounds.Width;
41             //調用 win32Api 設定指定位置為“AppBar”
42             SHAppBarMessage(ABM.ABM_NEW, ref data);
43             SHAppBarMessage(ABM.ABM_SETPOS, ref data);
44 
45             //使用 AppWindow 類將視窗設置為菜單模式(將標題欄去掉,並且設置為不可更改大小);
46             var wid = Win32Interop.GetWindowIdFromWindow(hwnd);
47             var op = OverlappedPresenter.CreateForContextMenu();
48             var appWindow = AppWindow.GetFromWindowId(wid);
49             appWindow.SetPresenter(op);//將視窗設置為菜單模式
50 
51             //使用win32Api 的SetWindowLong 函數將任務欄裡面的應用圖標去掉
52             var style = (User32.SetWindowLongFlags)User32.GetWindowLong(hwnd, User32.WindowLongIndexFlags.GWL_EXSTYLE);
53             style |= User32.SetWindowLongFlags.WS_EX_TOOLWINDOW;
54             User32.SetWindowLong(hwnd, User32.WindowLongIndexFlags.GWL_EXSTYLE, style);
55             //使用 win32Api MoveWindow 函數 更改視窗的大小和位置
56             User32.MoveWindow(hwnd, data.rc.Left, data.rc.top, width, data.rc.Height, true);
57         }
58 
59         /// <summary>
60         /// 從邊緣中取消停靠視窗
61         /// </summary>
62         void Detach(Window window)
63         {
64             //獲取視窗句柄
65             var hwnd = WindowNative.GetWindowHandle(window);
66             var data = new APPBARDATA();
67             data.hWnd = hwnd;
68             data.cbSize = (uint)Marshal.SizeOf(data);
69             var d = SHAppBarMessage(ABM.ABM_REMOVE, ref data);
70         
71             //將視窗的模式設置為普通的模式,將標題欄顯示出來
72             OverlappedPresenter op = OverlappedPresenter.Create();
73             var wid = Win32Interop.GetWindowIdFromWindow(hwnd);
74             var aw = AppWindow.GetFromWindowId(wid);
75             aw.SetPresenter(op);
76             //在任務欄上顯示圖標
77             var style = User32.SetWindowLongFlags.WS_VISIBLE;
78             User32.SetWindowLong(hwnd, User32.WindowLongIndexFlags.GWL_EXSTYLE, style);
79             //設置視窗大小和位置
80             User32.MoveWindow(hwnd, 20, 200, 400, 400, true);
81         }
82     }
後臺代碼

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2022-09-29 問題描述: 在“setting.py”的配置文件中修改資料庫引擎中,將系統預設的"sqlite3"尾碼改為了“sql”。出現問題。 原因分析: 問題查看: 修改後: 上述問題修改後,在“setting”中設置資料庫的其他內容(主機、埠、用戶、密碼、使用的指定數據名的資料庫), ...
  • 在Word中創建報告時,我們經常會遇到這樣的情況:我們需要將數據從Excel中複製和粘貼到Word中,這樣讀者就可以直接在Word中瀏覽數據,而不用打開Excel文檔 ...
  • 近期在處理一個將NVR錄像機上的錄像下載到伺服器並通過瀏覽器播放的需求。 梳理記錄下過程,做個備忘,同時遇到的一些細節問題解決,也供需要的同學參考。 需求比較簡單,就是把指定時間段的錄像上傳到伺服器保存,並且允許用戶通過web頁面web瀏覽器,進行播放, 並且可以拖動控制播放進度。效果如。 一、 視 ...
  • Python常用的英語單詞就那麼幾個,多打就熟悉了 說來好笑,我壓根就沒記英語單詞… 真的就是純靠多打多練, 畢竟打多了之後肌肉記憶就在那裡了 下麵就給大家帶來常用python清單彙總~ 一、互動式環境與print輸出(https://jq.qq.com/?_wv=1027&k=2Q3YTfym) ...
  • 一、線程的概念線程是CPU分配資源的基本單位。當一程式開始運行,這個程式就變成了一個進程,而一個進程相當於一個或者多個線程。當沒有多線程編程時,一個進程相當於一個主線程;當有多線程編程時,一個進程包含多個線程(含主線程)。使用線程可以實現程式大的開發。 多個線程可以在同一個程式中運行,並且每一個線程 ...
  • 一、整型數據類型 1、整型數據類型名稱及關鍵詞 2、為什麼要定義不同的整型類型? 因為不同的數據類型所占用的記憶體大小是不同的,他們可表示的數據範圍也是不同的。那麼char,short,int,long,long long,分別占用幾個位元組?具體的數值範圍又是多少?C語言並未規定數據類型的大小範圍,具 ...
  • 在上一篇文章`《驅動開發:內核字元串轉換方法》`中簡單介紹了內核是如何使用字元串以及字元串之間的轉換方法,本章將繼續探索字元串的拷貝與比較,與應用層不同內核字元串拷貝與比較也需要使用內核專用的API函數,字元串的拷貝往往伴隨有內核記憶體分配,我們將首先簡單介紹內核如何分配堆空間,然後再以此為契機簡介字... ...
  • 一、什麼是AOP AOP為Aspect Oriented Programming的縮寫,意為:面向切麵編程,通過預編譯方式和運行期間動態代理實現程式功能的統一維護的一種技術。AOP是OOP的延續,是軟體開發中的一個熱點,也是Spring框架中的一個重要內容,是函數式編程的一種衍生範型。利用AOP可以 ...
一周排行
    -Advertisement-
    Play Games
  • 1.部署歷史 猿友們好,作為初來實習的我,已經遭受社會的“毒打”,所以請容許我在下麵環節適當吐槽,3Q! 傳統部署 ​ 回顧以往在伺服器部署webapi項目(非獨立發佈),dotnet環境、守護進程兩個逃都逃不掉,正常情況下還得來個nginx代理。不僅僅這仨,可能牽扯到yum或npm。node等都要 ...
  • 隨著技術的進步,跨平臺開發已經成為了標配,在此大背景下,ASP.NET Core也應運而生。本文主要基於ASP.NET Core+Element+Sql Server開發一個校園圖書管理系統為例,簡述基於MVC三層架構開發的常見知識點,前一篇文章,已經簡單介紹瞭如何搭建開發框架,和登錄功能實現,本篇... ...
  • 這道題只要會自定義cmp恰當地進行排序,其他部分沒有什麼大問題。 上代碼: 1 #include<bits/stdc++.h> 2 using namespace std; 3 int n,s,h1,h2,cnt; 4 struct apple{ 5 int height,ns;//height為蘋 ...
  • 這篇文章主要描述RPC的路由策略,包括為什麼需要請求隔離,為什麼不在註冊中心中實現請求隔離以及不同粒度的路由策略。 ...
  • 簡介: 中介者模式,屬於行為型的設計模式。用一個中介對象來封裝一系列的對象交互。中介者是各對象不需要顯式地相互引用,從而使其耦合鬆散,而且可以獨立地改變他們之間的交互。 適用場景: 如果平行對象間的依賴複雜,可以使用中介者解耦。 優點: 符合迪米特法則,減少成員間的依賴。 缺點: 不適用於系統出現對 ...
  • 【前置內容】Spring 學習筆記全系列傳送門: Spring學習筆記 - 第一章 - IoC(控制反轉)、IoC容器、Bean的實例化與生命周期、DI(依賴註入) Spring學習筆記 - 第二章 - 註解開發、配置管理第三方Bean、註解管理第三方Bean、Spring 整合 MyBatis 和 ...
  • 簡介: 享元模式,屬於結構型的設計模式。運用共用技術有效地支持大量細粒度的對象。 適用場景: 具有相同抽象但是細節不同的場景中。 優點: 把公共的部分分離為抽象,細節依賴於抽象,符合依賴倒轉原則。 缺點: 增加複雜性。 代碼: //用戶類 class User { private $name; fu ...
  • 這次設計一個通用的多位元組SPI介面模塊,特點如下: 可以設置為1-128位元組的SPI通信模塊 可以修改CPOL、CPHA來進行不同的通信模式 可以設置輸出的時鐘 狀態轉移圖和思路與多位元組串口發送模塊一樣,這裡就不給出了,具體可看該隨筆。 一、模塊代碼 1、需要的模塊 通用8位SPI介面模塊 `tim ...
  • AOP-03 7.AOP-切入表達式 7.1切入表達式的具體使用 1.切入表達式的作用: 通過表達式的方式定義一個或多個具體的連接點。 2.語法細節: (1)切入表達式的語法格式: execution([許可權修飾符] [返回值類型] [簡單類名/全類名] [方法名]([參數列表]) 若目標類、介面與 ...
  • 測試一、虛繼承與繼承的區別 1.1 單個繼承,不帶虛函數 1>class B size(8): 1> + 1> 0 | + (base class A) 1> 0 | | _ia //4B 1> | + 1> 4 | _ib //4B 有兩個int類型數據成員,占8B,基類邏輯存在前面 1.2、單個 ...