Android Design Support Library系列之九:Bottom sheets

Android Support Library 23.2开始,添加了一个新特性Bottom sheets.

那么这个Bottom sheets有什么作用呢,观看官方示例图:


官方介绍.
A bottom sheet is a sheet of material that slides up from the bottom edge of the screen.

Bottom Sheets是一个从屏幕底部边缘向上滑出的一个面板.

惯例,使用前添加依赖:

compile 'com.android.support:design:25.3.1'

一、BottomSheet

1、简单使用
<?xml version="1.0" encoding="utf-8"?>
<android.support.design.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">

    <android.support.v4.widget.NestedScrollView
        android:id="@+id/bottomsheet"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:behavior_hideable="true"
        app:behavior_peekHeight="48dp"
        app:layout_behavior="@string/bottom_sheet_behavior">

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:divider="@android:drawable/divider_horizontal_bright"
            android:orientation="vertical"
            android:showDividers="middle">

            <TextView
                android:layout_width="match_parent"
                android:layout_height="48dp"
                android:background="@android:color/holo_blue_dark"
                android:gravity="center"
                android:text="收藏"
                android:textColor="@android:color/white" />

            <TextView
                android:layout_width="match_parent"
                android:layout_height="48dp"
                android:background="@android:color/holo_blue_dark"
                android:gravity="center"
                android:text="分享"
                android:textColor="@android:color/white" />

            <TextView
                android:layout_width="match_parent"
                android:layout_height="48dp"
                android:background="@android:color/holo_blue_dark"
                android:gravity="center"
                android:text="取消"
                android:textColor="@android:color/white" />
        </LinearLayout>
    </android.support.v4.widget.NestedScrollView>


</android.support.design.widget.CoordinatorLayout>
2、常用属性、方法介绍

1)与BottomShee相关的最重要的3条属性

使用BottomSheetBehavior,表明这是一个BottomSheet 
    app:layout_behavior="@string/bottom_sheet_behavior"

Tip:必须用在CoordinatorLayout的直接子View上
BottomSheet 预显的高度
    app:behavior_peekHeight=" "

Tip:BottomSheet 默认是关闭的,使用该属性可以预显一部分内容
可以在代码中通过:mBottomSheetBehavior.setPeekHeight()设置
BottomSheet是否可以完全隐藏,默认false
    app:behavior_hideable=" "

Tip:当我们设置了预显高度时,如果不设为true,向下拖动BottomSheet,预显内容隐藏不了
可以在代码中通过:mBottomSheetBehavior.setHideable()设置

2)BottomSheet设置回调

BottomSheetBehavior.setBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() {
    /**
     *  BottomSheet状态的改变时的回调
     * @param newState    BottomSheet此时的状态
     */
    @Override
    public void onStateChanged(@NonNull View bottomSheet, int newState) {
    }

    /**
     *   拖拽BottomSheet时的回调
     */
    @Override
    public void onSlide(@NonNull View bottomSheet, float slideOffset) {
    }
});

下面分别来测一下BottomSheet的5种状态

package com.my;

import android.os.Bundle;
import android.support.annotation.NonNull;
import android.support.design.widget.BottomSheetBehavior;
import android.support.v4.widget.NestedScrollView;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

    private NestedScrollView mBottomSheet;
    private BottomSheetBehavior mBottomSheetBehavior;

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

        /**
         * 从NestedScrollView上获取到它的Behavior
         */
        mBottomSheet = (NestedScrollView) findViewById(R.id.bottomsheet);
        mBottomSheetBehavior = BottomSheetBehavior.from(mBottomSheet);

        /**
         * 设置回调
         */
        mBottomSheetBehavior.setBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() {
            @Override
            public void onStateChanged(@NonNull View bottomSheet, int newState) {
                switch (newState){
                    case BottomSheetBehavior.STATE_COLLAPSED:
                        Toast.makeText(MainActivity.this, "BottomSheet关闭时", Toast.LENGTH_SHORT).show();
                        break;
                    case BottomSheetBehavior.STATE_DRAGGING:
//                        Toast.makeText(MainActivity.this, "拖拽BottomSheet时", Toast.LENGTH_SHORT).show();
                        break;
                    case BottomSheetBehavior.STATE_SETTLING:
//                        Toast.makeText(MainActivity.this, "拖拽松开手指时", Toast.LENGTH_SHORT).show();
                        break;
                    case BottomSheetBehavior.STATE_EXPANDED:
//                        Toast.makeText(MainActivity.this, "完全展开的状态", Toast.LENGTH_SHORT).show();
                        break;
                    case BottomSheetBehavior.STATE_HIDDEN:
//                        Toast.makeText(MainActivity.this, "完全隐藏时的状态", Toast.LENGTH_SHORT).show();
                        break;
                }
            }

            @Override
            public void onSlide(@NonNull View bottomSheet, float slideOffset) {

            }
        });
    }

}
STATE_COLLAPSED
STATE_DRAGGING
STATE_SETTLING
STATE_EXPANDED
STATE_HIDDEN

