01月25, 2017

使用辅助服务监听系统按键

简介

为什么需要监听系统按键呢?因为我们在做Bigbang应用的时候,需要给用户提供一些功能入口,让用户能够触发全局复制、截屏OCR等功能和开关分词功能。一开始我们实现了通知栏开关和悬浮球开关两套方案,基本上满足了用户的需要。不过为了精益求精,我又想到了通过组合键的形式触发开关,于是就用到了本文的内容。

先看看效果图

音量增+音量键触发菜单

也可以下载全能分词体验

1. 如何使用辅助服务

这部分和通过辅助模式获取点击的文字基本一样,但是需要注意的是xml中canRequestFilterKeyEvents必须设置成true,否则无法按键事件。

2. 如何获取系统按键

其实只要实现了AccessibilityService的onKeyEvent方法就可以获得按键事件了,很简单。代码如下:

public class BigBangMonitorService extends AccessibilityService {

    private AccessibilityServiceInfo mAccessibilityServiceInfo;

    String back ;
    String home ;
    String recent ;

    @Override
    public void onCreate() {
        super.onCreate();
        back = getVitualNavigationKey(this, "accessibility_back", "com.android.systemui", "");
        home = getVitualNavigationKey(this, "accessibility_home", "com.android.systemui", "");
        recent = getVitualNavigationKey(this, "accessibility_recent", "com.android.systemui", "");

        mAccessibilityServiceInfo=new AccessibilityServiceInfo();
        mAccessibilityServiceInfo.feedbackType=FEEDBACK_GENERIC;
        mAccessibilityServiceInfo.eventTypes=AccessibilityEvent.TYPE_VIEW_CLICKED|AccessibilityEvent.TYPE_VIEW_LONG_CLICKED|AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED;
        int flag=0;
        if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.LOLLIPOP){
            flag=flag|AccessibilityServiceInfo.FLAG_RETRIEVE_INTERACTIVE_WINDOWS;
        }
        if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.JELLY_BEAN_MR2){
            flag=flag|AccessibilityServiceInfo.FLAG_REQUEST_FILTER_KEY_EVENTS;
        }
        mAccessibilityServiceInfo.flags=flag;
        mAccessibilityServiceInfo.notificationTimeout=100;
        setServiceInfo(mAccessibilityServiceInfo);

    }

    @Override
    protected boolean onKeyEvent(KeyEvent paramKeyEvent) {
        KeyPressedTipViewController.getInstance().onKeyEvent(paramKeyEvent);
        return false;//如果返回true,就会导致其他应用接收不到事件了,但是对KeyEvent的修改是不会分发到其他应用中的!
    }

    @Override
    public void onAccessibilityEvent(AccessibilityEvent event) {
        //虚拟手机按键处理,优先级高于是否点击分词的判断
        if ((event.getEventType() == TYPE_VIEW_LONG_CLICKED) && ("com.android.systemui".equals(event.getPackageName())))
        {
            if (TextUtils.isEmpty(event.getContentDescription())){
                return;
            }
            //长按虚拟机触发的,需要转到按键处理去
            if (!TextUtils.isEmpty(back) && event.getContentDescription().equals(back)){
                KeyPressedTipViewController.getInstance().onKeyLongPress(KeyEvent.KEYCODE_BACK);
            }else if (!TextUtils.isEmpty(home) && event.getContentDescription().equals(home)){
                KeyPressedTipViewController.getInstance().onKeyLongPress(KeyEvent.KEYCODE_HOME);
            }else if (!TextUtils.isEmpty(recent) && event.getContentDescription().equals(recent)){
                KeyPressedTipViewController.getInstance().onKeyLongPress(KeyEvent.KEYCODE_APP_SWITCH);
            }
        }
    }
}

稍微讲解一下下面几个注意点:

  1. 在onCreate中需要设置flag,从flag的名字上就能看出是用于声明接收key事件的,其实等同于在xml中的accessibilityFlags中添加。
  2. 跟onAccessibilityEvent中的点击事件不一样,这里收到的不是异步的回调,如果onKey返回true,则会导致其他应用收不到key事件。但是对这些paramKeyEvent进行的修改不会影响到应用中接收事件。
  3. 你可能注意到了onAccessibilityEvent中做的处理,这是为了兼容具有NavigationBar(虚拟导航栏)的手机。因为在这些手机上,点击返回、Home、多任务三个按钮的时候,是作为普通的点击事件来处理的,也就是onAccessibilityEvent中的回调。至于代码中我只处理了TYPE_VIEW_LONG_CLICKED,则是因为单击这些键触发菜单显然是不合适的,所以没有做单击的触发方式。
  4. KeyPressedTipViewController中的处理是跟业务逻辑相关的处理,可以不用关心,反正已经得到按键事件了,做爱做的事就行了。

源码

完整代码可以参考Bigbang项目的BigBangMonitorService类。

ps:BigBangMonitorService中还包含了全局复制的功能和监听点击的文字的功能,阅读的时候不要被干扰了,感兴趣的可以看——通过辅助模式获取点击的文字使用辅助服务实现全局复制这两篇文章

转载注明出处:十个雨点

本文链接:http://www.siki.space/post/monitor_system_key_through_accessibility_service.html

-- EOF --

Comments