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