translucentsystembar的最佳实践

weizhxa weizhxa     2022-09-03     520

关键词:

转自:http://www.jianshu.com/p/0acc12c29c1b

近几天准备抽空总结Android一些系统UI的实践使用,于是开始动手建了一个库 AndroidSystemUiTraining ,边撸代码边写总结

今天开写第一篇,对 Translucent System Bar 的实践做一些总结。说起 Translucent System Bar 的特性,可能有些朋友还比较陌生,这里做一下简单的介绍。

技术分享
Android 4.3豌豆荚

看上图,Android 4.4之前,即使我们打开手机app,我们还总是能看到系统顶部那条黑乎乎的通知栏,这样会使得app稍显突兀。于是Android 4.4开始,便引入了Translucent System Bar的系特性,用于弥补系统通知栏突兀之处。(估计也是向ios学习,因为ios一大早就有这个特性)。我们先来看看 Translucent System Bar 新特性引入后,发生了什么样的变化。下面截取了 中华万年历的天气预报界面 和 QQ音乐主界面 的效果(两个界面的效果实现 Translucent System Bar 的方式有些区别,下文会细讲)

技术分享
中华万年历
技术分享
QQ音乐

可以看到,系统的通知栏和app界面融为一体,妈妈再也不用面对黑乎乎的通知栏了。有关 Translucent System Bar 的特性就暂且介绍到此。

工程简介

先简单介绍一下工程的结构,核心部分已经圈出,待我逐一讲解

技术分享
工程结构
  • 主要的操作都在style.xml 和 AndroidManifest.xml 中,Activity里面没有任何涉及到Translucent System Bar设置的代码,所以可以忽略不看。

  • ColorTranslucentBarActivity 和 ImageTranslucentBarActivity 分别用于展示两种不同实现方式的效果

  • 要在Activity中使用 Translucent System Bar 特性,首先需要到AndroidManifest中为指定的Activity设置Theme。但是需要注意的是,我们不能直接在values/style.xml直接去自定义 Translucet System Bar 的Theme,因为改特性仅兼容 Android 4.4 开始的平台,所以直接在values/style.xml声明引入,工程会报错。有些开发者朋友会在代码中去判断SDK的版本,然后再用代码设置Theme。虽然同样可以实现效果,但个人并不推崇这种做法。我所采取的方法则是建立多个SDK版本的values文件夹,系统会根据SDK的版本选择合适的Theme进行设置。大家可以看到上面我的工程里面有values、values-v19、values-v21。

第一种方式

第一种方式,需要做下面三步设置

1、在values、values-v19、values-v21的style.xml都设置一个 Translucent System Bar 风格的Theme

values/style.xml

<style name="ImageTranslucentTheme" parent="AppTheme">
    <!--在Android 4.4之前的版本上运行,直接跟随系统主题-->
</style>

values-v19/style.xml

<style name="ImageTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="android:windowTranslucentStatus">true</item>
    <item name="android:windowTranslucentNavigation">true</item>
</style>

values-v21/style.xml

<style name="ImageTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="android:windowTranslucentStatus">false</item>
    <item name="android:windowTranslucentNavigation">true</item>
    <!--Android 5.x开始需要把颜色设置透明,否则导航栏会呈现系统默认的浅灰色-->
    <item name="android:statusBarColor">@android:color/transparent</item>
</style>

上面需要注意的地方是,无论你在哪个SDK版本的values目录下,设置了主题,都应该在最基本的values下设置一个同名的主题。这样才能确保你的app能够正常运行在 Android 4.4 以下的设备。否则,肯定会报找不到Theme的错误。

2、在AndroidManifest.xml中对指定Activity的theme进行设置

<activity
    android:name=".ui.ImageTranslucentBarActivity"
    android:label="@string/image_translucent_bar"
    android:theme="@style/ImageTranslucentTheme" />

3、在Activity的布局文件中设置背景图片,同时,需要把android:fitsSystemWindows设置为true

activity_image_translucent_bar.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@mipmap/env_bg"
    android:fitsSystemWindows="true">

</RelativeLayout>

到此,第一种实现方式完成,大家可以看看下面的效果

技术分享
ImageTranslucentTheme效果

就跟中华万年历的天气预报效果界面一样,系统的整个导航栏都融入了app的界面中,背景图片填满了整个屏幕,看起来舒服很多。这里还有一个android:fitsSystemWindows设置需要注意的地方,后面会在细讲。接下来看第二种实现。

