原文连接: Introduction to MotionLayout (part II)php
这是系列文章「MotionLayout 介绍」的 part II。阅读以前请先查看 part I! (中文点这)android
在文本中,咱们将继续经过各类示例揭露基本的 MotionLayout 特性,介绍自定义属性 (custom attribute)、图像操做 (image operations) 和关键帧 (keyframes)。git
在part I的最后,咱们建立了一个引用MotionScene
的MotionLayout
(self-contained MotionScene)。咱们能够进一步利用它,来实现其余属性的过渡。github
事实上,最初的ConstraintSet
只封装了布局规则;可是为了丰富动画效果,咱们常常须要作些别的事(好比背景颜色)。在 ConstraintLayout 2.0,ConstraintSet
也能够存储自定义属性状态。查看下方动画,背景颜色跟着移动过程发生了改变。app
之前,你不得不在代码中处理此问题。如今,你能够直接经过 XML 指定属性:ide
<Constraint android:id="@+id/button" ...>
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#D81B60"/>
</Constraint>
复制代码
这是这个动画修改后的MotionScene
文件工具
<?xml version="1.0" encoding="utf-8"?>
<MotionScene xmlns:android="http://schemas.android.com/apk/res/android" xmlns:motion="http://schemas.android.com/apk/res-auto">
<Transition motion:constraintSetEnd="@+id/end" motion:constraintSetStart="@+id/start" motion:duration="1000" motion:interpolator="linear">
<OnSwipe motion:dragDirection="dragRight" motion:touchAnchorId="@+id/button" motion:touchAnchorSide="right" />
</Transition>
<ConstraintSet android:id="@+id/start">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginStart="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintStart_toStartOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#D81B60" />
</Constraint>
</ConstraintSet>
<ConstraintSet android:id="@+id/end">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginEnd="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintEnd_toEndOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#9999FF" />
</Constraint>
</ConstraintSet>
</MotionScene>
复制代码
自定义属性是用属性名字 (attributeName) 来指定的,属性名字须要和对象中的getter/setter
方法对应:布局
getName
(e.g. getBackgroundColor)setName
(e.g. setBackgroundColor)另外还须要指定属性值的类型:post
customColorValue
customIntegerValue
customFloatValue
customStringValue
customDimension
customBoolean
最后当咱们定义一个自定义属性,你须要同时定义开始 (start) 和结束 (end) 的ConstraintSet
动画
当咱们处理复杂的过渡时,经常须要对图像进行一些操做,而且对他们进行动画处理。ConstraintLayout2.0 引入了名为 ImageFilterView (继承与AppCompatImageView)的一个有用的工具类来轻松实现这一点。
下面是咱们在两张图片之间作的淡入淡出 (cross-fade) 效果:
首先咱们须要建立一个包含 ImageFilterView 的 MotionLayout 文件。
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.motion.MotionLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/motionLayout" app:layoutDescription="@xml/scene_04" android:layout_width="match_parent" android:layout_height="match_parent">
<android.support.constraint.utils.ImageFilterView android:id="@+id/image" android:background="@color/colorAccent" android:src="@drawable/roard" app:altSrc="@drawable/hoford" android:layout_width="64dp" android:layout_height="64dp"/>
</android.support.constraint.motion.MotionLayout>
复制代码
它与 ImageView 的主要区别在于altSrc
属性
<android.support.constraint.image.ImageFilterView android:id="@+id/image" ... android:src="@drawable/roard" app:altSrc="@drawable/hoford"/>
复制代码
在MotionScene 文件中使用对应的淡入淡出属性(crossfade
)
<?xml version="1.0" encoding="utf-8"?>
<MotionScene xmlns:android="http://schemas.android.com/apk/res/android" xmlns:motion="http://schemas.android.com/apk/res-auto">
<Transition motion:constraintSetEnd="@+id/end" motion:constraintSetStart="@+id/start" motion:duration="1000" motion:interpolator="linear">
<OnSwipe motion:dragDirection="dragRight" motion:touchAnchorId="@+id/image" motion:touchAnchorSide="right" />
</Transition>
<ConstraintSet android:id="@+id/start">
<Constraint android:id="@+id/image" android:layout_width="100dp" android:layout_height="100dp" android:layout_marginStart="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintStart_toStartOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="crossfade" motion:customFloatValue="0" />
</Constraint>
</ConstraintSet>
<ConstraintSet android:id="@+id/end">
<Constraint android:id="@+id/image" android:layout_width="100dp" android:layout_height="100dp" android:layout_marginEnd="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintEnd_toEndOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="crossfade" motion:customFloatValue="1" />
</Constraint>
</ConstraintSet>
</MotionScene>
复制代码
ImageFilterView 也提供了更多的功能:
饱和度 saturation : 0 = grayscale, 1 = original, 2 = hyper saturated 对比度 contrast : 1 = unchanged, 0 = gray, 2 = high contrast 色温 warmth : 1 = neutral, 2 = warm (red tint), 0.5 = cold (blue tint) 淡入淡出 crossfade (with
app:altSrc
)
这里有另一个例子显示怎么使用滤镜饱和度:
简单的指定自定义属性就能够操做饱和度:
<CustomAttribute motion:attributeName="saturation" motion:customFloatValue="1" />
复制代码
这里是这个例子使用的 MotionLayout 文件:
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.motion.MotionLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/motionLayout" app:layoutDescription="@xml/scene_05" android:layout_width="match_parent" android:layout_height="match_parent">
<android.support.constraint.utils.ImageFilterView android:id="@+id/image" android:src="@drawable/sunset2" android:scaleType="centerCrop" android:layout_width="match_parent" android:layout_height="300dp" />
</android.support.constraint.motion.MotionLayout>
复制代码
这里是对应的场景(Scene)文件:
<?xml version="1.0" encoding="utf-8"?>
<MotionScene xmlns:android="http://schemas.android.com/apk/res/android" xmlns:motion="http://schemas.android.com/apk/res-auto">
<Transition motion:constraintSetStart="@+id/start" motion:constraintSetEnd="@+id/end" motion:duration="1000">
<OnSwipe motion:touchAnchorId="@+id/image" motion:touchAnchorSide="top" motion:dragDirection="dragUp" />
</Transition>
<ConstraintSet android:id="@+id/start">
<Constraint android:id="@+id/image" android:layout_width="match_parent" android:layout_height="300dp" motion:layout_constraintStart_toStartOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="saturation" motion:customFloatValue="1" />
</Constraint>
</ConstraintSet>
<ConstraintSet android:id="@+id/end">
<Constraint android:id="@+id/image" android:layout_width="match_parent" android:layout_height="300dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintEnd_toEndOf="parent">
<CustomAttribute motion:attributeName="saturation" motion:customFloatValue="0" />
</Constraint>
</ConstraintSet>
</MotionScene>
复制代码
大多数状况下,MotionLayout 是使用“静止状态” (resting states) 的 ConstraintSets 实现的。这种方式下,咱们知道最终的布局结果将适应不一样的屏幕尺寸:从本质上来讲,MotionLayout的表现像一个典型的 ConstraintLayout。
原文:The general idea for MotionLayout is that “resting states” are implemented as ConstraintSets. This way, we know that the resulting layouts will correctly adapt to different screen sizes: essentially, MotionLayout will behave like a typical ConstraintLayout.
在某些状况,你可能但愿有一个中间状态——一个要通过的状态(a state to go through),而不是要停留的状态(not a state to stay in)。固然你能够指定两个以上的 ConstraintSets 来解决,可是更好的作法是使用 Keyframes。
关键帧能够应用于位置或属性值。它们主要让你在转换期间指定一个时间节点上的改变。
举个栗子,你可能想让一个控件在过渡进行到 25% 的时候转换变成红色。或者在过渡中的 50% 时,改成向上移动。
有多种设置位置关键帧 (KeyPosition) 的方法 (pathRelative、deltaRelative、parentRelative),咱们将在本系列的第4部分详细介绍。
简单介绍位置关键帧 (position keyframes),这里指定了在过渡进行到 50% 的时候,位置在屏幕高度的 25%处。
<Transition ...>
<KeyFrameSet>
<KeyPosition motion:keyPositionType="parentRelative" motion:percentY="0.25" motion:framePosition="50" motion:target="@+id/button"/>
</KeyFrameSet>
</Transition>
复制代码
最终的效果以下:
和以往同样,MotionLayout 文件仍然很是简单:
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.motion.MotionLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/motionLayout" app:layoutDescription="@xml/scene_06" android:layout_width="match_parent" android:layout_height="match_parent">
<View android:id="@+id/button" android:background="@color/colorAccent" android:layout_width="64dp" android:layout_height="64dp" />
</android.support.constraint.motion.MotionLayout>
复制代码
MotionScene 文件与咱们以前看到的很是类似,只是添加了一个KeyPosition
元素:
<?xml version="1.0" encoding="utf-8"?>
<MotionScene xmlns:android="http://schemas.android.com/apk/res/android" xmlns:motion="http://schemas.android.com/apk/res-auto">
<Transition motion:constraintSetStart="@+id/start" motion:constraintSetEnd="@+id/end" motion:duration="1000" motion:interpolator="linear">
<OnSwipe motion:touchAnchorId="@+id/button" motion:touchAnchorSide="right" motion:dragDirection="dragRight" />
<KeyFrameSet>
<KeyPosition motion:keyPositionType="parentRelative" motion:percentY="0.25" motion:framePosition="50" motion:target="@+id/button"/>
</KeyFrameSet>
</Transition>
<ConstraintSet android:id="@+id/start">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginStart="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintStart_toStartOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#D81B60"/>
</Constraint>
</ConstraintSet>
<ConstraintSet android:id="@+id/end">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginEnd="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintEnd_toEndOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#9999FF"/>
</Constraint>
</ConstraintSet>
</MotionScene>
复制代码
相似位置关键帧,你能够在过渡中的特定的指定属性值(使用KeyAttribute
)。
例如,咱们可能须要操纵对象在 50% 位置的时候,指定执行缩放和旋转,效果以下:
这能够经过在KeyFrameSet
中添加一个KeyAttribute
元素来实现:
<KeyFrameSet>
<KeyAttribute android:scaleX="2" android:scaleY="2" android:rotation="-45" motion:framePosition="50" motion:target="@id/button" />
</KeyFrameSet>
复制代码
MotionLayout 文件与前一个例子相同,惟一不一样的是 MotionScene 文件中添加了KeyAttribute
:
<?xml version="1.0" encoding="utf-8"?>
<MotionScene xmlns:android="http://schemas.android.com/apk/res/android" xmlns:motion="http://schemas.android.com/apk/res-auto">
<Transition motion:constraintSetStart="@+id/start" motion:constraintSetEnd="@+id/end" motion:duration="1000" motion:interpolator="linear">
<OnSwipe motion:touchAnchorId="@+id/button" motion:touchAnchorSide="right" motion:dragDirection="dragRight" />
<KeyFrameSet>
<KeyAttribute android:scaleX="2" android:scaleY="2" android:rotation="-45" motion:framePosition="50" motion:target="@id/button" />
<KeyPosition motion:keyPositionType="screenRelative" motion:percentY="0.2" motion:framePosition="50" motion:target="@id/button"/>
</KeyFrameSet>
</Transition>
<ConstraintSet android:id="@+id/start">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginStart="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintStart_toStartOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#D81B60"/>
</Constraint>
</ConstraintSet>
<ConstraintSet android:id="@+id/end">
<Constraint android:id="@+id/button" android:layout_width="64dp" android:layout_height="64dp" android:layout_marginEnd="8dp" motion:layout_constraintBottom_toBottomOf="parent" motion:layout_constraintEnd_toEndOf="parent" motion:layout_constraintTop_toTopOf="parent">
<CustomAttribute motion:attributeName="backgroundColor" motion:customColorValue="#9999FF"/>
</Constraint>
</ConstraintSet>
</MotionScene>
复制代码
第二章介绍了 MotionLayout 更高级的功能,给出了如何利用自定义属性和关键帧建立更引入注目的动画示例。
你能够在ConstraintLayout examples github repository查看这些示例的源码。
在本系列文章中还有更多内容: