Android CoordinatorLayout打造的頂部欄

CoordinatorLayout常常與AppBarLayout和CollapsingToolbarLayout一起使用,用于打造各種炫酷效果的頂部欄。之前寫過一篇Android 實現(xiàn)酷炫的頂部欄,不過要注意的是這次我們用到Androidx,原因是Google 發(fā)布了 Android support library 28,同時也發(fā)布了 androidx 1.0.0 第一個正式版本,然后支持庫的 "28.0.0" 將會是最后一次更新,之后的更新都會遷移到 Androidx 中,所以我們這里也使用Androidx 了。
一、遷移到 Androidx
我們順便講一下遷移到 Androidx的遷移步驟,首先在 gradle.properties 文件中添加

# 表示使用 androidx
android.useAndroidX=true
# 表示將第三方庫遷移到 androidx
android.enableJetifier=true

遷移后如果會報錯,需要 Flie -> Invalidate Caches /Restart 一下就可以了。
具體可以參考遷移到 AndroidX

二、按照慣例,我們先看看效果圖:

image

從效果圖來看,當設(shè)置了layout_behavior的控件響應(yīng)起了CollapsingToolbarLayout中的layout_scrollFlags事件時,ImageView會有視差效果的向上滾動移除屏幕,當開始折疊時,我們代碼上來設(shè)置 toolbar 背景透明圖。

三、我們先來看看布局文件:

<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <!--第一部分:伸縮工具欄-->
    <com.google.android.material.appbar.AppBarLayout
        android:id="@+id/app_bar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:elevation="0dp">

        <com.google.android.material.appbar.CollapsingToolbarLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:layout_scrollFlags="scroll|exitUntilCollapsed"
            app:title=""
            app:titleEnabled="false">

            <ImageView
                android:layout_width="match_parent"
                android:layout_height="220dp"
                android:background="@mipmap/icon_bg"
                android:scaleType="fitXY" />

            <androidx.appcompat.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="46dp"
                android:background="@drawable/bg_toolbar_gardient"
                android:minHeight="46dp"
                app:layout_collapseMode="pin"
                app:navigationIcon="@mipmap/icon_common_back"
                app:title="">

                <TextView
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center"
                    android:text="個人主頁"
                    android:textColor="@android:color/white"
                    android:textSize="18dp"
                    android:textStyle="bold" />
            </androidx.appcompat.widget.Toolbar>

        </com.google.android.material.appbar.CollapsingToolbarLayout>

    </com.google.android.material.appbar.AppBarLayout>

    <!--第二部分:主要內(nèi)容-->
    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/recyclerview"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />

</androidx.coordinatorlayout.widget.CoordinatorLayout>

(1)我們在CollapsingToolbarLayout中設(shè)置了一個ImageView和一個Toolbar,并將這個CollapsingToolbarLayout作為一個整體放在AppBarLayout中;
(2)在CollapsingToolbarLayout中,我們設(shè)置了app:layout_scrollFlags="scroll|enterAlwaysCollapsed",它的值還包括:

  • scroll - 想滾動就必須設(shè)置這個,也就是說值設(shè)為scroll的View會跟隨滾動事件一起滾動
  • enterAlways - 值設(shè)為enterAlways的View,當RecyclerView往下滾動時,該View會直接往下滾動
  • exitUntilCollapsed - 值設(shè)為exitUntilCollapsed的View,當這個View要往上逐漸“消逝”時,會一直往上滑動,直到剩下的的高度達到它的最小高度后,再響應(yīng)RecyclerView的內(nèi)部滑動事件。
  • enterAlwaysCollapsed - 當值設(shè)為enterAlwaysCollapsed 的View已經(jīng)設(shè)置minHeight屬性又使用此標志時,這個View只能以最小高度進入,只有當滾動視圖到達頂部時才擴大到完整高度