3)获取BottomSheet状态、设置BottomSheet状态

mBottomSheetBehavior.getState();
mBottomSheetBehavior.setState();

有时我们不希望预显内容,希望通过点击事件来显示和隐藏BottomSheet,此时可以通过上面两个方法获取此时的状态以及设置此时的状态

在布局文件中添加了一个Button,通过android:onClick="click1"设置点击事件

    public void click1(View v) {
        if (mBottomSheetBehavior.getState() == BottomSheetBehavior.STATE_EXPANDED) {
            mBottomSheetBehavior.setState(BottomSheetBehavior.STATE_COLLAPSED);
        } else if (mBottomSheetBehavior.getState() == BottomSheetBehavior.STATE_COLLAPSED) {
            mBottomSheetBehavior.setState(BottomSheetBehavior.STATE_EXPANDED);
        }
    }

二、BottomSheetDialog

BottomSheetDialog的使用就更加简单了
第一步:实例化

BottomSheetDialog dialog = new BottomSheetDialog(MainActivity.this);

第二步:设置对话框内容

View view = View.inflate(MainActivity.this, R.layout.dialog_layout, null);
dialog.setContentView(view);

或者
dialog.setContentView(R.layout.dialog_layout);

这个对话框的内容完全由你自己决定

第三步:show

dialog.show();

布局文件dialog_layout:

<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.NestedScrollView 
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="12dp"
        android:layout_marginRight="12dp"
        android:orientation="vertical">

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/a"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="设置备注及标签"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/b"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="标为星标朋友"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/c"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="设置朋友圈权限"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/d"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="发送该名片"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/e"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="投诉"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/f"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="加入黑名单"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/g"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="删除"
            android:textColor="#555555"
            android:textSize="16sp" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:drawableLeft="@drawable/h"
            android:drawablePadding="24dp"
            android:padding="12dp"
            android:text="添加到桌面"
            android:textColor="#555555"
            android:textSize="16sp" />
    </LinearLayout>
</android.support.v4.widget.NestedScrollView>

和一般的dialog一样,BottomSheetDialog可以通过dismiss()消失

dialog.dismiss(); dialog消失
dialog.setCanceledOnTouchOutside(false);  触摸dialog之外的地方,dialog不消失
dialog.setCancelable(false);  dialog无法取消,按返回键都取消不了

三、BottomSheetDialogFragment

第一步:继承BottomSheetDialogFragment ,复写onCreateView()方法

package com.my;

import android.os.Bundle;
import android.support.design.widget.BottomSheetBehavior;
import android.support.design.widget.BottomSheetDialogFragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class MyFragment extends BottomSheetDialogFragment {

    public static MyFragment newInstance() {
        return new MyFragment();
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.dialog_layout, container, false);
        return v;
    }
}

第二步:实例化

MyFragment myFragment = MyFragment.newInstance();

第三步:show

myFragment.show(getSupportFragmentManager(), "bottom sheet tag");

查看BottomSheetDialogFragment继承结构,发现它其实就是一个DialogFragment


四、设计规范

1、Bottom Sheets特别适合有三个或者三个以上的操作需要提供给用户选择、并且不需要对操作有额外解释的情景.如果只有两个或者更少的操作,或者需要详加描述的,可以考虑使用菜单(Menu)或者对话框替代.

2、在一个标准的列表样式的Bottom Sheets中,每一个操作应该有一句描述和一个左对齐的icon,并且这些图标大小、间距都有标准.

更多设计规范,请观看官方指南.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容