如何控制android系统中NavigationBar 的显示与隐藏

时间:2022-12-12 14:21:15

我们使用的大多数android手机上的Home键,返回键以及menu键都是实体触摸感应按键。如果你用Google的Nexus4或Nexus5话,你会发现它们并没有实体按键或触摸感应按键,取而代之的是在屏幕的下方加了一个小黑条,在这个黑条上有3个按钮控件,这种设置无疑使得手机的外观的设计更加简约。但我遇到身边用Nexus 4手机的人都吐槽这种设计,原因很简单:好端端的屏幕,被划出一块区域用来显示3个按钮(如下图所示):Back, Home, Recent。并且它一直用在那里占用着。

如何控制android系统中NavigationBar 的显示与隐藏

在android源码中,那一块区域被叫做NavigationBar。同时,google在代码中也预留了标志,用来控制它的显示与隐藏。NavigationBar的显示与隐藏的控制是放在SystemU中的,具体的路径是:\frameworks\base\packages\SystemUI。对android4.0以上的手机而言,SystemUi包含两部分:StatusBar和NavigationBar。在SystemUI的工程下有一个类PhoneStatusBar.java,在该类中可以发现关于控制NavigationBar的相关代码:

在start()方法里可以看到NavigationBar是在那时候被添加进来,但只是添加,决定它显示还是隐藏是在后面控制的。

  1.  
    <span style="font-size:18px;">@Override
  2.  
    public void start() {
  3.  
    mDisplay = ((WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE))
  4.  
    .getDefaultDisplay();
  5.  
    updateDisplaySize();
  6.  
     
  7.  
    /// M: Support Smartbook Feature.
  8.  
    if (SIMHelper.isMediatekSmartBookSupport()) {
  9.  
    /// M: [ALPS01097705] Query the plug-in state as soon as possible.
  10.  
    mIsDisplayDevice = SIMHelper.isSmartBookPluggedIn(mContext);
  11.  
    Log.v(TAG, "start, mIsDisplayDevice=" + mIsDisplayDevice);
  12.  
    }
  13.  
     
  14.  
    super.start(); // calls createAndAddWindows()
  15.  
     
  16.  
    addNavigationBar();
  17.  
     
  18.  
    // Lastly, call to the icon policy to install/update all the icons.
  19.  
    mIconPolicy = new PhoneStatusBarPolicy(mContext);
  20.  
     
  21.  
    mHeadsUpObserver.onChange(true); // set up
  22.  
    if (ENABLE_HEADS_UP) {
  23.  
    mContext.getContentResolver().registerContentObserver(
  24.  
    Settings.Global.getUriFor(SETTING_HEADS_UP), true,
  25.  
    mHeadsUpObserver);
  26.  
    }
  27.  
    }</span>

其中的addNavigationBar()具体的实现方法如下:

  1.  
    <span style="font-size:18px;"> // For small-screen devices (read: phones) that lack hardware navigation buttons
  2.  
    private void addNavigationBar() {
  3.  
    if (DEBUG) Slog.v(TAG, "addNavigationBar: about to add " + mNavigationBarView);
  4.  
    if (mNavigationBarView == null) return;
  5.  
     
  6.  
    prepareNavigationBarView();
  7.  
     
  8.  
    mWindowManager.addView(mNavigationBarView, getNavigationBarLayoutParams());
  9.  
    }</span>

可以看到Navigationbar实际上windowmanager向window窗口里添加一个view。在调用addNavigationBar()方法之前会回调start()的父方法super.start()来判断是否要添加NavigationBar。在super.start()的调用父类方法里会调用createAndAddWindows(),该方法内会判断是否需要添加显示NavigationBar,然后决定是否要实例化NavigationBarView.

  1.  
    <span style="font-size:18px;">try {
  2.  
    boolean showNav = mWindowManagerService.hasNavigationBar();
  3.  
    if (DEBUG) Slog.v(TAG, "hasNavigationBar=" + showNav);
  4.  
    if (showNav) {
  5.  
    mNavigationBarView =
  6.  
    (NavigationBarView) View.inflate(context, R.layout.navigation_bar, null);
  7.  
     
  8.  
    mNavigationBarView.setDisabledFlags(mDisabled);
  9.  
    mNavigationBarView.setBar(this);
  10.  
    }
  11.  
    } catch (RemoteException ex) {
  12.  
    // no window manager? good luck with that
  13.  
    }</span>

