掌握CoordinatorLayout

2023-12-05 05:59
文章标签 掌握 coordinatorlayout

本文主要是介绍掌握CoordinatorLayout,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接

Mastering the Coordinator Layout


在Google I/0 2015大会上,Google发布了一个与Material Design相关的控件支持库,在这个控件库中我们可以找到很多新的ViewGroup控件,例如 AppBarLayout,CollapsingToolbarLayout和CoordinatorLayout.
考虑到这些新的控件非常有用,因此我决定写一篇文章来介绍其相关的配置和使用技巧.


CoordinatorLayout

这个控件的作用和它的名字描述一样,是用来协调它里面所有子view行为的控件.
考虑下面的例子:
simple_coordinator

我们可以看到View之间是相互配合的,View会根据其他View的滑动而做出相应的变化.

举一个使用CoordinatorLayout的简单例子:

<?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:background="@android:color/background_light"android:fitsSystemWindows="true"><android.support.design.widget.AppBarLayout
        android:id="@+id/id_main_appbar"android:layout_width="match_parent"android:layout_height="300dp"android:fitsSystemWindows="true"android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"><android.support.design.widget.CollapsingToolbarLayout
            android:id="@+id/id_main_collapsing"android:layout_width="match_parent"android:layout_height="match_parent"android:fitsSystemWindows="true"app:contentScrim="?attr/colorPrimary"app:expandedTitleMarginEnd="64dp"app:expandedTitleMarginStart="48dp"app:layout_scrollFlags="scroll|exitUntilCollapsed"><ImageView
                android:id="@+id/id_main_backdrop"android:layout_width="match_parent"android:layout_height="match_parent"android:contentDescription="@null"android:fitsSystemWindows="true"android:scaleType="centerCrop"android:src="@mipmap/ic_launcher"app:layout_collapseMode="parallax" /><android.support.v7.widget.Toolbar
                android:id="@+id/id_main_toolbar"android:layout_width="match_parent"android:layout_height="?attr/actionBarSize"app:layout_collapseMode="pin"app:popupTheme="@style/ThemeOverlay.AppCompat.Light" /></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"app:layout_behavior="@string/appbar_scrolling_view_behavior"><TextView
            android:layout_width="match_parent"android:layout_height="wrap_content"android:lineSpacingExtra="8dp"android:padding="@dimen/activity_horizontal_margin"android:text="@string/lorem"android:textSize="20sp" /></android.support.v4.widget.NestedScrollView><android.support.design.widget.FloatingActionButton
        android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_margin="@dimen/activity_horizontal_margin"android:src="@mipmap/ic_launcher"app:layout_anchor="@id/id_main_appbar"app:layout_anchorGravity="bottom|right|end" /></android.support.design.widget.CoordinatorLayout>

在上面的layout布局文件中,CoordinatorLayout直接包含了3个子控件,分别是:
AppBarLayout,NestedScrollView和FloatingActionBar.

<CoordinatorLayout><AppbarLayout/><scrollableView/><FloatingActionButton/>
</CoordinatorLayout>

AppBarLayout

AppBarLayout是一个继承LinearLayout的ViewGroup,默认是垂直方向.它可以控件它里面子控件的滚动行为.直接听起来可能比较疑惑,我用一个git图片来描述一下:
AppBarLayout

在上面的gif图片中,AppBarLayout是那个整体的蓝色View,它里面包含了可以收缩的图片,一个Toolbar,一个LinearLayout用来包含标题和副标题,以及一个TabLayout.

我们可以通过设置layout_scrollFlags来直接控制AppBarLayout中的子控件.在这个例子中,大部分子View的layout_scrollFlags属性都设置了scroll,如果没有设置的话,当可滚动的View进行滚动时,这些没有设置scroll属性的View的位置会保持不变.

layout_scrollFlags设置值为snap,可以避免进入动画中间状态,这意味着动画会一直持续到View完全显示或者完全隐藏为止.

