如何使用AccessibilityService?

7
我正在学习AccessibilityService。我已经学习了什么是Accessibility Service以及为什么我们要使用它,以及Accessibility Service的优点。因此,我尝试制作一个Accessibility service的演示,以了解它的工作原理,即当收到电子邮件或任何whatsapp、微信消息时,它会进入哪个方法。但是,我没有成功。
当我开始这个项目时,我在StartService(i)行上遇到了异常。所以请您检查一下我的代码,并告诉我我犯了什么错误。您的评论和建议对我和AccessibilityService初学者将非常有帮助。
我使用的代码如下:
在MainActivity中...
public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent i = new Intent(MainActivity.this, MyService.class);
        startService(i);
    }
}

在AccessibilityService类中...
public class MyService extends AccessibilityService{


    @Override
    public void onAccessibilityEvent(AccessibilityEvent event) {
        // TODO Auto-generated method stub
        Log.i("Accessibility", "onAccessibilityEvent");
        Toast.makeText(getApplicationContext(), "onAccessibilityEvent", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onInterrupt() {
        // TODO Auto-generated method stub
        Log.i("Interrupt", "Interrupt");
        Toast.makeText(getApplicationContext(), "onInterrupt", Toast.LENGTH_SHORT).show();

    }

    @Override
    protected void onServiceConnected() {
        // TODO Auto-generated method stub
        super.onServiceConnected();
        Log.i("Service", "Connected");
        Toast.makeText(getApplicationContext(), "onServiceConnected", Toast.LENGTH_SHORT).show();
    }

}

在 Manifest 文件中...
<uses-permission android:name="android.permission.BIND_ACCESSIBILITY_SERVICE" />
<application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.notifications.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <service
        android:name="MyService"
        android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE" >
        <intent-filter>
            <action android:name="android.accessibilityservice.AccessibilityService" />
        </intent-filter>

        <meta-data
            android:name="android.accessibilityservice"
            android:resource="@xml/accessibilityservice" />
    </service>

    </application>

在res/xml文件夹中,即accessibilityservice.xml文件中...
<?xml version="1.0" encoding="utf-8"?>
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
    android:accessibilityEventTypes="typeNotificationStateChanged"
    android:accessibilityFeedbackType="feedbackAllMask"
    android:notificationTimeout="100" />

我获取到的日志信息是...

06-29 01:58:40.798: E/AndroidRuntime(21707): FATAL EXCEPTION: main
06-29 01:58:40.798: E/AndroidRuntime(21707): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.notifications/com.example.notifications.MainActivity}: java.lang.SecurityException: Not allowed to start service Intent { cmp=com.example.notifications/.MyService } without permission android.permission.BIND_ACCESSIBILITY_SERVICE
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1659)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1675)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread.access$1500(ActivityThread.java:121)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:943)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.os.Handler.dispatchMessage(Handler.java:99)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.os.Looper.loop(Looper.java:130)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread.main(ActivityThread.java:3701)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at java.lang.reflect.Method.invokeNative(Native Method)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at java.lang.reflect.Method.invoke(Method.java:507)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:866)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:624)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at dalvik.system.NativeStart.main(Native Method)
06-29 01:58:40.798: E/AndroidRuntime(21707): Caused by: java.lang.SecurityException: Not allowed to start service Intent { cmp=com.example.notifications/.MyService } without permission android.permission.BIND_ACCESSIBILITY_SERVICE
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ContextImpl.startService(ContextImpl.java:867)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.content.ContextWrapper.startService(ContextWrapper.java:336)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at com.example.notifications.MainActivity.onCreate(MainActivity.java:15)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
06-29 01:58:40.798: E/AndroidRuntime(21707):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1623)
06-29 01:58:40.798: E/AndroidRuntime(21707):    ... 11 more

https://dev59.com/iGAg5IYBdhLWcg3wpMSz - CommonsWare
@CommonsWare 我没有看到这个对话框,很遗憾你的应用程序已停止。我已经更新了上面的日志记录。 - nawaab saab
2个回答

10

你不能像其他服务一样控制AccessibilityService的启动/停止,因此无法“启动”它们。相反,Android系统根据用户所选择的设置来控制它们的启动和停止。

你能做的最好的事情是启动辅助功能设置页面,并让用户为你启用它:

Intent openSettings = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
openSettings.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_NO_HISTORY);
startActivity(openSettings);

希望能帮助到您!

3
您正在尝试启动自己的AccessibilityService。但这是不可能的,因为您没有android.permission.BIND_ACCESSIBILITY_SERVICE权限。此外,这也是不必要的,因为系统会在适当的时候(例如用户在设置中启用服务)启动您的服务。
如果您的AccessibilityService中有业务逻辑,您也想在其他地方使用它,请将其移动到一些可同时被AccessibilityService和其他代码使用的中央位置。

感谢commonsware的回答,但我想在收到电子邮件或来自WhatsApp、微信等应用的消息时,在Toast或Logcat中显示一条消息。我该如何做到这一点?是否可以使用辅助功能服务来获取每当电子邮件或来自WhatsApp、微信等应用的消息时的Toast?如果可以,我该如何应用辅助功能服务? - nawaab saab
1
在哪些情况下会调用“onInterrupt”?它不是和“onDestroy”一样吗? 我以为操作系统会尽可能让它保持活动状态,是吗? - android developer
@androiddeveloper:我不知道,抱歉。 - CommonsWare
@CommonsWare 的问题在于它并没有很好地记录下来... 相反,它更像是这样的:https://miro.medium.com/max/501/1*Pyxsc7Uixbitv5myywaA_Q.jpeg - android developer

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接