1

我无法动态更改 QStackedWidget 的多个实例的布局/元素。下面的完整代码使人们能够切换一个按钮,单击该按钮将创建一个具有被切换按钮的相应名称的 QFrame,并且在取消切换时将因此删除具有与该按钮关联的相应名称的框架。但是,我希望能够使用它旁边的 QComboBox 来修改通过左侧按钮创建的 QFrame 的布局。在前面的示例中,我已经能够取消单击所有按钮,从而删除框架,然后通过组合框选择正确的布局,然后在通过重新切换按钮重新创建框架时查看框架的调整布局。

from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
import sys



class Box(QFrame):
    def __init__ (self,title):
        super().__init__()
        self.title = title

        self.frame1_UI = QWidget()
        self.frame2_UI = QWidget()
        self.frame3_UI= QWidget()

        self.frame1()
        self.frame2()
        self.frame3()

        self.Stack = QStackedWidget(self)
        self.Stack.addWidget(self.frame1_UI)
        self.Stack.addWidget(self.frame2_UI)
        self.Stack.addWidget(self.frame3_UI)

        name = QLabel(title)  
        task_h_box = QHBoxLayout()
        task_h_box.addWidget(name)
        task_h_box.addWidget(self.Stack)
        self.setLayout(task_h_box)

    def text(self):
        return self.title

    def frame1(self):
        label = QLabel('frame1')
        task_h_box = QHBoxLayout()
        task_h_box.addWidget(label)
        self.setStyleSheet("QFrame{background-color: rgb(0,0,0); border-radius: 10px;}")
        self.frame1_UI.setLayout(task_h_box)
    
    def frame2(self):
        
        label = QLabel('frame2')
        task_h_box = QHBoxLayout()
        task_h_box.addWidget(label)
        self.frame2_UI.setLayout(task_h_box)
        self.setStyleSheet("QFrame{background-color: rgb(100,100,100); border-radius: 10px;}")

    def frame3(self):
        label = QLabel('frame3')
        task_h_box = QHBoxLayout()
        task_h_box.addWidget(label)
        self.frame3_UI.setLayout(task_h_box)
        self.setStyleSheet("QFrame{background-color: rgb(150,150,150); border-radius: 10px;}")


class window(QMainWindow):
    def __init__(self):
        super().__init__()
        

        widget = QWidget(self)
        
        #Widgets
        creator_button = QPushButton('Creator')
        creator_button.setCheckable(True)
        creator_button.clicked.connect(self.creation)

        creator_button2 = QPushButton('Creator 2')
        creator_button2.setCheckable(True)
        creator_button2.clicked.connect(self.creation)
        
        self.control_button = QComboBox()
        self.control_button.addItem('frame1')
        self.control_button.addItem('frame2')
        self.control_button.addItem('frame3')
        
        self.control_button.currentIndexChanged.connect(self.transition)
        
        #Layouts
        self.layout = QHBoxLayout(widget)
        self.creator_layout = QVBoxLayout()
        self.control_layout = QVBoxLayout()
        self.box_layout = QVBoxLayout()
        self.layout.addLayout(self.creator_layout)
        self.layout.addLayout(self.control_layout)
        self.layout.addLayout(self.box_layout)
        
        #Placements
        self.creator_layout.addWidget(creator_button)
        self.creator_layout.addWidget(creator_button2)
        self.control_layout.addWidget(self.control_button)

        self.setCentralWidget(widget)
        self.boxList = []
        self.show()

    def creation(self, checked):
        sender = self.sender()
        title = str('I am son of ' + sender.text())
        if checked:
            widget = Box(title)
            self.box_layout.addWidget(widget)
            self.boxList.append(widget.text())
        elif not checked:
            x = self.boxList.index(title)
            self.box_layout.itemAt(x).widget().deleteLater()
            self.boxList.remove(title)

    
    def transition(self, i): 
        title = ('I am son of Creator 2') 
        self.Box = Box(title)
        self.Box.Stack.setCurrentIndex(i)

        
        

app = QApplication(sys.argv)
window = window()
sys.exit(app.exec_())
4

1 回答 1

0

您不需要在 中创建另一个 Box transition,只需在现有的中更改 QStackWidget 的索引。

def transition(self, i): 
    for box in self.findChildren(Box):
        box.Stack.setCurrentIndex(i)

在创建 Box 时设置索引也是有意义的。

def creation(self, checked):
    sender = self.sender()
    title = str('I am son of ' + sender.text())
    if checked:
        widget = Box(title)
        widget.Stack.setCurrentIndex(self.control_button.currentIndex())
        self.box_layout.addWidget(widget)
        self.boxList.append(widget.text())
    elif not checked:
        x = self.boxList.index(title)
        self.box_layout.itemAt(x).widget().deleteLater()
        self.boxList.remove(title)

这应该通过对代码的最小更改来做到这一点。但是,如果您在切换按钮时显示/隐藏 Box 小部件,而不是每次都创建/销毁新实例,则可以简化整个事情。

class window(QMainWindow):
    def __init__(self):
        super().__init__()
        widget = QWidget(self)

        self.box = Box('I am son of Creator', visible=False)
        self.box2 = Box('I am son of Creator 2', visible=False)

        creator_button = QPushButton('Creator', checkable=True)
        creator_button.toggled[bool].connect(self.box.setVisible)
        creator_button2 = QPushButton('Creator 2', checkable=True)
        creator_button2.toggled[bool].connect(self.box2.setVisible)
        
        control_button = QComboBox()
        control_button.addItems(['frame1', 'frame2', 'frame3'])
        control_button.currentIndexChanged.connect(self.transition)
        
        grid = QGridLayout(widget)
        grid.addWidget(creator_button, 0, 0)
        grid.addWidget(creator_button2, 1, 0)
        grid.addWidget(control_button, 0, 1, 2, 1)
        grid.addWidget(self.box, 0, 2)
        grid.addWidget(self.box2, 1, 2)
        
        self.setCentralWidget(widget)
        self.show()
    
    def transition(self, i):
        self.box.Stack.setCurrentIndex(i)
        self.box2.Stack.setCurrentIndex(i)

并且只需要为 Box 构造函数添加 unpack 关键字参数visible=False

class Box(QFrame):
    def __init__ (self, title, **kwargs):
        super().__init__(**kwargs)
于 2021-02-21T06:03:54.143 回答