方式二

相比中华万年历,QQ音乐采用的是另外一种实现的方式,它将app的Tab栏和系统导航栏分开来设置。

技术分享
QQ音乐效果风格

由于它的Tab栏是纯色的,所以只要把系统通知栏的颜色设置和Tab栏的颜色一致即可,实现上相比方法一要简单很多。同样要到不同SDK版本的values下,创建一个同名的theme,在values-v21下,需要设置系统导航栏的颜色:

values-v21/style.xml

<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="android:windowTranslucentStatus">false</item>
    <item name="android:windowTranslucentNavigation">true</item>
    <item name="android:statusBarColor">@color/color_31c27c</item>
</style>

再到ColorTranslucentBarActivity的布局文件activity_color_translucent_bar.xml中设置Tab栏的颜色

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    android:orientation="vertical">

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="55dp"
        android:background="@color/color_31c27c">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="QQ Music"
            android:textColor="@android:color/white"
            android:textSize="20sp" />

    </RelativeLayout>
</LinearLayout>

到此,我们就可以得到和QQ音乐主界面一样的效果了。

技术分享
QQ音乐界面实现效果

到此,就大体介绍完了 Translucent System Bar 的两种实现方式了。

android:fitsSystemWindows的“踩坑”

通过前面的两种方式,大家估计会留意到一个地方,就是所有实现 Translucent System Bar 效果的Activity,都需要在根布局里设置 android:fitsSystemWindows="true" 。设置了该属性的作用在于,不会让系统导航栏和我们app的UI重叠,导致交互问题。这样说可能比较抽象,看看下面两个效果图的对比就知道了。

技术分享
有fitsSystemWindows设置
技术分享
没有fitsSystemWindows设置

注:上面的演示效果,是借助了我的另一个开源项目,详情请戳:AndroidAlbum

这样的话,如果我有10个Activity要实现这种效果,就要在10个布局文件中做设置,非常麻烦。所以,想到一种方法,在theme中加上如下的android:fitsSystemWindows设置:

<item name="android:fitsSystemWindows">true</item>

发现果真可以了。所有要实现 Translucent System Bar 的Activity,只需要设置了这个theme即可,改起来也很方便。可惜,后来出现了一个BUG,让我还是得老老实实的回去布局文件中设置。

技术分享
Toast文字错位

Toast打印出来的文字都往上偏移了。这里也是我疏忽的地方,因为在布局文件中设置是对View生效,而到了theme进行设置则是对Window生效了,两者在实现上就不一样了。所以,最终只能改回原来的方式去实现。

实践总结

最后做一下小小的总结:

  • 方式一适用于app中没有导航栏,且整体的背景是一张图片的界面;
  • 方式二适用于app中导航栏颜色为纯色的界面;
  • android:fitsSystemWindows设置要在布局文件中,不要到theme中设置;

怎样,介绍到这里,你会使用 Translucent System Bar 了吗?赶快到你的app中引入吧!

补充更新(2016-02-19)

一些热心的网友反馈,在Android 4.4平台上使用第二种方法失效。我立马跑到Android4.4的真机运行一遍,果真出现下面的bug,顶部变成黑白渐变了。

技术分享
4.4平台上第二种方案出现的BUG

在此,先为自己的疏忽向广大读者说声抱歉。以后会最大程度的避免这种低级错误的产生。下面给出此Bug的修复方案:

第一步:去到 ColorTranslucentBarActivity 的布局文件中,将布局划分成为标题布局和内容布局两部分;

第二步:将 ColorTranslucentBarActivity 的根布局颜色设置与标题布局的颜色一致,并将内容布局设置为白色;

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/color_31c27c"
    android:fitsSystemWindows="true"
    android:orientation="vertical">

    <!--标题布局-->
    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="55dp"
        android:background="@color/color_31c27c">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="QQ Music"
            android:textColor="@android:color/white"
            android:textSize="20sp" />

    </RelativeLayout>

    <!--内容布局-->
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@android:color/white"
        android:orientation="vertical">

        <Button
            android:id="@+id/btn_show_toast"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="Show a toast" />
    </LinearLayout>

</LinearLayout>

经过以上两步,即可在 4.4 平台上实现 Translucent System Bar 的效果 。最后附上修复bug后的效果图一张。

技术分享
Android 4.4平台bug修复后的效果图

补充更新(2016-02-22)

