窗口居中 & 变更触发机制

解决:

1。单实例窗口,窗口每次隐藏后再显示时,位置居中显示

2。多屏幕下单实例窗口,当父窗口移动到其它屏幕时,单实例窗口再次弹出时,位置才更新到父窗口屏幕。

3。子窗口每次唤醒时,都居中显示。

窗口首次显示的位置 - WindowStartupLocation

windows的启动时位置显示,WindowStartupLocation

  • CenterOwner --显示在父窗口的中间(设置Owner)
  • CenterScreen --显示在当前屏幕中间
  • Manual --默认位置

当第一次window.ShowDialog时,window显示如上设置。

变更触发机制

上面只涉及到了首次显示位置,之后,窗口的位置会继续保留

  • 如何设置窗口隐藏之后再次弹出时,显示在中间(CenterOwner/CenterScreen)?
  • 如何设置窗口一直停留在显示在中间?

我们先了解一下,有哪些触发机制

  1. Activated 窗口激活
    窗口变更为前台窗口时(即显示在最前面),会触发
  2. IsVisibleChanged 显示变更
    当我们设置窗口隐藏Hide()时,IsVisibile=false.窗口再次ShowDialog时,IsVisibile=true;

利用如上俩种机制,下面就可以搞事情了。

首先定义几个枚举:

     /// <summary>
/// 窗口显示变更触发时机
/// </summary>
public enum WindowLocationInvokeOccasion
{
/// <summary>
/// 只要Activated就显示在中间
/// </summary>
Activated = , /// <summary>
/// 只在第一次Activated时,显示在中间一次,之后的变化就不修改
/// </summary>
FirstActivated, /// <summary>
/// 窗口每次显示时,窗口居中
/// <para>可以解决单实例窗口弹出不居中问题</para>
/// </summary>
Visibile, /// <summary>
/// 窗口每次显示时,如父窗口与当前窗口不在同一屏幕时,窗口居中
/// <para>可以解决单实例窗口弹出不居中问题</para>
/// </summary>
VisibileInDifferentScreen, /// <summary>
/// 不触发
/// </summary>
Defatult
}

如上枚举包含了4种触发机制。

我们再定义个附加属性,通过附加属性去设置窗口的额外功能-居中显示触发机制

 /// <summary>
/// 窗口显示居中触发时机
/// <para>另:居中显示设置,请使用<see cref="Window"/>的<see cref="WindowStartupLocation"/>属性</para>
/// </summary>
public static readonly DependencyProperty InvokeOccasionProperty = DependencyProperty.RegisterAttached(
"InvokeOccasion", typeof(WindowLocationInvokeOccasion), typeof(WindowLocationOptions),
new PropertyMetadata(default(WindowLocationInvokeOccasion), InvokeOccasionProperty_ChangedCallback));

在属性更改触发事件中,根据不同的触发条件,设置不同的居中显示。

    • Activated --只要Activated就显示在中间
      每次触发,直接显示窗口即可
    • 首次Activated
      通过设置window.Activated -= ShowInCenter_Activated;禁用下次触发进入
    • Visibile
    • VisibileInDifferentScreen
      窗口显示时,如父窗口与当前窗口不在同一屏幕时,窗口居中.
      怎么判断当前子窗口与父窗口是否在同一屏幕?
 var screen = Screen.FromHandle(new WindowInteropHelper(parentWindow).Handle);

 Graphics currentGraphics = Graphics.FromHwnd(new WindowInteropHelper(parentWindow).Handle);
double dpiXRatio = currentGraphics.DpiX / ;
double dpiYRatio = currentGraphics.DpiY / ; //当子窗口与父窗口所在屏幕相同时,不作处理
var isSubWindowInSameScreen = subWindow.Left > screen.Bounds.Left / dpiXRatio &&
subWindow.Left < screen.Bounds.Left / dpiXRatio + screen.Bounds.Width / dpiXRatio &&
subWindow.Top > screen.Bounds.Top / dpiYRatio &&
subWindow.Top < screen.Bounds.Top / dpiYRatio + screen.Bounds.Height / dpiYRatio;
return isSubWindowInSameScreen;

介绍完成触发条件,下面说下窗口居中显示。
居中显示,分为当前屏幕内居中/主窗口内居中,直接上代码

1.在主窗口中居中显示-CenterOwner

