想搞点自动化,想了下市面上PC的自动化用RPA比较多,但RPA限制太多了。
就想着自己用Python写一个能操作微信,可以做到自动回复,自动读取聊天记录,等等的一些操作的自动化代码。
目前我利用这个Python实现的功能有,使用微信PC端,搜索群聊,发送内容,检测内容是否发送成功,以及读取群聊天记录的。
运行效果如下。


使用Python自动化操作电脑微信,实现自动回复,发送图片功能


实现原理

这里主要使用到了uiautomation库,还有UISpy.exe这些工具包括源代码我都会打包给各位的。

具体代码最终我都会放出来给各位的,如果各位对这个代码有疑问,可以咨询我微3957165

自动化需要发送的图片如下。

python 抓取群聊天记录 python读取微信群聊天记录_控件

自动化操作微信页面代码。

import time
import uiautomation as auto
import pyperclip
import os
import subprocess
#引入打印错误日志信息包
import traceback


def weChat(groupName,index=0):
    if index == None or index == 0:
        index = 0
    # 连接到微信应用的主窗口
    wechat_window = auto.WindowControl(searchDepth=1, Name="微信", ClassName="WeChatMainWndForPC")
    # 检查微信窗口是否存在
    if not wechat_window.Exists(5,1):
        print("微信窗口没有找到")
        return   
    def getS():
        wechat_window.SetActive() # 激活窗口            
        # 寻找同级下面的
        # 定位到 ToolBarControl
        tab_control = wechat_window.ToolBarControl(searchDepth=3)
        print(tab_control)
        # 获取toolbar的父控件
        parent_control = tab_control.GetParentControl()

        # 检查是否找到父级控件
        if not parent_control.Exists():
            print("没有找到父级控件。")
            return
        # 获取父控件的所有子控件
        sibling_controls = parent_control.GetChildren()    
        target_control = sibling_controls[1]    
        children = target_control.GetChildren()    
        first_child_control = children[0]
        
        
        text_control = first_child_control.TextControl(searchDepth=3,ClassName="", AutomationId="", Name="")  # 按条件修改
        
        # 设置焦点
        text_control.SetFocus()
        
        time.sleep(0.5)  # 稍微等待一下,让操作生效
        text_control.Click()  # 点击以获取焦点
        
        
        # 最后定位到搜索编辑框
        search_box = first_child_control.EditControl(searchDepth=3, Name="搜索")
        print(search_box)
        # 检查搜索框是否存在
        if not search_box.Exists(3):
            print("未找到搜索框")
            return
        print("找到搜索框")
        # 设置焦点
        search_box.SetFocus()
            # 全选搜索框中的内容
        search_box.SendKeys('{Ctrl}a')
        time.sleep(0.5)  # 稍微等待一下,让操作生效
        # 删除选中的内容
        search_box.SendKeys('{Delete}')
        time.sleep(0.5)  # 稍微等待一下,让操作生效


        # 输入想要搜索的内容
        search_box.SendKeys(groupName)     
        print("搜索框的信息数据值是:")
        print(search_box)
        # 这里的第二个是父级的第二个子元素
        print("父级的第二个子元素的信息数据值是:")
        print(children[1])
        return children[1]
    listPart = getS()
    print(listPart)
    listC = listPart.ListControl(searchDepth=3)  # 按条件修改
    print("listC的信息数据值是:",listC)
    # 获取第一个子元素,可能需要根据实际的控件类型调整GetFirstChildControl的参数
    first_item = listC.GetFirstChildControl()

    # 确保第一个元素存在
    if not first_item.Exists(2, 1):
        print("未找到第一个列表项")
        return
    print("找到第一个列表项")
    
    # 例如,获取第一个元素的文本  默认是第一个完全搜索结果的群昵称
    first_item = listC.ListItemControl(foundIndex=1)

    # 对第一个元素进行操作,例如点击
    if not first_item.Exists():
        print("First item not found or not clickable")
        return
    time.sleep(0.5)  # 稍微等待一下,让操作生效
    first_item.Click()
    time.sleep(0.5)  # 稍微等待一下,让操作生效
    return  wechat_window
    