LinearLayout其中包含了一个标题和一个副标题,当用户向上移动时LinearLayout是一直显示的,直到移出屏幕.而TabLayout是会一直可见的,因为我们没有在TabLayout上设置任何滚动属性.

正如你所见,AppbarLayout的强大管理能力是通过给子View设置不同的Scroll属性来实现的.

<AppBarLayout><CollapsingToolbarLayout
        app:layout_scrollFlags="scroll|snap"/><Toolbar
        app:layout_scrollFlags="scroll|snap"/><LinearLayout
        android:id="+id/title_container"app:layout_scrollFlags="scroll|enterAlways"/><TabLayout /> <!-- no flags -->
</AppBarLayout>

这些参数的设置需要参考Google Developers docs.我建议大家通过代码练习来掌握它们,我会在文章的末尾提供几个Github上的例子来供大家学习参考.


AppbarLayout flags

SCROLL_FLAG_ENTER_ALWAYS: 当任何向下滚动事件发生时,View都会移入,不管scrolling view是否正杂滚动.
SCROLL_FLAG_ENTER_ALWAYS_COLLAPSED: 它是”enterAlways”的附件属性,它是的View恢复到指定的最小高度后来开始显示,然后再慢慢展开.
SCROLL_FLAG_EXIT_UNTIL_COLLAPSED:当向上移出屏幕时,View会一直收缩到最小高度后,再移出屏幕.
SCROLL_FLAG_SCROLL:View会根据指定方向进行滚动.
SCROLL_FLAG_SNAP:当滚动结束时,如果这个View只有部分可见,它将会自动滚动到最近的边界(完全可见或者完全隐藏).


CoordinatorLayout Behaviors

让我们来做一个小测试,打开Android Studio(大于1.4版本),使用Scrolling Activity模板来创建一个Activity,编译运行之后如下所示:
appbarlayout

当我们来review生成的代码,不过是layout还是java类中我们都找不到FloatingActionBar滚动变化的源码.为什么呢?

答案在FloatingActionBar的源代码里,从Android Studio1.2版本开始,增加了ctrl+click可以查看源代码,源码如下:

/** Copyright (C) 2015 The Android Open Source Project**  Floating action buttons are used for a*  special type of promoted action. *  They are distinguished by a circled icon *  floating above the UI and have special motion behaviors *  related to morphing, launching, and the transferring anchor point.* *  blah.. blah.. */
@CoordinatorLayout.DefaultBehavior(FloatingActionButton.Behavior.class)
public class FloatingActionButton extends ImageButton {...public static class Behavior extends CoordinatorLayout.Behavior<FloatingActionButton> {private boolean updateFabVisibility(CoordinatorLayout parent, AppBarLayout appBarLayout, FloatingActionButton child {if (a long condition) {// If the anchor's bottom is below the seam, // we'll animate our FAB outchild.hide();} else {// Else, we'll animate our FAB back inchild.show();}}}...
}

负责缩放动画的是design library引入的Behavior新元素,在这个例子中是CoordinatorLayout.Behavior.它会根据一些滚动条件,来判断是否显示FloatingActionButton.


SwipeDismissBehavior

继续深入理解源码,我们可以在design support library库的源码中发现一个public类SwipeDismissBehavior.通过这个Behavior,我们可以通过CoordinatorLayout轻松实现滑动删除的效果.

swipe_dismiss

@Override
public void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_swipe_behavior);mCardView = (CardView) findViewById(R.id.swype_card);final SwipeDismissBehavior<CardView> swipe = new SwipeDismissBehavior();swipe.setSwipeDirection(SwipeDismissBehavior.SWIPE_DIRECTION_ANY);swipe.setListener(new SwipeDismissBehavior.OnDismissListener() {@Override public void onDismiss(View view) {Toast.makeText(SwipeBehaviorExampleActivity.this,"Card swiped !!", Toast.LENGTH_SHORT).show();}@Override public void onDragStateChanged(int state) {}});LayoutParams coordinatorParams = (LayoutParams) mCardView.getLayoutParams();coordinatorParams.setBehavior(swipe);
}

Custom Behaviors

创建一个自定义的Behavior,不像大家想象的那么难.在开始之前,我们必须明确两个元素: child和dependency.

dog

Childs and dependencies

child是指使用behavior的view,dependency是behavior的触发器,并和child进行互动.在这个例子中,child是一个ImageView,dependency是Toolbar,当Toolbar发生了移动,ImageView也跟随移动.

child_dependency

现在我们已经明确了child和dependency的概念,接下来我们讲一下如何实现.

第一步我们需要继承CoordinatorLayout.Behavoir<\T>,其中T是View对象,在上面的例子中是ImageView.接下来,我们必须重写两个方法:

