Android dialog,activity 屏蔽Home键的教程详解

相信在Android应用上,很多时候逻辑是需要屏蔽Home键的,但这个用户体验是否需要,就看各位的需求了。

一般的方法屏蔽Home键,大家一定看过不少文章了。我总结一下,先说一下一般情况下Activity的屏蔽按键和Home键吧。

屏蔽其他键,重写onKeyDown


@Override

public boolean onKeyDown(int keyCode, KeyEvent event) {

Log.i(TAG,"keycode="+keyCode + "   isBan="+isBan);

switch (keyCode) {

  case KeyEvent.KEYCODE\_BACK:

  Log.i(TAG,"KEYCODE\_BACK");

  return true;

}

return super.onKeyDown(keyCode, event);

}

大家会发现,这里屏蔽Home键是捕捉不到的,因为大家的权限一般是User所以是无效的。

而其实android处理Home键等系统级按键是有一定的处理的。

引用

看看源码是怎样处理的 \frameworks\policies\base\phone\com\android\internal\policy\impl\PhoneWindowManager.java


        // First we always handle the home key here, so applications

        // can never break it, although if keyguard is on, we do let

        // it handle it, because that gives us the correct 5 second

        // timeout.

        if (code == KeyEvent.KEYCODE\_HOME) {

            // If a system window has focus, then it doesn't make sense

            // right now to interact with applications.

            WindowManager.LayoutParams attrs = win != null ? win.getAttrs() : null;

            if (attrs != null) {

                final int type = attrs.type;

                if (type == WindowManager.LayoutParams.TYPE\_KEYGUARD

                   || type == WindowManager.LayoutParams.TYPE\_KEYGUARD\_DIALOG) {

                    // the "app" is keyguard, so give it the key

                    return false;

                }

                final int typeCount = WINDOW\_TYPES\_WHERE\_HOME\_DOESNT\_WORK.length;

                for (int i=0; i<typeCount; i++) {

                    if (type == WINDOW\_TYPES\_WHERE\_HOME\_DOESNT\_WORK\[i\]) {

                        // don't do anything, but also don't pass it to the app

                        return true;

                    }

                }

            }

通过源码,我们不难发现两个的参数 WindowManager.LayoutParams.TYPE_KEYGUARD和

WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG

**借鉴于此,重写onAttachedToWindow,以实现屏蔽Home键

**


public void onAttachedToWindow() {

this.getWindow().setType(WindowManager.LayoutParams.TYPE\_KEYGUARD);

super.onAttachedToWindow();

}

- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 华丽的分界线,以下内容更精彩- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

轮到dialog了,如果在Activity弹出dialog,在Activity设置以上2个方法是没办法屏蔽的。

其实,原理是一样的,只是地方不一样而已


final Dialog dialog = new Dialog(this);

dialog.setContentView(R.layout.mydailog);

dialog.getWindow().setType(WindowManager.LayoutParams.TYPE\_KEYGUARD);

dialog.show();

dialog.setOnKeyListener(new android.content.DialogInterface.OnKeyListener(){

@Override

public boolean onKey(DialogInterface dialog, int keyCode,KeyEvent event) {

  switch (keyCode) {

   case KeyEvent.KEYCODE\_BACK:

   Log.i(TAG,"KEYCODE\_BACK");

   return true;

  }

  return false;

}

}); 

这样运行后,出错如下:


10-18 13:27:06.380: ERROR/AndroidRuntime(4684): Caused by: android.view.WindowManager$BadTokenException: Unable to add window   

\[url=mailto:android.view.ViewRoot$W@2b046d68\]android.view.ViewRoot$W@2b046d68\[/url\] -- permission denied for this window type

其实,只需要把dialog.getWindow().setType的位置放在show后面就可以了

正确答案


dialog.show();

dialog.getWindow().setType(WindowManager.LayoutParams.TYPE\_KEYGUARD);

这么,就完成了Back键的屏蔽 和Home键盘的屏蔽了!

总结:

1:)在以上用WindowManager.LayoutParams.TYPE_KEYGUARD的地方改用

WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG 效果一样。至于两者的具体差别,得以后再研究研究。

2:)其实,在源码里是这样调用的。


final AlertDialog dialog = new AlertDialog.Builder(mContext)

.setTitle(null)

.setMessage(message)

.setNeutralButton(R.string.ok, null)

.create();

dialog.getWindow().setType(WindowManager.LayoutParams.TYPE\_KEYGUARD\_DIALOG);

dialog.show();

但我们如果这样调用就会出现之前的那个error:permission denied for this window type 这就显而易见了吧~~

3:)ProgressDialog 默认屏蔽 Back键,Dialog,AlertDialog则需setOnKeyListener

4:)其实屏蔽Home键,在页面的某个地方,例如一个Button的onClick里,去设置setType就可以了,如:


button.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

  getWindow().setType(WindowManager.LayoutParams.TYPE\_KEYGUARD);

}

});

但前提是重载Activity的onAttachedToWindow(),哪怕只是一个空实现,然后返回父类方法。


@Override  

public void onAttachedToWindow() {

  super.onAttachedToWindow();

}

5:)其实它们,都是常用的~


switch (keyCode) {

case KeyEvent.KEYCODE\_HOME:

  Log.i(TAG,"KEYCODE\_HOME");

  return true;

case KeyEvent.KEYCODE\_BACK:

  Log.i(TAG,"KEYCODE\_BACK");

  return true;

case KeyEvent.KEYCODE\_CALL:

  Log.i(TAG,"KEYCODE\_CALL");

  return true;

case KeyEvent.KEYCODE\_SYM:

  Log.i(TAG,"KEYCODE\_SYM");

  return true;

case KeyEvent.KEYCODE\_VOLUME\_DOWN:

  Log.i(TAG,"KEYCODE\_VOLUME\_DOWN");

  return true;

case KeyEvent.KEYCODE\_VOLUME\_UP:

  Log.i(TAG,"KEYCODE\_VOLUME\_UP");

  return true;

case KeyEvent.KEYCODE\_STAR:

  Log.i(TAG,"KEYCODE\_STAR");

  return true;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值