1. 程式人生 > >說說 Android 的 Material Design 設計(五)——可摺疊式標題欄

說說 Android 的 Material Design 設計(五)——可摺疊式標題欄

1 CollapsingToolbarLayout 佈局

CollapsingToolbarLayout 是基於 Toolbar 的佈局。它可以讓 Toolbar 的效果變得更加華麗。

**注意:**CollapsingToolbarLayout 只能作為 AppBarLayout 的直接子佈局。

現在我們建立一個空活動來顯示貓的詳情:

然後在 activity_cat.xml 中編寫介面佈局,主要分為兩個部分,一個是標題欄,另一個是內容:

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:material="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <!--標題欄佈局-->
    <android.support.design.widget.AppBarLayout
        android:id="@+id/app_bar_layout"
        android:layout_width="match_parent"
        android:layout_height="250dp">

        <android.support.design.widget.CollapsingToolbarLayout
            android:id="@+id/collapsing_toolbar_layout"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
            material:contentScrim="?attr/colorPrimary"
            material:layout_scrollFlags="scroll|exitUntilCollapsed">

            <ImageView
                android:id="@+id/cat_image_view"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:scaleType="centerCrop"
                material:layout_collapseMode="parallax" />

            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                material:layout_collapseMode="pin" />

        </android.support.design.widget.CollapsingToolbarLayout>
    </android.support.design.widget.AppBarLayout>

    <!--內容佈局-->
    <android.support.v4.widget.NestedScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        material:layout_behavior="@string/appbar_scrolling_view_behavior">

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">

            <android.support.v7.widget.CardView
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_marginBottom="15dp"
                android:layout_marginLeft="15dp"
                android:layout_marginRight="15dp"
                android:layout_marginTop="35dp"
                material:cardCornerRadius="4dp">

                <TextView
                    android:id="@+id/cat_text_view"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_margin="10dp" />
            </android.support.v7.widget.CardView>

        </LinearLayout>

    </android.support.v4.widget.NestedScrollView>

    <!--用於評論的懸浮按鈕-->
    <android.support.design.widget.FloatingActionButton
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="16dp"
        android:src="@drawable/comment"
        material:layout_anchor="@id/app_bar_layout"
        material:layout_anchorGravity="bottom|end" />
</android.support.design.widget.CoordinatorLayout>
  1. 整體佈局使用 CoordinatorLayout。
  2. 定義了 xmlns:material="http://schemas.android.com/apk/res-auto" 名稱空間。

標題欄佈局:

  1. 標題欄佈局使用 AppBarLayout,內層嵌套了 CollapsingToolbarLayout。
  2. CollapsingToolbarLayout 中的 contentScrim 可指定當標題欄趨於摺疊以及摺疊之後的背景色;而 layout_scrollFlags 中的 scroll 表示 CollapsingToolbarLayout 會隨著內容的滾動而滾動; layout_scrollFlags 中的 exitUntilCollapsed 表示摺疊後會保留在介面中(不會被移出螢幕)。
  3. 在 CollapsingToolbarLayout 中,定義了 ImageView 與 Toolbar。即這個標題欄是由普通標題欄加圖片組合而成的。其中有一個 layout_collapseMode 屬性,它用於設定摺疊模式。pin 表示在摺疊過程中使用保持,就像被大頭針定住了一樣;parallax 表示在摺疊過程中會產生視差。

內容佈局:

  1. 外層使用 NestedScrollView,它不僅支援使用滾動的方式來檢視螢幕外的資料,還能巢狀響應滾動事件。我們還通過 layout_behavior 來指定佈局行為。
  2. 因為 NestedScrollView 內部只允許存在一個直接的子佈局,所以我們可以先巢狀一個 LinearLayout,然後再其內部放入更多內容。
  3. 這裡使用 TextView 來顯示內容,然後把它放入 CardView (卡片式佈局)中。
  4. 我們為 CardView 設定了外邊距,還設定了圓角。這些工作都是為了讓頁面變得更好看。

最後我們還加了一個懸浮按鈕(FloatingActionButton):

  1. 首先把按鈕圖片 png ,放在 drawable 資料夾下。
  2. 通過 layout_anchor 來設定錨點,讓懸浮按鈕出現在標題欄區域內。
  3. 通過 layout_anchorGravity 讓按鈕出現在標題欄區域的右下角。

2 活動類

public class CatActivity extends AppCompatActivity {


    /**
     * 貓名
     */
    public static final String CAT_NAME = "cat_name";

    /**
     * 貓圖片 ID
     */
    public static final String CAT_IMAGE_ID = "cat_image_id";


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

        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        ActionBar actionBar = getSupportActionBar();
        if (actionBar != null) {//啟用 HomeAsUp 按鈕(返回箭頭)
            actionBar.setDisplayHomeAsUpEnabled(true);
        }

        Intent intent = getIntent();

        //設定標題
        CollapsingToolbarLayout layout = (CollapsingToolbarLayout) findViewById(R.id.collapsing_toolbar_layout);
        String catName = intent.getStringExtra(CAT_NAME);
        layout.setTitle(catName);

        //設定圖片
        int catImageId = intent.getIntExtra(CAT_IMAGE_ID, 0);
        ImageView imageView = (ImageView) findViewById(R.id.cat_image_view);
        Glide.with(this).load(catImageId).into(imageView);//載入圖片