设置窗口的依靠位置Location(Left,Top)(左上角)

  • 子窗口最大化时 --WindowState=“Maximized”最大化窗口,固定的弹出到主屏幕,因此需额外处理,根据屏幕Location设置位置;
  • 父窗口最大化时 --父窗口最大化时,父窗口的location,因窗口设置margin,有可能不准确,故取屏幕位置
  • CenterOwner窗口居中显示 --直接取父窗口的位置/大小和子窗口的大小,进行计算即可

PS:窗口的位置Left/Top可能为负

 /// <summary>
/// 在主窗口中居中显示
/// </summary>
/// <param name="subWindow"></param>
/// <param name="parentWindow"></param>
private static void SetWindowInCenterOwner(Window subWindow, Window parentWindow)
{
//最大化窗口,固定的弹出到主屏幕,因此需额外处理
if (subWindow.WindowState == WindowState.Maximized)
{
//子窗口最大化时,需要根据屏幕设置位置;
var screen = Screen.FromHandle(new WindowInteropHelper(parentWindow).Handle); Graphics currentGraphics = Graphics.FromHwnd(new WindowInteropHelper(parentWindow).Handle);
double dpiXRatio = currentGraphics.DpiX / ;
double dpiYRatio = currentGraphics.DpiY / ; subWindow.Left = screen.Bounds.Left / dpiXRatio;
subWindow.Top = screen.Bounds.Top / dpiYRatio;
}
if (parentWindow.WindowState == WindowState.Maximized)
{
//父窗口最大化时,父窗口的location,因窗口设置margin,有可能不准确,故取屏幕位置
var screen = Screen.FromHandle(new WindowInteropHelper(parentWindow).Handle); Graphics currentGraphics = Graphics.FromHwnd(new WindowInteropHelper(parentWindow).Handle);
double dpiXRatio = currentGraphics.DpiX / ;
double dpiYRatio = currentGraphics.DpiY / ; //窗口居中显示
subWindow.Left = screen.Bounds.Left / dpiXRatio +
(screen.Bounds.Width / dpiXRatio - subWindow.ActualWidth) / ;
subWindow.Top = screen.Bounds.Top / dpiYRatio +
(screen.Bounds.Height / dpiYRatio - subWindow.ActualHeight) / ;
}
else
{
//窗口居中显示
subWindow.Left = parentWindow.Left + (parentWindow.ActualWidth - subWindow.ActualWidth) / ;
subWindow.Top = parentWindow.Top + (parentWindow.ActualHeight - subWindow.ActualHeight) / ;
}
}

2.当前屏幕内居中-CenterScreen

窗口位置设置和上面的一样,值得注意的是DPI。

通过win的显示设置,调整文本显示比例,屏幕的位置转换(X,Y),得考虑DPI的换算

 /// <summary>
/// 在父窗口所在屏幕居中显示
/// </summary>
/// <param name="subWindow"></param>
/// <param name="parentWindow"></param>
private static void SetWindowInCenterScreen(Window subWindow, Window parentWindow)
{
SetWindowLocationInScreen(subWindow, parentWindow, subWindow.WindowState);
} private const int DpiPercent = ; private static void SetWindowLocationInScreen(Window subWindow, Window parentWindow, WindowState windowState)
{
var intPtr = new WindowInteropHelper(parentWindow).Handle;
var screen = Screen.FromHandle(intPtr); using (Graphics currentGraphics = Graphics.FromHwnd(intPtr))
{
double dpiXRatio = currentGraphics.DpiX / DpiPercent;
double dpiYRatio = currentGraphics.DpiY / DpiPercent; if (windowState == WindowState.Maximized)
{
//设置全屏Location
subWindow.Left = screen.Bounds.Left / dpiXRatio;
subWindow.Top = screen.Bounds.Top / dpiYRatio;
}
else
{
//设置居中Location
subWindow.Left = screen.Bounds.Left / dpiXRatio +
(screen.Bounds.Width / dpiXRatio - subWindow.ActualWidth) / ;
subWindow.Top = screen.Bounds.Top / dpiYRatio +
(screen.Bounds.Height / dpiYRatio - subWindow.ActualHeight) / ;
}
}
}

关键字:单实例窗口,窗口居中,CenterOwner,CenterScreen,当前屏幕DPI

