赞
踩
想搞点自动化,想了下市面上PC的自动化用RPA比较多,但RPA限制太多了。
就想着自己用Python写一个能操作微信,可以做到自动回复,自动读取聊天记录,等等的一些操作的自动化代码。
目前我利用这个Python实现的功能有,使用微信PC端,搜索群聊,发送内容,检测内容是否发送成功,以及读取群聊天记录的。
运行效果如下。
使用Python自动化操作电脑微信,实现自动回复,发送图片功能
这里主要使用到了uiautomation库,还有UISpy.exe这些工具包括源代码我都会打包给各位的。
具体代码最终我都会放出来给各位的,如果各位对这个代码有疑问,可以咨询我微3957165。
自动化需要发送的图片如下。
自动化操作微信页面代码。
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("全部发送完成")
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。