PyQt6 删除带有嵌套 class 的自定义小部件导致程序崩溃

PyQt6 deleting custom widget with nested class causes the program to crash

我正在使用 Python 3.9.5.

我在我的项目中遇到了一些严重的问题,这里是一个最小可重现的示例代码,以及一些描述。

from PyQt6.QtCore import *
from PyQt6.QtGui import *
from PyQt6.QtWidgets import *

class Editor(QTextEdit):
    doubleClicked = pyqtSignal(QTextEdit)
    def __init__(self):
        super().__init__()
        self.setReadOnly(True)
    
    def mouseDoubleClickEvent(self, e: QMouseEvent) -> None:
        self.doubleClicked.emit(self)

class textcell(QGroupBox):
    def __init__(self, text):
        super().__init__()
        self.setAlignment(Qt.AlignmentFlag.AlignLeft | Qt.AlignmentFlag.AlignTop)
        self.label = QLabel(text)
        self.apply = makebutton('Apply')
        self.apply.hide()
        self.editor = Editor()
        self.editor.doubleClicked.connect(lambda: self.editor.setReadOnly(False))
        self.editor.doubleClicked.connect(self.apply.show)
        self.hbox = QHBoxLayout()
        self.hbox.addSpacerItem(spacer)
        self.hbox.addWidget(self.apply)
        self.vbox = QVBoxLayout()
        self.vbox.addWidget(self.label)
        self.vbox.addWidget(self.editor)
        self.vbox.addLayout(self.hbox)
        self.setLayout(self.vbox)
        self.apply.clicked.connect(self.on_ApplyClick)
    
    def on_ApplyClick(self):
        self.editor.setReadOnly(True)
        self.apply.hide()


def makebutton(text):
    button = QPushButton()
    button.setFixedSize(60, 20)
    button.setText(text)
    return button


class songpage(QGroupBox):
    def __init__(self, texts):
        super().__init__()
        self.init(texts)
        self.setCheckable(True)
        self.setChecked(False)
    
    def init(self, texts):
        self.vbox = QVBoxLayout()
        artist = textcell('Artist')
        artist.editor.setText(texts[0])
        album = textcell('Album')
        album.editor.setText(texts[1])
        title = textcell('Title')
        title.editor.setText(texts[2])
        self.vbox.addWidget(artist)
        self.vbox.addWidget(album)
        self.vbox.addWidget(title)
        self.setLayout(self.vbox)

spacer = QSpacerItem(0, 20, QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Minimum)

class Ui_MainWindow(QMainWindow):
    def __init__(self):
        super().__init__()
        self.resize(405, 720)
        self.setWindowTitle('example')
        frame = self.frameGeometry()
        center = self.screen().availableGeometry().center()
        frame.moveCenter(center)
        self.move(frame.topLeft())
        self.centralwidget = QWidget(self)
        vbox = QVBoxLayout(self.centralwidget)
        hbox = QHBoxLayout()
        add = makebutton('Add')
        delete = makebutton('Delete')
        hbox.addWidget(add)
        hbox.addSpacerItem(spacer)
        hbox.addWidget(delete)
        vbox.addLayout(hbox)
        self.scrollArea = QScrollArea(self.centralwidget)
        self.scrollArea.setWidgetResizable(True)
        self.scrollAreaWidgetContents = QWidget()
        self.scrollArea.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAlwaysOn)
        self.verticalLayout = QVBoxLayout(self.scrollAreaWidgetContents)
        self.verticalLayout.setAlignment(Qt.AlignmentFlag.AlignTop)
        self.scrollArea.setWidget(self.scrollAreaWidgetContents)
        self.scrollArea.setAlignment(Qt.AlignmentFlag.AlignLeft | Qt.AlignmentFlag.AlignTop)
        vbox.addWidget(self.scrollArea)
        self.setCentralWidget(self.centralwidget)
        add.clicked.connect(self.addObj)
        delete.clicked.connect(self.deleteObj)
    def addObj(self):
        Obj = songpage(('AAA', 'BBB', 'CCC'))
        self.verticalLayout.addWidget(Obj)
    def deleteObj(self):
        item = self.verticalLayout.itemAt(0)
        widget = item.widget()
        self.verticalLayout.removeItem(item)
        self.verticalLayout.removeWidget(widget)


app = QApplication([])
window = Ui_MainWindow()
window.show()
app.exec()

问题很简单,如果我点击添加按钮,小部件将被添加并且一切正常,如果我双击一个QTextEdit,它的应用按钮将显示并且它将从只读变为可编辑。

点击应用按钮后,按钮会隐藏,对应的QTextEdit又会变成只读状态

我终于设法将双击信号添加到 QTextEdit。

而且,问题是,如果我单击删除按钮,而不是按预期删除内容,它会使整个应用程序崩溃。

很抱歉,最小可重现示例太长了,但我只设法重现了所有代码的问题,我真的不知道出了什么问题。

那么如何解决呢?

好吧,我想通了,这是由我添加到每个使用固定大小按钮的水平布局的间隔项引起的。

所有此类布局都使用相同的间隔项,完全相同的间隔项,而不仅仅是相同。

根据我的观察,间隔项都是对同一对象的引用,它们都是同一对象的回声,该对象位于固定的内存地址。

老实说我不明白这是怎么回事,同一个对象不仅可以添加到多个布局并同时出现在所有布局中,还可以多次添加到同一个布局,但它总是仍然是原始对象,而不是其自身的副本。

我想当我将相同的间隔项添加到多个布局时,我没有添加原始间隔项,而是添加了原始项的副本,它们相同但位于不同的内存地址,很明显事实并非如此Python 是如何工作的。

所以当我删除一个小部件时,它里面的所有东西,它的布局里面的所有东西都会被删除,间隔项目也会被删除。

因为所有布局中的所有间隔项都是对原始间隔项的引用,当我删除其中一个布局时,原始间隔项也被删除,间隔项从所有其他布局中删除,但是它的阴影仍然存在,并且该项目没有从所有其他布局中正确删除,布局包含对不再存在的对象的引用,因此应用程序崩溃了。

通过删除间隔项的定义并将添加间隔项替换为 .addStretch(),错误已修复。