def sendImg(wechat_window,groupName,fileUrl,index=0,iterIndex=0):
    time.sleep(0.5)  # 稍微等待一下,让操作生效
    tab_control = wechat_window.ToolBarControl(searchDepth=3)
    
    if not tab_control.Exists(3, 1):
        print(f"未找到名为 '{groupName}' 的编辑控件")
        return
    # 可以正确找到群聊消息控件的数据信息
    mesgContList = wechat_window.ListControl(searchDepth=13, Name="消息")
    # print(mesgContList)
    # 打印控件信息
    # dump_tree(mesgContList)
    mgsDataList = []
    # 要搜索谁的消息 默认搜索这个群里面自己发的聊天记录,这里自己自定义。
    target_name = None
    first_child = tab_control.GetFirstChildControl()
    if first_child:
        target_name = first_child.Name
    
    max_depth = 5  # 您可以设置您希望搜索的最大深度,并且返回所有子集

    # 调用函数查找所有包含特定Name的ListItemControl控件
    matching_list_items = find_list_items_with_name(mesgContList, target_name, max_depth)

    mlistIndex = matching_list_items.__len__()
    print("上一条信息的值是",matching_list_items[mlistIndex-1].Name)
    print("当前信息的值是",index)
    if index != 0 :        
        name_str = matching_list_items[mlistIndex-1].Name
        result = None
        # 尝试将字符串转换为整数
        try:
            name_int = int(name_str)
            result = name_int + 1  # 现在可以执行减法操作
            print("改变后的值是",result)
        except ValueError:
            print(f"Cannot convert '{name_str}' to an integer.")
        if str(index).strip() == str(result).strip():
            print("上一条信息已经发送完成")
        else:
            print("上一条信息未发送完成,重新执行发送消息操作")
            if iterIndex > 3:
                print("上一条信息未发送完成,重新执行发送消息操作超过3次,退出")
                return
            return sendImg(wechat_window,groupName,fileUrl,index,iterIndex+1)

    # 打印或处理匹配的列表项
    # for list_item in matching_list_items:
    #     print(list_item.Name)  
    
    time.sleep(1)
    edit_control  = wechat_window.EditControl(searchDepth=20, Name=groupName)  # 按条件修改
    time.sleep(1)
    edit_control.SetFocus() 
    time.sleep(1)  
    edit_control.Click()
    time.sleep(1)
    # 清除当前值
    edit_control .SendKeys('{Ctrl}a{Delete}')
    # 将文件路径复制到剪贴板
    time.sleep(0.3)
    # 获取焦点
    edit_control .SetFocus()
    time.sleep(0.3)

    tmp_clipboard_image_path = os.path.join(os.getenv('TEMP'), 'clipboard_image.png')
    os.system(f'copy "{fileUrl}" "{tmp_clipboard_image_path}"')
    time.sleep(1)
    # 使用命令行工具将图片放入剪贴板
    subprocess.run(['nircmd.exe', 'clipboard', 'copyimage', tmp_clipboard_image_path])
    time.sleep(1)
    edit_control .SendKeys('{Ctrl}v')
    time.sleep(1)
    # 发送图片
    edit_control .SendKeys('{Enter}')
    time.sleep(30)
    edit_control .SendKeys('{Ctrl}a{Delete}')
    time.sleep(0.3)
    edit_control .SetFocus()
    time.sleep(1)
    edit_control .SendKeys(str(index))
    time.sleep(1)
    # 发送文字
    edit_control .SendKeys('{Enter}')
    #防止出现网络延迟
    time.sleep(1)

    #校验一下文本还有图片是否发送完成 重新获取一下消息列表
    matching_list_items = find_list_items_with_name(mesgContList, target_name, max_depth)
    mlistIndex = matching_list_items.__len__()
    print("上一条信息的值是",matching_list_items[mlistIndex-1].Name)
    print("当前信息的值是",index)
    if "[图片]" == matching_list_items[mlistIndex-1].Name:
        time.sleep(1)
        edit_control.SetFocus() 
        time.sleep(1)  
        edit_control.Click()
        time.sleep(1)
        print("上一条信息是图片,不是文本,重新执行发送文本的操作。")
        edit_control .SendKeys('{Ctrl}a{Delete}')
        time.sleep(0.3)
        edit_control .SetFocus()
        time.sleep(1)
        edit_control .SendKeys(str(index))
        time.sleep(1)
        # 发送文字
        edit_control .SendKeys('{Enter}')
        #防止出现网络延迟
        time.sleep(1)


    time.sleep(3)
    # 发送完成图片后,删除图片
    try:

        if os.path.exists(tmp_clipboard_image_path):
            os.remove(tmp_clipboard_image_path)
    except OSError as e:
        print(f"错误: {fileUrl} : {e.strerror}")
        traceback.print_exc()
        e.with_traceback()
    except Exception as e:
        print(f"错误: {fileUrl} : {e}")
        traceback.print_exc()
        e.with_traceback()
        