WindowManagerService类实现了WindowManagerPolicy的接口,所以WindowManagerService会回调WindowManagerPolicy 的hasNavigationBar()接口,

  1.  
    <span style="font-size:18px;"> @Override
  2.  
    public boolean hasNavigationBar() {
  3.  
    return mPolicy.hasNavigationBar();
  4.  
    }</span>

Policy向下调用实际上调用的是PhoneWindowManager实现的hasNavigationBar方法,下面代码是PhoneWindowManager中的hasNavigationBar()方法。

  1.  
    <span style="font-size:18px;">// Use this instead of checking config_showNavigationBar so that it can be consistently
  2.  
    // overridden by qemu.hw.mainkeys in the emulator.
  3.  
    public boolean hasNavigationBar() {
  4.  
    return mHasNavigationBar;
  5.  
    }</span>

而mHasNavigationBar的赋值可以在PhoneWindowManager中的setInitialDisplaySize(Display display, int width, int height, int density)方法中找到,

  1.  
    <span style="font-size:18px;"> if (!mHasSystemNavBar) {
  2.  
    mHasNavigationBar = mContext.getResources().getBoolean(
  3.  
    com.android.internal.R.bool.config_showNavigationBar);
  4.  
    // Allow a system property to override this. Used by the emulator.
  5.  
    // See also hasNavigationBar().
  6.  
    String navBarOverride = SystemProperties.get("qemu.hw.mainkeys");
  7.  
    if (! "".equals(navBarOverride)) {
  8.  
    if (navBarOverride.equals("1")) mHasNavigationBar = false;
  9.  
    else if (navBarOverride.equals("0")) mHasNavigationBar = true;
  10.  
    }
  11.  
    } else {
  12.  
    mHasNavigationBar = false;
  13.  
    }</span>

从上面代码可以看到mHasNavigationBar的值的设定是由两处决定的:
1.首先从系统的资源文件中取设定值config_showNavigationBar, 这个值的设定的文件路径是frameworks/base/core/res/res/values/config.xml

  1.  
    <!-- Whether a software navigation bar should be shown. NOTE: in the future this may be
  2.  
    autodetected from the Configuration. -->
  3.  
    <bool name="config_showNavigationBar">false</bool>

2.然后系统要获取“qemu.hw.mainkeys”的值,这个值可能会覆盖上面获取到的mHasNavigationBar的值。如果“qemu.hw.mainkeys”获取的值不为空的话,不管值是true还是false,都要依据后面的情况来设定。

所以上面的两处设定共同决定了NavigationBar的显示与隐藏。