很多童鞋反应,在每个布局文件中都要写上 android:fitsSystemWindows="true" ,有没有更佳方便的方法,本人当时没有思路。今天收到coder_sharp 童鞋反馈的一种更为简便的思路

技术分享
coder_sharp童鞋提供的新思路

个人把他的思路,整理成代码,如下:

public abstract class TranslucentBarBaseActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        supportRequestWindowFeature(Window.FEATURE_NO_TITLE);

        setContentView(getLayoutResId());//把设置布局文件的操作交给继承的子类

        ViewGroup contentFrameLayout = (ViewGroup) findViewById(Window.ID_ANDROID_CONTENT);
        View parentView = contentFrameLayout.getChildAt(0);
        if (parentView != null && Build.VERSION.SDK_INT >= 14) {
            parentView.setFitsSystemWindows(true);
        }
    }

    /**
     * 返回当前Activity布局文件的id
     *
     * @return
     */
    abstract protected int getLayoutResId();
}

所有需要实现效果的界面继承以上的父类,并实现 getLayoutResId 抽象方法即可,就可以不用在布局文件中不断做重复操作了,具体代码详见工程中的 TranslucentBarBaseActivity 和 BestTranslucentBarActivity。

补充更新(2016-02-25)

近几天在琢磨 Material Design 的一些新控件效果,意外的发现上面提到的第二种方式,在将原 values-v21/style.xml

<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    ....
    .... 
    <item name="android:statusBarColor">@color/color_31c27c</item>

</style>

换成

<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    ....
    .... 
    <item name="android:statusBarColor">@android:color/transparent</item>

</style>

之后,依旧可以实现同样的效果。那么,到了这里你就可以发现,上面提到的两种方式从本质上其实是殊途同归(最终总结得到的就是一种方式)!

我是热爱技术,喜欢开源和分享的Clock,很享受写文和其他开发者们探讨问题的乐趣,其中很多bug都是其他细心的开发者发现的。如果你对我写的内容感兴趣,欢迎关注我的简书,我很乐意把我开发中一些有趣的东西分享到简书中来(虽然目前仅仅是Android,但相信以后肯定会有其他内容的),希望与其他开发者共同探讨,共同进步!

分享即美德,最后附上源代码地址:
https://github.com/D-clock/AndroidSystemUiTraining

欢迎关注我的简书,以及:
















制作容器镜像的最佳实践(代码片段)

...的关于镜像制作的相关文章总结出来的.包括通用的容器最佳实践,java,nginx,python容器最佳实践.最佳实践的目的一方面保证镜像是可复用的,提升DevOps效率,另一方面是为了提高安全性.希望对各位有所帮助.本文分为四部分内容,分别... 查看详情

DAO(数据访问对象)最佳实践 - 我看到的示例同时使用 DAO 和服务对象,这里的最佳实践是啥?

】DAO(数据访问对象)最佳实践-我看到的示例同时使用DAO和服务对象,这里的最佳实践是啥?【英文标题】:DAO(dataaccessobject)bestpractices-examplesIseeuseaDAOandaServicesobjectboth,whatisthebestpracticehere?DAO(数据访问对象)最佳实践-我看到的... 查看详情

activity的最佳实践(代码片段)

文章目录Activity的最佳实践知晓当前是在哪一个Activiy随时随地退出程序启动Activity的最佳写法Activity的最佳实践知晓当前是在哪一个Activiy创建一个BaseActivity类,继承AppCompatActivity类.重写onCreate方法openclassBaseActivity:AppCompatActivity()overr... 查看详情

设置布尔标志的最佳实践

】设置布尔标志的最佳实践【英文标题】:Bestpracticeforsettingbooleanflags【发布时间】:2015-07-1411:49:54【问题描述】:我正在寻找有关拥有一个具有标志的类的最佳实践,该标志决定是否可以对该类执行操作。整个类将在WCFREST服务... 查看详情

分析网页的最佳实践

】分析网页的最佳实践【英文标题】:Bestpracticeforprofilingwebpages【发布时间】:2010-10-1813:52:13【问题描述】:我正在尝试在我的应用程序中配置页面。我也想请教一些最佳实践来完成这项工作早期开发者使用工具作为Chrome开发者... 查看详情

有哪些 OOoBasic 最佳实践?

】有哪些OOoBasic最佳实践?【英文标题】:WhataresomeOOoBasicbestpractices?【发布时间】:2009-06-0514:51:46【问题描述】:编写OOoBasic程序的最佳实践有哪些?到目前为止,我知道你应该使用XRay,但其中也有很多奇怪的小怪癖。【问题讨... 查看详情

