Android模拟返回、Home、多任务、菜单键

一般的安卓设备底部都有三大按键(当然,也可以改成全面屏手势)——返回、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、多任务、菜单按键的点击。

  • 7
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android中,我们可以通过重写Activity的onBackPressed()方法屏蔽返回的功能,代码如下: ``` @Override public void onBackPressed() { // 屏蔽返回 // super.onBackPressed(); // 如果需要默认的返回功能,需要注释掉这行代码 } ``` 对于HOME的屏蔽,由于Android系统限制,我们无法屏蔽HOME的功能。但是我们可以监听HOME的按下事件,并在按下事件发生时做一些特定的操作,例如弹出对话框或者跳转到其他页面。代码如下: ``` private static final String SYSTEM_DIALOG_REASON_KEY = "reason"; private static final String SYSTEM_DIALOG_REASON_HOME_KEY = "homekey"; @Override public void onAttachedToWindow() { super.onAttachedToWindow(); // 监听HOME getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG); } @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_HOME) { // HOME按下事件 return true; } return super.onKeyDown(keyCode, event); } @Override public void onWindowFocusChanged(boolean hasFocus) { super.onWindowFocusChanged(hasFocus); if (!hasFocus) { // 如果Activity失去了焦点,则判断是否是HOME按下事件 Intent closeDialog = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS); sendBroadcast(closeDialog); } } @Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); // 判断是否是HOME按下事件 String reason = intent.getStringExtra(SYSTEM_DIALOG_REASON_KEY); if (SYSTEM_DIALOG_REASON_HOME_KEY.equals(reason)) { // HOME按下事件 } } ``` 需要注意的是,由于Android系统限制,无法屏蔽HOME的功能,因此以上代码只能监听HOME的按下事件,并在按下事件发生时做一些特定的操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值