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

QT实现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_spacingself.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_rowelse: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_rowself.col = i % self.buttons_per_rowself.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_rowelse: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_rowself.col = i % self.buttons_per_rowself.gridLayout_2.addWidget(button, self.row, self.col)

注意事项

  1. 按钮尺寸:DockWidget下的所有Qwidget应具有相同或相似的尺寸,否则布局可能会不均匀
  2. 容差设置:代码中的+10像素容差是经验值,可能需要根据实际情况调整

END 😃

http://www.gsyq.cn/news/27558.html

相关文章:

  • 2025 蛋白/8秒液体/发膜推荐榜:玛丝兰 5 星领跑,这些修护力出众的品牌值得囤!西安悦己容凭技术实力登顶
  • 2025年知名的雕塑推荐TOP品牌企业 - Di
  • 2025 聚焦重庆标书制作服务品质:重庆睿标通领衔,工程/建筑/市政/物业服务标书制作专业机构推荐​
  • Spring - 教程
  • 2025 年最新推荐!集装箱拖车供应厂家权威榜单重磅发布,全方位解析优质厂家实力助企业选对合作伙伴
  • 实战案例 | 利用山海鲸可视化软件,构建制造业数字孪生监控大屏
  • 【IEEE出版 | 往届4年稳定EI检索 | 高录用、稳定检索】第五届无线通信、网络与物联网国际学术会议(WCNIoT 2025)
  • SS251021B. 箱客思 做题记录 - 邻补角
  • 1.51.0 mm LTCC低通,DC-3.7 GHz,带内插损≤0.6 dB,军工温宽——国产HT-LFCG-3700+(Pin-to-Pin替代LFCG-3700+)
  • Pandas 深入学习【3】材料标准化处理 StandardScaler
  • web预览tif格式文件踩坑
  • 电网不平衡条件下DFIG风力发电机动态建模与控制
  • C#实现CRC8、CRC16、CRC32校验算法
  • 完整教程:leetcode_138 随机链表的复制
  • 成功案例分享|ArmSoM CM5赋能海洋保育,边缘AI守护鲸豚之声
  • 复矩阵的QR分解
  • Maven-继承与聚合 - 实践
  • 千疮百孔的心被恨与悲彻底剥离 Kill my memory 让我将快乐全忘记
  • 权威调研榜单:天津全屋定制整体橱柜方案TOP4榜单好评深度解析
  • 单时段机组组合优化的粒子群算法实现(MATLAB)
  • SketchUp 2022-2025 坯子插件库 v3.2.6官方正式版下载安装教程
  • 2025 年固化剂生产厂家最新推荐排行榜:聚焦国内优质厂商,助力选购高性价比混凝土及厂房用固化剂
  • 广义串并联图
  • 第八章 内存马分析-java01-nacos
  • 2025 种植棚/养殖棚/工程/羊肚菌/保温/园林/加厚/绿化/草苫子推荐榜:济宁泽萌草制品 5 星领跑,适配大棚 / 混凝土 / 园艺多场景需求
  • AI不是魔法,而是算力+算法+数据的平衡艺术!
  • 雪碧图动画实例 - 教程
  • 2025/10/22
  • 2025 年钢管厂家最新推荐榜:覆盖精密钢管、汽车钢管、高强钢钢管等品类,为下游采购企业提供权威选品参考
  • 生成函数入门