标签: Android 隐藏导航栏

Android 7.0隐藏导航栏、状态栏的一些方法

Android 7.0隐藏导航栏和状态栏的一些方法

近期笔者在某款app时,需求要对导航栏(返回,home键,recent键的那部分)进行隐藏,此外还要做相关操作的屏蔽避免导航栏恢复显示。原本的方案是通过发广播给system ui,在system ui做remove导航栏的操作实现。后来架构设计阶段评估该方案对于外部模块的耦合太强,因此希望对方案进行优化,要求只在app自身做改动就可以实现。经过一番折腾完成之后,特将一些心得整理出来供参考。

 

心得1. 如何在不拉伸主要显示区域情况下,隐藏导航栏和状态栏。

笔者原本采用如下代码实现隐藏效果:

View decorView = activity.getWindow().getDecorView();

decorView.setSystemUiVisibility(

View.SYSTEM_UI_FLAG_LAYOUT_STABLE                        |View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION

|View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

|View.SYSTEM_UI_FLAG_HIDE_NAVIGATION

| View.SYSTEM_UI_FLAG_FULLSCREEN

|View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY);

实践后发现虽然能实现隐藏状态栏和导航栏,但会出现原有显示区域的布局被拉伸的情况。这与我们的要求不相符,详细研究View的setSystemUiVisibility方法可以传入的flag的含义和用法如下:

1. View.SYSTEM_UI_FLAG_VISIBLE:显示状态栏,Activity不全屏显示(恢复到有状态的正常情况)。

2. View.INVISIBLE:隐藏状态栏,同时Activity会伸展全屏显示。

3. View.SYSTEM_UI_FLAG_FULLSCREEN:Activity全屏显示,且状态栏被隐藏覆盖掉。

4. View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN:Activity全屏显示,但状态栏不会被隐藏覆盖,状态栏依然可见,Activity顶端布局部分会被状态遮住。

5. View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION:效果同View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

6. View.SYSTEM_UI_LAYOUT_FLAGS:效果同View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

7. View.SYSTEM_UI_FLAG_HIDE_NAVIGATION:隐藏虚拟按键(导航栏)。有些手机会用虚拟按键来代替物理按键。

8. View.SYSTEM_UI_FLAG_LOW_PROFILE:状态栏显示处于低能显示状态(low profile模式),状态栏上一些图标显示会被隐藏。

9.View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY:粘性沉浸模式,向内滑动的操作会让系统栏临时显示,并处于半透明的状态。此时没有标签会被清除,系统UI可见性监听器也不会被触发。如果用户没有进行操作,系统栏会在一段时间内自动隐藏。

10.View.SYSTEM_UI_FLAG_IMMERSIVE:非粘性沉浸模式,它是基于其他设置过的标签(SYSTEM_UI_FLAG_HIDE_NAVIGATION和SYSTEM_UI_FLAG_FULLSCREEN)来隐藏系统栏的。当用户向内滑动,系统栏重新显示并保持可见。

 

通过上面的分析可以发现,带有LAYOUT的flag,表示Activity会延伸到原本属于状态栏或导航栏的位置,所以导致了布局拉伸;在要求隐藏状态栏的地方,只需要用View.SYSTEM_UI_FLAG_FULLSCREEN即可。这时候,导航栏存在,状态栏隐藏并且不会拉伸。

若要求导航栏和状态栏都隐藏,采用View.SYSTEM_UI_FLAG_FULLSCREEN和 View.SYSTEM_UI_FLAG_HIDE_NAVIGATION配合即可。也可以使用View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY和View.SYSTEM_UI_FLAG_HIDE_NAVIGATION配合,效果是一样的。

心得2. 如何屏蔽触摸手势滑动调出导航栏。

上述的设置还是无法解决点击屏幕导航栏恢复的问题。经过查找资料了解到滑动屏幕主要会被PhoneWindowManager的SystemGesturesPointerEventListener所截获,而且研究发现,笔者当前采用的mtk方案在framework有针对设置了FULL_SCREEN的界面无法禁止状态栏下拉的问题,做了一些扩展,包括在View类中增加了public static final int SYSTEM_UI_FLAG_IMMERSIVE_GESTURE_ISOLATED = 0x00002000; 并在SystemGesturesPointerEventListener的onSwipeFromTop,onSwipeFromBottom等方法里添加isGestureIsolated()方法判断,isGestureIsolated代码如下:

1.   private boolean isGestureIsolated() {

2.  +                        WindowState win = mFocusedWindow != null ? mFocusedWindow : mTopFullscreenOpaqueWindowState;

3.  +                        if (win != null && (win.getSystemUiVisibility() & View.SYSTEM_UI_FLAG_IMMERSIVE_GESTURE_ISOLATED) != 0)

4.  +                            return true;

5.  +                        else

6.  +                            return false;

7.  +                    }