        //設定內容
        TextView contentView = (TextView) findViewById(R.id.cat_text_view);
        contentView.setText(generateContent(catName));

    }

    /**
     * 生成內容
     *
     * @param catName 貓名
     * @return
     */
    private String generateContent(String catName) {
        //實踐中,會根據引數名稱,返回對應的內容
        return "俄羅斯藍貓(Russian Blue),歷史上曾被稱做阿契安吉藍貓,曾有過三種不同的名字。最初是阿契安吉藍貓,直到20世紀40年代才有現在的名字,另外有段時間它則叫做馬耳他貓。證據顯示,這種貓確實原產於俄羅斯,因為已在俄羅斯寒帶地區發現了同種的貓。俄羅斯藍貓體型細長,大而直立的尖耳朵,腳掌小而圓,走路像是用腳尖在走。身上披著銀藍色光澤的短被毛,配上修長苗條的體型和輕盈的步態,盡顯一派貓中的貴族風度。";
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()){
            case android.R.id.home://點選 HomeAsUp 按鈕時
                finish();//關閉當前活動(即返回上一個活動)
                return true;
        }
        return super.onOptionsItemSelected(item);
    }
}
  1. 我們在標題欄啟用 HomeAsUp 按鈕(返回箭頭),用於關閉當前活動。
  2. 將 CollapsingToolbarLayout 傳遞過來的貓型別,作為標題。
  3. 接著使用 Glide 來載入圖片。
  4. 然後設定內容。
  5. 最後在 onOptionsItemSelected 中處理 HomeAsUp 按鈕的點選事件。

現在我們要處理首頁活動 RecyclerView 的點選事件,當點選某隻貓時,開啟它的詳情活動頁:

public class CatAdapter extends RecyclerView.Adapter<CatAdapter.ViewHolder> {

    private Context context;

    private List<Cat> cats = new ArrayList<>();

    public CatAdapter(List<Cat> cats) {
        this.cats = cats;
    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        if (context == null) {//設定上下文環境
            context = parent.getContext();
        }
        View view = LayoutInflater.from(context).inflate(R.layout.cat_item, parent, false);

        //處理 RecyclerView 點選事件
        final ViewHolder holder = new ViewHolder(view);
        holder.cardView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                int position = holder.getAdapterPosition();
                Cat cat = cats.get(position);
                Intent intent = new Intent(context, CatActivity.class);
                intent.putExtra(CatActivity.CAT_NAME, cat.getType());
                intent.putExtra(CatActivity.CAT_IMAGE_ID, cat.getImgId());
                context.startActivity(intent);
            }
        });

        return holder;
    }

   ...
}

執行效果:

3 融合系統頂部狀態列

利用 fitsSystemWindows 屬性我們可以將標題欄中的圖片與系統頂部狀態列融合起來。

注意: 因為我們這裡的 ImageView 巢狀的很深,所以必須把它以及它的祖先佈局中的 fitsSystemWindows 都設定為 true,才能真正生效:

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:material="http://schemas.android.com/apk/res-auto"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   android:fitsSystemWindows="true"
   >

   <!--標題欄佈局-->
   <android.support.design.widget.AppBarLayout
       android:id="@+id/app_bar_layout"
       android:layout_width="match_parent"
       android:layout_height="250dp"
       android:fitsSystemWindows="true"
       >

       <android.support.design.widget.CollapsingToolbarLayout
           android:id="@+id/collapsing_toolbar_layout"
           android:layout_width="match_parent"
           android:layout_height="match_parent"
           android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
           material:contentScrim="?attr/colorPrimary"
           material:layout_scrollFlags="scroll|exitUntilCollapsed"
           android:fitsSystemWindows="true"
           >

           <ImageView
               android:id="@+id/cat_image_view"
               android:layout_width="match_parent"
               android:layout_height="match_parent"
               android:scaleType="centerCrop"
               material:layout_collapseMode="parallax"
               android:fitsSystemWindows="true"
               />

       ...
       </android.support.design.widget.CollapsingToolbarLayout>
   </android.support.design.widget.AppBarLayout>

  

...
</android.support.design.widget.CoordinatorLayout>

接著把狀態列的顏色設定為透明,即

<item name="android:statusBarColor">@android:color/transparent</item>

因為這個屬性從 API21 (Android 5.0 +)才出現,所以我們必須做差異化實現。

在 res 目錄下新建一個 values-v21 目錄,然後再新建一個 styles.xml:

<resources>
    <!--Android 5.0+-->
    <style name="catActivityTheme" parent="AppTheme">
        <!--狀態列顏色指定為透明-->
        <item name="android:statusBarColor">@android:color/transparent</item>
    </style>
</resources>

這樣設計是因為只有 Android 5.0+ 才會去讀取該檔案。

我們還需要在 values/styles.xml 中新建一個不包含實際內容的 catActivityTheme,確保 Android 5.0 之前的版本不會拋錯:

<!--小於 Android 5.0-->
<style name="catActivityTheme" parent="AppTheme">
</style>

最後在 AndroidManifest.xml 中的 CatActivity 活動,設定剛剛定義的主題:

 <activity android:name=".CatActivity"
            android:theme="@style/catActivityTheme"
            ></activity>

執行效果:

就算是現在的劉海屏,也可以適配的很好哦O(∩_∩)O~