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

1 CollapsingToolbarLayout 布局

CollapsingToolbarLayout 是基于 Toolbar 的布局。它可以讓 Toolbar 的效果變得更加華麗褂萧。

注意:CollapsingToolbarLayout 只能作為 AppBarLayout 的直接子布局闷串。

現(xiàn)在我們創(chuàng)建一個空活動來顯示貓的詳情:

然后在 activity_cat.xml 中編寫界面布局,主要分為兩個部分铅鲤,一個是標題欄,另一個是內(nèi)容:

<?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>

    <!--內(nèi)容布局-->
    <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,內(nèi)層嵌套了 CollapsingToolbarLayout径玖。
  2. CollapsingToolbarLayout 中的 contentScrim 可指定當標題欄趨于折疊以及折疊之后的背景色痴脾;而 layout_scrollFlags 中的 scroll 表示 CollapsingToolbarLayout 會隨著內(nèi)容的滾動而滾動; layout_scrollFlags 中的 exitUntilCollapsed 表示折疊后會保留在界面中(不會被移出屏幕)梳星。
  3. 在 CollapsingToolbarLayout 中赞赖,定義了 ImageView 與 Toolbar。即這個標題欄是由普通標題欄加圖片組合而成的冤灾。其中有一個 layout_collapseMode 屬性前域,它用于設(shè)置折疊模式。pin 表示在折疊過程中使用保持韵吨,就像被大頭針定住了一樣匿垄;parallax 表示在折疊過程中會產(chǎn)生視差。

內(nèi)容布局:

  1. 外層使用 NestedScrollView,它不僅支持使用滾動的方式來查看屏幕外的數(shù)據(jù)椿疗,還能嵌套響應(yīng)滾動事件漏峰。我們還通過 layout_behavior 來指定布局行為。
  2. 因為 NestedScrollView 內(nèi)部只允許存在一個直接的子布局届榄,所以我們可以先嵌套一個 LinearLayout浅乔,然后再其內(nèi)部放入更多內(nèi)容。
  3. 這里使用 TextView 來顯示內(nèi)容铝条,然后把它放入 CardView (卡片式布局)中靖苇。
  4. 我們?yōu)?CardView 設(shè)置了外邊距,還設(shè)置了圓角班缰。這些工作都是為了讓頁面變得更好看贤壁。

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

  1. 首先把按鈕圖片 png ,放在 drawable 文件夾下埠忘。
  2. 通過 layout_anchor 來設(shè)置錨點脾拆,讓懸浮按鈕出現(xiàn)在標題欄區(qū)域內(nèi)。
  3. 通過 layout_anchorGravity 讓按鈕出現(xiàn)在標題欄區(qū)域的右下角给梅。

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();

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

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

        //設(shè)置內(nèi)容
        TextView contentView = (TextView) findViewById(R.id.cat_text_view);
        contentView.setText(generateContent(catName));

    }

    /**
     * 生成內(nèi)容
     *
     * @param catName 貓名
     * @return
     */
    private String generateContent(String catName) {
        //實踐中假丧,會根據(jù)參數(shù)名稱,返回對應(yīng)的內(nèi)容
        return "俄羅斯藍貓(Russian Blue)动羽,歷史上曾被稱做阿契安吉藍貓包帚,曾有過三種不同的名字。最初是阿契安吉藍貓运吓,直到20世紀40年代才有現(xiàn)在的名字渴邦,另外有段時間它則叫做馬耳他貓。證據(jù)顯示拘哨,這種貓確實原產(chǎn)于俄羅斯谋梭,因為已在俄羅斯寒帶地區(qū)發(fā)現(xiàn)了同種的貓。俄羅斯藍貓體型細長倦青,大而直立的尖耳朵瓮床,腳掌小而圓,走路像是用腳尖在走产镐。身上披著銀藍色光澤的短被毛隘庄,配上修長苗條的體型和輕盈的步態(tài),盡顯一派貓中的貴族風度癣亚。";
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()){
            case android.R.id.home://點擊 HomeAsUp 按鈕時
                finish();//關(guān)閉當前活動(即返回上一個活動)
                return true;
        }
        return super.onOptionsItemSelected(item);
    }
}
  1. 我們在標題欄啟用 HomeAsUp 按鈕(返回箭頭)丑掺,用于關(guān)閉當前活動。
  2. 將 CollapsingToolbarLayout 傳遞過來的貓類型述雾,作為標題街州。
  3. 接著使用 Glide 來加載圖片兼丰。
  4. 然后設(shè)置內(nèi)容。
  5. 最后在 onOptionsItemSelected 中處理 HomeAsUp 按鈕的點擊事件唆缴。

