接觸WinUI3是我第一次知道原來微軟有這么多前端技術:Winforms、WPF、UWP、MAUI、WinUI3、Win32 API、WinRT。前五個都是Visual Studio提供的模板,Win32是Windows底層的C/C++API接口,WinRT是Windows提供的一種跨平臺運行時API接口。
WinUI3是新推出的一些列的Windows UI平臺組件(根據我的理解也就是一堆UI組件了,基本上啥高級封裝都沒有),功能實現基本上是靠Win32和WinRT實現。
實現步驟可以參照:Example: Use Mica in a Windows AppSDK/WinUI 3 app
或者WinUI-Gallery
首先,必須明確WinUI3是通過WinRT中的DesktopAcrylicController類提供了為COM組件設置亞克力筆刷的方法,所以先創建對象,然后指定COM組件
DesktopAcrylicController m_acrylicController = new();
m_acrylicController.AddSystemBackdropTarget(m_Window.As());
非常理論,非常正確。運行會拋出一個不知所云的C++異常
然后如果仔細查看AddSystemBackdropTarget的文檔,會發現該方法必須要通過線程調度隊列(DispatcherQueue)去調用
這里有個坑,開始我以為這個DispatcherQueue是指Window的隊列(Microsoft.UI.Dispatching.DispatcherQueue),但實際上指的是由WinRT創建的DispatcherQueue(Windows.System.DispatcherQueue)
詳細可以看我在Stackoverflow上的提問:How to set the acrylic style to window?
對于這一部分,需要通過CreateDispatcherQueueController()函數來實現。參考MS的示例,可以通過創建WindowsSystemDispatcherQueueHelper去實現
首先引入函數,根據官方文檔對CreateDispatcherQueueController和DispatcherQueueOptions的描述,需要對參數中的結構體DispatcherQueueOptions做映射
struct DispatcherQueueOptions
{internal int dwSize;
internal int threadType;
internal int apartmentType;
}
之后引入函數
[DllImport("CoreMessaging.dll")]
private static extern int CreateDispatcherQueueController(
[In] DispatcherQueueOptions options,
[In, out, MarshalAs(UnmanagedType.IUnknow)] ref object dispatcherQueueController // MarshalAs指示應該將這個object參數封裝成IUnknown類型傳遞給C++函數
)
最后檢查并確保調度線程控制器存在
object m_dispatcherQueueController = null;
public void EnsureWindowsSystemDispatcherQueueController()
{if (DispatcherQueue.GetForCurrentThread() != null)
{return;
}
if (m_dispatcherQueueController == null)
{DispatcherQueueOptions options;
options.dwSize = Marshal.SizeOf(typeof(DispatcherQueueOptions));
options.threadType = 2;
options.apartmentType = 2;
CreateDispatcherQueueController(options, ref m_dispatcherQueueController);
}
}
之后在調用AddSystemBackdropTarget()函數前先確保WinRT顯示層的調度線程存在再運行
WindowsSystemDispatcherQueueHelper m_wsdqHelper = new();
m_wsdqHelper.EnsureWindowsSystemDispatcherQueueController();
DesktopAcrylicController m_backdropController = new();
m_backdropController.AddSystemBackdropTarget(m_window.As());
運行發現程序沒有報錯并成功啟動,但是并沒有實現亞克力的效果。通過看MS的示例我發現我沒有為acrylicController設置SystemBackdropConfiguration。所以實例化一個SystemBackdropConfiguration對象然后為acrylicController配置上就解決了。
ps:看了一下SystemBackdropConfiguration的源碼,不太理解這個函數在實例化時注冊和初始化了一些什么東西。
最終成功代碼如下:
WindowsSystemDispatcherQueueHelper m_wsdqHelper = new();
m_wsdqHelper.EnsureWindowsSystemDispatcherQueueController();
SystemBackdropConfiguration m_configurationSource = new();
// m_configurationSource.IsInputActive = true;
DesktopAcrylicController m_backdropController = new();
m_backdropController.AddSystemBackdropTarget(m_window.As());
m_backdropController.SetSystemBackdropConfiguration(m_configurationSource);
但是他有個問題:窗口在失去焦點時不會更新背景,我看WinUI-Gallery的處理辦法好像是讓失去焦點的窗口暫時不啟用亞克力效果,不知道有沒有更好的解決辦法,歡迎分享。
你是否還在尋找穩定的海外服務器提供商?創新互聯www.cdcxhl.cn海外機房具備T級流量清洗系統配攻擊溯源,準確流量調度確保服務器高可用性,企業級服務器適合批量采購,新人活動首月15元起,快前往官網查看詳情吧
新聞標題:[WinUI3]如何設置亞克力窗口?-創新互聯
本文來源:http://vcdvsql.cn/article2/ccshoc.html
成都網站建設公司_創新互聯,為您提供靜態網站、網站建設、外貿建站、響應式網站、手機網站建設、網站收錄
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