  • layoutDependsOn
  • onDependentViewChanged

layoutDependsOn方法在layout布局发生变化时被回调,我们需要在dependency发生变化时返回true.在上面的例子中,当用户滑动屏幕时这个方法被自动调用,我们需要让child做出相应的反应.

@Override
public boolean layoutDependsOn(CoordinatorLayout parent, CircleImageView child, View dependency) {return dependency instance of Toolbar;
}

当layoutDependsOn返回true时,onDependentViewChanged函数就会被调用.在这个方法中,我们需要实现动画效果.

public boolean onDependentViewChanged(
CoordinatorLayout parent, CircleImageView avatar, View dependency) {
modifyAvatarDependingDependencyState(avatar, dependency);
}private void modifyAvatarDependingDependencyState(
CircleImageView avatar, View dependency) {//  avatar.setY(dependency.getY());//  avatar.setBlahBlat(dependency.blah / blah);
} 

完整的自定义Behavior代码:

public static class AvatarImageBehavior extendsCoordinatorLayout.Behavior<CircleImageView> {@Overridepublic boolean layoutDependsOn(
CoordinatorLayout parent, CircleImageView, child, View dependency) {return dependency instanceof Toolbar; } 

public boolean onDependentViewChanged(
CoordinatorLayout parent, CircleImageView avatar, View dependency) {
modifyAvatarDependingDependencyState(avatar, dependency);}private void modifyAvatarDependingDependencyState(CircleImageView avatar, View dependency) {//  avatar.setY(dependency.getY());//  avatar.setBlahBlah(dependency.blah / blah);}    
}

Resources

Coordinator Behavior Example - Github
Coordinator Examples - Github
Introduction to coordinator layout on Android - Grzesiek Gajewski

这篇关于掌握CoordinatorLayout的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/456424

相关文章

轻松掌握python的dataclass让你的代码更简洁优雅

《轻松掌握python的dataclass让你的代码更简洁优雅》本文总结了几个我在使用Python的dataclass时常用的技巧,dataclass装饰器可以帮助我们简化数据类的定义过程,包括设置默... 目录1. 传统的类定义方式2. dataclass装饰器定义类2.1. 默认值2.2. 隐藏敏感信息

生信代码入门:从零开始掌握生物信息学编程技能

少走弯路,高效分析;了解生信云,访问 【生信圆桌x生信专用云服务器】 : www.tebteb.cc 介绍 生物信息学是一个高度跨学科的领域,结合了生物学、计算机科学和统计学。随着高通量测序技术的发展,海量的生物数据需要通过编程来进行处理和分析。因此,掌握生信编程技能,成为每一个生物信息学研究者的必备能力。 生信代码入门,旨在帮助初学者从零开始学习生物信息学中的编程基础。通过学习常用

如何掌握面向对象编程的四大特性、Lambda 表达式及 I/O 流:全面指南

这里写目录标题 OOP语言的四大特性lambda输入/输出流(I/O流) OOP语言的四大特性 面向对象编程(OOP)是一种编程范式,它通过使用“对象”来组织代码。OOP 的四大特性是封装、继承、多态和抽象。这些特性帮助程序员更好地管理复杂的代码,使程序更易于理解和维护。 类-》实体的抽象类型 实体(属性,行为) -》 ADT(abstract data type) 属性-》成

JAVA初级掌握的J2SE知识(二)和Java核心的API

/** 这篇文章送给所有学习java的同学,请大家检验一下自己,不要自满,你们正在学习java的路上,你们要加油,蜕变是个痛苦的过程,忍受过后,才会蜕变! */ Java的核心API是非常庞大的,这给开发者来说带来了很大的方便,经常人有评论,java让程序员变傻。 但是一些内容我认为是必须掌握的,否则不可以熟练运用java,也不会使用就很难办了。 1、java.lang包下的80%以上的类

JAVA初级掌握的J2SE知识(一)

时常看到一些人说掌握了Java,但是让他们用Java做一个实际的项目可能又困难重重,在这里,笔者根据自己的一点理解斗胆提出自己的一些对掌握Java这个说法的标准,当然对于新手,也可以提供一个需要学习哪些内容的参考。另外这个标准仅限于J2SE部分,J2EE部分的内容有时间再另说。 1、语法:必须比较熟悉,在写代码的时候IDE的编辑器对某一行报错应该能够根据报错信息知道是什么样的语法错误并且知道

作为刚从事Java开发的小白,需要掌握哪些技能

作为一个刚踏入Java开发世界的小白,面对各种技术和工具,你可能会觉得有点不知所措。但是别担心,我会给你一个简单清晰的路线图,让你可以有条不紊地掌握基本技能,逐步成长为一名Java开发者。 1. 扎实的Java基础 Java的基础是你迈向高级开发的重要基石,建议从以下几个方面着手: 语法和基础概念:比如变量、条件语句、循环、方法、数组、面向对象编程(OOP)等等。这些基础如同建房子的地基,越

掌握Hive函数[2]:从基础到高级应用

目录 高级聚合函数 多进一出 1. 普通聚合 count/sum... 2. collect_list 收集并形成list集合,结果不去重 3. collect_set 收集并形成set集合,结果去重  案例演示 1. 每个月的入职人数以及姓名  炸裂函数  概述  案例演示 1. 数据准备 1)表结构 2)建表语句 3)装载语句 2. 需求 1)需求说明 2)答

