【Android】MotionLayout实现动画效果

【Android】MotionLayout实现开场动画

在移动应用开发中,动画不仅仅是美化界面的工具,它更是提升用户体验的关键手段。Android 平台一直以来都提供了丰富的动画框架,但随着应用复杂性的增加,开发者对动画的需求也变得更加复杂和多样化。为了更好地应对这些需求,Google 在 ConstraintLayout 的基础上推出了 MotionLayout

效果展示

先来看看MotionLayout可以实现的效果吧~

basic-horizontal-mot -original-original

来看看笔者用MotionLayout做的蒟蒻效果:

6748a7b1411f56f328fa -original-original

关于MotionLayout

如何去制作这样一个流畅的动画效果呢?我们直接进入本篇博客的主题:MotionLayout

MotionLayout简述

首先来简单介绍一下MotionLayout。MotionLayout 是 Android ConstraintLayout 的扩展,集布局和动画于一体,旨在简化复杂的界面过渡和动画效果的实现。与传统的动画框架相比,MotionLayout 更加声明性、更易于可视化,并且能够处理非常复杂的动画场景。通过 MotionLayout,开发者可以在布局文件中定义多个状态,并在这些状态之间进行平滑的过渡,从而实现丰富的动画效果,例如转场动画、导航菜单的展开与收起等。

QQ_1725026153876

这是Google官方文档中MotionLayout的介绍,我们可以看到它继承于ConstraintLayout,这在本文的后半部分,具体的代码操作中会有所体现,学习MotionLayout的使用可能需要对ConstraintLayout有一定的了解。

那么,MotionLayout是如何去实现动画的呢?所谓的声明性,可视性又是什么呢?又该如何定义多个状态,进行平滑过渡呢?我们接着往下看。

MotionScene

要做到动画效果的实现,我们离不开一个关键的配置文件:MotionScene。

它是一个 XML 文件,定义了动画的所有相关信息,包括状态、过渡、关键帧等。MotionScene 通过描述 ConstraintSet(状态集)、Transition(过渡)、KeyFrame(关键帧)等来组织和管理动画的逻辑。

MotionScene 的整体结构可以分为以下几个主要部分:

  1. ConstraintSet: 定义不同的布局状态(开始状态和结束状态)。
  2. Transition: 定义状态之间的过渡及其控制方式。
  3. KeyFrame: 定义动画过程中的关键变化点。
  4. OnSwipe: 定义与用户手势交互的动画触发器。

ConstraintSet

ConstraintSet 是 MotionScene 的基础部分,用于定义组件在不同状态下的布局约束。在 MotionScene 中,通常至少定义两个 ConstraintSet,分别对应 startend 状态。

每一个 ConstraintSet 都包含了布局中所有控件的约束信息。通过定义多个 ConstraintSet,你可以控制动画在不同状态下的表现。(一个ConstraintSet即为一个状态下,UI控件的集合)

以下是一些常用的属性:

ConstraintSet 中每个 Constraint 的属性基本与 ConstraintLayout 的属性相同。以下是一些常用属性:

  • layout_constraintStart_toStartOf: 将组件的开始边缘与另一个组件的开始边缘对齐。
  • layout_constraintEnd_toEndOf: 将组件的结束边缘与另一个组件的结束边缘对齐。
  • layout_constraintTop_toTopOf: 将组件的顶部与另一个组件的顶部对齐。
  • layout_constraintBottom_toBottomOf: 将组件的底部与另一个组件的底部对齐。
  • layout_widthlayout_height: 定义组件的宽度和高度。
  • rotation: 设置组件的旋转角度。
  • alpha: 设置组件的透明度。

开始状态与结束状态(Start and End States)

MotionLayout 的动画本质上是在两个状态之间进行转换:开始状态(start)和结束状态(end)。这两个状态定义了动画的起点和终点,分别通过 ConstraintSet 在 XML 中描述。

  • 开始状态(Start State):这是动画的初始状态,定义了组件的布局、大小、位置、旋转角度等属性。
  • 结束状态(End State):这是动画的最终状态,描述了动画结束时组件应达到的目标属性。

在动画开始时,组件处于 start 状态,随着用户的操作或代码控制,MotionLayout 会将组件逐步过渡到 end 状态。这个转换过程是通过 Transition 来定义和控制的。

