微信跳一跳自动运行python脚本


Posted in Python onJanuary 08, 2018

本文实例为大家分享了微信小程序跳一跳自动运行脚本,供大家参考,具体内容如下

1、压缩包带了adb等必须工具,配置一下环境变量即可

2、Python 直接运行即可 (Python3.6)

代码:

wechat_jump_auto.py

# coding: utf-8
'''
# === 思路 ===
# 核心:每次落稳之后截图,根据截图算出棋子的坐标和下一个块顶面的中点坐标,
# 根据两个点的距离乘以一个时间系数获得长按的时间
# 识别棋子:靠棋子的颜色来识别位置,通过截图发现最下面一行大概是一条直线,就从上往下一行一行遍历,
# 比较颜色(颜色用了一个区间来比较)找到最下面的那一行的所有点,然后求个中点,
# 求好之后再让 Y 轴坐标减小棋子底盘的一半高度从而得到中心点的坐标
# 识别棋盘:靠底色和方块的色差来做,从分数之下的位置开始,一行一行扫描,由于圆形的块最顶上是一条线,
# 方形的上面大概是一个点,所以就用类似识别棋子的做法多识别了几个点求中点,
# 这时候得到了块中点的 X 轴坐标,这时候假设现在棋子在当前块的中心,
# 根据一个通过截图获取的固定的角度来推出中点的 Y 坐标
# 最后:根据两点的坐标算距离乘以系数来获取长按时间(似乎可以直接用 X 轴距离)
'''
import os
import sys
import subprocess
import time
import math
from PIL import Image
import random
from six.moves import input
import debug, config
import numpy as np



VERSION = "1.1.1"


debug_switch = False # debug 开关,需要调试的时候请改为:True
config = config.open_accordant_config()

# Magic Number,不设置可能无法正常执行,请根据具体截图从上到下按需设置,设置保存在 config 文件夹中
under_game_score_y = config['under_game_score_y']
press_coefficient = config['press_coefficient'] # 长按的时间系数,请自己根据实际情况调节
piece_base_height_1_2 = config['piece_base_height_1_2'] # 二分之一的棋子底座高度,可能要调节
piece_body_width = config['piece_body_width'] # 棋子的宽度,比截图中量到的稍微大一点比较安全,可能要调节


screenshot_way = 2


def pull_screenshot():
 '''
 新的方法请根据效率及适用性由高到低排序
 '''
 global screenshot_way
 if screenshot_way == 2 or screenshot_way == 1:
 process = subprocess.Popen('adb shell screencap -p', shell=True, stdout=subprocess.PIPE)
 screenshot = process.stdout.read()
 if screenshot_way == 2:
 binary_screenshot = screenshot.replace(b'\r\n', b'\n')
 else:
 binary_screenshot = screenshot.replace(b'\r\r\n', b'\n')
 f = open('autojump.png', 'wb')
 f.write(binary_screenshot)
 f.close()
 elif screenshot_way == 0:
 os.system('adb shell screencap -p /sdcard/autojump.png')
 os.system('adb pull /sdcard/autojump.png .')


def set_button_position(im):
 '''
 将 swipe 设置为 `再来一局` 按钮的位置
 '''
 global swipe_x1, swipe_y1, swipe_x2, swipe_y2
 w, h = im.size
 left = int(w / 2)
 top = int(1584 * (h / 1920.0))
 left = int(random.uniform(left-50, left+50))
 top = int(random.uniform(top-10, top+10)) # 随机防 ban
 swipe_x1, swipe_y1, swipe_x2, swipe_y2 = left, top, left, top


def jump(distance):
 '''
 跳跃一定的距离
 '''
 press_time = distance * press_coefficient
 press_time = max(press_time, 200) # 设置 200ms 是最小的按压时间
 press_time = int(press_time)
 cmd = 'adb shell input swipe {x1} {y1} {x2} {y2} {duration}'.format(
 x1=swipe_x1,
 y1=swipe_y1,
 x2=swipe_x2,
 y2=swipe_y2,
 duration=press_time
 )
 print(cmd)
 os.system(cmd)
 return press_time


