Python & PyQt。剪贴板内容到 listwidget

Python & PyQt. Clipboard contents to listwidget

我正在用 PyQt4 编写我的第一个 python 应用程序。我是 python 的初学者,通常是编程新手。基本上这个应用程序所做的是监视剪贴板数据变化和 put/display 这作为 QlistWidget 中的一个项目。您还可以通过复制粘贴数据并单击添加按钮来手动添加项目。至于监控数据并自动将其添加到列表小部件中,我添加了一个条件,需要切换单选按钮才能让 "monitoring feature" 执行它的操作。添加项目的手动方式有效,但监控功能似乎不起作用。它不会将数据(保存在剪贴板中)添加到列表小部件。我想我似乎错过了什么。 我正在使用 Python 2.7 和 PyQt 4.11

这是我的代码。我删除了额外的代码行(样式表等~)。

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

try:
    _fromUtf8 = QtCore.QString.fromUtf8
except AttributeError:
    def _fromUtf8(s):
        return s

try:
    _encoding = QtGui.QApplication.UnicodeUTF8
    def _translate(context, text, disambig):
        return QtGui.QApplication.translate(context, text, disambig, _encoding)
except AttributeError:
    def _translate(context, text, disambig):
        return QtGui.QApplication.translate(context, text, disambig)

rbs = False
class Ui_Form(QtGui.QWidget):
    def __init__(self):
        QtGui.QWidget.__init__(self)
        self.setupUi(self)
    def setupUi(self, Form):
        Form.setObjectName(_fromUtf8("Form"))
        Form.resize(385, 277)
        self.gridLayout = QtGui.QGridLayout(Form)
        self.gridLayout.setObjectName(_fromUtf8("gridLayout"))
        self.radioButton = QtGui.QRadioButton(Form)
        self.radioButton.setObjectName(_fromUtf8("radioButton"))
        self.gridLayout.addWidget(self.radioButton, 0, 1, 1, 1)

        self.radioButton.toggled.connect(self.toggledRB)

        self.lineEdit = QtGui.QLineEdit(Form)
        self.lineEdit.setObjectName(_fromUtf8("lineEdit"))
        self.gridLayout.addWidget(self.lineEdit, 1, 0, 1, 1)

        self.pushButton = QtGui.QPushButton(Form)
        self.pushButton.setObjectName(_fromUtf8("pushButton"))
        self.gridLayout.addWidget(self.pushButton, 1, 1, 1, 1)

        self.pushButton.clicked.connect(self.addItem)

        self.listWidget = QtGui.QListWidget(Form)
        self.listWidget.setEditTriggers(QtGui.QAbstractItemView.DoubleClicked|QtGui.QAbstractItemView.EditKeyPressed|QtGui.QAbstractItemView.SelectedClicked)
        self.listWidget.setSelectionBehavior(QtGui.QAbstractItemView.SelectItems)
        self.listWidget.setMovement(QtGui.QListView.Snap)
        self.listWidget.setUniformItemSizes(False)
        self.listWidget.setObjectName(_fromUtf8("listView"))
        self.gridLayout.addWidget(self.listWidget, 2, 0, 1, 2)


        self.retranslateUi(Form)
        QtCore.QMetaObject.connectSlotsByName(Form)

    def retranslateUi(self, Form):
        Form.setWindowTitle(_translate("Form", "Form", None))

        self.radioButton.setText(_translate("Form", "monitor clipboard", None))
        self.pushButton.setText(_translate("Form", "Add", None))

    def toggledRB(self, bool):
        global rbs
        if self.radioButton.isChecked():
            rbs = True
        else:
            rbs = False
        print rbs

    def addItem(self):
        yyy = self.lineEdit.text()
        self.listWidget.addItem(yyy)

    def dataReceiver(self, intel):
        print 'data is here!'
        self.listWidget.addItem(intel)

class clipboardListener(QObject):  

    @pyqtSlot()  
    def changedSlot(self):
        global rbs
        print 'clipboard data detected!'
        if rbs == True:
            intel = QApplication.clipboard().text()
            print intel
            xxx = Ui_Form()
            xxx.dataReceiver(intel)
        else:
            print 'False Alarm. There was no data. I was just kidding.'

if __name__ == "__main__":
    import sys
    app = QtGui.QApplication(sys.argv)
    Form = QtGui.QWidget()
    ui = Ui_Form()
    ui.setupUi(Form)
    Form.show()

    listener = clipboardListener()
    QObject.connect(QApplication.clipboard(),SIGNAL("dataChanged()"),listener,SLOT("changedSlot()"))

    sys.exit(app.exec_()) 

在您的 clibpboardListener 中,您创建了一个 Ui_Form 的新实例,并在该实例上调用 dataReceiver 方法,而不是您之前创建的 ui 实例。

要使这项工作正常进行,您需要将调用 dataReceiver 方法的对象传递给 clipboardListener

另一个建议:您应该使用 new style signals 而不是调用 QObject.connect(...)。在 python 中,你不应该使用 ==True 进行比较,要么直接使用布尔值作为条件,要么使用 is True 如果你想让它显式检查True 不适用于比较对象的隐式真值。

这个有效:

...

class clipboardListener(QObject):  

    def __init__(self, target):
        QObject.__init__(self)
        self.target = target

    @pyqtSlot()  
    def changedSlot(self):
        global rbs
        print 'clipboard data detected!'
        if rbs:
            intel = QApplication.clipboard().text()
            print intel
            self.target.dataReceiver(intel)
        else:
            print 'False Alarm. There was no data. I was just kidding.'

if __name__ == "__main__":
    import sys
    app = QtGui.QApplication(sys.argv)
    Form = QtGui.QWidget()
    ui = Ui_Form()
    ui.setupUi(Form)
    Form.show()

    listener = clipboardListener(ui)

    QApplication.clipboard().dataChanged.connect(listener.changedSlot)

    sys.exit(app.exec_())