例如:笔者制作的动画效果的配置文件如下:

    <ConstraintSet android:id="@+id/start"><Constraintandroid:id="@+id/toolbar"android:layout_width="match_parent"android:layout_height="wrap_content"android:alpha="0.7"motion:layout_constraintEnd_toEndOf="parent"motion:layout_constraintStart_toStartOf="parent"motion:layout_constraintBottom_toTopOf="parent" /><Constraintandroid:id="@+id/rv_msg"android:layout_width="match_parent"android:layout_height="wrap_content"motion:layout_constraintStart_toEndOf="parent"motion:layout_constraintTop_toTopOf="@id/linearLayout" /><Constraintandroid:id="@+id/linearLayout"android:layout_width="match_parent"android:layout_height="60dp"android:layout_marginBottom="10dp"motion:layout_constraintTop_toBottomOf="parent" /></ConstraintSet><ConstraintSet android:id="@+id/end"><Constraintandroid:id="@+id/toolbar"motion:layout_constraintEnd_toEndOf="parent"android:layout_width="match_parent"android:layout_height="wrap_content"android:alpha="0.7"motion:layout_constraintTop_toTopOf="parent"motion:layout_constraintStart_toStartOf="parent" /><Constraintandroid:id="@+id/rv_msg"android:layout_width="match_parent"android:layout_height="0dp"motion:layout_constraintBottom_toTopOf="@id/linearLayout"motion:layout_constraintTop_toBottomOf="@+id/toolbar"motion:layout_constraintStart_toStartOf="parent" /><Constraintandroid:id="@+id/linearLayout"android:layout_width="match_parent"android:layout_height="60dp"android:layout_marginBottom="10dp"motion:layout_constraintBottom_toBottomOf="parent"motion:layout_constraintStart_toStartOf="parent" /></ConstraintSet>
  • start 状态下,所有元素(toolbarrv_msglinearLayout)都在屏幕外部,因此不可见。
  • end 状态下,所有元素都移入屏幕内,toolbar 位于顶部,rv_msgtoolbar 下面,并占据屏幕的中间部分,linearLayout 位于屏幕底部。

当 MotionLayout 执行从 startend 的过渡时,toolbarrv_msglinearLayout 将从屏幕外部滑动到它们在 end 状态下的位置,创建一个滑动进入的动画效果。

Transition

Transition 是 MotionLayout 中的关键概念,用于定义从一个状态(ConstraintSet)到另一个状态的过渡动画。Transition 不仅控制动画的时间和插值器,还可以定义触发动画的条件和动画的具体行为(例如响应触摸事件或特定的时间)。

basic-horizontal-mot -original-original

例如,在这个实例中,可以通过滑动来控制方块控件的位置,这一操作可以定义在Transition中。

在笔者制作的动画效果中,Transition如下:

    <Transitionmotion:constraintSetEnd="@+id/end"motion:constraintSetStart="@id/start"motion:duration="2000"motion:autoTransition="animateToEnd"></Transition>	
  • app:constraintSetStart: 定义动画的起始状态,这个属性引用了一个 ConstraintSet
  • app:constraintSetEnd: 定义动画的结束状态,这个属性也引用了一个 ConstraintSet
  • app:duration: 动画的持续时间,以毫秒为单位。

除了基本属性,Transition 还提供了一些高级属性和功能,帮助你更细致地控制动画行为。

触摸事件(OnSwipe 和 OnClick)

你可以通过 OnSwipeOnClick 来定义触摸事件,触发 Transition 的执行。

OnSwipe: 定义了在滑动手势下如何触发 Transition,例如上下滑动或左右滑动。

<OnSwipeapp:dragDirection="dragUp"app:touchAnchorId="@id/imageView"app:touchAnchorSide="bottom" />
  • app:dragDirection: 指定滑动的方向,例如 dragUp 表示向上滑动。
  • app:touchAnchorId: 触摸事件的锚点,即滑动时要跟踪的视图。
  • app:touchAnchorSide: 指定锚点的哪一侧被用作触摸参考点。

OnClick: 定义点击事件触发 Transition,例如点击某个按钮时触发动画。

<OnClickapp:targetId="@id/button"app:clickAction="transitionToEnd" />
  • app:targetId: 触发点击事件的目标视图 ID。
  • app:clickAction: 定义点击后的动作,例如 transitionToEnd 表示触发从 startend 的过渡。
自动过渡(AutoTransition)

