說說 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>
xmlns:material="http://schemas.android.com/apk/res-auto"
標題欄佈局:
- 標題欄佈局使用 AppBarLayout,內層嵌套了 CollapsingToolbarLayout。
- CollapsingToolbarLayout 中的 contentScrim 可指定當標題欄趨於摺疊以及摺疊之後的背景色;而 layout_scrollFlags 中的 scroll 表示 CollapsingToolbarLayout 會隨著內容的滾動而滾動; layout_scrollFlags 中的 exitUntilCollapsed 表示摺疊後會保留在介面中(不會被移出螢幕)。
- 在 CollapsingToolbarLayout 中,定義了 ImageView 與 Toolbar。即這個標題欄是由普通標題欄加圖片組合而成的。其中有一個 layout_collapseMode 屬性,它用於設定摺疊模式。pin 表示在摺疊過程中使用保持,就像被大頭針定住了一樣;parallax 表示在摺疊過程中會產生視差。
內容佈局:
- 外層使用 NestedScrollView,它不僅支援使用滾動的方式來檢視螢幕外的資料,還能巢狀響應滾動事件。我們還通過 layout_behavior 來指定佈局行為。
- 因為 NestedScrollView 內部只允許存在一個直接的子佈局,所以我們可以先巢狀一個 LinearLayout,然後再其內部放入更多內容。
- 這裡使用 TextView 來顯示內容,然後把它放入 CardView (卡片式佈局)中。
- 我們為 CardView 設定了外邊距,還設定了圓角。這些工作都是為了讓頁面變得更好看。
最後我們還加了一個懸浮按鈕(FloatingActionButton):
- 首先把按鈕圖片 png ,放在 drawable 資料夾下。
- 通過 layout_anchor 來設定錨點,讓懸浮按鈕出現在標題欄區域內。
- 通過 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); } }
- 我們在標題欄啟用 HomeAsUp 按鈕(返回箭頭),用於關閉當前活動。
- 將 CollapsingToolbarLayout 傳遞過來的貓型別,作為標題。
- 接著使用 Glide 來載入圖片。
- 然後設定內容。
- 最後在 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~