记录 PyQt6 / PySide 6 自定义边框窗口的 Bug 及可能可行的解决方案:窗口抖动和添加 DWM 环绕阴影的大致原理

本文主要是介绍记录 PyQt6 / PySide 6 自定义边框窗口的 Bug 及可能可行的解决方案:窗口抖动和添加 DWM 环绕阴影的大致原理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

本篇文章将要讨论我在前不久发表的关于 PyQt6 / PySide6 自定义边框窗口代码及内容中的问题:

(终)PyQt6 / PySide 6 + Pywin32 自定义标题栏窗口 + 完全还原 Windows 原生窗口边框特效_pyside6 win32 无边框窗口-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/2402_84665876/article/details/141535937?spm=1001.2014.3001.5501

问题:

1. 窗口右边框和下边框的抖动问题:

这算是 FramelessWindowHint 的通病,在 UI 基于 Qt 的应用程序中,大多数去掉系统边框的应用都会发生在拖拽窗口顶部,右上方或左侧调整大小时窗口右侧和底部发生抖动的情况,比如 123 云盘的 Windows 客户端,OpenShot 和 QtFramelessWindow 第三方库,这是我推测的调整大小的信号传输步骤:

 无论如何,当你调整一个自定义窗口的大小时,NativeEvent 和 resizeEvent 事件在同时进行,但由于两个事件不是平级关系,NativeEvent 将比 resizeEvent 先一步收到系统通知,这导致窗口调整大小时会出现延迟现象。

2.添加 DWM 阴影不等于窗口绘制:

在【前言】所提到的文章中,我曾备注过这样一段话:

注:使用此方法给窗口添加 DWM 阴影实际上就是新建了一个空白的带阴影的窗口,再将自己创建的窗口绘制到该窗口上(覆盖原有标题栏),所以修改后的窗口拖动和改变大小时占用的 CPU 和内存资源大概是未添加阴影时的 2~3 倍甚至更多,当系统刚启动时或外部应用占用的 CPU 和内存资源过多时,拖动和缩放窗口会出现明显的卡顿。

 其实这种说法是错误的,窗口图形的 ”边界“ 和系统给窗口的 ”边框“ 并不是一回事,添加阴影的原理是利用 Windows 的桌面窗口管理器 (DWM) 来扩展窗口的框架区域,使得阴影效果能够显示在窗口周围。这种效果是 DWM 在窗口外框绘制阴影,而不是实际改变窗口本身的边界或形状。

问题原因和可能可行的解决方案:

* 这篇文章并没有给出解决问题后的代码,因为我也正在尝试用此方法解决问题。

** 答案来源:ChatGPT 4

原问题: 为什么给窗口设置 FramelessWindowHint 后,调整大小时窗口的右侧或底部边界会发生“抖动”?有没有办法解决这个问题?
当设置 FramelessWindowHint 时,Qt 窗口会失去原生的 Windows 标题栏和边框。此时,Qt 本身负责处理窗口的边界调整和绘制。这可能导致调整大小时的抖动问题,尤其是在应用 DwmExtendFrameIntoClientArea 时,因为这会修改窗口的视觉区域。
  1. 抖动的原因:抖动问题通常由以下几个原因导致:

    • 不一致的窗口区域更新:DWM 扩展区域和 Qt 本身的窗口边界之间可能出现同步问题。DWM 更新边界时,窗口内容有时可能会滞后于实际的框架阴影扩展,导致“抖动”或闪烁。
    • 多线程或异步绘制冲突:窗口边界调整和重新绘制过程中的线程不安全操作或异步更新可能导致这种抖动。
  2. 解决方法

    • 同步更新:确保 DwmExtendFrameIntoClientArea 和窗口大小调整同步。你可以在大小调整事件中调用 DwmExtendFrameIntoClientArea 来重新设置边界。
    • 使用 Win32 API 的原生调整大小:使用 Windows API 的原生窗口调整大小处理,而不是 Qt 的事件。你可以捕获窗口的 WM_NCHITTEST 消息来处理拖动和调整大小区域,从而保留原生行为。
    • 避免频繁调用 DWM 扩展:在窗口大小调整过程中,避免频繁调用 DwmExtendFrameIntoClientArea。可以尝试只在初始化时调用一次。

这篇关于记录 PyQt6 / PySide 6 自定义边框窗口的 Bug 及可能可行的解决方案:窗口抖动和添加 DWM 环绕阴影的大致原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1132809

相关文章

Vite 打包目录结构自定义配置小结

《Vite打包目录结构自定义配置小结》在Vite工程开发中,默认打包后的dist目录资源常集中在asset目录下,不利于资源管理,本文基于Rollup配置原理,本文就来介绍一下通过Vite配置自定义... 目录一、实现原理二、具体配置步骤1. 基础配置文件2. 配置说明(1)js 资源分离(2)非 JS 资

ShardingProxy读写分离之原理、配置与实践过程

《ShardingProxy读写分离之原理、配置与实践过程》ShardingProxy是ApacheShardingSphere的数据库中间件,通过三层架构实现读写分离,解决高并发场景下数据库性能瓶... 目录一、ShardingProxy技术定位与读写分离核心价值1.1 技术定位1.2 读写分离核心价值二

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

C#文件复制异常:"未能找到文件"的解决方案与预防措施

《C#文件复制异常:未能找到文件的解决方案与预防措施》在C#开发中,文件操作是基础中的基础,但有时最基础的File.Copy()方法也会抛出令人困惑的异常,当targetFilePath设置为D:2... 目录一个看似简单的文件操作问题问题重现与错误分析错误代码示例错误信息根本原因分析全面解决方案1. 确保

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

SpringBoot3匹配Mybatis3的错误与解决方案

《SpringBoot3匹配Mybatis3的错误与解决方案》文章指出SpringBoot3与MyBatis3兼容性问题,因未更新MyBatis-Plus依赖至SpringBoot3专用坐标,导致类冲... 目录SpringBoot3匹配MyBATis3的错误与解决mybatis在SpringBoot3如果

Python中的filter() 函数的工作原理及应用技巧

《Python中的filter()函数的工作原理及应用技巧》Python的filter()函数用于筛选序列元素,返回迭代器,适合函数式编程,相比列表推导式,内存更优,尤其适用于大数据集,结合lamb... 目录前言一、基本概念基本语法二、使用方式1. 使用 lambda 函数2. 使用普通函数3. 使用 N

聊聊springboot中如何自定义消息转换器

《聊聊springboot中如何自定义消息转换器》SpringBoot通过HttpMessageConverter处理HTTP数据转换,支持多种媒体类型,接下来通过本文给大家介绍springboot中... 目录核心接口springboot默认提供的转换器如何自定义消息转换器Spring Boot 中的消息