文章标题

5.0新特性常用布局

  • MD-CoordinatorLayout (协调者布局)
  • MD-FloatingActionButton (悬浮操作按钮)
  • Snackbar 为用户提供简单的反馈信息
  • MD-Toolbar 工具栏(替代之前的ActionBar)
  • MD-AppBarLayout (应用标题栏容器)
  • MD-CollapsingToolbarLayout 折叠效果的布局容器
  • MD-TabLayout 标签导航
  • MD-TextInputLayout 输入框控件的悬浮标签
  • MD-NavigationView 抽屉导航

一 创建一个ScrollingActivity 效果如图

这里写图片描述

如果要实现上图效果,步骤如下
首先需要修改theme—NoActionBar

这里写图片描述

打开values-v21 如下,下面两行是绘制系统栏的颜色,但系统栏颜色只能在5.0之后

这里写图片描述

二 FloatActionButton的使用

这里写图片描述

三 FloatActionButton的进一步使用

app:backgroundTint - 设置FAB的背景颜色。
app:rippleColor - 设置FAB点击时的背景颜色。
app:borderWidth - FAB边缘宽度, 在有的4.1的sdk上FAB会显示为正方形,或在5.0以后的sdk没有阴影效果。可以设置为borderWidth="0dp"解决此问题。
app:elevation - 默认状态下FAB的阴影大小。
app:pressedTranslationZ - 点击时候FAB的阴影大小。
app:fabSize - 设置FAB的大小,该属性有两个值,分别为normal和mini,对应的FAB大小分别为56dp和40dp。
src - 设置FAB的图标,Google建议符合Design设计的该图标大小为24dp。

在CoordinatorLayout的子控件中, 通过app:layout_anchor(布局锚) 和 app:layout_anchorGravity属性创造出悬浮效果
app:layout_anchor: 把指定的锚点控件作为当前Button的目标视图。
app:layout_anchorGravity: 在目标视图中的位置。值由bottom、center、right、left、top等

-这里写图片描述

四 ToolBar的使用

这里写图片描述

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
    android:fitsSystemWindows="true">
    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:fitsSystemWindows="true"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#50f000">
    </android.support.v7.widget.Toolbar>
</android.support.design.widget.CoordinatorLayout>
public class ToolBar extends AppCompatActivity{
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_toolbar);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);//通过toolbar代替actionbar
        getSupportActionBar().setTitle("我是标题");//设置标题
        getSupportActionBar().setDisplayHomeAsUpEnabled(true);//设置返回按键
        getSupportActionBar().setLogo(R.mipmap.ic_launcher);//设置标题
        getSupportActionBar().setSubtitle("二级标题");
        //监听返回按键的点击
        toolbar.setNavigationOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Toast.makeText(ToolBar.this, "点击了返回按钮", Toast.LENGTH_SHORT).show();
            }
        });
    }
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.menu_toolbar,menu);
        return true;
    }
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()){
            case R.id.email:
                Toast.makeText(this, "email被点击了", Toast.LENGTH_SHORT).show();
                break;
            case R.id.btn1:
                Toast.makeText(this, "btn1被点击了", Toast.LENGTH_SHORT).show();
                break;
            case R.id.btn2:
                Toast.makeText(this, "btn2被点击了", Toast.LENGTH_SHORT).show();
                break;
            case R.id.btn3:
                Toast.makeText(this, "btn3被点击了", Toast.LENGTH_SHORT).show();
                break;
        }
        return true;
    }
}

这里写图片描述
这里写图片描述

五 AppBarLayout (应用标题栏容器)

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
                                                 android:layout_width="match_parent"
                                                 android:layout_height="match_parent"
                                                 xmlns:app="http://schemas.android.com/apk/res-auto"
                                                 android:fitsSystemWindows="true"
                                                 android:orientation="vertical">
    <android.support.design.widget.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:fitsSystemWindows="true">
        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            app:layout_scrollFlags="scroll"
            android:layout_width="match_parent"
            android:layout_height="wrap_content">
        </android.support.v7.widget.Toolbar>
        <ImageView
            app:layout_scrollFlags="scroll|enterAlways"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:fitsSystemWindows="true"
            android:scaleType="centerCrop"
            android:src="@drawable/mm"/>
    </android.support.design.widget.AppBarLayout>
    <!--滚动-->
    <android.support.v4.widget.NestedScrollView
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior"
        tools:context="com.arvin.materialapp.ScrollingActivity"
        tools:showIn="@layout/activity_scrolling">
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_margin="@dimen/text_margin"
            android:text="@string/large_text"/>
    </android.support.v4.widget.NestedScrollView>
