欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

浅谈如何使用Python控制手机(二)

程序员文章站 2022-03-15 20:41:56
目录1. 序言2. 前置工作3. 打开app4. 获取app的包名5. 打开特定页面6. 其它细节7. 总结1. 序言每年淘宝双十一的时候,总是要刷各种各样的浏览页面,收集能量或者喵币或者什么。那既然...

1. 序言

每年淘宝双十一的时候,总是要刷各种各样的浏览页面,收集能量或者喵币或者什么。

那既然如此,我就总想着,能否通过python自动调用的方式来刷网页。

2. 前置工作

本文是基于使用python控制手机(一),默认已经安装了adb并配置了环境变量,安装了python环境,且在python中安装了uiautomator2和weditor等包。

3. 打开app

当我们使用uiautomator2包来打开某个app时,可以通过点击屏幕特殊位置的方式来实现。但是其中存在的问题便是,可能由于我们app图标的移动,而使得程序无法运行。健壮性和通用性不高。

其实在uiautomator2这个包中,提供了一种可以通过app包名就可以打开特定app的方式,例如打开和关闭淘宝。

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
d.app_start("com.taobao.taobao")  # 打开淘宝
time.sleep(10)  # 等待10秒钟
d.app_stop("com.taobao.taobao")  # 关闭淘宝

再比如打开和关闭微信:

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
d.app_start("com.tencent.mm")  # 打开微信
time.sleep(10)  # 等待10秒钟
d.app_stop("com.tencent.mm")  # 关闭微信

4. 获取app的包名

有的时候,我们是不太清楚一个app的包名的,这时我们可以通过打印设备当前信息的方式来获取app的包名。首先我们需要将要获取的app打开,并且保持在手机最前台

执行代码:

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
print(d.info)  # 打印设备信息

输出结果如下:

{'currentpackagename': 'com.taobao.taobao', 'displayheight': 2111, 'displayrotation': 0, 'displaysizedpx': 393, 'displaysizedpy': 851, 'displaywidth': 1080, 'productname': 'cannon', 'screenon': true, 'sdkint': 29, 'naturalorientation': true}
process finished with exit code 0

在所打印的json键值对中,键currentpackagename对应的值,即为此时正在最前台的app的包名,上述结果操作时,正在最前的app为淘宝。

5. 打开特定页面

一般来说,如果页面切换按钮含有特定文字,我们直接通过文字进行定位是最方便的,也是最准确的,比如打开微信朋友圈:

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
d.app_start("com.tencent.mm")  # 打开微信
time.sleep(2)  # 等待2秒钟
d(text='发现').click()  # 点击文字为“发现”的控件
time.sleep(2)  # 等待2秒钟
d(text='朋友圈').click()  # 点击文字为“朋友圈”的控件

因为可能存在的,app的加载时间和对点击操作的响应时间,尽量在每次点击操作之后,为app和手机留有足够的反应时间。值得注意的是,如果打开微信之后,恰好有个常用联系人的昵称叫做“发现”,那就可能会被误点,这种情况下我们需要使用别的定位方式来定位特定控件。

需要点击的文字如果是固定的,就可以使用d(text="xxx")来选择控件元素,其中xxx为特定的文字。如果部分文字是固定的,比如第一次元素显示文字为“我是第11932位访客”,第二次显示文字为“我是第12111位访客”,那我们可以通过d(textcontains="我是第").click()来点击这个控件,或者通过d(textcontains="位访客").click()来点击这个控件,这种方式就可以通过子字符串来定位特定的元素控件。

还是使用进入朋友圈举例:

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
d.app_start("com.tencent.mm")  # 打开微信
time.sleep(2)  # 等待2秒钟
# 点击“发现”,三选一
d(text='发现').click()  # 点击文字为“发现”控件
d(textcontains='发').click()  # 点击带“发”的控件
# 通过weditor获得的xpath定位
d.xpath('//*[@resource-id="com.tencent.mm:id/e8y"]/android.widget.linearlayout[1]/android.widget.relativelayout[3]/android.widget.linearlayout[1]').click()  
time.sleep(2)  # 等待2秒钟
# 点击“朋友圈”,三选一
d(text='朋友圈').click()  # 点击文字为“朋友圈”控件
d(textcontains='朋').click()  # 点击带“朋”的控件
# 通过weditor获得的xpath定位
d.xpath('//*[@resource-id="android:id/list"]/android.widget.linearlayout[1]/android.widget.linearlayout[1]/android.widget.linearlayout[1]/android.widget.linearlayout[1]/android.widget.linearlayout[1]/android.widget.linearlayout[1]/android.widget.linearlayout[1]').click()