(3)在Toolbar控件中,我們設(shè)置了app:layout_collapseMode="pin",layout_collapseMode (折疊模式) - 有兩個值:

  • pin - 設(shè)置為這個模式時,當CollapsingToolbarLayout完全收縮后,Toolbar還可以保留在屏幕上
  • parallax - 設(shè)置為這個模式時,在內(nèi)容滾動時,CollapsingToolbarLayout中的View(比如ImageView)也可以同時滾動,實現(xiàn)視差滾動效果,通常和layout_collapseParallaxMultiplier(設(shè)置視差因子)搭配使用

四、我們先來看看代碼

package per.juan.coordinatorlayoutdome;

import android.content.res.Resources;
import android.graphics.Color;
import android.os.Build;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.view.WindowManager;
import android.widget.TextView;
import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.google.android.material.appbar.AppBarLayout;
import java.lang.reflect.Field;

public class MainActivity extends AppCompatActivity {
    private Toolbar toolbar;
    private AppBarLayout app_bar;
    private RecyclerView mRecyclerView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initToolBar();
        initView();
    }

    private void initView() {
        mRecyclerView = findViewById(R.id.recyclerview);
        toolbar = findViewById(R.id.toolbar);
        app_bar = findViewById(R.id.app_bar);

        final int alphaMaxOffset = dpToPx(150);
        toolbar.getBackground().setAlpha(0);
        app_bar.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
            @Override
            public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
                // 設(shè)置 toolbar 背景
                if (verticalOffset > -alphaMaxOffset) {
                    toolbar.getBackground().setAlpha(255 * -verticalOffset / alphaMaxOffset);
                } else {
                    toolbar.getBackground().setAlpha(255);
                }
            }
        });

        mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
        mRecyclerView.setAdapter(new ContentAdapter());
    }

    protected void initToolBar() {
        try {
            Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
            if (toolbar != null) {
                // 沉浸模式
                int statusBarHeight = getStatusBarHeight();
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
                    openAndroidLStyle();

                    toolbar.setPadding(0, statusBarHeight, 0, 0);
                    toolbar.getLayoutParams().height = dpToPx(46) + statusBarHeight;
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 開啟沉浸式模式支持
     */
    public void openAndroidLStyle() {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            Window window = getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setStatusBarColor(Color.TRANSPARENT);
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
    }

    /**
     * 獲取狀態(tài)欄高度
     */
    public int getStatusBarHeight() {
        try {
            Class<?> c = Class.forName("com.android.internal.R$dimen");
            Object obj = c.newInstance();
            Field field = c.getField("status_bar_height");
            int x = Integer.parseInt(field.get(obj).toString());
            return getResources().getDimensionPixelSize(x);
        } catch (Exception e) {
            e.printStackTrace();
        }

        return 0;
    }

    /**
     * dp轉(zhuǎn)換為px
     */
    public static int dpToPx(float dp) {
        return (int) (dp * Resources.getSystem().getDisplayMetrics().density + 0.5f);
    }

    private class ContentAdapter extends RecyclerView.Adapter<ContentAdapter.ContentHolder> {
        @Override
        public ContentAdapter.ContentHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            return new ContentHolder(LayoutInflater.from(MainActivity.this).inflate(android.R.layout.simple_list_item_1, parent, false));
        }

        @Override
        public void onBindViewHolder(ContentAdapter.ContentHolder holder, int position) {
            holder.itemTv.setText("item");
        }

        @Override
        public int getItemCount() {
            return 35;
        }

        class ContentHolder extends RecyclerView.ViewHolder {

            private TextView itemTv;

            public ContentHolder(View itemView) {
                super(itemView);
                itemTv = (TextView) itemView.findViewById(android.R.id.text1);
            }
        }
    }
}

(1)這里我們需要開啟沉浸式模式的支持;
(2)通過AppBarLayout.addOnOffsetChangedListener來監(jiān)聽折疊過程中AppBarLayout垂直方向上的偏移量的改變,我們代碼上來設(shè)置 toolbar 背景透明圖。

好了,本篇文章就這樣啦,存在總結(jié)不到位的地方還望指導,感謝 _
源碼下載

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容