平均堆栈身份验证 - 最佳实践

】平均堆栈身份验证-最佳实践【英文标题】:meanstackauthentication-bestpractice【发布时间】:2015-03-0103:04:35【问题描述】:我正在尝试找出构建身份验证系统的最佳实践,最简单的一个(基于用户名和密码)。我搜索了最佳实践并... 查看详情

JSF 的最佳实践是啥? [关闭]

】JSF的最佳实践是啥?[关闭]【英文标题】:WhatarethebestpracticesforJSF?[closed]JSF的最佳实践是什么?[关闭]【发布时间】:2010-09-0702:40:20【问题描述】:我过去做过Java和JSP编程,但我是JavaServerFaces的新手,想知道是否有一套JSF开发... 查看详情

NSURLSession 的最佳实践是啥?

】NSURLSession的最佳实践是啥?【英文标题】:What‘sthebestpracticesofNSURLSession?NSURLSession的最佳实践是什么?【发布时间】:2020-12-2309:53:48【问题描述】:先订阅我的应用:大部分场景使用AFNetworking,一小部分场景使用NSURLSession.share... 查看详情

NSPersistentContainer newBackgroundContext 的最佳实践是啥?

】NSPersistentContainernewBackgroundContext的最佳实践是啥?【英文标题】:What\'sthebestpracticeforNSPersistentContainernewBackgroundContext?NSPersistentContainernewBackgroundContext的最佳实践是什么?【发布时间】:2018-06-2421:22:00【问题描述】:我正在熟悉N... 查看详情

angular中订阅方法的最佳实践是啥?

】angular中订阅方法的最佳实践是啥?【英文标题】:Whatisthebestpracticeforsubscribemethodinangular?angular中订阅方法的最佳实践是什么?【发布时间】:2022-01-0102:42:05【问题描述】:我只是想知道哪个是Angular的最佳实践。这是一个可观... 查看详情

生成 OAuth 令牌的最佳实践?

】生成OAuth令牌的最佳实践?【英文标题】:BestpracticesaroundgeneratingOAuthtokens?【发布时间】:2010-12-1005:45:40【问题描述】:我意识到OAuthspec没有指定有关ConsumerKey、ConsumerSecret、AccessToken、RequestToken、TokenSecret或验证程序代码来源的... 查看详情

类型断言的最佳实践

Golang语言中的类型断言的最佳实践利用函数typejudge来判断传入的实参是什么类型,并打印出它的类型及数值packagemainimport("fmt"_"sort"_"math/rand")typeWisdomstruct namestringfuncTypeJuDge(Items...interface) forindex,value:=rangeItems switch 查看详情

模型中字段的最佳实践

】模型中字段的最佳实践【英文标题】:BestPracticeforfieldsinthemodel【发布时间】:2013-01-2421:42:30【问题描述】:我想知道是否有最佳实践来设置模型将检索的字段。在这两种不同的方法之间(伪代码):在控制器中定义值并将它... 查看详情

项目版本管理的最佳实践:云效飞流flow篇

简介:飞流Flow的最佳实践(使用阿里云云效)为了更好地使用飞流Flow,接下来将结合阿里云云效来讲解飞流Flow的最佳实践目录一、分支规约二、版本号规约2.1主版本号(首位版本号)2.2次版本号(迭代号)2.3小版本号三、云效... 查看详情

配置 SORM 的最佳实践

】配置SORM的最佳实践【英文标题】:BestpracticesforconfiguringSORM【发布时间】:2013-10-0723:32:25【问题描述】:我想知道是否有人可以分享一些使用SORM的最佳实践,参考在参与者系统中使用和自动化测试(单元/功能)。你好吗:从... 查看详情

Firestore 数据结构的最佳实践是啥?

】Firestore数据结构的最佳实践是啥?【英文标题】:Whatisthebestpracticeoffirestoredatastructure?Firestore数据结构的最佳实践是什么?【发布时间】:2018-07-1700:13:10【问题描述】:我正在使用firebase制作博客应用。我想知道数据结构的最佳... 查看详情

目前持久性的最佳实践是啥?

】目前持久性的最佳实践是啥?【英文标题】:Whatisthebestpracticeforpersistencerightnow?目前持久性的最佳实践是什么?【发布时间】:2008-12-1210:19:57【问题描述】:我来自java背景。但我想从跨平台的角度了解什么是持久对象的最佳实... 查看详情