因此我们只需要在setSystemUiVisibility中加上SYSTEM_UI_FLAG_IMMERSIVE_GESTURE_ISOLATED就可以屏蔽在屏幕上的滑动了。

注:该方法仅针对MTK手机平台适用,若是高通平台,则需要framework自行扩展类似接口。

 

心得3. 如何在对话框调起时不显示状态栏。

setSystemUiVisibility隐藏状态栏后,弹出Dialog或输入法键盘时,状态栏又会跑出来.

在请教了资深开发者之后发现原因,如下是framework中Diaolgshow的处理:

public void show() {

……

WindowManager.LayoutParams l = mWindow.getAttributes();
……

mWindowManager.addView(mDecor, l);
……

}

dialog是在窗体addview,并且沿用了原窗体的参数.

因此如果原窗体没有设置full_screen的flag,则addview刷新窗体时,也是非full_screen而把状态栏调出来.

(输入法键盘的显示也是类似情况)

 

解决方案:如果要隐藏状态栏并且避免dialog弹出时状态栏跑出来,需要改变Window的参数即可:

getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN)

Android 沉浸式状态栏与隐藏导航栏

Android 沉浸式状态栏与隐藏导航栏

1 前言
一般我们在Android的APP开发中,APP的界面如下:

%title插图%num
可以看到,有状态栏、ActionBar(ToolBar)、导航栏等,一般来说,APP实现沉浸式有三种需求:沉浸式状态栏,隐藏导航栏,APP全屏
沉浸式状态栏是指状态栏与ActionBar颜色相匹配,
隐藏导航栏不用多说,就是将导航栏隐藏,去掉下面的黑条。
APP全屏是指将状态栏与导航栏都隐藏,例如很多游戏界面,都是APP全屏。
所以,在做这一步时,关键要问清楚产品狗的需求,免得白费功夫。
下面,分别来介绍这三种方式的实现。

2 沉浸式状态栏
沉浸式状态栏效果一般如下:
顺便在网上找的图
%title插图%num
关于沉浸式状态栏网上的方案很多,比如android 5.0 以上的MD设计,或者修改activiyty的window的setStatusBarColor()方法,设置颜色。需要说明一点的时,沉浸式状态栏只对API19以上有效。
这里我依然采用的是设置Activity的Window设置setStatusBarColor()的方法。代码如下:

/**
* 设置状态栏的颜色
*/
@TargetApi(Build.VERSION_CODES.KITKAT)
public static void statusBarTintColor(Activity activity, int color) {
// 代表 5.0 及以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
activity.getWindow().setStatusBarColor(color);
return;
}

// versionCode > 4.4 and versionCode < 5.0
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT && Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
//透明状态栏
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
ViewGroup androidContainer = (ViewGroup) activity.findViewById(android.R.id.content);
// 留出高度 setFitsSystemWindows true代表会调整布局,会把状态栏的高度留出来
View contentView = androidContainer.getChildAt(0);
if (contentView != null) {
contentView.setFitsSystemWindows(true);
}
// 在原来的位置上添加一个状态栏
View statusBarView = createStatusBarView(activity);
androidContainer.addView(statusBarView, 0);
statusBarView.setBackgroundColor(color);
}
}

/**
* 创建一个需要填充statusBarView
*/
private static View createStatusBarView(Activity activity) {
View statusBarView = new View(activity);
ViewGroup.LayoutParams statusBarParams = new ViewGroup.LayoutParams(
ViewGroup.LayoutParams.MATCH_PARENT, getStatusBarHeight(activity));
statusBarView.setLayoutParams(statusBarParams);
return statusBarView;
}

/**
* 获取状态栏的高度
*/
public static int getStatusBarHeight(Context context) {
int result = 0;
int resourceId = context.getResources().getIdentifier(“status_bar_height”, “dimen”, “android”);
if (resourceId > 0) {
result = context.getResources().getDimensionPixelSize(resourceId);
}
return result;
}

3 隐藏导航栏
隐藏导航栏就是使用了UI Flag

/**
*
* @param activity
* @param
*/
public static void setNavigationBar(Activity activity,int visible){
View decorView = activity.getWindow().getDecorView();
//显示NavigationBar
if (View.GONE == visible){
int option = SYSTEM_UI_FLAG_HIDE_NAVIGATION;
decorView.setSystemUiVisibility(option);
}
}

4 APP全屏
这里的APP全屏又分为隐藏状态栏与ActionBar,与隐藏导航栏,状态栏。
隐藏状态栏:

