相信在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键。至于其中到底哪一代码导致样式改变呢,那就以后再去看源代码了~