仿全历史——全沉浸时间轴实现

前言

我又来了!前段时间由于夏天的燥热与自身的懒惰,一直没有更新博客哈哈哈。今天良心发现,来更一个比较有意思的东西,如题——仿全历史APP的全沉浸时间轴实现。
话不多说,先上图


Screenshot_20200530_225923_com.allhistory.dls.mar.jpg

这张图呢,就是全历史App中全古古迹功能的界面图。
显然,它通过沉浸状态栏、透明背景、recyclerView的自定义Item等,实现了一个很优秀的界面效果。
今天,我要做的就是猜测这效果背后的实现原理,并仿制一个类似的界面

正文

解构

我们先看看结构一下这效果背后的组成部分:

Screenshot_20200530_225923_com.allhistory.dls.mar_副本.jpg

总的来讲,实现这样的效果主要需要四个部分:
1.沉浸状态栏、透明toolbar与背景
2.TabLayout样式自定义
3.TextSwitcher文字切换
(个人猜测是有可能是使用TextSwitcher实现的,实际今天的仿制里不涉及这个哈,今天的重点是沉浸与时间轴。如果这个也包含,篇幅就控制不住了......)
4.ItemDecoration定制,绘图画出时间轴效果

实现

因为涉及到的内容比较多,所以只会放出部分关键的代码哈。

1.沉浸状态栏与透明背景

状态栏的沉浸涉及到不同版本、不同情景状态下的适配,相对来讲比较复杂。想要研究的比较深的朋友,可以查看https://www.jianshu.com/p/dc20e98b9a90这篇博客,内容充实,讲的也非常详细,本文的的沉浸适配也对其进行了参考。
我们今天涉及到的沉浸状态栏实际非常简单,准确的讲应该叫透明状态栏,也并不包含交互效果下状态栏的变换。
本文采用了透明主题的配置,如下所示:

API21 之后(也就是 android 5.0 之后)的状态栏,会默认覆盖一层半透明遮罩。且为了保持4.4以前系统正常使用,故需要三份 style 文件,即默认的values(不设置状态栏透明)、values-v19、values-v21(解决半透明遮罩问题)>

//valuse
<style name="TranslucentTheme" parent="AppTheme">
</style>

// values-v19。v19 开始有 android:windowTranslucentStatus 这个属性
<style name="TranslucentTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <item name="android:windowTranslucentStatus">true</item>
        <item name="android:windowTranslucentNavigation">true</item>
</style>

// values-v21。5.0 以上提供了 setStatusBarColor()  方法设置状态栏颜色。
<style name="TranslucentTheme" parent="Theme.AppCompat.Light.NoActionBar">
    <item name="android:windowTranslucentStatus">false</item>
    <item name="android:windowTranslucentNavigation">true</item>
    <!--Android 5.x开始需要把颜色设置透明,否则导航栏会呈现系统默认的浅灰色-->
    <item name="android:statusBarColor">@android:color/transparent</item>
</style>

设置了透明主题之后,实际会布局会顶到状态栏,因而需要为toolbar设置一个paddingTop

  /**
     * 设置顶部statusBar与顶部View同背景
     *
     * @param activity 需要设置的activity
     * @param topView  顶部需要设置padding的view
     */
    public static void  setStatusBarMergeWithTopView(Activity activity, final View topView) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            final int statusBarHeight = getStatusBarHeight(activity);
            topView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
                @Override
                public void onGlobalLayout() {
                    topView.setPadding(0, statusBarHeight, 0, 0);
                    topView.getLayoutParams().height = topView.getHeight() + statusBarHeight;
                    topView.getViewTreeObserver().removeGlobalOnLayoutListener(this);
                }
            });
        }
    }

对应Acitivity设置:
StatusBarUtil.setStatusBarMergeWithTopView(this, tbHistory);//tbHistory为需要设置的透明toolbar,会在下面的布局中看到。
Acitivty布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:background="@drawable/historytest"
    android:layout_width="match_parent"
    android:orientation="vertical"
    android:layout_height="match_parent">
            <androidx.appcompat.widget.Toolbar
                android:id="@+id/tb_history"
                android:layout_width="match_parent"
                android:layout_height="?actionBarSize"
                app:layout_collapseMode="pin"
                android:background="@android:color/transparent"
                android:minHeight="?actionBarSize"
                app:navigationIcon="@drawable/ic_personal_center_back"
                app:title="">
                <TextView
                    android:id="@+id/tv_title"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center"
                    android:text="全古迹"
                    android:textColor="@android:color/white"
                    android:textSize="@dimen/text_size_xlarge_18"
                    android:visibility="visible"
                    android:textStyle="bold" />
            </androidx.appcompat.widget.Toolbar>
            <com.google.android.material.tabs.TabLayout
                android:id="@+id/tl_history"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:tabMode="scrollable"
                app:tabGravity="fill"
                app:tabTextAppearance="@style/HistoryTestTabLayoutTextStyle"
                app:tabIndicatorHeight="0dp"
                app:tabPadding="@dimen/height_2"
                app:tabSelectedTextColor="@color/white"
                app:tabTextColor="@color/normal_grey"/>

            <androidx.viewpager.widget.ViewPager
                android:id="@+id/vp_fg"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_marginTop="@dimen/margin_2"
                android:layout_marginBottom="@dimen/margin_min_2" />
