WPF 修改屏幕DPI,会触发控件重新加载Unload/Load
修改屏幕DPI,会触发控件的Unloaded/Loaded
现象/重现案例
这里简单介绍下,修改屏幕DPI,触发Unloaded/Loaded的神奇案例
1. 我们新建一个窗口,添加一个UserControl1,然后在UserControl1中添加UserControl2
<Window x:Class="WPFUnloadedTriggerTest.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:WPFUnloadedTriggerTest"
mc:Ignorable="d"
Title="MainWindow" Height="450" Width="800">
<local:UserControl1></local:UserControl1>
</Window>
------------------------------我是分隔线-----------------------------------
<UserControl x:Class="WPFUnloadedTriggerTest.UserControl1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local="clr-namespace:WPFUnloadedTriggerTest"
mc:Ignorable="d"
d:DesignHeight="450" d:DesignWidth="800">
<local:UserControl2></local:UserControl2>
</UserControl>

2. 显示窗口后,修改DPI比例

3. 设置完后,会触发Unloaded/Loaded重新加载
Unloaded的触发顺序是UserControl1-->UserControl2,Window并不会触发Unloaded事件!
是不是诡异?我们继续。。。
4. Window我们添加一个ControlTemplate模块
<Window.Template>
<ControlTemplate TargetType="Window">
<Border>
<AdornerDecorator>
<ContentPresenter />
</AdornerDecorator>
</Border>
</ControlTemplate>
</Window.Template>
再重复2、3步骤,Unloaded的触发顺序变了:
触发UserControl2的Unloaded,Window、UserControl1并不会触发Unloaded事件!
问题分析
第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
WPF 修改屏幕DPI,会触发控件重新加载Unload/Load的更多相关文章
- 重新想象 Windows 8.1 Store Apps (81) - 控件增强: 加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Contract 分享 WebView 中的内容, 为 WebView 截图
原文:重新想象 Windows 8.1 Store Apps (81) - 控件增强: 加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Contract 分享 Web ...
- Delphi7 第三方控件1stClass4000的TfcImageBtn按钮控件动态加载jpg图片例子
Delphi7 第三方控件1stClass4000的TfcImageBtn按钮控件动态加载jpg图片例子 procedure TForm1.Button1Click(Sender: TObject); ...
- easyui控件的加载顺序
使用easyui做布局时,会模仿窗口程序界面,做出一些较复杂的布局.按由外层到内层的顺序: (最外层)panel->tabs->tabs1 ->tabs2->layout-&g ...
- 使用DevExpress.XtraTabbedMdi.XtraTabbedMdiManager控件来加载MDI窗体
使用DevExpress.XtraTabbedMdi.XtraTabbedMdiManager控件来加载MDI窗体 [csharp] view plaincopyprint? <SPAN ...
- 在WinForm应用程序中,使用选项卡控件来加载不同的Form界面!
TabPage tp=new TabPage(); your选项卡控件.Controls.Add(tp); From1 frm=new Form1(); frm.TopLevel = false; f ...
- Asp.net 的DropDownList 控件动态加载
在做连接数据库表,在页面上用DropDownList 下拉框查询某条数据时,因为数据库里的数据,随时都有可能增加或减少,而下拉框关联的某个字段的值并不一定是固定的. 表信息: CREATE TABLE ...
- 【高德地图API】Pivot控件中加载地图并禁止Pivot手势
如题,解决方案,参考[Windows phone应用开发[20]-禁止Pivot手势]http://www.cnblogs.com/chenkai/p/3408658.html. xaml代码清单 ...
- WPF中不规则窗体与WindowsFormsHost控件的兼容问题完美解决方案
首先先得瑟一下,有关WPF中不规则窗体与WindowsFormsHost控件不兼容的问题,网上给出的解决方案不能满足所有的情况,是有特定条件的,比如 WPF中不规则窗体与WebBrowser控件的兼 ...
- Jquery如何序列化form表单数据为JSON对象 C# ADO.NET中设置Like模糊查询的参数 从客户端出现小于等于公式符号引发检测到有潜在危险的Request.Form 值 jquery调用iframe里面的方法 Js根据Ip地址自动判断是哪个城市 【我们一起写框架】MVVM的WPF框架(三)—数据控件 设计模式之简单工厂模式(C#语言描述)
jquery提供的serialize方法能够实现. $("#searchForm").serialize();但是,观察输出的信息,发现serialize()方法做的是将表单中的数 ...
随机推荐
- 云计算 docker 容器部署
什么是docker容器: 容器就是在隔离的环境中运行的一个进程,如果进程停止,容器就会退出.隔离的环境拥有自己的系统文件,ip地址,主机名等kvm虚拟机,linux,系统文件 容器和虚拟化的区别 : ...
- zookeeper - 通过java代码连接zookeeper(2)
首先创建一个Maven项目 <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&qu ...
- Charles抓包工具的使用(一)
前提:charles的说明 Charles其实是一款代理服务器,通过过将自己设置成系统(电脑或者浏览器)的网络访问代理服务器,然后截取请求和请求结果达到分析抓包的目的.该软件是用Java写的,能够在W ...
- 原版_打字游戏.html
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- python 爬取网页简单数据---以及详细解释用法
一.准备工作(找到所需网站,获取请求头,并用到请求头) 找到所需爬取的网站(这里举拉勾网的一些静态数据的获取)----------- https://www.lagou.com/zhaopin/Pyt ...
- Pandas 分组聚合
# 导入相关库 import numpy as np import pandas as pd 创建数据 index = pd.Index(data=["Tom", "Bo ...
- [考试反思]0821NOIP模拟测试28:沉默
这次不能把我前面的分数段都列出来了,因为实在太多了. 这次也不能把我后面的分数段列出来了,因为我后面没有了. yxm,mikufun,Pairs170100分第10.50分第29. 我:爆零,倒数第一 ...
- 手把手教你定制标准Spring Boot starter,真的很清晰
写在前面 我们每次构建一个 Spring 应用程序时,我们都不希望从头开始实现具有「横切关注点」的内容:相反,我们希望一次性实现这些功能,并根据需要将它们包含到任何我们要构建的应用程序中 横切关注点 ...
- postman发送json参数到后端controller报错415的解决方案
警告 [http-nio-8080-exec-9] org.springframework.web.servlet.handler.AbstractHandlerExceptionResolver.l ...
- 原生JS实现队结构及利用队列模拟‘击鼓传花’游戏
1. 前言 队列,是一种遵从先进先出(FIFO,First-In-First-Out)原则的有序集合.队列在尾部添加新元素,并从顶部移除元素,最新添加的元素必须排在队列的末尾. 2.功能说明 enqu ...