0

我有一个 MainWindow,我从按钮调用子窗口(弹出窗口),我无法访问 def updateTime(self),它给了我一个属性错误:

AttributeError:“MainWindow”对象没有属性“updateTime”

如果我取出 MainWindow 部分,它工作正常,所以我真的不明白问题是什么。任何帮助将不胜感激。

from PyQt4 import QtGui, QtCore
from PyQt4 import *
from PyQt4.QtCore import *
import sys

CurrentTime = 0

class MainWindow(QtGui.QMainWindow):
    def __init__(self, parent=None):
        QtGui.QMainWindow.__init__(self, parent)
        self.central = QtGui.QWidget(self)
        self.setCentralWidget(self.central)

        self.hboxPB = QtGui.QHBoxLayout()
        self.vboxPB = QtGui.QVBoxLayout()
        self.MyButton = QtGui.QPushButton(self.central)
        self.MyButton.setText("Push Me")
        self.hboxPB.addWidget(self.MyButton)
        self.vboxPB.addLayout(self.hboxPB)
        self.MyButton.resize(90,90)
        self.MyButton.clicked.connect(lambda: widgetWindow.start(self))

class widgetWindow(QtGui.QWidget):
    def __init__(self, parent = None):
        QtGui.QWidget.__init__(self,parent)
        super(widgetWindow, self).__init__()
        widgetWindow.start(self)

    def start(self):
        window = QtGui.QMainWindow(self)
        window.setAttribute(QtCore.Qt.WA_DeleteOnClose)
        CentralWidget = QtGui.QWidget()
        self.timeSlider = QtGui.QSlider(QtCore.Qt.Horizontal, self)
        CentralWidgetLayout = QtGui.QHBoxLayout()
        VBox = QtGui.QVBoxLayout()
        CentralWidgetLayout.addWidget(self.timeSlider)
        VBox.addLayout(CentralWidgetLayout)
        CentralWidget.setLayout(VBox)
        window.setCentralWidget(CentralWidget)
        window.show()

        self.timer = QtCore.QTimer()
        self.timer.timeout.connect(self.updateTime)
        self.timer.start(1000)

    def updateTime(self):
        global CurrentTime
        CurrentTime = CurrentTime + 1
        self.timeSlider.setValue(CurrentTime)

def main():
    app = QtGui.QApplication(sys.argv)
    win = MainWindow()
    win.show()
    win.resize(800,450)
    sys.exit(app.exec_())

if __name__ == '__main__':
    main()
4

2 回答 2

2

你的问题从这条线开始

self.MyButton.clicked.connect(lambda: widgetWindow.start(self))

让我们解构它在做什么。

  1. 你有一个按钮,self.MyButton它存在于类的实例中MainWindow(你在main()函数中实例化类)

  2. 您正在连接到clicked按钮的信号。

  3. 您连接到此信号的lambda函数是调用类中的函数的函数widgetWindow。请注意,这与调用类实例的方法不同。您没有在这里实例化一个类(您没有创建一个对象)。您是说,在类中使用方法定义startwidgetWindow但使其作用于类实例self所在的对象。selfMainWindow

希望您现在开始看到您所做的事情的问题。您没有创建widgetWindow类的实例,而是尝试使用 from 的方法,widgetWindow就好像它是MainWindow. 如果您仍然遇到问题,我建议您阅读更多关于 Python 中面向对象编程的内容(特别是如果您不清楚类和对象之间的区别)

因此,解决方案是创建一个实例widgetWindow(而不是直接访问类的方法),并将您的按钮连接到该实例的方法。我已经修改了您的代码来执行此操作,该代码发布在下面。我已经评论了我更改的部分。如果您对我所做的事情有任何疑问,请告诉我。

from PyQt4 import QtGui, QtCore
from PyQt4 import *
from PyQt4.QtCore import *
import sys

CurrentTime = 0

class MainWindow(QtGui.QMainWindow):
    def __init__(self, parent=None):
        QtGui.QMainWindow.__init__(self, parent)
        self.central = QtGui.QWidget(self)
        self.setCentralWidget(self.central)

        self.hboxPB = QtGui.QHBoxLayout()
        self.vboxPB = QtGui.QVBoxLayout()
        self.MyButton = QtGui.QPushButton(self.central)
        self.MyButton.setText("Push Me")
        self.hboxPB.addWidget(self.MyButton)
        self.vboxPB.addLayout(self.hboxPB)
        self.MyButton.resize(90,90)

        # instantiate the widgetWindow (pass this window as the parent)
        self.widgetwindow = widgetWindow(self)
        # connect the button to the start method of this instance we created above
        self.MyButton.clicked.connect(self.widgetwindow.start)