現(xiàn)在我們要處理首頁活動 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) {//設(shè)置上下文環(huán)境
            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 融合系統(tǒng)頂部狀態(tài)欄

利用 fitsSystemWindows 屬性我們可以將標題欄中的圖片與系統(tǒng)頂部狀態(tài)欄融合起來。

注意: 因為我們這里的 ImageView 嵌套的很深,所以必須把它以及它的祖先布局中的 fitsSystemWindows 都設(shè)置為 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>

接著把狀態(tài)欄的顏色設(shè)置為透明闯第,即

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

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

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

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

這樣設(shè)計是因為只有 Android 5.0+ 才會去讀取該文件。

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

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

最后在 AndroidManifest.xml 中的 CatActivity 活動唧躲,設(shè)置剛剛定義的主題:

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

運行效果:


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

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末碱璃,一起剝皮案震驚了整個濱河市弄痹,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌嵌器,老刑警劉巖肛真,帶你破解...
    沈念sama閱讀 211,884評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異爽航,居然都是意外死亡蚓让,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評論 3 385
  • 文/潘曉璐 我一進店門讥珍,熙熙樓的掌柜王于貴愁眉苦臉地迎上來历极,“玉大人,你說我怎么就攤上這事衷佃√诵叮” “怎么了?”我有些...
    開封第一講書人閱讀 157,435評論 0 348
  • 文/不壞的土叔 我叫張陵氏义,是天一觀的道長锄列。 經(jīng)常有香客問我,道長觅赊,這世上最難降的妖魔是什么右蕊? 我笑而不...
    開封第一講書人閱讀 56,509評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮吮螺,結(jié)果婚禮上饶囚,老公的妹妹穿的比我還像新娘帕翻。我一直安慰自己,他們只是感情好萝风,可當我...
    茶點故事閱讀 65,611評論 6 386
  • 文/花漫 我一把揭開白布嘀掸。 她就那樣靜靜地躺著,像睡著了一般规惰。 火紅的嫁衣襯著肌膚如雪睬塌。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,837評論 1 290
  • 那天歇万,我揣著相機與錄音揩晴,去河邊找鬼。 笑死贪磺,一個胖子當著我的面吹牛硫兰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播寒锚,決...
    沈念sama閱讀 38,987評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼劫映,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了刹前?” 一聲冷哼從身側(cè)響起泳赋,我...
    開封第一講書人閱讀 37,730評論 0 267
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎喇喉,沒想到半個月后祖今,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,194評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡轧飞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,525評論 2 327
  • 正文 我和宋清朗相戀三年衅鹿,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片过咬。...
    茶點故事閱讀 38,664評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡大渤,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出掸绞,到底是詐尸還是另有隱情泵三,我是刑警寧澤,帶...
    沈念sama閱讀 34,334評論 4 330
  • 正文 年R本政府宣布衔掸,位于F島的核電站烫幕,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏敞映。R本人自食惡果不足惜较曼,卻給世界環(huán)境...
    茶點故事閱讀 39,944評論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望振愿。 院中可真熱鬧捷犹,春花似錦弛饭、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至枪孩,卻和暖如春憔晒,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蔑舞。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評論 1 266
  • 我被黑心中介騙來泰國打工拒担, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人斗幼。 一個月前我還...
    沈念sama閱讀 46,389評論 2 360
  • 正文 我出身青樓澎蛛,卻偏偏與公主長得像抚垄,于是被迫代替她去往敵國和親蜕窿。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,554評論 2 349

推薦閱讀更多精彩內(nèi)容