一般的安卓设备底部都有三大按键(当然,也可以改成全面屏手势)——返回、Home、多任务。有些时候,我们不想让这些按钮在屏幕底部占用空间,而是可以自由拖动,那么应当如何实现?
答案:辅助功能。安卓的辅助功能提供了相关接口用于模拟这些按键的点击。
首先,我们创建工程,写下一个类继承 AccessibilityService 类:
class NavigationBarService : AccessibilityService() {
override fun onCreate() {
super.onCreate()
// 辅助功能开启
}
override fun onDestroy() {
super.onDestroy()
// 辅助功能关闭
}
override fun onAccessibilityEvent(event: AccessibilityEvent) {
}
override fun onInterrupt() {
}
}
然后在清单文件里注册服务:
<service
android:name=".service.NavigationBarService"
android:exported="false"
android:label="@string/service_navigation_bar_label"
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/accessibility_service_navigation_bar" />
</service>
strings.xml
<string name="service_navigation_bar_label">菜单键-导航键模拟</string>
<string name="service_navigation_bar_desc">激活后可以模拟菜单键、导航键。</string>
accessibility_service_navigation_bar.xml
<?xml version="1.0" encoding="utf-8"?>
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:description="@string/service_navigation_bar_desc" />
这样一来应用就会出现在系统辅助功能的应用列表。接下来主动跳转到辅助功能页:
val intent = Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS)
startActivityForResult(intent, 0)
点击授权:
授权成功后,AccessibilityService 会创建。我们可以调用其中的 performGlobalAction 方法来执行按钮的点击:
// 点击返回
performGlobalAction(AccessibilityService.GLOBAL_ACTION_BACK)
// 点击Home
performGlobalAction(AccessibilityService.GLOBAL_ACTION_HOME)
// 点击多任务
performGlobalAction(AccessibilityService.GLOBAL_ACTION_RECENTS)
说完以上按键,再来谈谈菜单键。菜单键以前是手机的三大按键之一,后来由于谷歌的要求被取消了,改成了多任务键。但是这样一来,好多应用的菜单选项就无法弹出了(例如QQ、微博)。如何模拟菜单按键?
我们可以使用 InputMethodService 类(输入法服务)。
class KeyInputService : InputMethodService() {
override fun onCreate() {
super.onCreate()
// 当前输入法被选中时
}
override fun onDestroy() {
super.onDestroy()
// 输入法被换成其他输入法时
}
}
注册清单文件:
<service
android:name=".service.KeyInputService"
android:permission="android.permission.BIND_INPUT_METHOD"
android:exported="false">
<intent-filter>
<action android:name="android.view.InputMethod" />
</intent-filter>
<meta-data
android:name="android.view.im"
android:resource="@xml/im_service_key_input" />
</service>
im_service_key_input.xml(settingsActivity可以指定在设置页点击输入法时应跳转到哪个Activity,如下图)
<?xml version="1.0" encoding="utf-8"?>
<input-method xmlns:android="http://schemas.android.com/apk/res/android"
android:settingsActivity="com.magicianguo.menukey.activity.MainActivity" />
判断当前应用输入法是否启用:
fun isCurrentIMEnabled(context: Context): Boolean {
val manager = context.getSystemService(Context.INPUT_METHOD_SERVICE)
as InputMethodManager
manager.enabledInputMethodList.forEach {
if (it.packageName == context.packageName) {
return true
}
}
return false
}
判断当前应用是否为默认输入法:
fun isCurrentIMSelected(context: Context): Boolean {
val name = Settings.Secure.getString(context.contentResolver,
Settings.Secure.DEFAULT_INPUT_METHOD)
return name.contains("${context.packageName}/")
}
之后应用可以请求切换输入法:
val manager = context.getSystemService(Context.INPUT_METHOD_SERVICE)
as InputMethodManager
manager.showInputMethodPicker()
监听输入法的变更,可以监听 Intent.ACTION_INPUT_METHOD_CHANGED 广播。
当应用被选中为默认输入法后,可以在 InputMethodService 中调用 getCurrentInputConnection() 方法,然后发送按键事件即可。
val keyCode = KeyEvent.KEYCODE_MENU
val inputConnection = getCurrentInputConnection()
inputConnection.sendKeyEvent(KeyEvent(KeyEvent.ACTION_DOWN, keyCode))
inputConnection.sendKeyEvent(KeyEvent(KeyEvent.ACTION_UP, keyCode))
测试效果:
QQ:
微博:
源代码可以到Github上面看:GitHub - MagicianGuo/Android-MenuKey: 能够模拟返回、Home、多任务、菜单按键的点击。