利用辅助服务实现微信自动抢红包,其实原理很简单:就是模拟点击操作。这里记录下来,方便以后查询。
实现大致思路:
- 检测微信红包
- 打开聊天界面,找到红包
- 拆红包
(当然其中有很多细节需要去优化,这里列出基本方法)
1.检测红包
这里是通过AccessibilityEvent事件来检测,关于AccessibilityEvent的详细使用情况,我会在后边的blog中持续更新。这里主要检测3种事件。
如果是通知栏事件直接打开到聊天详情界面,开始找红包,如果是界面改变事件测直接启动红包。核心代码:
public void onAccessibilityEvent(AccessibilityEvent event) {
LogUtil.d("事件---->onAccessibilityEvent " + event);
switch (event.getEventType()) {
case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED://通知栏状态改变
List<CharSequence> texts = event.getText();
if (!texts.isEmpty()) {
for (CharSequence t : texts) {
if (!StringUtil.isEmpty(t)) {
if (t.toString().contains(WECHAT_NOTIFICATION_TIP)) {
LogUtil.d("事件----> notifychanged " + t);
openNotify(event);
isNotifyClicked = true;
isOPenRedPacketViewClicked = false;
nickName = getNickName(t.toString());
startRedPaceketTime = TimeUtil.formatDateToHMSS(System.currentTimeMillis());
endRedPacketTime = null;
redPacketMoney = "0.00";
redPacketStatus = 0;
break;
}
}
}
}
break;
case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED://window界面状态改变
case AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED://连接改变
LogUtil.i("通知栏点击事件 isNotifyClicked = " + isNotifyClicked);
startRedPacket(event);
break;
default:
break;
}
}
2.找红包
找红包关系到抢红包的效率,如果越快找到红包,那么抢红包耗时就越短,所以这个地方优化效率很重要。这个地方是检查是否包含 查看红包和领红包。核心代码:
private void openRedPacketView(AccessibilityEvent event) {
LogUtil.d("事件----> start openRedPacketView " + event);
AccessibilityNodeInfo rootNodeInfo = event.getSource();
if (rootNodeInfo != null) {
List<AccessibilityNodeInfo> nodes = findAccessibilityNodeInfosByTexts(rootNodeInfo, new String[]{WECHAT_VIEW_OTHERS_CH, WECHAT_VIEW_SELF_CH});
if (nodes != null && nodes.size() > 0) {
int size = nodes.size();
String id = getRedPacketText(nodes.get(size - 1));
long now = System.currentTimeMillis();
//不过不应该返回,并且通知栏没有被点过,则执行打开红包视图
if (!shouldReturn(id, now - lastFetchedTime) && isNotifyClicked) {
AccessibilityNodeInfo cellNode = nodes.get(size - 1);
cellNode.getParent().performAction(AccessibilityNodeInfo.ACTION_CLICK);
lastFetchedRedPackeId = id;
lastFetchedTime = now;
isOPenRedPacketViewClicked = true;
isNotifyClicked = false;//再聊天详情界面,执行完点击事件以后通知栏点击状态就可以置为false
LogUtil.d("事件----> end openRedPacketView" + event);
}
}
}
}
3.拆红包
拆红包相对就简单了。如果前边的步骤已经执行完成,那么就已经打开了红包界面,拆红包就是点击拆按钮就可以了。核心代码:
private void openRedPacket(AccessibilityEvent event) {
LogUtil.d("事件----> start openRedPacket " + event);
AccessibilityNodeInfo rootNodeInfo = event.getSource();
if (rootNodeInfo != null) {
List<AccessibilityNodeInfo> nodes = new ArrayList<AccessibilityNodeInfo>();
for (int i = 0; i < rootNodeInfo.getChildCount(); i++) {
CharSequence name = rootNodeInfo.getChild(i).getClassName();
if (!StringUtil.isEmpty(name) && "android.widget.Button".equals(name.toString())) {//匹配拆红包按钮
nodes.add(rootNodeInfo.getChild(i));
break;
}
}
if (nodes.isEmpty()) {
//拆红包
nodes = findAccessibilityNodeInfosByTexts(rootNodeInfo, new String[]{WECHAT_OPEN_CH, WECHAT_OPEN_EN});
}
if (nodes != null && nodes.size() > 0) {
AccessibilityNodeInfo cellNode = nodes.get(nodes.size() - 1);
cellNode.performAction(AccessibilityNodeInfo.ACTION_CLICK);
endRedPacketTime = TimeUtil.formatDateToHMSS(System.currentTimeMillis());
redPacketStatus = 1;//成功状态
LogUtil.d("事件----> end openRedPacket " + event);
} else {
List<AccessibilityNodeInfo> nodes3 = findAccessibilityNodeInfosByTexts(rootNodeInfo, new String[]{WECHAT_BETTER_LUCK_CH, WECHAT_BETTER_LUCK_EN, WECHAT_DETAILS_EN, WECHAT_DETAILS_CH});
if (nodes3 != null) {//如果匹配到手慢或者查看红包详情,说明红包已经抢完,保存失败数据
redPacketStatus = 2;//手慢了失败状态
endRedPacketTime = null;
redPacketMoney = "0.00";
//保存红包记录
try {
RedPacketEntity entity = new RedPacketEntity();
entity.setStartTime(startRedPaceketTime);
entity.setEndTime(endRedPacketTime);
entity.setNickName(nickName);
entity.setMoney(redPacketMoney);
entity.setStatus(redPacketStatus);
x.getDb(DaoConfig.getInstance()).save(entity);
} catch (Exception e) {
}
保存完成清空数据
nickName = null;
startRedPaceketTime = null;
endRedPacketTime = null;
redPacketMoney = "0.00";
redPacketStatus = 0;
}
}
}
}
至此核心代码就讲解完成,当然这里只贴出的核心代码,具体实现demo后续更新。