当前位置: 首页 > news >正文

基于PySide6实现DockWidget内部组件自动换行布局

DockWidget窗口大小改变时,内部的按钮能够自动重新排列,以最佳方式利用可用空间。具体表现为:

1. 当水平空间足够时,按钮排成一行

2. 当水平空间不足时,按钮自动换行

程序环境

Python 3.8.9

pyside6==6.1.3

pip install pyside6==6.1.3

设计结构图解

微信图片_2025-10-22_152148_073

实现效果

20251022_155729

demo代码获取

Gitee:dockwidget-demo

百度网盘:https://pan.baidu.com/s/1PRAjVGBtLQFZkWnZsJ2f2A?pwd=eiti

image

代码实现

以下是完整的实现代码:

import re

import sys

from PySide6.QtWidgets import QApplication

from ui_main_windowtest import *

class MainWindow(QMainWindow, Ui_MainWindow):

def __init__(self):

super().__init__()

self.setupUi(self)

self.row = 0 # 行

self.col = 0 # 列

self.buttons_per_row = 0 # 每行按钮数量

self.scrollArea.widget().installEventFilter(self)

# 获取布局参数

self.h_spacing = self.gridLayout_2.horizontalSpacing()

self.v_spacing = self.gridLayout_2.verticalSpacing()

self.margins = self.gridLayout_2.contentsMargins()

# 获取第一个按钮的参考尺寸

if self.gridLayout_2.count() > 0:

first_button = self.gridLayout_2.itemAt(0).widget()

self.button_width = first_button.sizeHint().width() + self.h_spacing

self.button_height = first_button.sizeHint().height() + self.v_spacing + 10 # 根据测试这里+10容差可以防止出现程序无限执行rearrangeButtons方法的情况

def eventFilter(self, obj, event):

"""监控ScrollArea内widget的resize事件"""

if event.type() == QEvent.Resize:

# 获取当前可用宽高

available_height = self.scrollArea.widget().size().height()

available_width = self.scrollArea.widget().size().width()

# 检查是否需要重新排列按钮

if available_height > (self.row + 1) * self.button_height and available_width > self.button_width:

self.rearrangeButtons()

elif available_width > (self.buttons_per_row + 1) * self.button_width and available_height > self.button_height:

self.rearrangeButtons()

return super().eventFilter(obj, event)

def rearrangeButtons(self):

"""重新排列按钮以适应新的窗口大小"""

# 计算新的每行按钮数量

available_width = self.scrollArea.widget().width() - self.margins.left() - self.margins.right()