</android.support.design.widget.CoordinatorLayout>

AppBarLayout只能作为CoordinatorLayout里的第一个子view。
在RecyclerView或者任意支持嵌套滚动的view如NestedScrollView上添加app:layout_behavior属性
app:layout_behavior
属性值配置为@string/appbar_scrolling_view_behavior,该值在com.android.support的design包下, 用来通知AppBarLayout, 界面内容发生了滚动事件.
在AppBarLayout中的子View中配置app:layout_scrollFlags属性,就可以在滚动事件发生的时候自动执行自己动画.

app:layout_scrollFlags

属性里面必须至少添加scroll这个flag,这样这个View及其子控件才会滚动出屏幕,否则它最终将一直固定在顶部。
app:layout_scrollFlags的几个属性值及其意义

scroll

列表在顶端时,可以跟着滑动方向滑动(只有配置了scroll, 其他属性才能生效)
enterAlways

不管滑动列表到哪里, 只要往下拉, 当前控件就跟着往下滑出.
enterAlwaysCollapsed

跟enterAlways类似, 但是最大只会滑出折叠后的大小.要配合android:minHeight=”50dp”属性使用.最小高度即为折叠后的高度 (配置方式:”scroll|enterAlways|enterAlwaysCollapsed”)
exitUntilCollapsed

不管滑动列表到哪里, 只要向上拉,这个View会跟着滑动直到折叠。配合android:minHeight=”50dp”属性使用.最小高度即为折叠后的高度(配置方式:”scroll|exitUntilCollapsed”)
snap

滑动结束的时候,如果这个View部分显示,它就会滑动到离它最近的上边缘或下边缘。

布局结构如下:

<CoordinatorLayout>
    <AppbarLayout>
        <ImageView/>
        <Toolbar/>
    </AppbarLayout>
    <NestedScrollView/>
</CoordinatorLayout>

六 CollapsingToolbarLayout 折叠效果的布局容器

如果需要Toolbar有折叠效果, 可以让CollapsingToolbarLayout包裹Toolbar
相当于让CollapsingToolbarLayout临时托管了Toolbar的标题内容, 执行平移和缩放动画
Toolbar可以随着CollapsingToolbarLayout向上滑动
给Toolbar添加app:layout_collapseMode属性可以配置向上滑动时的效果

app:layout_collapseMode=”none” 默认效果,随父控件一起滑动
app:layout_collapseMode=”pin” 图钉效果,开始滑动时自己不动, 到顶端才离开
app:layout_collapseMode=”parallax” 视察效果,随父控件进行高度缩放
布局结构如下:

<CoordinatorLayout>
    <AppbarLayout>
        <CollapsingToolbarLayout>
            <ImageView/>
            <Toolbar/>
        </CollapsingToolbarLayout>
        ...
    </AppbarLayout>
    <NestedScrollView/>
</CoordinatorLayout>

七 TabLayout 标签导航

Tab 是在 Android 应用程序中用户体验(UX)最佳实践的一部分。以前,如果需要使用新的材料设计风格的 Tab,需要自己去Github下载 SlidingTabLayout 和 SlidingTabStrip.

布局配置

<android.support.design.widget.TabLayout
    android:id="@+id/tabs"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />

代码配置:

TabLayout tabLayout = (TabLayout) findViewById(R.id.tabs);
// 关联ViewPager
tabLayout.setupWithViewPager(mViewPager);

这里写图片描述