def find_piece_and_board(im):
 '''
 寻找关键坐标
 '''
 w, h = im.size

 piece_x_sum = 0
 piece_x_c = 0
 piece_y_max = 0
 board_x = 0
 board_y = 0
 scan_x_border = int(w / 8) # 扫描棋子时的左右边界
 scan_start_y = 0 # 扫描的起始 y 坐标
 im_pixel = im.load()
 # 以 50px 步长,尝试探测 scan_start_y
 for i in range(int(h / 3), int(h*2 / 3), 50):
 last_pixel = im_pixel[0, i]
 for j in range(1, w):
 pixel = im_pixel[j, i]
 # 不是纯色的线,则记录 scan_start_y 的值,准备跳出循环
 if pixel[0] != last_pixel[0] or pixel[1] != last_pixel[1] or pixel[2] != last_pixel[2]:
 scan_start_y = i - 50
 break
 if scan_start_y:
 break
 print('scan_start_y: {}'.format(scan_start_y))

 # 从 scan_start_y 开始往下扫描,棋子应位于屏幕上半部分,这里暂定不超过 2/3
 for i in range(scan_start_y, int(h * 2 / 3)):
 for j in range(scan_x_border, w - scan_x_border): # 横坐标方面也减少了一部分扫描开销
 pixel = im_pixel[j, i]
 # 根据棋子的最低行的颜色判断,找最后一行那些点的平均值,这个颜色这样应该 OK,暂时不提出来
 if (50 < pixel[0] < 60) and (53 < pixel[1] < 63) and (95 < pixel[2] < 110):
 piece_x_sum += j
 piece_x_c += 1
 piece_y_max = max(i, piece_y_max)

 if not all((piece_x_sum, piece_x_c)):
 return 0, 0, 0, 0
 piece_x = int(piece_x_sum / piece_x_c)
 piece_y = piece_y_max - piece_base_height_1_2 # 上移棋子底盘高度的一半

 # 限制棋盘扫描的横坐标,避免音符 bug
 if piece_x < w/2:
 board_x_start = piece_x
 board_x_end = w
 else:
 board_x_start = 0
 board_x_end = piece_x

 for i in range(int(h / 3), int(h * 2 / 3)):
 last_pixel = im_pixel[0, i]
 if board_x or board_y:
 break
 board_x_sum = 0
 board_x_c = 0

 for j in range(int(board_x_start), int(board_x_end)):
 pixel = im_pixel[j, i]
 # 修掉脑袋比下一个小格子还高的情况的 bug
 if abs(j - piece_x) < piece_body_width:
 continue

 # 修掉圆顶的时候一条线导致的小 bug,这个颜色判断应该 OK,暂时不提出来
 if abs(pixel[0] - last_pixel[0]) + abs(pixel[1] - last_pixel[1]) + abs(pixel[2] - last_pixel[2]) > 10:
 board_x_sum += j
 board_x_c += 1
 if board_x_sum:
 board_x = board_x_sum / board_x_c
 last_pixel = im_pixel[board_x, i]

 # 从上顶点往下 +274 的位置开始向上找颜色与上顶点一样的点,为下顶点
 # 该方法对所有纯色平面和部分非纯色平面有效,对高尔夫草坪面、木纹桌面、药瓶和非菱形的碟机(好像是)会判断错误
 for k in range(i+274, i, -1): # 274 取开局时最大的方块的上下顶点距离
 pixel = im_pixel[board_x, k]
 if abs(pixel[0] - last_pixel[0]) + abs(pixel[1] - last_pixel[1]) + abs(pixel[2] - last_pixel[2]) < 10:
 break
 board_y = int((i+k) / 2)

 # 如果上一跳命中中间,则下个目标中心会出现 r245 g245 b245 的点,利用这个属性弥补上一段代码可能存在的判断错误
 # 若上一跳由于某种原因没有跳到正中间,而下一跳恰好有无法正确识别花纹,则有可能游戏失败,由于花纹面积通常比较大,失败概率较低
 for l in range(i, i+200):
 pixel = im_pixel[board_x, l]
 if abs(pixel[0] - 245) + abs(pixel[1] - 245) + abs(pixel[2] - 245) == 0:
 board_y = l+10
 break

 if not all((board_x, board_y)):
 return 0, 0, 0, 0

 return piece_x, piece_y, board_x, board_y


def check_screenshot():
 '''
 检查获取截图的方式
 '''
 global screenshot_way
 if os.path.isfile('autojump.png'):
 os.remove('autojump.png')
 if (screenshot_way < 0):
 print('暂不支持当前设备')
 sys.exit()
 pull_screenshot()
 try:
 Image.open('./autojump.png').load()
 print('采用方式 {} 获取截图'.format(screenshot_way))
 except Exception:
 screenshot_way -= 1
 check_screenshot()


def yes_or_no(prompt, true_value='y', false_value='n', default=True):
 default_value = true_value if default else false_value
 prompt = '%s %s/%s [%s]: ' % (prompt, true_value, false_value, default_value)
 i = input(prompt)
 if not i:
 return default
 while True:
 if i == true_value:
 return True
 elif i == false_value:
 return False
 prompt = 'Please input %s or %s: ' % (true_value, false_value)
 i = input(prompt)