Transition 还支持自动过渡,即在满足特定条件时自动触发动画。你可以通过以下属性控制自动过渡:

  • app:autoTransition
    设置自动过渡的类型。
    • jumpToEnd: 动画直接跳到结束状态。
    • animateToEnd: 动画自动播放到结束状态。
    • jumpToStart: 动画直接跳回开始状态。
    • animateToStart: 动画自动回到开始状态。

KeyFrame

KeyFrame 是 MotionLayout 中用于精确控制动画过程的工具。它允许你在动画的特定时间点(关键帧)对视图的属性进行自定义设置,以实现复杂的动画效果。KeyFrame 可以让你在动画过程中定义位置、属性变化、周期性动画等,从而获得更细致的动画表现。

其实就是实现帧动画

在 MotionLayout 中,KeyFrame 用于指定在动画过程中的某些关键时间点上视图的状态。这些关键时间点通常是动画的百分比位置(0% 到 100%),即动画开始到结束的过程中的特定位置。

可以实现如下的效果:

动画效果

KeyFrame 类型

MotionLayout 提供了几种不同类型的 KeyFrame,每种类型用于不同的动画控制需求:

  1. KeyPosition
  2. KeyAttribute
  3. KeyCycle
1. KeyPosition

KeyPosition 主要用于控制视图的位置变化。通过 KeyPosition,你可以定义在动画的特定时刻视图的位置(xy 坐标)。

属性

  • app:framePosition: 关键帧的位置,范围从 0 到 100,表示动画的百分比。
  • app:motionTarget: 关键帧作用的目标视图。
  • app:percentX: 目标视图在动画时刻的相对 X 坐标(0.0 到 1.0)。
  • app:percentY: 目标视图在动画时刻的相对 Y 坐标(0.0 到 1.0)。

示例

<KeyPositionapp:framePosition="50"app:motionTarget="@id/targetView"app:percentX="0.5"app:percentY="0.5" />

在这个示例中,KeyPosition 定义了在动画的 50% 位置上,targetView 的位置应为相对父视图的中心(50% X 和 50% Y)。

2. KeyAttribute

KeyAttribute 用于控制视图的属性变化,例如尺寸、透明度、旋转等。你可以在关键帧中定义这些属性在特定时刻的值。

属性

  • app:framePosition: 关键帧的位置,范围从 0 到 100,表示动画的百分比。
  • app:motionTarget: 关键帧作用的目标视图。
  • app:attributeName: 要设置的属性名称(如 alpharotationscaleXscaleY 等)。
  • app:attributeValue: 属性在关键帧时刻的值。

示例

<KeyAttributeapp:framePosition="50"app:motionTarget="@id/targetView"app:attributeName="alpha"app:attributeValue="0.5" />
3. KeyCycle

KeyCycle 用于控制周期性动画,即动画过程中的重复效果。它允许你创建周期性的动画效果,如摆动或波动。

属性

  • app:framePosition: 关键帧的位置,范围从 0 到 100,表示动画的百分比。
  • app:motionTarget: 关键帧作用的目标视图。
  • app:cycle: 循环次数或周期。
  • app:waveShape: 波形形状,例如 sinetriangle 等。
  • app:wavePeriod: 波形周期。

示例

<KeyCycleapp:framePosition="50"app:motionTarget="@id/targetView"app:waveShape="sine"app:wavePeriod="1"app:rotation="360" />

怎么创建MotionLayout

可以像写ConstraintLayout一样,直接在创建的时候就选择使用MotionLayout,但是笔者更加推荐先写一个ConstraintLayout,再转换为MotionLayout的方法。(对喜欢改UI的人比较友好)

首先我们拿到一个ConstraintLayout

QQ_1725028634488

右侧工具栏,启动!ComponentTree,启动!右键main然后点击Convert to MotionLayout,就可以自动转换成MotionLayout啦,同时也会自动创建一个MotionScene配置文件。

不想写代码怎么办

那么这个时候可能就有同学会问了,虽然已经把需要写的代码简化成这样了,但我还是不想写,怎么办?(谁问你了?)

不要慌,请看VCR:

QQ_1725028790092

我们把右侧工具栏拉出来,发现了…一个AndroidStudio自带的图形化工具!

可以在这里自由注册Constraint中的组件,也可以自由设置start以及end状态中组件的位置,大小,角度等等属性。

QQ_1725028892570

点击左上角的箭头,还可以直接设置Transition的属性,添加关键帧等等。