new_buttons_per_row = max(1, available_width // self.button_width)

# 如果每行按钮数量没有变化,则不需要重新排列

if new_buttons_per_row != self.buttons_per_row:

self.buttons_per_row = new_buttons_per_row

else:

return

# 收集所有按钮

buttons = []

for i in range(self.gridLayout_2.count()):

item = self.gridLayout_2.itemAt(i)

if item.widget():

buttons.append(item.widget())

# 按按钮名称自然排序(1, 2, 3, ..., 10, 11),不排序每次重启程序顺序都会不一样

buttons.sort(key=lambda btn: [

int(text) if text.isdigit() else text.lower()

for text in re.split('([0-9]+)', btn.objectName())

])

# 清除当前布局

while self.gridLayout_2.count():

item = self.gridLayout_2.takeAt(0)

if item.widget():

item.widget().setParent(None)

# 重新排列按钮

for i, button in enumerate(buttons):

self.row = i // self.buttons_per_row

self.col = i % self.buttons_per_row

self.gridLayout_2.addWidget(button, self.row, self.col)

if __name__ == "__main__":

app = QApplication(sys.argv)

window = MainWindow()

window.show()

sys.exit(app.exec())

代码解析

1. 初始化布局参数

获取了布局的关键参数,这些参数用于准确计算可用空间和按钮尺寸:

self.h_spacing = self.gridLayout_2.horizontalSpacing()

self.v_spacing = self.gridLayout_2.verticalSpacing()

self.margins = self.gridLayout_2.contentsMargins()

2. 计算按钮尺寸

我们以scrollArea第一个Qwidget为参考,计算Qwidget的宽度和高度(所有Qwidget宽高必须统一):

ps:这里变量名写成了button,其实获取的是Qwidget的宽度和高度

first_button = self.gridLayout_2.itemAt(0).widget()

self.button_width = first_button.sizeHint().width() + self.h_spacing

self.button_height = first_button.sizeHint().height() + self.v_spacing + 10

注意这里加了10像素的容差,这是为了避免在某些边界情况下出现无限循环的问题。

3. 事件过滤器

通过eventFilter监控ScrollArea内widget的resize事件:

def eventFilter(self, obj, event):

if event.type() == QEvent.Resize:

# 获取当前可用宽高

available_height = self.scrollArea.widget().size().height()

available_width = self.scrollArea.widget().size().width()

# 检查是否需要重新排列

if available_height > (self.row + 1) * self.button_height and available_width > self.button_width:

self.rearrangeButtons()

elif available_width > (self.buttons_per_row + 1) * self.button_width and available_height > self.button_height:

self.rearrangeButtons()

return super().eventFilter(obj, event)

4. 重新排列按钮

rearrangeButtons方法是核心逻辑所在:

def rearrangeButtons(self):

# 计算新的每行按钮数量

available_width = self.scrollArea.widget().width() - self.margins.left() - self.margins.right()

new_buttons_per_row = max(1, available_width // self.button_width)

# 如果每行按钮数量没有变化,则不需要重新排列

if new_buttons_per_row != self.buttons_per_row:

self.buttons_per_row = new_buttons_per_row

else:

return

# 收集并排序按钮

buttons = []

for i in range(self.gridLayout_2.count()):

item = self.gridLayout_2.itemAt(i)

if item.widget():

buttons.append(item.widget())

# 使用正则表达式实现按钮名称的自然排序,可以通过命名的方式强制规定Qwidget组件顺序

buttons.sort(key=lambda btn: [

int(text) if text.isdigit() else text.lower()

for text in re.split('([0-9]+)', btn.objectName())

])

# 清除当前布局

while self.gridLayout_2.count():

item = self.gridLayout_2.takeAt(0)

if item.widget():

item.widget().setParent(None)

# 重新排列

for i, button in enumerate(buttons):

self.row = i // self.buttons_per_row

self.col = i % self.buttons_per_row

self.gridLayout_2.addWidget(button, self.row, self.col)

注意事项

按钮尺寸:DockWidget下的所有Qwidget应具有相同的尺寸,否则布局可能会不均匀

容差设置:代码中的+10像素容差是经验值,可能需要根据实际情况调整

http://www.rkmt.cn/news/131966.html

相关文章:

  • 通达信好公式个股突破
  • 【开题答辩全过程】以 基于Vue的爱心公益募捐平台的设计与实现为例,包含答辩的问题和答案
  • Thinkphp和Laravel教务-评教,教学质量反馈系统 教师评价系统vue
  • 通达信老司机上车
  • 通达信共振指标成功率80%
  • Spring(循环依赖,spring与springboot的区别)
  • 打造三菱PLC自动售货机系统:从搭建到实现
  • Thinkphp和Laravel考研资料信息分享共享交流平台 rm8u3-vue
  • cesium126,240311,Ce for Ue 加载天地图P1-了解天地图的地图API:
  • 通达信随心买
  • Thinkphp和Laravel酒店客房 vue
  • Windows系统文件efswrt.dll丢失或损坏 下载修复方法
  • 【计算机毕业设计案例】基于springboot的云与糖蛋糕购物平台系统的设计与实现烘焙行业线上线下融合、个性化定制服务(程序+文档+讲解+定制)
  • 线性代数:多维世界的变形工具箱
  • 【计算机毕业设计案例】基于springboot的实验室预约系统的设计与实现实验室使用情况设备预约、时段管控、使用记录及安全监管的数字化服务(程序+文档+讲解+定制)
  • 计算机Java毕设实战-基于SpringBoot+Vue实现的前后端分离在线考试系统基于springboot的在线考试系统的设计与实现【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • Java毕设选题推荐:基于javaweb在线考试系统设计与实现基于springboot的在线考试系统的设计与实现【附源码、mysql、文档、调试+代码讲解+全bao等】
  • 解读一个简单的x64程序的反汇编结果
  • 你是你吃出来的
  • 最新ASNT七大无损检测技巧全解析(RT/UT/PA/MT/PT/ECT/VT实操指南)- 上海欧鑫 ASNT 认证培训
  • 基于深度学习的草莓健康度检测系统演示与介绍(YOLOv12/v11/v8/v5模型+Pyqt5界面+训练代码+数据集)
  • LLM基础
  • MATLAB电力系统继电保护之自动重合闸
  • CVE-2025-68079:ThemeNectar Salient Shortcodes插件中的跨站脚本漏洞解析
  • OKR 要不要考核?
  • 开题报告反复被打回?虎贲等考 AI:让研究起点精准踩中学术规范
  • 写论文软件哪个好?虎贲等考 AI:毕业论文创作的 “全能通关神器”
  • 9 款 AI 写论文哪个好?实测对比后,这款全流程神器成学术党首选! 毕业论文写作季,AI 写作工具已成学子 “救命稻草”。市面上百度智能云千帆大模型、科大讯
  • win32创建内存映射文件
  • 并发模式下的内存压力:如果同时启动 100 个 Transition 任务,React 堆内存会爆炸吗?