找回密码
 立即注册
首页 业界区 业界 QT实现DockWidget内部组件自动换行布局

QT实现DockWidget内部组件自动换行布局

挽幽 3 天前
主要功能概述

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

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

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

程序环境

Python 3.8.9
pyside6==6.1.3
  1. pip install pyside6==6.1.3
复制代码
设计结构图解

1.png

实现效果

2.gif

demo代码获取

Gitee:dockwidget-demo

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

3.png

代码实现

以下是完整的实现代码:
  1. import re
  2. import sys
  3. from PySide6.QtWidgets import QApplication
  4. from ui_main_windowtest import *
  5. class MainWindow(QMainWindow, Ui_MainWindow):
  6.     def __init__(self):
  7.         super().__init__()
  8.         self.setupUi(self)
  9.         self.row = 0    # 行
  10.         self.col = 0    # 列
  11.         self.buttons_per_row = 0    # 每行按钮数量
  12.         self.scrollArea.widget().installEventFilter(self)
  13.         # 获取布局参数
  14.         self.h_spacing = self.gridLayout_2.horizontalSpacing()
  15.         self.v_spacing = self.gridLayout_2.verticalSpacing()
  16.         self.margins = self.gridLayout_2.contentsMargins()
  17.         # 获取第一个按钮的参考尺寸
  18.         if self.gridLayout_2.count() > 0:
  19.             first_button = self.gridLayout_2.itemAt(0).widget()
  20.             self.button_width = first_button.sizeHint().width() + self.h_spacing
  21.             self.button_height = first_button.sizeHint().height() + self.v_spacing + 10 # 根据测试这里+10容差可以防止出现程序无限执行rearrangeButtons方法的情况
  22.     def eventFilter(self, obj, event):
  23.         """监控ScrollArea内widget的resize事件"""
  24.         if event.type() == QEvent.Resize:
  25.             # 获取当前可用宽高
  26.             available_height = self.scrollArea.widget().size().height()
  27.             available_width = self.scrollArea.widget().size().width()
  28.             # 检查是否需要重新排列按钮
  29.             if available_height > (self.row + 1) * self.button_height and available_width > self.button_width:
  30.                 self.rearrangeButtons()
  31.             elif available_width > (self.buttons_per_row + 1) * self.button_width and available_height > self.button_height:
  32.                 self.rearrangeButtons()
  33.         return super().eventFilter(obj, event)
  34.     def rearrangeButtons(self):
  35.         """重新排列按钮以适应新的窗口大小"""
  36.         # 计算新的每行按钮数量
  37.         available_width = self.scrollArea.widget().width() - self.margins.left() - self.margins.right()
  38.         new_buttons_per_row = max(1, available_width // self.button_width)
  39.         # 如果每行按钮数量没有变化,则不需要重新排列
  40.         if new_buttons_per_row != self.buttons_per_row:
  41.             self.buttons_per_row = new_buttons_per_row
  42.         else:
  43.             return
  44.         # 收集所有按钮
  45.         buttons = []
  46.         for i in range(self.gridLayout_2.count()):
  47.             item = self.gridLayout_2.itemAt(i)
  48.             if item.widget():
  49.                 buttons.append(item.widget())
  50.         # 按按钮名称自然排序(1, 2, 3, ..., 10, 11),不排序每次重启程序顺序都会不一样
  51.         buttons.sort(key=lambda btn: [
  52.             int(text) if text.isdigit() else text.lower()
  53.             for text in re.split('([0-9]+)', btn.objectName())
  54.         ])
  55.         # 清除当前布局
  56.         while self.gridLayout_2.count():
  57.             item = self.gridLayout_2.takeAt(0)
  58.             if item.widget():
  59.                 item.widget().setParent(None)
  60.         # 重新排列按钮
  61.         for i, button in enumerate(buttons):
  62.             self.row = i // self.buttons_per_row
  63.             self.col = i % self.buttons_per_row
  64.             self.gridLayout_2.addWidget(button, self.row, self.col)
  65. if __name__ == "__main__":
  66.     app = QApplication(sys.argv)
  67.     window = MainWindow()
  68.     window.show()
  69.     sys.exit(app.exec())
复制代码
代码解析

1. 初始化布局参数

获取了布局的关键参数,这些参数用于准确计算可用空间和按钮尺寸:
  1. self.h_spacing = self.gridLayout_2.horizontalSpacing()
  2. self.v_spacing = self.gridLayout_2.verticalSpacing()
  3. self.margins = self.gridLayout_2.contentsMargins()
复制代码
2. 计算按钮尺寸

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

ps:这里变量名写成了button,其实获取的是Qwidget的宽度和高度
  1. first_button = self.gridLayout_2.itemAt(0).widget()
  2. self.button_width = first_button.sizeHint().width() + self.h_spacing
  3. self.button_height = first_button.sizeHint().height() + self.v_spacing + 10
复制代码
注意这里加了10像素的容差,这是为了避免在某些边界情况下出现无限循环的问题。

3. 事件过滤器

通过eventFilter监控ScrollArea内widget的resize事件:
  1. def eventFilter(self, obj, event):
  2.     if event.type() == QEvent.Resize:
  3.         # 获取当前可用宽高
  4.         available_height = self.scrollArea.widget().size().height()
  5.         available_width = self.scrollArea.widget().size().width()
  6.         # 检查是否需要重新排列
  7.         if available_height > (self.row + 1) * self.button_height and available_width > self.button_width:
  8.             self.rearrangeButtons()
  9.         elif available_width > (self.buttons_per_row + 1) * self.button_width and available_height > self.button_height:
  10.             self.rearrangeButtons()
  11.     return super().eventFilter(obj, event)
复制代码
4. 重新排列按钮

rearrangeButtons方法是核心逻辑所在:
  1. def rearrangeButtons(self):
  2.     # 计算新的每行按钮数量
  3.     available_width = self.scrollArea.widget().width() - self.margins.left() - self.margins.right()
  4.     new_buttons_per_row = max(1, available_width // self.button_width)
  5.     # 如果每行按钮数量没有变化,则不需要重新排列
  6.     if new_buttons_per_row != self.buttons_per_row:
  7.         self.buttons_per_row = new_buttons_per_row
  8.     else:
  9.         return
  10.     # 收集并排序按钮
  11.     buttons = []
  12.     for i in range(self.gridLayout_2.count()):
  13.         item = self.gridLayout_2.itemAt(i)
  14.         if item.widget():
  15.             buttons.append(item.widget())
  16.     # 使用正则表达式实现按钮名称的自然排序,可以通过命名的方式强制规定Qwidget组件顺序
  17.     buttons.sort(key=lambda btn: [
  18.         int(text) if text.isdigit() else text.lower()
  19.         for text in re.split('([0-9]+)', btn.objectName())
  20.     ])
  21.     # 清除当前布局
  22.     while self.gridLayout_2.count():
  23.         item = self.gridLayout_2.takeAt(0)
  24.         if item.widget():
  25.             item.widget().setParent(None)
  26.     # 重新排列
  27.     for i, button in enumerate(buttons):
  28.         self.row = i // self.buttons_per_row
  29.         self.col = i % self.buttons_per_row
  30.         self.gridLayout_2.addWidget(button, self.row, self.col)
复制代码
注意事项


  • 按钮尺寸:DockWidget下的所有Qwidget应具有相同的尺寸,否则布局可能会不均匀
  • 容差设置:代码中的+10像素容差是经验值,可能需要根据实际情况调整
END
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

您需要登录后才可以回帖 登录 | 立即注册