美文网首页
微信跳一跳辅助制作详解

微信跳一跳辅助制作详解

作者: handsomeslow | 来源:发表于2018-01-05 17:59 被阅读1101次

前言

微信跳一跳目前很火,然后由于大家想获得朋友圈较高的排名,这和一开始的微信运动步数一样,出现了手机挂狗脖子上刷步数的现象,这回微信跳一跳也有外挂了,本文和你分享一下本人制作辅助思路

分析

首先玩过游戏的都知道,小人会根据按压的时长来跳相应的距离,这个按压时长和距离一定是成正比的,为啥这么说?应该这是这个游戏的核心玩法,就是要通过玩家掌握好这个按压时长才能得到较准确的跳跃距离,若不成正比,那这个游戏简直太难掌握了,根本不能很好的操作小人。

这样的话,只要能够准确计算出两个点之间的距离,既能计算出按压时长,计算距离,那就要找到起点和终点,这是做这个辅助的重点。

与游戏交互

做为一个Android开发者一定知道:ADB,我们则可以通过ADB对手机进行截屏即拿到了游戏界面:

进行屏幕截图
screencap 命令是一个用于对设备显示屏进行屏幕截图的 shell 实用程序。在 shell 中,此语法为:

screencap filename
要从命令行使用 screencap,请输入以下命令:

$ adb shell screencap /sdcard/screen.png
以下屏幕截图会话示例向您展示使用 adb shell 捕获屏幕截图,并使用 pull 命令从设备下载此文件:

$ adb shell
shell@ $ screencap /sdcard/screen.png
shell@ $ exit
$ adb pull /sdcard/screen.png

游戏界面拿到之后需要去分析图片,拿到像素点信息,很显然在python里要用到了PIL(python imaging library)这个库,通过PIL去获取各个像素点信息:

im = Image.open('image_file_path')
pix = im.load()
r,g,b,a = pix[x,y]

图像中一个像素点用32位表示,RGBA 各8位,所以我们用0~255来表示色值,RGBA组合呈现不同的颜色,但是一般常见的PNG是有alpha通道,而JPG是没有alpha通道的。

基础讲完了,我们开始来找起点和终点了!

找起点

首先我们都知道方块的中心就是一个终点,观察一下起始状态:

小人中心点

小人起始时肯定是在中心点的,通过对角线可以看出小人圆台底部圆心即为起点。那么小人的起点就是:紫色最宽的区域的中点即为小人的横坐标;紫色的最低点即为小人的纵坐标。

紫色小人

代码:

im = Image.open('./temp_screen.png')
pix = im.load()

min_point_x = 65535
max_point_x = 0
max_point_y = 0

global start_x
global start_y
    
for x in range(190,910):
        for y in range(1000,1250):
            r,g,b,a = pix[x,y]
            if abs(r - 56) < 5 and abs(g-59) < 5  and abs(b-102) < 5:
                min_point_x = min(min_point_x,x)
                max_point_x = max(max_point_x,x)
                start_y = y
    start_x = (min_point_x + max_point_x) / 2
    return start_x,start_y

找终点

对于终点我们可以观察得出一个简单的规律,终点和起点总是在一条直线上,那么我们就可以算出直线函数:

直线函数

(x0,y0)为起点,斜率 k 算出来是1.6(PS量一下两个点的坐标,纵坐标之差与横坐标之差比),这样的话终点在小人的左侧直线斜率即为-1.6,在右侧直线斜率即为1.6。然后只通过斜率算出终点的横纵坐标是不现实的,至少知道横坐标或者纵坐标。

那我们分析一下横坐标好算还是纵坐标好算呢?显然是横坐标!

  • 对于菱形来说终点的横坐标不就是最高的那个点么?
  • 对于圆来说最上面可能是一条线段,那横坐标不就是线段的中点么?
  • 好像没有其它图形了...

至此我们的问题转换成查找方块最上方的点的横坐标,继而计算出终点的纵坐标。

从上至下一排排的扫描,直至找到最高点。而背景是一个渐变色,就需要我们每一排都要去更新一次背景色,再找这一行是否存在与背景色不一样的像素点。

因为方块颜色和背景图相差较大,这里用灰度值来进行比较,RGB转灰度图:

gray = 0.2989 * r + 0.5870 * g + 0.1140 * b

如果被比较的像素点与背景的灰度值之差小于5即认为非背景点,并保持本行所有非背景点,算出横坐标,进而算出纵坐标即可,代码如下:

im = Image.open('./temp_screen.png')
img_L = im.convert("L")
list_x = []
is_end = False

for y in xrange(400,1100):
    base_pix = img_L.getpixel((1,y))
    for x in xrange(100,910):
        pix = img_L.getpixel((x,y))
        if abs(base_pix - pix) > 5 :
            if abs(x - start_x) > 100:
                list_x.append(x)
                is_end = True
    if is_end:
        end_x = sum(list_x)/len(list_x)
        if end_x < start_x:
            end_y = start_y-(start_x-end_x)/1.6
        else:
            end_y = start_y-(end_x-start_x)/1.6
        return end_x,end_y

操作

既然我们已经算出起点和终点了,则可以算出两点之间的距离:

sx,sy = cal_start()
ex,ey = cal_end()
distance = (sx - ex)**2 + (sy - ey)**2 
distance = distance ** 0.5

操作小人

这个还是利用了ADB的shell命令:

def jump(distance):
    press_time = int(distance * 1.35)
    cmd = 'adb shell input swipe 320 410 320 410 ' + str(press_time)
    os.system(cmd)

循环连续跳

最终加上循环代码:

def start():
    while True:
        pull_screenshot()
        time.sleep(1.0)
        im = Image.open('./temp_screen.png')
        img_L = im.convert("L")
        ll = img_L.getpixel((5,5))
        if abs(ll - 47) < 5:
            return
        sx,sy = cal_start()
        ex,ey = cal_end()
        distance = (sx - ex)**2 + (sy - ey)**2 
        distance = distance ** 0.5
        print 'distance =', distance
        jump(distance)
        time.sleep(2.0)

结果

跳一跳测试结果

写在最后:

第一次想到做辅助脚本还是看到了 wangshub 在知乎上发布的第一个版本,那时候还利用了matplotlib去手动点击起点和终点,实现跳一跳,由于在上班并没有去改代码,但是做为一个有图像知识基础的Android开发者,很快想好思路,回去即着手写本文的代码,晚上写好后,朋友说人家已经更新了,刷新一看,各个版本已经push上去了。然后看star数一天一天涨到了8K+,并且图片识别、AI技术层出不穷,完全看到一个python学习热潮。比对发现本文方法略为简单,深深感受到学习的道路很长很长,还需不断的去努力,争取做那个引领风骚的人!

相关文章

网友评论

      本文标题:微信跳一跳辅助制作详解

      本文链接:https://www.haomeiwen.com/subject/iykagxtx.html