def main():
 '''
 主函数
 '''
 op = yes_or_no('请确保手机打开了 ADB 并连接了电脑,然后打开跳一跳并【开始游戏】后再用本程序,确定开始?')
 if not op:
 print('bye')
 return
 print('程序版本号:{}'.format(VERSION))
 debug.dump_device_info()
 check_screenshot()

 i, next_rest, next_rest_time = 0, random.randrange(3, 10), random.randrange(5, 10)
 while True:
 pull_screenshot()
 im = Image.open('./autojump.png')
 # 获取棋子和 board 的位置
 piece_x, piece_y, board_x, board_y = find_piece_and_board(im)
 ts = int(time.time())
 print(ts, piece_x, piece_y, board_x, board_y)
 set_button_position(im)
 jump(math.sqrt((board_x - piece_x) ** 2 + (board_y - piece_y) ** 2))
 if debug_switch:
 debug.save_debug_screenshot(ts, im, piece_x, piece_y, board_x, board_y)
 debug.backup_screenshot(ts)
 i += 1
 if i == next_rest:
 print('已经连续打了 {} 下,休息 {}s'.format(i, next_rest_time))
 for j in range(next_rest_time):
 sys.stdout.write('\r程序将在 {}s 后继续'.format(next_rest_time - j))
 sys.stdout.flush()
 time.sleep(2)
 print('\n继续')
 i, next_rest, next_rest_time = 0, random.randrange(30, 100), random.randrange(10, 60)
 time.sleep(np.random.uniform(0.6,0.9)) # 为了保证截图的时候应落稳了,多延迟一会儿,随机值防 ban


if __name__ == '__main__':
 main()

simple.py

# -*- coding: utf-8 -*-

import numpy as np
import cv2
import os
import time
import re

# 屏幕截图
def pull_screenshot(path):
 os.system('adb shell screencap -p /sdcard/%s' % path)
 os.system('adb pull /sdcard/%s .' % path)

# 根据x距离跳跃
def jump(distance, alpha):
 press_time = max(int(distance * alpha), 200)

 cmd = 'adb shell input swipe {} {} {} {} {}'.format(bx1, by1, bx2, by2, press_time)
 os.system(cmd)

screenshot = 'screenshot.png'
alpha = 0
bx1, by1, bx2, by2 = 0, 0, 0, 0
chess_x = 0
target_x = 0

fix = 1.6667

# 检查分辨率是否是960x540
size_str = os.popen('adb shell wm size').read()
if size_str:
 m = re.search(r'(\d+)x(\d+)', size_str)
 if m:
 hxw = "{height}x{width}".format(height=m.group(2), width=m.group(1))
 if hxw == "960x540":
 fix = 3.16

while True:
 pull_screenshot(screenshot)
 image_np = cv2.imread(screenshot)
 image_np = cv2.cvtColor(image_np, cv2.COLOR_BGR2RGB)
 gray = cv2.Canny(image_np, 20, 80)

 HEIGHT = image_np.shape[0]
 WIDTH = image_np.shape[1]

 bx1 = WIDTH / 2
 bx2 = WIDTH / 2
 by1 = HEIGHT * 0.785
 by2 = HEIGHT * 0.785
 alpha = WIDTH * fix

 # 获取棋子x坐标
 linemax = []
 for i in range(int(HEIGHT * 0.4), int(HEIGHT * 0.6)):
 line = []
 for j in range(int(WIDTH * 0.15), int(WIDTH * 0.85)):
 if image_np[i, j, 0] > 40 and image_np[i, j, 0] < 70 and image_np[i, j, 1] > 40 and image_np[i, j, 1] < 70 and image_np[i, j, 2] > 60 and image_np[i, j, 2] < 110:
 gray[i, j] = 255
 if len(line) > 0 and j - line[-1] > 1:
 break
 else:
 line.append(j)

 if len(line) > 5 and len(line) > len(linemax):
 linemax = line
 if len(linemax) > 20 and len(line) == 0:
 break

 chess_x = int(np.mean(linemax))

 # 获取目标x坐标
 for i in range(int(HEIGHT * 0.3), int(HEIGHT * 0.5)):
 flag = False
 for j in range(WIDTH):
 # 超过朋友时棋子上方的图案
 if np.abs(j - chess_x) < len(linemax):
 continue
 if not gray[i, j] == 0:
 target_x = j
 flag = True
 break
 if flag:
 break

 # 修改检测图
 gray[:, chess_x] = 255
 gray[:, target_x] = 255
 # 保存检测图
 cv2.imwrite('detection.png', gray)

 print(chess_x, target_x)
 jump(float(np.abs(chess_x - target_x)) / WIDTH, alpha)

 # 等棋子落稳
 time.sleep(np.random.random() + 1)

