Python3.5+PyQt5多线程+itchat实现微信防撤回桌面版代码

Posted CrossPython

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python3.5+PyQt5多线程+itchat实现微信防撤回桌面版代码相关的知识,希望对你有一定的参考价值。

weChatThread线程类

之前一直不会python多线程,写这个程序的时候,发现不用多线程会陷入无限未响应状态。于是学了半天python多线程,但是在主函数里写的时候,发现一个问题,Ui主线程和工作线程没有分离,使用itchat等库的时候会堵塞主线程,换句话说PyQt中子线程不能操作GUI界面。之前写的多线程仍然属于Ui主线程,是其子线程,所以才造成未响应。
既然知道问题了,那就查资料解决问题,后来,在几篇博客上找到了解决办法

然后仿照第一篇博客,重写了QThread类,并借鉴第三篇博客,学会了PyQt多线程中的信号/槽机制,用来传递参数。

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
__author__ = \'memgq\'

from PyQt5.QtCore import QThread,pyqtSignal
import itchat
import time,os
import shutil
import re

from itchat.content import *

class weChatWord(QThread):
    getMsgSignal = pyqtSignal(str) #pyqtSignal()必须写在__init__前面,里面可接收的参数类型挺多的,str,list,dict都支持
    def __init__(self,parent=None):
        super(weChatWord,self).__init__(parent)
        self.msg_list=[]
        self.type_list=[\'Picture\',\'Recording\', \'Attachment\',\'Video\']



    def clearList(self):
        \'\'\'
        清空缓存消息和文件
        :return: 
        \'\'\'
        tm_now=time.time()
        len_list=len(self.msg_list)
        if len_list>0:
            for i in range(len_list):
                if tm_now-self.msg_list[i][\'msg_time\']>121:
                    if self.msg_list[i][\'msg_type\'] in self.type_list:
                        try:
                            os.remove(".\\\\BackUp\\\\"+self.msg_list[i][\'msg_content\'])
                        except Exception as e:
                            print(e)
                        finally:
                            pass
                else:break
            self.msg_list=self.msg_list[i:]



    def run(self):
        \'\'\'
        重写run()函数,
        :return:
        \'\'\'
        @itchat.msg_register([TEXT, PICTURE, MAP, CARD, SHARING, RECORDING, ATTACHMENT, VIDEO, FRIENDS], isFriendChat=True,
                      isGroupChat=True)
        def getMsg(msg):
            \'\'\'
            注册消息类型,并对不同类型的消息执行不用的操作
            :param msg: 
            :return: 
            \'\'\'
            msg_dict={}
            # pprint.pprint(msg)
            msg_id = msg[\'MsgId\']  # 消息ID
            msg_time = msg[\'CreateTime\']
            msg_url=None
            msg_group=""
            if (itchat.search_friends(userName=msg[\'FromUserName\'])):
                if itchat.search_friends(userName=msg[\'FromUserName\'])[\'RemarkName\']:
                    msg_from = itchat.search_friends(userName=msg[\'FromUserName\'])[\'RemarkName\']  # 消息发送人备注
                elif itchat.search_friends(userName=msg[\'FromUserName\'])[\'NickName\']:  # 消息发送人昵称
                    msg_from = itchat.search_friends(userName=msg[\'FromUserName\'])[\'NickName\']  # 消息发送人昵称
                else:
                    msg_from = r"读取发送消息好友失败"
            else:
                msg_group = msg[\'User\'][\'NickName\']
                msg_from = msg[\'ActualNickName\']
            msg_type = msg[\'Type\']
            if msg_type in [\'Text\', \'Friends\',\'Sharing\']:
                msg_content = msg[\'Text\']
                msg_url = msg[\'Url\']
            elif msg_type in self.type_list:
                msg_content=msg[\'FileName\']
                msg[\'Text\'](msg[\'FileName\'])
                shutil.move(msg_content,r\'.\\\\BackUp\\\\\')
            elif msg[\'Type\'] == \'Card\':
                msg_content = msg[\'RecommendInfo\'][\'NickName\'] + r" 的名片"
            elif msg[\'Type\'] == \'Map\':
                x, y, location = re.search("<location x=\\"(.*?)\\" y=\\"(.*?)\\".*label=\\"(.*?)\\".*",
                                           msg[\'OriContent\']).group(1,
                                                                    2,
                                                                    3)
                if location is None:
                    msg_content = r"纬度->" + x.__str__() + " 经度->" + y.__str__()
                else:
                    msg_content = r"" + location

            msg_dict={\'msg_id\':msg_id,\'msg_time\':msg_time,\'msg_from\':msg_from,\'msg_group\':msg_group,
                      \'msg_content\':msg_content,\'msg_type\':msg_type,\'msg_url\':msg_url}
            self.msg_list.append(msg_dict)
            self.clearList()


        @itchat.msg_register([NOTE],isFriendChat=True, isGroupChat=True)
        def recall(msg):
            \'\'\'
            当消息类型为通知类的时候,查找消息内容是否为撤回消息,如果是,则执行撤回后的防撤回操作
            :param msg: 
            :return: 
            \'\'\'
            # pprint.pprint(msg)
            msg_content=msg[\'Content\']
            if re.search(r\'\\<replacemsg\\>\\<!\\[CDATA\\[(.*)撤回了一条消息\\]\\]\\>\\<\\/replacemsg\\>\',msg_content):
                msg_note=re.search(r\'\\<replacemsg\\>\\<!\\[CDATA\\[(.*)\\]\\]\\>\\<\\/replacemsg\\>\',msg_content).group(1)
                old_msg_id=re.search(r\'\\<msgid\\>([0-9]+)\\</msgid\\>\',msg_content).group(1)
                for each in self.msg_list:
                    if each[\'msg_id\']==old_msg_id:
                        timeArray = time.localtime()
                        otherStyleTime = time.strftime("%Y-%m-%d %H:%M:%S,", timeArray)
                        msg_note = msg_note + \',撤回内容为:\' + each[\'msg_content\']
                        if each[\'msg_group\']!=\'\':
                            msg_note = "群组("+each[\'msg_group\']+")中"+msg_note
                        msg_note=otherStyleTime+msg_note
                        itchat.send(msg_note,toUserName=\'filehelper\')
                        self.msg_list.pop(self.msg_list.index(each))
                        self.getMsgSignal.emit(msg_note)
                        break

        #创建BuckUp文件夹
        if not os.path.exists(".\\\\BackUp\\\\"):
            os.mkdir(\'.\\\\BackUp\\\\\')
        #启动itchat()    
        itchat.auto_login(hotReload=True)
        itchat.run()

 

主程序类

 

class mainwindowapp(QMainWindow,wechatunrecall.Ui_MainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.createActions()
        self.createTrayIcon()
        self.pushButton.clicked.connect(self.saveLog)
        self.pushButton_2.clicked.connect(self.clearlog)
        self.pushButton_3.clicked.connect(self.houtai)
        self.trayIcon.activated.connect(self.iconActivated)
        timeArray = time.localtime()
        otherStyleTime = time.strftime("%Y-%m-%d %H:%M:%S", timeArray)
        self.setLog(otherStyleTime+",程序运行时,请用手机扫描弹出的二维码进行登录,并确保电脑上自带的Window照片查"
                                   "看器可用,撤回的图片文件等可下载附件连同运行日志保存在程序目录下BackUp文件夹中。\\n")
        self.weChatBigWord()


    def saveLog(self):
        \'\'\'
        保存日志
        :return: 
        \'\'\'
        if not os.path.exists(".\\\\BackUp\\\\"):
            os.mkdir(".\\\\BackUp\\\\")
        timeArray = time.localtime()
        otherStyleTime = time.strftime("%Y-%m-%d%H%M%S", timeArray)
        text=self.textBrowser.toPlainText()
        logPath=".\\\\BackUp\\\\"+otherStyleTime+\'.txt\'
        with open(logPath,\'w\') as f:
            f.write(text)

    def setLog(self,msg):
        \'\'\'
        往运行日志窗口写撤回消息的内容
        :param msg: 
        :return: 
        \'\'\'
        self.textBrowser.append(msg)

    def createTrayIcon(self):
        \'\'\'
        创建托盘图标,可以让程序最小化到windows托盘中运行
        :return: 
        \'\'\'
        self.trayIconMenu=QMenu(self)
        self.trayIconMenu.addAction(self.restoreAction)
        self.trayIconMenu.addSeparator()
        self.trayIconMenu.addAction(self.quitAction)
        self.trayIcon=QSystemTrayIcon(self)
        self.trayIcon.setContextMenu(self.trayIconMenu)
        self.trayIcon.setIcon(QIcon(\'./media/images/maincion.png\'))
        self.setWindowIcon(QIcon(\'./media/images/maincion.png\'))
        self.trayIcon.show()

    def createActions(self):
        \'\'\'
        为托盘图标添加功能
        :return: 
        \'\'\'
        self.restoreAction=QAction("恢复",self,triggered=self.showNormal)
        self.quitAction=QAction("退出",self,triggered=QApplication.instance().quit)


    def iconActivated(self,reason):
        \'\'\'
        激活托盘功能
        :param reason: 
        :return: 
        \'\'\'
        if reason in (QSystemTrayIcon.Trigger, QSystemTrayIcon.DoubleClick):
            self.showNormal()


    def houtai(self):
        self.hide()

    def clearlog(self):
        self.textBrowser.clear()


    def weChatBigWord(self):
        \'\'\'
        weChatThread类实例化,并启动线程
        :return: 
        \'\'\'
        from weChatThread import weChatWord
        self.wcBWThread=weChatWord()
        self.wcBWThread.getMsgSignal.connect(self.setLog)
        self.wcBWThread.start()

 

程序界面

程序界面仍然由Qtdesigner设计

后记

第一次尝试多线程编程,并且具体应用到实际项目中去,收获良多。

以上是关于Python3.5+PyQt5多线程+itchat实现微信防撤回桌面版代码的主要内容,如果未能解决你的问题,请参考以下文章

如何安装python3-pyqt5

PyQt5自学记录——PyQt5多线程实现详解

Python3.5+PYQT5:ModuleNotFoundError: No module named 'PyQt5'

PyQT5 多线程问题

改变QLabel的大小时,PyQt5“定时器无法从另一个线程启动”错误

PyQt5创建多线程