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

主要功能概述

当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

代码实现

以下是完整的实现代码:

python 复制代码
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. 初始化布局参数

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

python 复制代码
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的宽度和高度

python 复制代码
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事件:

python 复制代码
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方法是核心逻辑所在:

python 复制代码
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)

注意事项

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

END 😃