Android导航栏隐藏与浮现(一)

Posted by Codeboy on October 22, 2015

Android M已经发布了很久了,很多新的特性也非常的吸引人,比如Doze模式可以使导航时间更长,刷到nexus5上,体验了一下确实不错。但是导航栏(虚拟按键)一直存在,感觉不是特别的爽。今天我们就从android M(6.0)进行源码的修改,使nexus5能够方便的进行导航栏的隐藏与恢复。

修改源码前,看到了CSDN上的文章(http://blog.csdn.net/way_ping_li/article/details/45727335,记为文章A),写的很好,但是写的有些省略,整体的操作也较为复杂,本文的操作步骤如下:

① 下载并编译源码
② 长点击隐藏导航栏
③ 上滑显示导航栏
④ 编译部分修改代码,重新生成system.img

文章A的整体思路是在导航栏上添加了一个图标按钮,点击后隐藏导航栏,上滑显示导航栏。首先说一下隐藏导航栏,感觉官方的导航栏还是很完美的,所以不打算添加任何元素,这里对任务键(虚拟正方形按键)进行长点击操作进行修改;之后通过上滑来显示导航栏,文章A的思路是通过各种系统内的很多回调与消息的传递完成的,改动幅度蛮大的,自己尝试按照文章中的进行修改,需要对文章A中提到的部分进行修改外,还需对部分aidl以及与此相关的类进行修改等,比较的繁琐,当然最后实现了上滑显示,本文将通过广播来进行上滑操作的传递。

下载并编译源码

源码的下载编译可以参考google官方教程https://source.android.com/source/downloading.html,本文编译的源码分支android-6.0.1_r50。

nexus5源码下载后需要下载驱动https://developers.google.com/android/nexus/drivers,解压到根目录后执行,执行后产生vendor目录,之后编译代码

源码编译完成后,之后的framework的修改不用再次重新编译,只需使用mmm命令编译部分模块即可,最后使用make snod生成system.img,刷进手机即可。

修改framework后生成的system.img,在进行刷机时仅仅刷进system.img即可,首次刷机尽量把userdata.img、 boot.img都进行刷入

长点击隐藏导航栏

虚拟按键有3个,考虑到返回键一般有一定的作用,于是修改任务键(虚拟正方形按键)的长点击事件,使其在长点击后可以隐藏导航栏。

./frameworks/base/packages/SystemUI/src/com/android/systemui/statusbar/phone/PhoneStatusBar.java

private void prepareNavigationBarView() {
    mNavigationBarView.reorient();

    mNavigationBarView.getRecentsButton().setOnClickListener(mRecentsClickListener);
    mNavigationBarView.getRecentsButton().setOnTouchListener(mRecentsPreloadOnTouchListener);
    mNavigationBarView.getRecentsButton().setLongClickable(true);
    //去除长点击操作
    //mNavigationBarView.getRecentsButton().setOnLongClickListener(mLongPressBackRecentsListener);
    mNavigationBarView.getBackButton().setLongClickable(true);
    mNavigationBarView.getBackButton().setOnLongClickListener(mLongPressBackRecentsListener);
    mNavigationBarView.getHomeButton().setOnTouchListener(mHomeActionListener);
    mNavigationBarView.getHomeButton().setOnLongClickListener(mLongPressHomeListener);
    //添加长点击操作,长点击进行导航栏的删除
    mNavigationBarView.getRecentsButton().setOnLongClickListener(new View.OnLongClickListener() {
        @Override
        public boolean onLongClick(View view) {
            mWindowManager.removeView(mNavigationBarView);
            mNavigationBarView = null ;
            //isNavigationShow用于记录当前导航栏状态,定义在PhoneStatusBar的全局变量中
			isNavigationShow = false ;
            Log.d("===>LYD", "remove navigation bar");

            return true;
        }
    });
    mAssistManager.onConfigurationChanged();
}

此时进行长点击任务键将隐藏导航栏。

上滑显示导航栏

上滑显示导航栏借鉴了文章A中的一些技巧,在系统上滑时会回调./frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager.java的修改主要是实现onSwipeFromBottom(竖屏时)和onSwipeFromRight(横屏时)两个接口,在此两接口内发送广播。

./frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager.java

 // monitor for system gestures
    mSystemGestures = new SystemGesturesPointerEventListener(context,
            new SystemGesturesPointerEventListener.Callbacks() {
                @Override
                public void onSwipeFromTop() {
                    if (mStatusBar != null) {
                        requestTransientBars(mStatusBar);
                    }    
                }    
                @Override
                public void onSwipeFromBottom() {
                    if (mNavigationBar != null && mNavigationBarOnBottom) {
                        requestTransientBars(mNavigationBar);
                    }    
                    //开始发送广播
                    Intent intent = new Intent();
                    intent.setAction("LYD_SHOW_NAVIGATION_BAR");
                    mContext.sendBroadcast(intent);
                }    
                @Override
                public void onSwipeFromRight() {
                    if (mNavigationBar != null && !mNavigationBarOnBottom) {
                        requestTransientBars(mNavigationBar);
                    }    
                    //开始发送广播
                    Intent intent = new Intent();
                    intent.setAction("LYD_SHOW_NAVIGATION_BAR");
                    mContext.sendBroadcast(intent);
                }    
                //省略后续代码

下面我们需要做的是在之前的PhoneStatusBar中对广播进行处理,将状态栏添加到windowManager中。

 @Override
public void start() {
//省略代码,在此方法末尾动态注册广播监听器

    IntentFilter filter = new IntentFilter();
    filter.addAction("LYD_SHOW_NAVIGATION_BAR");
    mContext.registerReceiver(navBarBroadcastReceiver, filter);
}

private BroadcastReceiver navBarBroadcastReceiver = new LydShowNavigationBarBroadcast();

private static boolean isNavigationShow = true ;

//自定义广播
class LydShowNavigationBarBroadcast extends  BroadcastReceiver{
    public LydShowNavigationBarBroadcast() {
        super();
    }

    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d("====>LYD", "receiver show navigation bar broadcast");

		//防止多次被添加
        if(isNavigationShow){
            return ;
        }

        showNavigationBar();
    }

    @Override
    public IBinder peekService(Context myContext, Intent service) {
        return super.peekService(myContext, service);
    }
}  

//展示导航栏
public void showNavigationBar() {
    mNavigationBarView =(NavigationBarView) View.inflate(mContext, R.layout.navigation_bar, null);
    mNavigationBarView.setBar(this);
    prepareNavigationBarView();
    addNavigationBar();
    isNavigationShow = true;

    //防止在桌面时上拉出导航栏时,导航栏背景为黑色
    mNavigationBarView.setBackgroundColor(Color.TRANSPARENT);
    Log.d("===>LYD", "show navigation");
}

注意导入android.graphics.Color类

代码到此就修改完了,下面我们只需要对相应模块进行编译就行了。

编译部分修改代码,重新生成system.img

  • ①切换到源代码目录,运行

      source build/envsetup.sh  //初始化环境变量
      lunch //切换编译平台
    
  • ②编译PhoneWindowManager.java所在模块(core)

      mmm ./frameworks/base/services/core/
      mmm ./frameworks/base/services/
    
  • ③编译PhoneStatusBar.java所在模块(SystemUI)

      mmm ./frameworks/base/packages/SystemUI/
    
  • ④生成system.img.

       make snod
    
  • ⑤刷入system.img

      adb reboot bootloader
      fastboot flash system system.img
    

这样一个定制过导航栏的nexus5(hammerhead)系统已经制作完成了。

效果

解决了导航栏的隐藏与浮现,终于可以不用一直看着不搭配的导航栏了。下面是隐藏前后QQ音乐的截图:

后记

能够给该功能在设置中加一个开关就完美了,快来查看Android导航栏隐藏与浮现(二)吧。

由于笔者一般都是使用手机都是竖屏的,为了减少误操作,所以仅仅上滑时(swipeFromBottom)才显示导航栏,从右侧滑动(swipeFromRight)不触发操作。

刷机有风险,大家需谨慎,本文仅仅是提供一种思想,刷机造成的问题与本文以及作者无关。

如有任何知识产权、版权问题或理论错误,还请指正。

转载请注明原作者及以上信息。