相信在Android应用上,很多时候逻辑是需要屏蔽Home键的,但这个用户体验是否需要,就看各位的需求了。
    一般的方法屏蔽Home键,大家一定看过不少文章了。我总结一下,先说一下一般情况下Activity的屏蔽按键和Home键吧。
屏蔽其他键,重写onKeyDown


1.  
2.  @Override
3.  public boolean onKeyDown(int keyCode, KeyEvent event) {
4.  Log.i(TAG,"keycode="+keyCode + "   isBan="+isBan);
5.  switch (keyCode) {
6.    case KeyEvent.KEYCODE_BACK:
7.    Log.i(TAG,"KEYCODE_BACK");
8.    return true;
9.  }
10.  return super.onKeyDown(keyCode, event);
11.  }


复制代码

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


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


引用


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

1.  
2.          // First we always handle the home key here, so applications
3.          // can never break it, although if keyguard is on, we do let
4.          // it handle it, because that gives us the correct 5 second
5.          // timeout.
6.          if (code == KeyEvent.KEYCODE_HOME) {
7.              // If a system window has focus, then it doesn't make sense
8.              // right now to interact with applications.
9.              WindowManager.LayoutParams attrs = win != null ? win.getAttrs() : null;
10.              if (attrs != null) {
11.                  final int type = attrs.type;
12.                  if (type == WindowManager.LayoutParams.TYPE_KEYGUARD
13.                     || type == WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG) {
14.                      // the "app" is keyguard, so give it the key
15.                      return false;
16.                  }
17.                  final int typeCount = WINDOW_TYPES_WHERE_HOME_DOESNT_WORK.length;
18.                  for (int i=0; i<typeCount; i++) {
19.                      if (type == WINDOW_TYPES_WHERE_HOME_DOESNT_WORK[i]) {
20.                          // don't do anything, but also don't pass it to the app
21.                          return true;
22.                      }
23.                  }
24.              }


复制代码

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


WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG

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

1.  
2.  public void onAttachedToWindow() {
3.  this.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD);
4.  super.onAttachedToWindow();
5.  }
6.

复制代码

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



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


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

1.  final Dialog dialog = new Dialog(this);
2.  dialog.setContentView(R.layout.mydailog);
3.  dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD);
4.  dialog.show();
5.  dialog.setOnKeyListener(new android.content.DialogInterface.OnKeyListener(){
6.  @Override
7.  public boolean onKey(DialogInterface dialog, int keyCode,KeyEvent event) {
8.    switch (keyCode) {
9.     case KeyEvent.KEYCODE_BACK:
10.     Log.i(TAG,"KEYCODE_BACK");
11.     return true;
12.    }
13.    return false;
14.  }
15.  });

复制代码

这样运行后,出错如下:

    1.  
    2.  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
    3.


    复制代码

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


    正确答案

      1.  
      2.  dialog.show();
      3.  dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD);
      4.


      复制代码

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




      总结:


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


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




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


      1.  
      2.  final AlertDialog dialog = new AlertDialog.Builder(mContext)
      3.  .setTitle(null)
      4.  .setMessage(message)
      5.  .setNeutralButton(R.string.ok, null)
      6.  .create();
      7.  dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
      8.  dialog.show();


      复制代码

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




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




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


      1.  
      2.  button.setOnClickListener(new View.OnClickListener() {
      3.  @Override
      4.  public void onClick(View v) {
      5.    getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD);
      6.  }
      7.  });


      复制代码

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

      1.  
      2.  @Override  
      3.  public void onAttachedToWindow() {
      4.    super.onAttachedToWindow();
      5.  }

      复制代码

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

        1.  
        2.  switch (keyCode) {
        3.  case KeyEvent.KEYCODE_HOME:
        4.    Log.i(TAG,"KEYCODE_HOME");
        5.    return true;
        6.  case KeyEvent.KEYCODE_BACK:
        7.    Log.i(TAG,"KEYCODE_BACK");
        8.    return true;
        9.  case KeyEvent.KEYCODE_CALL:
        10.    Log.i(TAG,"KEYCODE_CALL");
        11.    return true;
        12.  case KeyEvent.KEYCODE_SYM:
        13.    Log.i(TAG,"KEYCODE_SYM");
        14.    return true;
        15.  case KeyEvent.KEYCODE_VOLUME_DOWN:
        16.    Log.i(TAG,"KEYCODE_VOLUME_DOWN");
        17.    return true;
        18.  case KeyEvent.KEYCODE_VOLUME_UP:
        19.    Log.i(TAG,"KEYCODE_VOLUME_UP");
        20.    return true;
        21.  case KeyEvent.KEYCODE_STAR:
        22.    Log.i(TAG,"KEYCODE_STAR");
        23.    return true;
        24.  }


        复制代码

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




        更新如下:


        总结1:)的问题,有答案了,时间问题我就简单写写吧:


            从功能上来说,是一样的,区别在样式。


            如果你喜欢用Theme.Dialog去把一个Activity装饰成一个Dialog去显示,你会发现。


        在Androidmanifest.xml代码

          1.  
          2.  android:theme="@android:style/Theme.Dialog"
          3.


          复制代码

          背景是透明的。


          如果在

            1.  
            2.  setTheme(android.R.style.Theme_Dialog);
            3.


            复制代码

            背景则是黑色的。


            这是为什么呢?。。。我不知道。


            治标不治本的方法来了!若你在Activity重写onAttachedToWindow

              1.  
              2.  public void onAttachedToWindow() {
              3.  this.getWindow().setType(WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
              4.  super.onAttachedToWindow();
              5.  }
              6.


              复制代码

              那么出来的效果,就是透明背景的dialog了,当然前提是你需要实现屏蔽Home键。至于其中到底哪一代码导致样式改变呢,那就以后再去看源代码了~