def find_matching_control(control, target_name, max_depth, current_depth=0):
    """递归搜索控件树,查找包含特定文本的控件"""
    if current_depth > max_depth:  # 到达最大搜索深度
        return None
    if control.Name == target_name:
        return control
    for child in control.GetChildren():
        result = find_matching_control(child, target_name, max_depth, current_depth + 1)
        if result:
            return result
    return None

def find_list_items_with_name(list_control, target_name, max_depth):
    """查找包含特定文本的所有ListItemControl"""
    matching_list_items = []
    for item in list_control.GetChildren():
        if isinstance(item, auto.ListItemControl):
            if find_matching_control(item, target_name, max_depth):
                matching_list_items.append(item)
    return matching_list_items

def sendText(wechat_window,groupName,text):
    time.sleep(1)  # 稍微等待一下,让操作生效
    tab_control = wechat_window.ToolBarControl(searchDepth=3)
    
    if not tab_control.Exists(3, 1):
        print(f"未找到名为 '{groupName}' 的编辑控件")
        return
    
    edit_control  = wechat_window.EditControl(searchDepth=20, Name=groupName)  # 按条件修改
    
    edit_control.SetFocus()   
    # 清除当前值
    edit_control .SendKeys('{Ctrl}a{Delete}')
    # 将文件路径复制到剪贴板
    
    # 获取焦点
    # edit_control .SetFocus()
    
    edit_control .SendKeys(text)
    # 发送
    edit_control .SendKeys('{Enter}')

    #防止出现网络延迟
    time.sleep(3)
    

def dump_tree(control, indent=0):
    if not control.Exists():  # 如果控件不存在则直接返回
        return
    # 打印当前控件的基本信息
    print(' ' * indent + f"ControlType: {control.ControlTypeName}, Name: {control.Name}, #{control.AutomationId}, ClassName: {control.ClassName}")
    # 对当前控件的子控件进行同样的操作
    for child in control.GetChildren():
        dump_tree(child, indent + 2)  # 增加缩进以反映层级关系

def lockW():
    # 获取微信窗口
    wechat_window = auto.WindowControl(Name="微信", ClassName="WeChatMainWndForPC")
        # 尝试逐步减小 searchDepth 的值
    for depth in range(1, 12):
        search_box = wechat_window.EditControl(searchDepth=depth, Name="搜索")
        if search_box.Exists(2):    
            print(search_box)
            print(f"成功找到消息列表控件,searchDepth={depth}")
            break
    else:
        print("在给定的层级范围内未找到消息列表控件")


#这里的图片操作我本人用的是循环读取我本地图片的。

def findFileImages():
    print("开始发送图片")
     # 获取当前脚本的完整路径(包括文件名)
    current_file_path = os.path.realpath(__file__)
    # 获取当前脚本所在的目录路径
    current_file_path = os.path.dirname(current_file_path)  
    print("当前脚本所在的目录路径:", current_file_path)
    directory_path  = current_file_path + '\\articles\\'

    now = datetime.now()
    # 将日期和时间格式化为字符串
    date_time_str = now.strftime("%Y-%m-%d %H:%M:%S")
    list = os.listdir(directory_path)
    if len(list) == 0:
        print("没有图片,等待下一次执行--",date_time_str)
        return
    print("图片数量:",len(list))
    # 读取微信群名称
    groupList = find_wx_group()
    # 微信群昵称
    for group in groupList:
        groupName = group['name']
        wechat_window = weChat(groupName)
        for index, item in enumerate(list):
            # 发送图片到微信
            id_info = os.path.splitext(item)[0]  # 移除扩展名,只保留文件名(即ID)
            print('提取的ID:', id_info)
            # curl =  directory_path +str(item['topicId'])+".png"
            # print(item)
            # curl =  directory_path + group
            curl =  directory_path +str(id_info)+".png"
            # print(curl)
            sendImg(wechat_window,groupName,curl,index)
            # 获取当前的日期和时间
        now = datetime.now()
        # 将日期和时间格式化为字符串
        date_time_str = now.strftime("%Y-%m-%d %H:%M:%S")
        text = f"{groupName}群,一共{list.__len__()}个内容,内容发送时间:{date_time_str}"
        print("图片内容发送完成--",groupName)
        sendText(wechat_window,groupName,text)
    #图片都发送完成 删除图片
    for index, item in enumerate(list):    
        fileUrl =  directory_path +str(id_info)+".png"
        if os.path.exists(fileUrl):
            os.remove(fileUrl)
    
    print("全部发送完成")