记录 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

相关文章

Windows Docker端口占用错误及解决方案总结

《WindowsDocker端口占用错误及解决方案总结》在Windows环境下使用Docker容器时,端口占用错误是开发和运维中常见且棘手的问题,本文将深入剖析该问题的成因,介绍如何通过查看端口分配... 目录引言Windows docker 端口占用错误及解决方案汇总端口冲突形成原因解析诊断当前端口情况解

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案

《Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案》:本文主要介绍Vue3组件中getCurrentInstance()获取App实例,但是返回nu... 目录vue3组件中getCurrentInstajavascriptnce()获取App实例,但是返回n

Spring Boot循环依赖原理、解决方案与最佳实践(全解析)

《SpringBoot循环依赖原理、解决方案与最佳实践(全解析)》循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系,:本文主要介绍SpringBoot循环依赖原理、解决方案与最... 目录一、循环依赖的本质与危害1.1 什么是循环依赖?1.2 核心危害二、Spring的三级缓存机制2.1 三

C#中async await异步关键字用法和异步的底层原理全解析

《C#中asyncawait异步关键字用法和异步的底层原理全解析》:本文主要介绍C#中asyncawait异步关键字用法和异步的底层原理全解析,本文给大家介绍的非常详细,对大家的学习或工作具有一... 目录C#异步编程一、异步编程基础二、异步方法的工作原理三、代码示例四、编译后的底层实现五、总结C#异步编程

在Spring Boot中浅尝内存泄漏的实战记录

《在SpringBoot中浅尝内存泄漏的实战记录》本文给大家分享在SpringBoot中浅尝内存泄漏的实战记录,结合实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录使用静态集合持有对象引用,阻止GC回收关键点:可执行代码:验证:1,运行程序(启动时添加JVM参数限制堆大小):2,访问 htt

Node.js 数据库 CRUD 项目示例详解(完美解决方案)

《Node.js数据库CRUD项目示例详解(完美解决方案)》:本文主要介绍Node.js数据库CRUD项目示例详解(完美解决方案),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考... 目录项目结构1. 初始化项目2. 配置数据库连接 (config/db.js)3. 创建模型 (models/

Vuex Actions多参数传递的解决方案

《VuexActions多参数传递的解决方案》在Vuex中,actions的设计默认只支持单个参数传递,这有时会限制我们的使用场景,下面我将详细介绍几种处理多参数传递的解决方案,从基础到高级,... 目录一、对象封装法(推荐)二、参数解构法三、柯里化函数法四、Payload 工厂函数五、TypeScript

MySQL高级查询之JOIN、子查询、窗口函数实际案例

《MySQL高级查询之JOIN、子查询、窗口函数实际案例》:本文主要介绍MySQL高级查询之JOIN、子查询、窗口函数实际案例的相关资料,JOIN用于多表关联查询,子查询用于数据筛选和过滤,窗口函... 目录前言1. JOIN(连接查询)1.1 内连接(INNER JOIN)1.2 左连接(LEFT JOI

MySQL 中查询 VARCHAR 类型 JSON 数据的问题记录

《MySQL中查询VARCHAR类型JSON数据的问题记录》在数据库设计中,有时我们会将JSON数据存储在VARCHAR或TEXT类型字段中,本文将详细介绍如何在MySQL中有效查询存储为V... 目录一、问题背景二、mysql jsON 函数2.1 常用 JSON 函数三、查询示例3.1 基本查询3.2