Android 4.0以上设备的虚拟按键中menu键的显示问题

在 Android 4.0以后,google添加了虚拟导航键来替换实体键,同时按键由原来的四大天王改为back、home、recent三个。研究源码可以发现是否显示菜单键是在 Window初始化的布局中判断的,也即PhoneWindow的generateLayout函数。

android 5.1 以下该函数部分代码是这样实现的:

[java]  view plain  copy
  1. final Context context = getContext();    
  2. final int targetSdk = context.getApplicationInfo().targetSdkVersion;    
  3. final boolean targetPreHoneycomb = targetSdk < android.os.Build.VERSION_CODES.HONEYCOMB;    
  4. final boolean targetPreIcs = targetSdk < android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH;    
  5. final boolean targetHcNeedsOptions = context.getResources().getBoolean(    
  6.         com.android.internal.R.bool.target_honeycomb_needs_options_menu);    
  7. final boolean noActionBar = !hasFeature(FEATURE_ACTION_BAR) || hasFeature(FEATURE_NO_TITLE);    
  8.     
  9. if (targetPreHoneycomb || (targetPreIcs && targetHcNeedsOptions && noActionBar)) {    
  10.     addFlags(WindowManager.LayoutParams.FLAG_NEEDS_MENU_KEY);    
  11. else {    
  12.     clearFlags(WindowManager.LayoutParams.FLAG_NEEDS_MENU_KEY);   

但是再看Android 6.0 的源码,PhoneWindow的generateLayout函数却稍微改变了点:

[java]  view plain  copy
  1. final Context context = getContext();    
  2. final int targetSdk = context.getApplicationInfo().targetSdkVersion;    
  3. final boolean targetPreHoneycomb = targetSdk < android.os.Build.VERSION_CODES.HONEYCOMB;    
  4. final boolean targetPreIcs = targetSdk < android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH;    
  5. final boolean targetPreL = targetSdk < android.os.Build.VERSION_CODES.LOLLIPOP;    
  6. final boolean targetHcNeedsOptions = context.getResources().getBoolean(    
  7.         R.bool.target_honeycomb_needs_options_menu);    
  8. final boolean noActionBar = !hasFeature(FEATURE_ACTION_BAR) || hasFeature(FEATURE_NO_TITLE);    
  9.     
  10. if (targetPreHoneycomb || (targetPreIcs && targetHcNeedsOptions && noActionBar)) {    
  11.     setNeedsMenuKey(WindowManager.LayoutParams.NEEDS_MENU_SET_TRUE);    
  12. else {    
  13.     setNeedsMenuKey(WindowManager.LayoutParams.NEEDS_MENU_SET_FALSE);    
  14. }  

而Android 5.1.1的generateLayout函数的那部分代码跟6.0基本上是一样的,所以暂时只看Android 6.0的代码。

在Android 6.0的Window类中多了一个方法,setNeedMenuKey 函数,该函数的作用就是设置是否显示虚拟菜单键,在Android 5.1.1之前是否显示菜单键是WindowManager.LayoutParams 中的一个flags,而在Android 5.1.1及以后,google把这个标记为改到了WindowManager.LayoutParams类中的needsMenuKey 字段去了,可以通过setNeedMenuKey 方法来修改。

说了这么多,也大概明白了怎么回事了,下面就给出一个通用的解决方法,利用反射来完成:

[java]  view plain  copy
  1. private void showNagivationMenuKey() {    
  2.   if (Build.VERSION.SDK_INT < Build.VERSION_CODES.ICE_CREAM_SANDWICH) {    
  3.       return;    
  4.   }    
  5.   if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.LOLLIPOP) {    
  6.       try {    
  7.           int flags = WindowManager.LayoutParams.class.getField("FLAG_NEEDS_MENU_KEY").getInt(null);    
  8.           getWindow().addFlags(flags);    
  9.       } catch (IllegalAccessException e) {    
  10.           e.printStackTrace();    
  11.       } catch (NoSuchFieldException e) {    
  12.           e.printStackTrace();    
  13.       }    
  14.   } else {    
  15.       try {    
  16.           Method setNeedsMenuKey = Window.class.getDeclaredMethod("setNeedsMenuKey"int.class);    
  17.           setNeedsMenuKey.setAccessible(true);    
  18.           int value = WindowManager.LayoutParams.class.getField("NEEDS_MENU_SET_TRUE").getInt(null);    
  19.           setNeedsMenuKey.invoke(getWindow(), value);    
  20.       } catch (NoSuchMethodException e) {    
  21.           e.printStackTrace();    
  22.       } catch (NoSuchFieldException e) {    
  23.           e.printStackTrace();    
  24.       } catch (IllegalAccessException e) {    
  25.           e.printStackTrace();    
  26.       } catch (InvocationTargetException e) {    
  27.           e.printStackTrace();    
  28.       }    
  29.   }    

该函数就可以强行设置在虚拟导航栏中显示菜单按钮,调用位置的话,我是放在setContentView函数之后使用的,亲测有效:
[java]  view plain  copy
  1. setContentView(R.layout.activity_main);    
  2. showNavigationMenuKey();   

在Activity中重写onKeyDown方法来监听Menu键:

[java]  view plain  copy
  1. public boolean onKeyDown(int keyCode, KeyEvent event) {  
  2.         if (keyCode == KeyEvent.KEYCODE_MENU) {  
  3.             // TODO: 处理菜单键的点击事件  
  4.             return true;  
  5.         }  
  6.         return super.onKeyDown(keyCode, event);  
  7. }  
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值