Back鍵的監聽
對於Back鍵的監聽比較容易,可以在多個系統回調處攔截,比如在activity的下列方法中都可以收到Back鍵按下的事件:
@Override public void onBackPressed() { // super.onBackPressed();//注釋掉這行,back鍵不退出activity Log.i(LOG_TAG, "onBackPressed"); } @Override public boolean dispatchKeyEvent(KeyEvent event) { Log.i(LOG_TAG, "dispatchKeyEvent: keyCode -- " + event.getKeyCode()); return super.dispatchKeyEvent(event); } @Override public boolean onKeyDown(int keyCode, KeyEvent event) { Log.i(LOG_TAG, "onKeyDown: keyCode -- " + keyCode); switch (keyCode) { case KeyEvent.KEYCODE_BACK: Log.i(LOG_TAG, "KeyEvent.KEYCODE_BACK"); break; case KeyEvent.KEYCODE_MENU: Log.i(LOG_TAG, "KeyEvent.KEYCODE_MENU"); break; case KeyEvent.KEYCODE_HOME: Log.i(LOG_TAG, "KeyEvent.KEYCODE_HOME"); // 收不到 break; case KeyEvent.KEYCODE_APP_SWITCH: Log.i(LOG_TAG, "KeyEvent.KEYCODE_APP_SWITCH"); // 收不到 break; default: break; } return super.onKeyDown(keyCode, event); }
Home鍵的廣播監聽
對於Home鍵的監聽不是那么容易,因為Home鍵可以將程序退出放在后台,所以這個事件是直接分發給系統,系統接收到之后做相應處理,Home鍵的事件不是直接傳遞到應用里面.所以在上述監聽Back鍵的代碼中,相應的回調中是收不到Home鍵的事件的.
參考文后的博客鏈接,對Home鍵的監聽主要通過注冊廣播接收器實現,攔截讓窗口關閉的系統動作,然后根據Intent里面的具體參數,分析當前到底是Home鍵, 應用切換鍵,還是其他功能按鍵.
接收器實現如下:
package com.mengdd.hellohome; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.util.Log; public class HomeWatcherReceiver extends BroadcastReceiver { private static final String LOG_TAG = "HomeReceiver"; private static final String SYSTEM_DIALOG_REASON_KEY = "reason"; private static final String SYSTEM_DIALOG_REASON_RECENT_APPS = "recentapps"; private static final String SYSTEM_DIALOG_REASON_HOME_KEY = "homekey"; private static final String SYSTEM_DIALOG_REASON_LOCK = "lock"; private static final String SYSTEM_DIALOG_REASON_ASSIST = "assist"; @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); Log.i(LOG_TAG, "onReceive: action: " + action); if (action.equals(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)) { // android.intent.action.CLOSE_SYSTEM_DIALOGS String reason = intent.getStringExtra(SYSTEM_DIALOG_REASON_KEY); Log.i(LOG_TAG, "reason: " + reason); if (SYSTEM_DIALOG_REASON_HOME_KEY.equals(reason)) { // 短按Home鍵 Log.i(LOG_TAG, "homekey"); } else if (SYSTEM_DIALOG_REASON_RECENT_APPS.equals(reason)) { // 長按Home鍵 或者 activity切換鍵 Log.i(LOG_TAG, "long press home key or activity switch"); } else if (SYSTEM_DIALOG_REASON_LOCK.equals(reason)) { // 鎖屏 Log.i(LOG_TAG, "lock"); } else if (SYSTEM_DIALOG_REASON_ASSIST.equals(reason)) { // samsung 長按Home鍵 Log.i(LOG_TAG, "assist"); } } } }
注意不同手機的按鍵不同,所以需要對不同理由做區分.
Home鍵監聽廣播注冊
廣播接收器的注冊有兩種方式,一種是靜態注冊,即寫在manifest里面聲明;另一種是動態注冊,即在Java代碼里面注冊.
上面對Home鍵實現監聽的這個receiver,靜態注冊如下:
<receiver android:name="com.mengdd.hellohome.HomeWatcherReceiver" > <intent-filter> <action android:name="android.intent.action.CLOSE_SYSTEM_DIALOGS" /> </intent-filter> </receiver>
但是發現靜態注冊不起作用,即收不到onReceive回調.
采用動態注冊:
private static HomeWatcherReceiver mHomeKeyReceiver = null; private static void registerHomeKeyReceiver(Context context) { Log.i(LOG_TAG, "registerHomeKeyReceiver"); mHomeKeyReceiver = new HomeWatcherReceiver(); final IntentFilter homeFilter = new IntentFilter(Intent.ACTION_CLOSE_SYSTEM_DIALOGS); context.registerReceiver(mHomeKeyReceiver, homeFilter); } private static void unregisterHomeKeyReceiver(Context context) { Log.i(LOG_TAG, "unregisterHomeKeyReceiver"); if (null != mHomeKeyReceiver) { context.unregisterReceiver(mHomeKeyReceiver); } }
在Activity的onResume和onPause里面分別調用:
@Override protected void onResume() { super.onResume(); registerHomeKeyReceiver(this); } @Override protected void onPause() { unregisterHomeKeyReceiver(this); super.onPause(); }
當然也可以根據需要在其他合適的時機注冊和注銷.
android app全屏終極方案
/** * 隱藏虛擬按鍵,並且全屏 */ protected void hideBottomUIMenu() { //隱藏虛擬按鍵,並且全屏 if (Build.VERSION.SDK_INT > 11 && Build.VERSION.SDK_INT < 19) { // lower api View v = this.getWindow().getDecorView(); v.setSystemUiVisibility(View.GONE); } else if (Build.VERSION.SDK_INT >= 19) { //for new api versions. View decorView = getWindow().getDecorView(); int uiOptions = View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY | View.SYSTEM_UI_FLAG_FULLSCREEN; decorView.setSystemUiVisibility(uiOptions); } } private void showBottomUIMenu(){ //恢復普通狀態 if (Build.VERSION.SDK_INT > 11 && Build.VERSION.SDK_INT < 19) { // lower api View v = this.getWindow().getDecorView(); v.setSystemUiVisibility(View.VISIBLE); } else if (Build.VERSION.SDK_INT >= 19) { //for new api versions. View decorView = getWindow().getDecorView(); int uiOptions = View.SCREEN_STATE_OFF; decorView.setSystemUiVisibility(uiOptions); } }