其实还有很多各不相同的定位方式,只要能定位到唯一的特定的控件进行点击即可。例如在双十一时,我在淘宝中切换到收集喵币页面的点击事件:

import uiautomator2 as u2
import time
d = u2.connect()  # 连接设备
d.app_start("com.taobao.taobao")  # 打开淘宝
time.sleep(5)  # 等待5秒钟
d.xpath('//*[@content-desc="双11超级喵糖"]').click()  # 打开喵糖页面

6. 其它细节

在双十一淘宝活动中,打开喵糖页面,会先弹出提示是否将此页面加入收藏的弹框,点击文字为“我再想想”的按钮。注意要判断此控件是否存在,点击不存在的控件将会报错。如果不确定一个控件是否存在,又不想判断的情况下,则需要使用 try......catch...... 来将其包裹。

if len(d(textcontains='我再想想')) > 0:  # 如果存在此控件
    d(textcontains='我再想想').click()  # 点击“我再想想”

点击“赚糖”控件,因为这个控件经常会被屏幕上出现的手指动画所挡住,因此需要等待:

while len(d(textcontains='赚糖')) <= 0:
    time.sleep(1)
d(textcontains='赚糖').click()

然后点击完后等会儿,再点击“去浏览”按钮:

while len(d(textcontains='去浏览')) > 0:
    print("检测到浏览按钮...")
    d(textcontains='去浏览').click()

等待15秒(算上反应时间,需要多等一会儿)返回即可:

d.press("back")  # 相当于手机返回键

7. 总结

其实具体的部分实现起来比较简单,在此总结一下uiautomator2 的其它一些功能。

关于按键:

d.press("home")         # 点击home键
d.press("back")         # 点击back键
d.press("left")         # 点击左键
d.press("right")        # 点击右键
d.press("up")           # 点击上键
d.press("down")         # 点击下键
d.press("center")       # 点击选中
d.press("menu")         # 点击menu按键
d.press("search")       # 点击搜索按键
d.press("enter")        # 点击enter键
d.press("delete")       # 点击删除按键
d.press("recent")       # 点击近期活动按键
d.press("volume_up")    # 音量+
d.press("volume_down")  # 音量-
d.press("volume_mute")  # 静音
d.press("camera")       # 相机
d.press("power")        # 电源键

关于锁屏与解锁:

# 一个设备信息字典中的布尔值,为true时代表当前屏幕亮起,为false代表当前屏幕熄灭
d.info.get('screenon')
# 仅点亮屏幕
d.screen_on()
# 点亮屏幕并解锁,注意如果有密码,则只能进入密码输入页面,需要输入密码才能解锁
d.unlock()  
# 关闭屏幕
d.screen_off()

关于点击等操作(支持百分比):

# 单击屏幕
d.click(x,y)  # x,y为点击坐标
# 双击屏幕
d.double_click(x, y)
d.double_click(x, y, 0.1)  # 默认两个单击之间间隔时间为0.1秒
# 长按
d.long_click(x, y)
d.long_click(x, y, 0.5)  # 长按0.5秒(默认)
# 滑动
d.swipe(sx, sy, ex, ey)
d.swipe(sx, sy, ex, ey, 0.5)  # 滑动0.5秒(默认)
#拖动
d.drag(sx, sy, ex, ey)
d.drag(sx, sy, ex, ey, 0.5)  # 拖动0.5秒(默认)
# 滑动点 多用于九宫格解锁,提前获取到每个点的相对坐标(这里支持百分比)
# 从点(x0, y0)滑到点(x1, y1)再滑到点(x2, y2)
# 两点之间的滑动速度是0.2秒
d.swipe((x0, y0), (x1, y1), (x2, y2), 0.2)
# 注意:单击,滑动,拖动操作支持百分比位置值。例:
d.long_click(0.5, 0.5) 表示长按屏幕中心

当然还有其它的一些功能,例如向上滑动屏幕,直到指定文字出现为止:

d(scrollable=true).scroll.to(text="3年级2班")

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注的更多内容!