如何控制android系统中NavigationBar 的显示与隐藏的更多相关文章

  1. 控制input输入框中提示信息的显示和隐藏的方法

    在运用html+css+javascrpt进行页面制作时,我们往往会遇到一些影响用户体验,而又容易被我们忽视的小细节.比如,input输入框中的提示信息,怎样才能根据对象获得和失去焦点而实现其显示和隐 ...

  2. Android系统中设置TextView的行间距(非行高)

    Android系统中TextView默认显示中文时会比较紧凑,不是很美观.为了让每行保持一定的行间距,可以设置属性android:lineSpacingExtra或android:lineSpacin ...

  3. Android系统中的广播(Broadcast)机制简要介绍和学习计划

    在Android系统中,广播(Broadcast)是在组件之间传播数据(Intent)的一种机制:这些组件甚至是可以位于不同的进程中,这样它就像Binder机制一样,起到进程间通信的作用:本文通过一个 ...

  4. Android系统中 setprop&comma;getprop&comma;watchprops命令的使用

    如:在frameworks/opt/net/ims/src/java/com/android/ims/ImsManager.java if (SystemProperties.get("pe ...

  5. 用adb pull命令从android系统中读取文件失败的原因及解决办法

    问题:使用adb pull命令从android系统中读取文件失败.显示:Permission denied   原因:是由于文件权限原因引起.       使用ls -l命令查看android系统中的 ...

  6. Android系统中setprop&comma;getprop&comma;watchprops命令的使用&lpar;转载&rpar;

    转自:http://blog.csdn.net/yao_guet/article/details/6531241 在android系统中,有一些初始化的配置文件,例如: /init.rc /defau ...

  7. 在Android系统中修改Android&period;mk使其同时编译rgb2565和rgb2888(向out&sol;host&sol;linux-x86&sol;bin&sol;下新增加一个工具命令)【转】

    本文转载自:http://blog.csdn.net/mu0206mu/article/details/7514559 在Android系统中修改android.mk使其同时编译rgb2565和rgb ...

  8. AIDL在android系统中的作用

    AIDL,Android Interface definition language的缩写,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口.最近看了下AIDL在A ...

  9. Android系统中的6种模式

    Android系统中的6种模式 1:一般启动模式(normal mode):    功能是正常启动手机,方法为关机状态下按电源键启动. 2:安全模式(safe mode):    此模式和正常启动一样 ...

随机推荐

  1. Go的String转码包

    https://github.com/qiniu/iconv https://github.com/djimenez/iconv-go 这是与go不相干的转码包:https://github.com/ ...

  2. AtomicInteger源码分析——基于CAS的乐观锁实

    1. 悲观锁与乐观锁 我们都知道,cpu是时分复用的,也就是把cpu的时间片,分配给不同的thread/process轮流执行,时间片与时间片之间,需要进行cpu切换,也就是会发生进程的切换.切换涉及 ...

  3. 【转】JDBC学习笔记&lpar;10&rpar;——调用函数&amp&semi;存储过程

    转自:http://www.cnblogs.com/ysw-go/ 如何使用JDBC调用存储在数据库中的函数或存储过程: * 1.通过COnnection对象的prepareCall()方法创建一个C ...

  4. Java8-2-Lambda表达式实战-一句话实现Map中按照Value排序

    在上一讲中, 我们着重的讲了表达式的一些基础知识和基本的使用, 今天我们来实战一把, 对Map的Value值排序进行简化. 在以前的思路我们的做法如下: /** * * Map根据value排序; * ...

  5. High Performance Networking in Google Chrome

    小结: 1. 小文件存储于一个文件中: 在内部,磁盘缓存(disk cache)实现了它自己的一组数据结构, 它们被存储在一个单独的缓存目录里.其中有索引文件(在浏览器启动时加载到内存中),数据文件( ...

  6. 《Java大学教程》--第1章 步入Java世界

    1.2 软件:用于计算机执行的指令的集合称之为程序(program).单个程序或者一组程序称之为软件(software)1.3 编译:计算机的语言称为机器码(machine code).用编译器(co ...

  7. 微信公众号使用LocalStorage解决返回缓存问题

    在开发微信公众号上应用程序时,遇到了一个普遍的问题,从A页跳转到B页后,再由B页跳转回A页,A要要保持跟跳转前一致,通过LocalStorage可以解决. LocalStorage,很好的解决了返回的 ...

  8. es6基础&lpar;7&rpar;--函数扩展

    { //有默认值的后面如果有参数必须要有默认值 function test(x,y="world"){ console.log(x,y) } test('hello');//hel ...

  9. Luogu2161 &lbrack;SHOI2009&rsqb;会场预约-线段树

    Solution 线段树维护 sum 表示区间内预约个数, L 表示区间最左边的预约, R 表示区间最右边的预约. $pushup$ 就是这样 : void up(int nd) { sum[nd] ...

  10. 如何查看DNS&quest;

    查看dns服务ip:    命令: ipconfig/all  查看dns缓存:    命令:ipconfig/displaydns     强制更新缓存:    命令:ipconfig /flush ...