# no need to subclass QWidget here. This is just a wrapper class to hold your main window
class widgetWindow(object):
    def __init__(self, parent = None):
        # Store the parent for use later
        self.parent = parent

    def start(self):
        # change the window parent to be the parent we stored in the __init__ method
        window = QtGui.QMainWindow(self.parent)
        window.setAttribute(QtCore.Qt.WA_DeleteOnClose)
        # Add a signal to stop the timer when the window is destroyed
        window.destroyed.connect(self.stopTimer)
        CentralWidget = QtGui.QWidget()
        # Change the parent to be the window we just created
        self.timeSlider = QtGui.QSlider(QtCore.Qt.Horizontal, window)
        CentralWidgetLayout = QtGui.QHBoxLayout()
        VBox = QtGui.QVBoxLayout()
        CentralWidgetLayout.addWidget(self.timeSlider)
        VBox.addLayout(CentralWidgetLayout)
        CentralWidget.setLayout(VBox)
        window.setCentralWidget(CentralWidget)
        window.show()

        self.timer = QtCore.QTimer()
        self.timer.timeout.connect(self.updateTime)
        self.timer.start(1000)

    def updateTime(self):
        global CurrentTime
        CurrentTime = CurrentTime + 1
        self.timeSlider.setValue(CurrentTime)

    # Method to stop the timer once the window is destroyed
    def stopTimer(self):
        self.timer.stop()

def main():
    app = QtGui.QApplication(sys.argv)
    win = MainWindow()
    win.show()
    win.resize(800,450)
    sys.exit(app.exec_())

if __name__ == '__main__':
    main()
于 2015-08-23T11:26:30.620 回答
-2

试试这个:

from PyQt4 import QtGui, QtCore
from PyQt4 import *
from PyQt4.QtCore import *
import sys

CurrentTime = 0
class widgetWindow(QtGui.QWidget):
    def __init__(self, parent = None):
        QtGui.QWidget.__init__(self,parent)
        super(widgetWindow, self).__init__()
        widgetWindow.start(self)

    def start(self):
        window = QtGui.QMainWindow(self)
        window.setAttribute(QtCore.Qt.WA_DeleteOnClose)
        CentralWidget = QtGui.QWidget()
        self.timeSlider = QtGui.QSlider(QtCore.Qt.Horizontal, self)
        CentralWidgetLayout = QtGui.QHBoxLayout()
        VBox = QtGui.QVBoxLayout()
        CentralWidgetLayout.addWidget(self.timeSlider)
        VBox.addLayout(CentralWidgetLayout)
        CentralWidget.setLayout(VBox)
        window.setCentralWidget(CentralWidget)
        window.show()

        self.timer = QtCore.QTimer()
        self.timer.timeout.connect(self.updateTime)
        self.timer.start(1000)

    def updateTime(self):
        global CurrentTime
        CurrentTime = CurrentTime + 1
        self.timeSlider.setValue(CurrentTime)



class MainWindow(QtGui.QMainWindow,widgetWindow):#here add subclass
    def __init__(self, parent=None):
        QtGui.QMainWindow.__init__(self, parent)
        self.central = QtGui.QWidget(self)
        self.setCentralWidget(self.central)

        self.hboxPB = QtGui.QHBoxLayout()
        self.vboxPB = QtGui.QVBoxLayout()
        self.MyButton = QtGui.QPushButton(self.central)
        self.MyButton.setText("Push Me")
        self.hboxPB.addWidget(self.MyButton)
        self.vboxPB.addLayout(self.hboxPB)
        self.MyButton.resize(90,90)
        self.MyButton.clicked.connect(lambda: widgetWindow.start(self))



def main():
    app = QtGui.QApplication(sys.argv)
    win = MainWindow()
    win.show()
    win.resize(800,450)
    sys.exit(app.exec_())

if __name__ == '__main__':
    main()

你有错误的命名空间。您必须确保使用正确的命名空间,否则解释器不知道查找类的正确位置。

于 2015-08-23T08:35:37.523 回答