掌握Go语言中的时间与日期操作

Go语言中的时间与日期操作 在编写程序时,处理时间和日期看似是一项无关紧要的任务,但在需要同步多个任务或从文本文件中读取时间时,它的重要性便凸显出来。Go语言中的time包为我们提供了丰富的时间与日期操作功能。本文将详细介绍如何在Go语言中解析时间与日期字符串、在不同的格式之间进行转换,以及如何按照所需的格式输出时间和日期。 初识time包 在学习如何解析字符串并将其转换为时间或日期之前,我

你读文献的方式可能错了!掌握这些技巧,让阅读事半功倍!

我是娜姐 @迪娜学姐 ,一个SCI医学期刊编辑,探索用AI工具提效论文写作和发表。 科研新手如何精读一篇论文? 很多科研新手,一上来就疯狂下载几十上百篇文献。囫囵吞枣看完了,还是什么都不知道,大脑一片空白。究竟该如何读文献收获最大? 大佬说,要积极阅读、频繁阅读。 什么是积极阅读? 相比被动阅读,积极阅读是指在阅读之前准备好问题、设置阅读目标、保持批判性,收获更多、进步更大的一种阅读

“掌握Linux网络编程艺术,构建跨平台的网络通信解决方案!“#Linux系统编程之网络编程

"掌握Linux网络编程艺术,构建跨平台的网络通信解决方案!"#Linux系统编程之网络 前言预备知识一、 网络编程概述1.1 网络编程概述图1.2 TCP/UDP对比(面试)1.3 端口号的作用 二、 字节序2.1 大端字节序(Big-Endian)2.2 小端字节序(Little-Endian)2.3 字节序的重要性2.4 字节序转换函数2.5 字节序示意图 三、 socket编程步骤四