首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >如何从不同的函数pyqt5调用线程

如何从不同的函数pyqt5调用线程
EN

Stack Overflow用户
提问于 2021-03-08 18:59:48
回答 1查看 56关注 0票数 0
代码语言:javascript
运行
复制
import sys
from PyQt5 import QtCore
from PyQt5 import QtWidgets
from PyQt5.QtWidgets import QMainWindow, QApplication,  QPlainTextEdit, QApplication, QMainWindow, QLabel, QComboBox
from PyQt5.QtGui import QPixmap
from pynput import keyboard
from pynput.keyboard import Listener, Controller
import pyperclip as pc 

keyboard = Controller()
class App(QMainWindow):
    def __init__(self, parent=None):
        super(App, self).__init__(parent)
        #super().__init__()
        label = QLabel(self)
        pixmap = QPixmap('E:/copycat/new.png')
        label.setPixmap(pixmap)
        label.setGeometry(0,0,900,400) 
        self.title = 'COPYCAT'
        self.left = 10
        self.top = 10
        self.width = 400
        self.height = 140
        self.initUI()
        self.key()

    def initUI(self):
        self.setWindowTitle(self.title)
        self.setGeometry(self.left, self.top, self.width, self.height)
        ###########
        combo = QComboBox(self)
        shotcut_list = ["Key.f9","Key.f2","Key.f3","Key.f4","Key.f5","Key.f6","Key.f7","Key.f8","Key.f1","Key.f10","Key.f11","Key.f12"]
        combo.addItems(shotcut_list)
        global shortcut
        global cptext
        shortcut = combo.currentText()
        combo.setGeometry(350, 120, 120, 30)
        combo.activated[str].connect(self.onChanged)  
        # Create textbox
        self.textbox = QPlainTextEdit(self)
        self.textbox.move(20, 160)
        self.textbox.setReadOnly(True)
        self.textbox.resize(500,205)
        self.setGeometry(70,70,540,388)
        self.show()

    def onChanged(self, text):
        global shortcut
        shortcut=text
        
    def print_key(self,key):
        if str(key) == shortcut:
            cptext = pc.paste() 
            keyboard.type(cptext)
            self.textbox.insertPlainText(cptext)
            self.textbox.insertPlainText("\n")

    def key(self):    
        listener = Listener(on_press=self.print_key)
        listener.start()
    
if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    ex = App()
    #ex.key()
    sys.exit(app.exec_())

当我从print_key函数更新文本框时,上面的代码显示了一个错误,它显示了这个错误:

代码语言:javascript
运行
复制
Cannot queue arguments of type 'QTextBlock'
(Make sure 'QTextBlock' is registered using qRegisterMetaType()
Cannot queue arguments of type 'QTextCursor'
(Make sure 'QTextCursor' is registered using qRegisterMetaType()
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2021-03-08 19:25:14

与on_press相关的回调是在辅助线程中执行的,所以您的实现是从Qt禁止的辅助线程更新图形用户界面,而您应该使用这些信号,因为它们是线程安全的。

代码语言:javascript
运行
复制
import sys
import threading

from PyQt5.QtCore import pyqtSignal, pyqtSlot, QObject
from PyQt5.QtGui import QPixmap
from PyQt5.QtWidgets import (
    QApplication,
    QComboBox,
    QLabel,
    QMainWindow,
    QPlainTextEdit,
)

from pynput.keyboard import Listener, Controller
import pyperclip as pc


class KeyboardListener(QObject):
    textChanged = pyqtSignal(str)

    def __init__(self, shortcut, parent=None):
        super().__init__(parent)
        self._shortcut = shortcut

        listener = Listener(on_press=self.handle_pressed)
        listener.start()
        self.mutex = threading.Lock()

    @property
    def shortcut(self):
        return self._shortcut

    def handle_pressed(self, key):
        with self.mutex:
            if str(key) == self.shortcut:
                cptext = pc.paste()
                self.textChanged.emit(cptext)

    @pyqtSlot(str)
    def update_shortcut(self, shortcut):
        with self.mutex:
            self._shortcut = shortcut


class App(QMainWindow):
    def __init__(self, parent=None):
        super(App, self).__init__(parent)
        # super().__init__()
        label = QLabel(self)
        pixmap = QPixmap("E:/copycat/new.png")
        label.setPixmap(pixmap)
        label.setGeometry(0, 0, 900, 400)
        self.title = "COPYCAT"
        self.left = 10
        self.top = 10
        self.width = 400
        self.height = 140
        self.initUI()

    def initUI(self):
        self.setWindowTitle(self.title)
        self.setGeometry(self.left, self.top, self.width, self.height)
        ###########
        combo = QComboBox(self)
        shotcut_list = [
            "Key.f9",
            "Key.f2",
            "Key.f3",
            "Key.f4",
            "Key.f5",
            "Key.f6",
            "Key.f7",
            "Key.f8",
            "Key.f1",
            "Key.f10",
            "Key.f11",
            "Key.f12",
        ]
        combo.addItems(shotcut_list)

        shortcut = combo.currentText()
        combo.setGeometry(350, 120, 120, 30)
        self.textbox = QPlainTextEdit(self)
        self.textbox.move(20, 160)
        self.textbox.setReadOnly(True)
        self.textbox.resize(500, 205)
        self.setGeometry(70, 70, 540, 388)

        self.keyboard = Controller()

        self.listener = KeyboardListener(combo.currentText())
        combo.activated[str].connect(self.listener.update_shortcut)
        self.listener.textChanged.connect(self.handle_text_changed)

    @pyqtSlot(str)
    def handle_text_changed(self, text):
        self.textbox.insertPlainText(text)
        self.textbox.insertPlainText("\n")
        self.keyboard.type(text)


def main():
    app = QApplication(sys.argv)
    ex = App()
    ex.show()
    sys.exit(app.exec_())


if __name__ == "__main__":
    main()
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/66528559

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档