QTreeWidget:如何动态更改sizeHint?

2024-06-02 12:04:06 发布

您现在位置:Python中文网/ 问答频道 /正文

我有一个QTreeWidget,其中TopLevelIteps被一个自定义小部件替换。所述小部件的最大和最小高度通过QStateMachine设置动画

QTreeWidget(GIF):
QTreeWidget(gif)

自定义小部件动画(GIF):
Custom Widget Animation(gif)

问题是,当自定义小部件展开时,这些行不会调整其高度以适应它:

行高固定大小(GIF):
Row height fixed size(gif)

导致小部件项目之间重叠,而不是像这样相互推开:

我想要的结果(GIF):
The results that I'm after(gif)

我尝试在顶级项上使用setSizeHint(),但它在项/小部件之间创建了一个很大的空白:

使用setSizeHint()(GIF):
Using setSizeHint()(gif)

我在想,也许我必须实现sizeHint(),但我真的不确定该放什么。还是有更好的方法解决这个问题

如果能给我一些提示,我将不胜感激

示例代码:

# Custom Widget
class ExpandableFrame(QFrame):

    def __init__(self):
        QFrame.__init__(self)

        # Default Properties
        self.setFixedSize(200,50)
        self.setStyleSheet('background-color: #2e4076; border: 2px solid black; border-radius: 10px;')
     
        # Setup expand button
        self.expandToggle = QToolButton()
        self.expandToggle.setText('[]')
        self.expandToggle.setMaximumSize(10,20)
        self.expandToggle.setCursor(Qt.PointingHandCursor)

        # Setup layout
        layout = QHBoxLayout()
        layout.setAlignment(Qt.AlignRight)
        layout.addWidget(self.expandToggle)
        self.setLayout(layout)

        self.expandArea()

    # Animates minimumHeight and maximumHeight
    def expandArea(self):

        heigth = self.height()
        newHeigth = 100

        machine = QStateMachine(self)

        state1 = QState()
        state1.assignProperty(self, b'minimumHeight', heigth)
        state1.assignProperty(self, b'maximumHeight', heigth)

        state2 = QState()
        state2.assignProperty(self, b'minimumHeight', newHeigth)
        state2.assignProperty(self, b'maximumHeight', newHeigth)

        # Create animations
        expandAnim = QPropertyAnimation(self, 'minimumHeight')
        closeAnim = QPropertyAnimation(self, 'maximumHeight')
        expandAnim.setDuration(125)
        closeAnim.setDuration(125)
        expandAnim.setEasingCurve(QEasingCurve.Linear)  

        # Create event transitions
        eventTransition1 = QEventTransition(self.expandToggle, QEvent.MouseButtonPress)
        eventTransition1.setTargetState(state2)
        eventTransition1.addAnimation(expandAnim)
        state1.addTransition(eventTransition1)

        eventTransition2 = QEventTransition(self.expandToggle, QEvent.MouseButtonPress)
        eventTransition2.setTargetState(state1)
        eventTransition2.addAnimation(closeAnim)
        state2.addTransition(eventTransition2)
        
        # Add the states to the machine
        machine.addState(state1)
        machine.addState(state2)
        machine.setInitialState(state1)
        machine.start()
    
# Tree Widget
class CustomTree(QTreeWidget):

    customWidget = []

    def __init__(self, parent=None):
        QTreeWidget.__init__(self, parent)

        # Create top level items
        itemCount = 3
        for element in range(itemCount):
            topLevelItem = QTreeWidgetItem()
            self.addTopLevelItem(topLevelItem)

            # Replace the top level items with the expandable custom widget:
            # Get the Model Index to each item
            modelIndex = self.indexFromItem(topLevelItem, 0)
            
            # Create the ExpandableFrame widgets for all the top level items
            self.customWidget.append(ExpandableFrame())

            # Set the widget to each top level item based on its index
            self.setIndexWidget(modelIndex, self.customWidget[element])
            
            # Create more items and add them as children of the top level items
            for x in range(itemCount):
                child = QTreeWidgetItem()
                child.setText(0,'Child')
                topLevelItem.addChild(child)

下面是示例的外观:

运行的最小代码示例:
Minimal Code Example running


Tags: theselfinit部件topcreateitemsmachine
1条回答
网友
1楼 · 发布于 2024-06-02 12:04:06

一种解决方案是,每次更改任何小部件的大小时,都会发出视图项委托的sizeHintChanged信号。这告诉视图应该更新项目的位置。要实现这一点,您可以覆盖ExpandableFrameresizeEvent并发出自定义信号,例如

class ExpandableFrame(QFrame):
    resized = pyqtSignal(QVariant)

    def resizeEvent(self, event):
        self.resized.emit(self.height())
        super().resizeEvent(event)

CustomTree中,您可以通过覆盖setIndexWidgetExpandableFrame.resized连接到自定义树的委托的sizeHintChanged信号,例如

class CustomTree(QTreeWidget):
    ...

    def setIndexWidget(self, index, widget):
        super().setIndexWidget(index, widget)
        if isinstance(widget, ExpandableFrame):
            widget.resized.connect(lambda: self.itemDelegate().sizeHintChanged.emit(index))

屏幕截图:

screen capture

这种方法的缺点是,如果移动或替换小部件,则需要更新连接

相关问题 更多 >