WPF 修改屏幕DPI,會觸發控件從新加載Unload/Load

修改屏幕DPI,會觸發控件的Unloaded/Loadedgit

現象/重現案例

這裏簡單介紹下,修改屏幕DPI,觸發Unloaded/Loaded的神奇案例github

1. 咱們新建一個窗口,添加一個UserControl1,而後在UserControl1中添加UserControl2express

 1 <Window x:Class="WPFUnloadedTriggerTest.MainWindow"
 2         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
 3         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
 4         xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
 5         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
 6         xmlns:local="clr-namespace:WPFUnloadedTriggerTest"
 7         mc:Ignorable="d"
 8         Title="MainWindow" Height="450" Width="800">
 9     <local:UserControl1></local:UserControl1>
10 </Window>
11 ------------------------------我是分隔線-----------------------------------
12 <UserControl x:Class="WPFUnloadedTriggerTest.UserControl1"
13              xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
14              xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
15              xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
16              xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
17              xmlns:local="clr-namespace:WPFUnloadedTriggerTest"
18              mc:Ignorable="d" 
19              d:DesignHeight="450" d:DesignWidth="800">
20     <local:UserControl2></local:UserControl2>
21 </UserControl>
View Code

2. 顯示窗口後,修改DPI比例ide

3. 設置完後,會觸發Unloaded/Loaded從新加載佈局

Unloaded的觸發順序是UserControl1-->UserControl2,Window並不會觸發Unloaded事件!spa

是否是詭異?咱們繼續。。。code

 4. Window咱們添加一個ControlTemplate模塊orm

1     <Window.Template>
2         <ControlTemplate TargetType="Window">
3             <Border>
4                 <AdornerDecorator>
5                     <ContentPresenter />
6                 </AdornerDecorator>
7             </Border>
8         </ControlTemplate>
9     </Window.Template>

 再重複二、3步驟,Unloaded的觸發順序變了:xml

觸發UserControl2的Unloaded,Window、UserControl1並不會觸發Unloaded事件!blog

問題分析

第2步驟中修改DPI後,Unloaded事件不必定觸發。如何須現呢?

將窗口靠近到任務欄上方,再修改文本比例。

 咱們查看調用堆棧,貌似是系統給窗口發送消息而後調用BroadcastUnloadedEvent事件,觸發Unload

 因此應該是修改DPI,窗口寬高超出了當前屏幕尺寸範圍,系統對UserControl的視覺樹進行從新加載佈局。

至於窗口沒有觸發Unloaded、以及在窗口添加以上模塊後下一級子控件也沒有觸發Unloaded事件的緣由,暫不瞭解

而對WPF-Unloaded/Loaded的已知狀況以下:

  • FrameworkElement, 第一次加載顯示時,會觸發Loaded。元素被釋放時,會觸發Unloaded。窗口Show/Close時,視覺樹變化都會觸發加載事件
  • MenuItem, 在FrameworkElement基礎上,每次和隱藏MenuItem時,會額外觸發Load/Unloaded
  • TabControl,當你選中一個tabItem時會觸發Loaded,當你取消選中一個tabItem時會觸發Unloaded,因此切換Tab時一定有一個Loaded一個Unloaded。
  • Expander,每次被Expanded擴展時會引起Loaded,但當隱藏時不會引起Unloaded。

 以上問題的解決方案?暫時沒有解決方案,只有規避措施,不要過於依賴於Unload/Loaded,並且使用了Unload/Loaded時也要添加註銷機制,防止重入

我在github提了個issue:After Modified screen dpi,Unloaded/Loaded is trigged unexpectedly

相關文章
相關標籤/搜索