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设计

后记

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

原文地址:https://www.cnblogs.com/pythonClub/p/10251939.html

时间: 2024-08-30 04:17:50

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

微信开发(1)---微信防撤回功能实现

手里负责的项目都是微信端开发,微信小程序,以及公众号相关,所以最近在看微信开发的书. 今天碰巧看到一篇知乎讲如何实现微信防撤回功能,所以自己去实现了下,颇为好玩. 微信防撤回功能轻松实现~ 先贴上知乎原帖: https://zhuanlan.zhihu.com/p/25689314?utm_source=zhihu&utm_medium=social 实现思路,将接收到的消息都存进一个字典(msg_dict),接收到撤回通知时,提取撤回消息的id,从字典中进行比对,然后发送.保存.那么字典能放得

用Python教你微信防撤回(文本、图片、语音、视频、名片等...)

大家在使用微信过程中,有时候消息还没看到,就被撤回了.毕竟好奇心大家都有,明知到消息被撤回了,就更想去看一下是什么内容心里想着万一是女神给我表白了呢.. 今天就用Python来做个微信防撤回的小功能.支持文本.图片.语音.视频.名片的防撤回. 思路:利用itchat库将每次接收到的消息进行缓存.监听撤回事件,当有消失被撤回时,通过msgId去缓存中取到信息再发送给文件传输助手.这样就实现了一个简单的防撤回功能. 效果图 在这里插入图片描述 将接收的消息缓存 import itchatfrom i

Exp10 Final 微信防撤回原理与实现

一.写在前面 1.为什么做免考? 相较于考试,免考更能锻炼自身创新和探索能力,更有挑战性. 2.选做微信推送防撤回原因? 微信已经成为日常沟通中必不可少的工具,如何知道别人在你没看微信的情况下偷偷说了什么就变得很重要(嘿嘿嘿) CTF我只研究过杂项.密码学和逆向,别的部分我也不擅长所以就不做了. 二.写在中间 1.什么是防撤回? 指对方发出撤回请求,系统应答后,对方微信显示成功撤回,而我方微信显示对方撤回提示但消息依然显示在界面. 2.实验环境 实验对象:wechatwin.dll (PC 2.

微信防撤回

1 # -*-encoding:utf-8-*- 2 import os 3 import re 4 import shutil 5 import time 6 import itchat 7 from itchat.content import * 8 9 # 说明:可以撤回的有文本文字.语音.视频.图片.位置.名片.分享.附件 10 11 # {msg_id:(msg_from,msg_to,msg_time,msg_time_rec,msg_type,msg_content,msg_sha

微信防撤回机器人

对方发送过来的消息,我们通过dict进行保存,记录消息id和对应的消息内容,当对方撤回消息的时候,我们根据检测到的消息id,找到对应的dict中的消息内容,实现撤回的消息精准复原. 撤回的消息发送到文件传输助手 1 # coding:utf-8 2 import itchat 3 from itchat.content import TEXT 4 from itchat.content import * 5 import sys 6 import time 7 import re 8 9 rel

利用Python制作微信小助手防撤回+自动回复二合一,非常强大!

导语大家新年好呀~今天在家闲来无事,想起来之前好多人吐槽那个微信防撤回的程序不能防止群消息撤回,于是优化了下,顺手把之前微信相关的一些小作品的代码整合了一下,写了个微信小助手,在这里分享给大家.开发工具Python版本:3.6.4相关模块:wxpy模块:itchat模块:argparse模块:pyecharts模块:以及一些Python自带的模块.环境搭建安装Python并添加到环境变量,pip安装需要的相关模块即可.简介微信小助手目前支持以下四项功能. 微信消息防撤回原理:就是把接收到的消息都

基于itchat的微信群聊小助手基础开发(一)

前段时间由于要管理微信群,基于itchat开发了一个简单的微信机器人 主要功能有: 图灵机器人功能 群聊昵称格式修改提示 消息防撤回功能 斗图功能 要开发一个基于itchat的最基本的聊天机器人,在github上作者已经给出基本的操作,本博文只对附加的功能进行说明总结. 确保已经了解itchat的基本操作,先对功能1和功能2进行说明: 1.图灵机器人功能 该功能也是最基础的功能之一,itchat的基础案例也有相关例子. 图灵机器人简单而言就是以一定的规则给图灵的服务器发送数据包,图灵的服务器会以

老司机带你攻破微信消息防撤回功能

前言: 微信聊天的时候,一疏忽竟然错过了什么,对方撤回了一条消息,一位小妹妹给你发了一张不可描述的照片,下一秒她又后悔选择了撤回.那么到哪去寻找撤回的消息呢?难道我们就不能做点什么? 就在昨天,我的一位好友壳子@ZKeeer 使用itchat库做了小脚本写了一篇文章:初学python--微信防撤回功能改进(一个用处不大的功能) - 知乎专栏,在得到壳子的授权之后,我对他的代码进行了修改,并把代码包装成了可执行的exe文件用来造福更多有需要的朋友们. 这个工具暂时我们教叫它:wMsg wMsg能实

itchat库微信自动回复祝福语

过年了,之前看到一些python文章介绍用itchat自动回复微信,我自己就写了一个. 官方文档https://itchat.readthedocs.io/zh/latest/,这个库挺简洁的,对着接口写就行了. 我的环境是win10+python3.7, 先pip install itchat 安装,用到一些别的库也自动装好了.发送的脚本如下 # -*- coding: utf-8 -*- import time,random import itchat as wx @wx.msg_regis