</LinearLayout>
 

这个布局文件除了Tablayout外并没有什么好讲的,所以我们直接进入第二部分

2.Tablayout自定义

全古迹中的tablayout主要就是进行了tab切换后字体的大小、粗细的变化,tab采用了滚动的模式,下划线Indicator是自定义的一个较短的下划线。
tablayout的下划线默认是与Tab等长的,后来官方提供了app:tabIndicatorFullWidth="true"这个属性来实现下划线与字体等长。
但是显然,达不到图中的那种效果。
想要实现图中的效果,主要有两种方法,一种是通过反射拿到该控件来设置长度,一种是自定义一个tabView。
这里我们选择了第二种方式。
1.注意在tablayout布局文件中设置app:tabIndicatorHeight="0dp"。
2.编写自定义TabView布局文件。

<?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="wrap_content"
    android:layout_gravity="center"
    android:orientation="horizontal">

    <TextView
        android:textSize="14sp"
        android:id="@+id/tv_tab_title"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_centerVertical="true"
        android:textColor="@color/white" />


    <View
        android:id="@+id/v_indicator"
        android:layout_width="14dp"
        android:layout_height="2dp"
        android:layout_below="@+id/tv_tab_title"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="5dp"
        android:visibility="invisible"
        android:background="#ffd700"/>


</RelativeLayout>
 

在对应的FragmentPagerAdapter中加上

   public View getTabView(int position) {
            View tabView = LayoutInflater.from(getApplicationContext()).inflate(R.layout.item_tab_layout_history, null);
            TextView tabTitle = (TextView) tabView.findViewById(R.id.tv_tab_title);
            tabTitle.setText(getPageTitle(position));//需要对应设置的tab标题
            return tabView;
        }

接下来就是设置viewPagerAdapter,以及监听tab选中事件。

ViewPagerAdapter viewPagerAdapter =new ViewPagerAdapter(getSupportFragmentManager());
        vpFg.setAdapter(viewPagerAdapter);
        vpFg.setOffscreenPageLimit(0);
        tlHistory.setupWithViewPager(vpFg);
        vpFg.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(tlHistory));
        for (int i = 0; i < tlHistory.getTabCount(); i++) {//设置自定义tabView
            View tabView = viewPagerAdapter.getTabView(i);
            if(i==0){//默认先加载第一个,所以第一个的字体等样式先设置一下。
                View indicator = tabView.findViewById(R.id.v_indicator);
                indicator.setVisibility(View.VISIBLE);
                TextView tabTitle = (TextView) tabView.findViewById(R.id.tv_tab_title);
                tabTitle.setTypeface(Typeface.defaultFromStyle(Typeface.BOLD));
                tabTitle.setTextSize(18);
                tabTitle.setTextColor(getResources().getColor(R.color.beige_yellow));
            }
            tlHistory.getTabAt(i).setCustomView(tabView);
        }
        tlHistory.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
            @Override
            public void onTabSelected(TabLayout.Tab tab) {
                //选中,则更改样式
                View tabView = tab.getCustomView();
                if(tabView!=null){
                    TextView tabTitle = (TextView) tabView.findViewById(R.id.tv_tab_title);
                    tabTitle.setTypeface(Typeface.defaultFromStyle(Typeface.BOLD));
                    tabTitle.setTextSize(18);
                    View indicator = tabView.findViewById(R.id.v_indicator);
                    indicator.setVisibility(View.VISIBLE);
                }
            }
            @Override
            public void onTabUnselected(TabLayout.Tab tab) {
                //未选中,则恢复样式
                View tabView = tab.getCustomView();
                if(tabView!=null){
                    TextView tabTitle = (TextView) tabView.findViewById(R.id.tv_tab_title);
                    tabTitle.setTypeface(Typeface.defaultFromStyle(Typeface.NORMAL));
                    tabTitle.setTextSize(14);
                    View indicator = tabView.findViewById(R.id.v_indicator);
                    indicator.setVisibility(View.INVISIBLE);
                }
            }

            @Override
            public void onTabReselected(TabLayout.Tab tab) {

            }
        });
        vpFg.setCurrentItem(0);
        vpFg.setOffscreenPageLimit(3);
        tlHistory.getTabAt(0).select();