/**
* 设置Activity的statusBar隐藏
* @param activity
*/
public static void statusBarHide(Activity activity){
// 代表 5.0 及以上
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
View decorView = activity.getWindow().getDecorView();
int option = View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN;
decorView.setSystemUiVisibility(option);
activity.getWindow().setStatusBarColor(Color.TRANSPARENT);
ActionBar actionBar = activity.getActionBar();
actionBar.hide();
return;
}

// versionCode > 4.4 and versionCode < 5.0
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT && Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
}

}

效果如下:

%title插图%num
这里先调用getWindow().getDecorView()方法获取到了当前界面的DecorView,然后调用它的setSystemUiVisibility()方法来设置系统UI元素的可见性。其中,SYSTEM_UI_FLAG_FULLSCREEN表示全屏的意思,也就是会将状态栏隐藏。另外,根据Android的设计建议,ActionBar是不应该独立于状态栏而单独显示的,因此状态栏如果隐藏了,我们同时也需要调用ActionBar的hide()方法将ActionBar也进行隐藏。

隐藏导航栏,状态栏:
一般游戏需要这种界面,代码如下:
在Activity的onWindowFocusChanged()中去设置界面完全全屏。

/**
* 导航栏,状态栏隐藏
* @param activity
*/
public static void NavigationBarStatusBar(Activity activity,boolean hasFocus){
if (hasFocus && Build.VERSION.SDK_INT >= 19) {
View decorView = activity.getWindow().getDecorView();
decorView.setSystemUiVisibility(
View.SYSTEM_UI_FLAG_LAYOUT_STABLE
| View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
| View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_FULLSCREEN
| View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY);
}
}

效果如下:

%title插图%num
可以看到,界面默认情况下是全屏的,状态栏和导航栏都不会显示。而当我们需要用到状态栏或导航栏时,只需要在屏幕顶部向下拉,或者在屏幕右侧向左拉,状态栏和导航栏就会显示出来,此时界面上任何元素的显示或大小都不会受影响。过一段时间后如果没有任何操作,状态栏和导航栏又会自动隐藏起来,重新回到全屏状态。

透明状态栏,导航栏:
另外,通过设置UI Flag,可以让导航栏,状态栏都透明化。

/**
* 导航栏,状态栏透明
* @param activity
*/
public static void setNavigationBarStatusBarTranslucent(Activity activity){
if (Build.VERSION.SDK_INT >= 21) {
View decorView = activity.getWindow().getDecorView();
int option = View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
| View.SYSTEM_UI_FLAG_LAYOUT_STABLE;
decorView.setSystemUiVisibility(option);
activity.getWindow().setNavigationBarColor(Color.TRANSPARENT);
activity.getWindow().setStatusBarColor(Color.TRANSPARENT);
}
ActionBar actionBar = activity.getActionBar();
actionBar.hide();
}

效果如下:

%title插图%num

友情链接: SITEMAP | 旋风加速器官网 | 旋风软件中心 | textarea | 黑洞加速器 | jiaohess | 老王加速器 | 烧饼哥加速器 | 小蓝鸟 | tiktok加速器 | 旋风加速度器 | 旋风加速 | quickq加速器 | 飞驰加速器 | 飞鸟加速器 | 狗急加速器 | hammer加速器 | trafficace | 原子加速器 | 葫芦加速器 | 麦旋风 | 油管加速器 | anycastly | INS加速器 | INS加速器免费版 | 免费vqn加速外网 | 旋风加速器 | 快橙加速器 | 啊哈加速器 | 迷雾通 | 优途加速器 | 海外播 | 坚果加速器 | 海外vqn加速 | 蘑菇加速器 | 毛豆加速器 | 接码平台 | 接码S | 西柚加速器 | 快柠檬加速器 | 黑洞加速 | falemon | 快橙加速器 | anycast加速器 | ibaidu | moneytreeblog | 坚果加速器 | 派币加速器 | 飞鸟加速器 | 毛豆APP | PIKPAK | 安卓vqn免费 | 一元机场加速器 | 一元机场 | 老王加速器 | 黑洞加速器 | 白石山 | 小牛加速器 | 黑洞加速 | 迷雾通官网 | 迷雾通 | 迷雾通加速器 | 十大免费加速神器 | 猎豹加速器 | 蚂蚁加速器 | 坚果加速器 | 黑洞加速 | 银河加速器 | 猎豹加速器 | 海鸥加速器 | 芒果加速器 | 小牛加速器 | 极光加速器 | 黑洞加速 | movabletype中文网 | 猎豹加速器官网 | 烧饼哥加速器官网 | 旋风加速器度器 | 哔咔漫画 | PicACG | 雷霆加速