(但是这个只能作为便利开发的一种手段,笔者亲测感觉这个东西不是太好用,主要体现在设置填入的时候总是不及时反馈,有时候设置了许多东西,运行发现并未设置上,叫人抓狂得很)

结语

参考文档:

突破传统动画:探索MotionLayout的独特优势-腾讯云开发者社区-腾讯云 (tencent.com)

MotionLayout examples | Views | Android Developers (google.cn)

Android | MotionLayout入门级使用教程(一)_motionlayout的使用-CSDN博客

希望这篇文章可以给大家的UI开发做一些思路上的扩展~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/52889.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何通过WinRAR软件有效禁止RAR压缩包内文件的修改

RAR压缩包作为一种广泛使用的文件格式&#xff0c;凭借其高压缩比和强大的功能&#xff0c;成为了许多用户保存和传输文件的首选。然而&#xff0c;在某些情况下&#xff0c;我们可能希望确保RAR压缩包内的文件不被随意修改或删除&#xff0c;以维护文件的安全性和完整性。本文…

【网络】数据链路层-MAC帧

数据链路层-以太网与ARP协议 文章目录 1.数据链路层2.以太网2.1什么是以太网2.2MAC帧格式 3.ARP协议3.1为什么有ARP协议&#xff1f;3.2ARP的定位3.3ARP协议工作流程3.4ARP数据格式 4.RARP协议 1.数据链路层 数据链路层是网络协议栈中最底层的内容&#xff0c;而在之前对其他…

python基础学习(最终篇)

文章目录 JSON的基础使用一. JSON简介二. JSON语法规则三. JSON数据类型四. JSON对象五. JSON数组六. JSON函数1. json.dumps2. json.loads3. json.dump4. json.load5. encode6. decode7. 参数说明 总结 JSON的基础使用 一. JSON简介 JSON(JavaScript Object Notation) 是一种…

IOC 概述

一、IOC 概述 控制反转&#xff08;Inversion of Control&#xff0c;简称 IOC&#xff09;是一种设计原则&#xff0c;它通过将对象的创建和管理权交给外部容器来实现对象之间的解耦。这种模式使得组件之间的依赖关系变得更加灵活和可维护。在 Spring 框架中&#xff0c;IOC …

恶劣天气下的目标检测新突破:多尺度退化建模与特征融合策略

更多优质内容&#xff0c;请关注公众号&#xff1a;智驾机器人技术前线 1.论文信息 论文标题&#xff1a;Degradation Modeling for Restoration-enhanced Object Detection in Adverse Weather Scenes 作者&#xff1a;Xiaofeng Wang, Xiao Liu, Hong Yang, Zhengyong Wang, …

阿里PAI-ChatLearn:大规模 Alignment高效训练框架正式开源

导读 ChatGPT是OpenAI开发的基于大型语言模型(LLM)的聊天机器人&#xff0c;以其令人惊叹的对话能力而迅速火爆并被广泛采用。ChatGPT 成功背后得益于大型语言模型生成领域的新训练范式&#xff1a;RLHF (Reinforcement Learning from Human Feedback)&#xff0c;即以强化学习…

PTA L1-028 判断素数

L1-028 判断素数&#xff08;10分&#xff09; 本题的目标很简单&#xff0c;就是判断一个给定的正整数是否素数。 输入格式&#xff1a; 输入在第一行给出一个正整数N&#xff08;≤ 10&#xff09;&#xff0c;随后N行&#xff0c;每行给出一个小于的需要判断的正整数。 …

利用clip模型实现text2draw

参考论文 实践 有数据增强的代码 import math import collections import CLIP_.clip as clip import torch import torch.nn as nn from torchvision import models, transforms import numpy as np import webp from PIL import Image import skimage import torchvision …

滚柱导轨:数控机床高效运行的驱动力

机床制造者最关心的莫过于机床的精度&#xff0c;刚性和使用寿命&#xff0c;对导轨系统的关注甚少。但导轨为机床功能的实现奠定了可靠的基础&#xff0c;各种类型的机床工作部件&#xff0c;都是利用控制轴在指定的导轨上运动。机床设计者根据机床的类型和用途选用各种不同形…

C++ 中 static 关键字

修饰普通变量&#xff0c;修改变量的存储区域和生命周期&#xff0c;使变量存储在静态区&#xff0c;在 main 函数运行前就分配了空间&#xff0c;如果有初始值就用初始值初始化它&#xff0c;如果没有初始值系统用默认值初始化它。 修饰普通函数&#xff0c;表明函数的作用范围…

