主要功能概述
当DockWidget窗口大小改变时,内部的按钮能够自动重新排列,以最佳方式利用可用空间。具体表现为:
1. 当水平空间足够时,按钮排成一行
2. 当水平空间不足时,按钮自动换行
程序环境
Python 3.8.9
pyside6==6.1.3
pip install pyside6==6.1.3
设计结构图解

实现效果

demo代码获取
Gitee:dockwidget-demo
百度网盘:https://pan.baidu.com/s/1PRAjVGBtLQFZkWnZsJ2f2A?pwd=eiti

代码实现
以下是完整的实现代码:
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像素容差是经验值,可能需要根据实际情况调整