下载:微信小程序跳一跳自动运行脚本

更多内容大家可以参考专题《微信跳一跳》进行学习。

更多关于python安装教程的文章请参考《python各版本安装教程》

更多精彩书单,请点击python编程必备书单

领取干货:零基础入门学习python视频教程

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python制作CSDN免积分下载器
Mar 10 Python
Python比较两个图片相似度的方法
Mar 13 Python
教你用Type Hint提高Python程序开发效率
Aug 08 Python
Python 备份程序代码实现
Mar 06 Python
程序员的七夕用30行代码让Python化身表白神器
Aug 07 Python
Python脚本操作Excel实现批量替换功能
Nov 20 Python
Python实现AI换脸功能
Apr 10 Python
Python数组拼接np.concatenate实现过程
Apr 18 Python
基于python实现matlab filter函数过程详解
Jun 08 Python
Python3安装模块报错Microsoft Visual C++ 14.0 is required的解决方法
Jul 28 Python
Python实现对word文档添加密码去除密码的示例代码
Dec 29 Python
一篇文章弄懂Python中的内建函数
Aug 07 Python
python3实现跳一跳点击跳跃
Jan 08 #Python
分数霸榜! python助你微信跳一跳拿高分
Jan 08 #Python
Python爬虫通过替换http request header来欺骗浏览器实现登录功能
Jan 07 #Python
ubuntu环境下python虚拟环境的安装过程
Jan 07 #Python
详解Python核心编程中的浅拷贝与深拷贝
Jan 07 #Python
用python实现的线程池实例代码
Jan 06 #Python
pip matplotlib报错equired packages can not be built解决
Jan 06 #Python
You might like
《星际争霸》各版本雷兽特点图文解析 雷兽不同形态一览
2020/03/02 星际争霸
PHP输出控制功能在简繁体转换中的应用
2006/10/09 PHP
详解php的魔术方法__get()和__set()使用介绍
2012/09/19 PHP
php和html的区别点详细总结
2019/09/24 PHP
js文字滚动停顿效果代码
2008/06/28 Javascript
jquery 弹出登录窗口实现代码
2009/12/24 Javascript
元素的内联事件处理函数的特殊作用域在各浏览器中存在差异
2011/01/12 Javascript
JavaScript定时器详解及实例
2013/08/01 Javascript
仿百度的关键词匹配搜索示例
2013/09/25 Javascript
使用Jquery实现每日签到功能
2015/04/03 Javascript
实现placeholder效果的方案汇总
2015/06/11 Javascript
js实现tab切换效果实例
2015/09/16 Javascript
js+html5实现的自由落体运动效果代码
2016/01/28 Javascript
JavaScript中有关一个数组中最大值和最小值及它们的下表的输出的解决办法
2016/07/01 Javascript
Web性能优化系列 10个提升JavaScript性能的技巧
2016/09/27 Javascript
jQuery实现的checkbox级联选择下拉菜单效果示例
2016/12/26 Javascript
前端面试知识点锦集(JavaScript篇)
2016/12/28 Javascript
vue.js树形组件之删除双击增加分支实例代码
2017/02/28 Javascript
Node.js中多进程模块Cluster的介绍与使用
2017/05/27 Javascript
关于vue-router的beforeEach无限循环的问题解决
2017/09/09 Javascript
vue 父组件调用子组件方法及事件
2018/03/29 Javascript
微信小程序授权登录及解密unionId出错的方法
2018/09/26 Javascript
使用ThinkJs搭建微信中控服务的实现方法
2019/08/08 Javascript
[03:08]迎霜节狂欢!2018年迎霜节珍藏Ⅰ一览
2018/12/25 DOTA
Python中的上下文管理器和with语句的使用
2018/04/17 Python
pycharm安装和首次使用教程
2018/08/27 Python
Python 2/3下处理cjk编码的zip文件的方法
2019/04/26 Python
python tkinter canvas使用实例
2019/11/04 Python
python实现人机五子棋
2020/03/25 Python
JD Sports澳洲官网:英国领先的运动鞋和运动时尚零售商
2020/02/15 全球购物
static全局变量与普通的全局变量有什么区别?static局部变量和普通局部变量有什么区别?static函数与普通函数有什么区别?
2015/02/22 面试题
C#面试常见问题
2013/02/25 面试题
食堂个人先进事迹
2014/01/22 职场文书
考试违纪检讨书
2014/02/02 职场文书
事业单位聘任报告
2015/03/02 职场文书
利用python进行数据加载
2021/06/20 Python