直到这里,已经完成了大半部分喽,还剩最后的时间轴的绘制。

3.时间轴绘制

时间轴的绘制呢,我们需要借用设置itemDecoration。
什么是itemDecoration?我们来看下官方的定义:

An ItemDecoration allows the application to add a special drawing and layout offset to specific item views from the adapter's data set. This can be useful for drawing dividers between items, highlights, visual grouping boundaries and more.>

ItemDecoration允许应用结合adapter的数据集,对特定的item添加绘制一个周边图案。可以用于给items之间添加分割线、高亮装饰效果或者分组边界等等。>

顾名思义,ItemDecoration就是对item起装饰作用,其最常见的用法就是添加分割线。
在这里的时间轴,需要画出圆环、圆点、线段、文字。
完整代码如下:

package com.example.ctccp.personalcenter.ui.adapter;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.view.View;

import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import java.util.List;


public class TimeAxisItemDecoration extends RecyclerView.ItemDecoration {
    private Paint LinePaint;
    private Paint TextPaint;
    private Paint PointPaint;
    private List<String> dynasty;
    private int divide_width = 66;

    public TimeAxisItemDecoration(Context context, List<String> dynasty) {
        LinePaint = new Paint();
        TextPaint = new Paint();
        PointPaint =new Paint();
        PointPaint.setStyle(Paint.Style.FILL);
        PointPaint.setStrokeWidth(8);
        PointPaint.setStrokeCap(Paint.Cap.ROUND);
        PointPaint.setColor(Color.WHITE);
        LinePaint.setColor(Color.WHITE);
        TextPaint.setColor(Color.WHITE);
        this.dynasty = dynasty;
    }

    @Override
    public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {
        super.onDraw(c, parent, state);
        int childCount = parent.getChildCount();
        RecyclerView.LayoutManager manager = parent.getLayoutManager();//为了动态获取
        for (int i = 0; i < childCount; i++) {
            View child = parent.getChildAt(i);
            int cx = manager.getLeftDecorationWidth(child) / 2;
            int cy = child.getTop()-8;
            float circleRadius = 14;
            int TextSize = 32;

            LinePaint.setStrokeWidth((float) 2.0);
            LinePaint.setStyle(Paint.Style.STROKE);

            c.drawCircle(cx,cy,circleRadius,LinePaint);//圆环
            c.drawPoint(cx,cy, PointPaint);//圆点
            c.drawLine(cx, child.getTop(), cx, child.getBottom()+child.getHeight(), LinePaint);//线

            TextPaint.setTextSize(TextSize);
        }
    }

    @Override
    public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) {
        super.onDrawOver(c, parent, state);
        RecyclerView.LayoutManager manager = parent.getLayoutManager();//为了动态获取
        int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            View child = parent.getChildAt(i);
            int cx = manager.getLeftDecorationWidth(child) ;
            int index = parent.getChildAdapterPosition(child);
            if(index<dynasty.size())
                c.drawText(dynasty.get(index), cx+8 , child.getTop() , TextPaint);
        }
    }


    @Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        super.getItemOffsets(outRect, view, parent, state);
        if(parent.getLayoutManager()instanceof LinearLayoutManager){
            outRect.set(divide_width,divide_width,divide_width/2,0);
        }

        //outRect outRect就是表示在item的上下左右所撑开的距离
        //View view:是指当前Item的View对象
        //RecyclerView parent: 是指RecyclerView 本身
        //RecyclerView.State state:通过State可以获取当前RecyclerView的状态,也可以通过State在RecyclerView各组件间传递参数
    }
}

成果

因为重点是沉浸和时间轴所以没有实现TextSwitcher的部分,数据也是统一填充的(害 ,懒就一个字),实际toolbar使用的是默认尺寸,toolbar高度再低点,应该视觉效果会更好一点。


Screenshot_20200531_175330_com.example.ctccp.jpg

后记

全历史实属一个良心、炫酷的app(害,全历史应该给我打5毛到卡里,我这还免费宣传了一波)。
溜了溜了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352