說說 Android 的 Material Design 設計(五)——可摺疊式標題欄
阿新 • • 發佈:2018-12-16
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>
- 整體佈局使用 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~