Python进阶04-网络编程

零、文章目录 Python进阶04-网络编程 1、计算机网络 网络相关知识请参考计算机网络详解 &#xff08;1&#xff09;IP地址的概念 IP 地址就是标识网络中设备的一个地址&#xff0c;好比现实生活中的家庭地址。 &#xff08;2&#xff09;IP地址的表现形式 IP 地址分为两类…

Javaweb学习之Vue数据绑定(五)

前期回顾 Javaweb学习之Vue实践小界面&#xff08;四&#xff09;-CSDN博客 认识数据绑定 Vue.js 中的数据绑定是一个核心概念&#xff0c;它极大地简化了前端开发中数据与视图&#xff08;DOM&#xff09;之间的同步问题。Vue.js 通过其响应式系统实现了数据的双向绑定&…

【Python Web开发】Flask+HTML学习笔记

目录 Flask框架一、安装flask库二、运行一个网页三、库函数及变量 HTML标签语言一、基本格式二、标签2.1 块级标签2.1.1 标题2.1.2 div2.1.3 图片2.1.4 列表2.1.5 表格 2.2 行内标签2.2.1 span2.2.2 超链接2.2.3 输入 2.3 其他标签2.3.1 提交表单 Flask框架 一、安装flask库 …

电阻柜柜体采用不锈钢材质有什么优点

电阻柜柜体采用不锈钢材质有什么优点&#xff1a; 1. **耐腐蚀性**&#xff1a;不锈钢材质具有极好的耐腐蚀性&#xff0c;能有效抵抗各种腐蚀物质的侵蚀&#xff0c;确保设备经久耐用。 2. **美观大方**&#xff1a;不锈钢表面光滑&#xff0c;质感细腻&#xff0c;呈现出大气…

Elasticsearch写入、读取、更新、删除以及批量操作(golang)

目录 1、背景 2、elasticsearch基础操作 2.1 创建es链接客户端 2.2 创建、删除索引 2.3 插入文档 2.3 查询文档 2.4 删除一条文档 2.5 更新文档 2.6 逻辑查询 2.7 滚动查询(批量查询) 2.8 批量插入数据 2.9 批量更新文档 2.10 批量删除文档 3、检索条件设置 4、测…

探索Unity与C#的无限潜能:从新手到高手的编程之旅

在数字创意与技术创新交织的今天&#xff0c;Unity游戏引擎凭借其强大的跨平台能力和灵活的编程接口&#xff0c;成为了无数开发者心中的首选。而C#&#xff0c;作为Unity的官方脚本语言&#xff0c;更是以其面向对象的特性和丰富的库支持&#xff0c;为游戏开发注入了无限可能…

Golang | Leetcode Golang题解之第375题猜数字大小II

题目&#xff1a; 题解&#xff1a; func getMoneyAmount(n int) int {f : make([][]int, n1)for i : range f {f[i] make([]int, n1)}for i : n - 1; i > 1; i-- {for j : i 1; j < n; j {f[i][j] j f[i][j-1]for k : i; k < j; k {cost : k max(f[i][k-1], f[…

关于Scrapy的那些事儿(四)Scrapy Shell

Scrapy Shell launch Scrapy shell 使用如下命令&#xff1a; scrapy shell <url>当运行scrapy shell的时候&#xff0c;它为我们提供了一些功能函数&#xff1a; shelp() :打印可用对象和快捷命令的帮助列表fetch&#xff08;request or url&#xff09;&#xff1a;…

CMake构建学习笔记11-minizip库的构建

准确来说&#xff0c;minizip其实是zlib提供的辅助工具&#xff0c;位于zlib库的contrib文件夹内。minizip提供了更为高级一点的接口&#xff0c;能直接操作文件进行压缩。不过&#xff0c;有点麻烦的是这个工具并没有提供CMake构建的方式。那么可以按照构建giflib的方式&#…

Java开发工程师-匹配性岗位(借鉴性质)

1.匹配性质 技能迁移:Java开发工程师通常具备较强的编程能力、逻辑思维和问题解决能力,这些技能可以迁移到其他领域。行业选择:考虑目前行业趋势以及未来发展方向,Java工程师可以转向大数据、人工智能、云计算等等领域。个人兴趣与职业发展:转行时个人的兴趣和职业发展规划…