八 TextInputLayout 输入框控件的悬浮标签

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical"
              android:padding="20dp">
    <android.support.design.widget.TextInputLayout
        android:id="@+id/layout_username"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        <EditText
            android:id="@+id/edit_username"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="用户名"/>
    </android.support.design.widget.TextInputLayout>
    <android.support.design.widget.TextInputLayout
        android:id="@+id/layout_pwd"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        <EditText
            android:id="@+id/edit_pwd"
            android:inputType="numberPassword"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:hint="密码"/>
    </android.support.design.widget.TextInputLayout>
    <Button
        android:id="@+id/bt_check"
        android:text="检查错误"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>
    <Button
        android:id="@+id/bt_count"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="计数"/>
</LinearLayout>
public class TextInputLayoutActivity extends AppCompatActivity implements View.OnClickListener {
    private TextInputLayout mLayout_username;
    private TextInputLayout mLayout_pwd;
    private EditText mEt_username;
    private EditText mEt_pwd;
    private Button mBt_check;
    private Button mBt_count;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_text_input);
        mLayout_username = (TextInputLayout) findViewById(R.id.layout_username);
        mLayout_pwd = (TextInputLayout) findViewById(R.id.layout_pwd);
        mEt_username = (EditText) findViewById(R.id.edit_username);
        mEt_pwd = (EditText)findViewById(R.id.edit_pwd);
        mBt_check = (Button) findViewById(R.id.bt_check);
        mBt_count = (Button) findViewById(R.id.bt_count);
        mBt_check.setOnClickListener(this);
        mBt_count.setOnClickListener(this);
    }
    @Override
    public void onClick(View view) {
        if(view.getId()==R.id.bt_check){
            mLayout_username.setErrorEnabled(true);
            mLayout_username.setError("用户名错误");
        }else {
            mLayout_pwd.setCounterEnabled(true);
        }
    }
}

九 NavigationView 抽屉导航

这里写图片描述

十 运行时权限

核心方法 (需Android 6.0 (API level 23)及以上)
检查是否已获得权限
context.checkPermission(permission, android.os.Process.myPid(), Process.myUid())
检查是否需要弹出本地提示
activity.shouldShowRequestPermissionRational e(permission);
fragment.shouldShowRequestPermissionRationale(perm);
请求指定权限
activity.requestPermissions(permissions, requestCode);
fragment.requestPermissions(perms, requestCode);
获取权限申请结果
onRequestPermissionsResult
权限申请官方文档

https://developer.android.com/training/permissions/index.html
开源项目

https://github.com/googlesamples/easypermissions
https://github.com/Karumi/Dexter
https://github.com/hotchemi/PermissionsDispatcher


public class RunningPermissionActivity extends AppCompatActivity implements View.OnClickListener {
    private Button mCamera;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_runningpermission);
        mCamera = (Button)findViewById(R.id.camera);
        mCamera.setOnClickListener(this);
    }
    @Override
    public void onClick(View view) {
        //检查是否获取到权限
        String permission= Manifest.permission.CAMERA;
        int getPermission=checkPermission(permission, Process.myPid(),Process.myUid());//true false
        if(getPermission== PackageManager.PERMISSION_GRANTED){
            Toast.makeText(this,"已经获得权限",Toast.LENGTH_SHORT).show();
        }else {
            //判断是否弹出友好提示
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                if(shouldShowRequestPermissionRationale(permission)){
                    AlertDialog.Builder builder=new AlertDialog.Builder(this);
                    builder.setTitle("友情提示");
                    builder.setMessage("需要获取拍照权限");
                    builder.setNegativeButton("取消",null);
                    builder.setPositiveButton("确定", new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialogInterface, int i) {
                            requestPermission();
                        }
                    });
                    builder.show();
                }else {
                    requestPermission();
                }
            }
        }
    }
    private void requestPermission() {
        Toast.makeText(this,"没有获得权限",Toast.LENGTH_SHORT).show();
        //一次请求多个权限
        String [] permissions=new String[]{Manifest.permission.CAMERA};
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            requestPermissions(permissions,1);
        }
    }
    //返回权限结果
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        Toast.makeText(this,permissions[0]+"授权结果为:"+grantResults[0],Toast.LENGTH_SHORT).show();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值