WPF 窗口居中 & 变更触发机制的更多相关文章

  1. 【深入浅出Linux网络编程】 “基础 -- 事件触发机制”

    回顾一下“"开篇 -- 知其然,知其所以然"”中的两段代码,第一段虽然只使用1个线程但却也只能处理一个socket,第二段虽然能处理成百上千个socket但却需要创建同等数量的线程 ...

  2. window.open窗口居中和窗口最大化

    1.window.open()参数 window.open(pageURL,name,parameters) 其中: pageURL为子窗口路径 name为子窗口句柄 parameters为窗口参数( ...

  3. [WPF疑难]如何禁用WPF窗口的系统菜单(SystemMenu)

    原文 [WPF疑难]如何禁用WPF窗口的系统菜单(SystemMenu) [WPF疑难]如何禁用WPF窗口的系统菜单(SystemMenu) 周银辉 点击窗口左上角图标时弹出来的菜单也就是这里所说的系 ...

  4. Example005控制弹出窗口居中显示

    <!-- 实例005控制弹出窗口居中显示 --> <head> <meta charset="UTF-8"> </head> < ...

  5. WPF C# 命令的运行机制

    1.概述 1.1 WPF C# 命令的本质 命令是 WPF 中的输入机制,它提供的输入处理比设备输入具有更高的语义级别. 例如,在许多应用程序中都能找到的“复制”.“剪切”和“粘贴”操作就是命令. W ...

  6. EventEmitter:nodeJs事件触发机制

    Node.js 所有的异步 I/O 操作在完成时都会发送一个事件到事件队列 Node.js 里面的许多对象都会分发事件:一个 net.Server 对象会在每次有新连接时触发一个事件, 一个 fs.r ...

  7. Wpf窗口中打开WinForm窗口

    获取wpf窗口对应的句柄窗口 using System; using System.Windows; using System.Windows.Interop; using IWin32Window ...

  8. win32程序之窗口程序,以及消息机制

    win32程序值窗口程序,以及消息机制 一丶简介 通过上一讲.我们了解了窗口其实是绘制出来的.而且是不断绘制的过程. 所以窗口的本质是绘制. 但是我们现在看到的窗口程序.都可以点击关闭按钮. 使用鼠标 ...

  9. tkinter窗口居中方法

    tkinter窗口居中 from tkinter import * class MyFrm(Frame): def __init__(self, master): self.root=master s ...

随机推荐

  1. 新概念英语(1-137)A pleasant dream

    Lesson 137 A pleasant dream 美好的梦 Listen to the tape then answer this question. What would Julie like ...

  2. python中导入模块的本质, 无法导入手写模块的解决办法

    最近身边一些朋友发生在项目当中编写自己模块,导入的时候无法导入的问题. 下面我来分享一下关于python中导入模块的一些基本知识. 1 导入模块时寻找路径 在每一个运行的python程序当中,都维护了 ...

  3. Python系列之 - 反射

    一.静态方法(staticmethod)和类方法(classmethod) 类方法:有个默认参数cls,并且可以直接用类名去调用,可以与类属性交互(也就是可以使用类属性) 静态方法:让类里的方法直接被 ...

  4. 《Java面向对象设计》

    <Java面向对象设计> 第一章 面向对象软件工程与UML p理解为什么需要软件工程 p掌握软件工程的基本概念 p掌握软件生命周期各个阶段的主要任务 p了解流行软件开发过程 p了解软件过程 ...

  5. 微信小程序:模板消息推送提示{“errcode”:41030,”errmsg”:”invalid page hint: [gP1eXXXXXX]”}

    在开发小程序 模板消息定时推送功能时,在开发版测试程序功能运行正常,但提交到线上后提示报错{“errcode”:41030,”errmsg”:”invalid page hint: [gP1eXXXX ...

  6. Spring源码分析:Spring IOC容器初始化

    概述: Spring 对于Java 开发来说,以及算得上非常基础并且核心的框架了,在有一定开发经验后,阅读源码能更好的提高我们的编码能力并且让我们对其更加理解.俗话说知己知彼,百战不殆.当你对Spri ...

  7. app 下载更新 file-downloader 文件下载库的简单介绍和使用

    app 下载更新 file-downloader 文件下载库的简单介绍和使用 今天介绍一个下载库:file-downloader 文件下载库 说明: * 本文内容来自原 file-downloader ...

  8. MongoDB最佳实践中文手册

    背景:查阅了一下MongoDB的相关文档,发现中文文档还是比较少的,工作中需要用到MongoDB,而这本<MongoDB最佳实践>是很好的选择,所以就把这本手册翻译了一下,其中生涩的专业用 ...

  9. [LeetCode] Shopping Offers 购物优惠

    In LeetCode Store, there are some kinds of items to sell. Each item has a price. However, there are ...

  10. [TJOI 2013]单词

    Description 题库链接 给出一篇文章的所有单词,询问每个单词出现的次数. 单词总长 \(\leq 10^6\) Solution 算是 \(AC\) 自动机的板子,注意拼成文章的时候要在单词 ...