Flutter开发进阶之动画

Flutter开发进阶之动画

在Flutter中,动画是至关重要的一个部分,它能够为应用程序提供更加丰富和生动的用户体验,Flutter中的动画系统是UI框架的核心功能之一,也是开发者学习Flutter框架的重要部分,由于动画原理在所有程序中都是相同的,即视觉暂留,因此理解这一原理对于更好地使用Flutter的动画系统至关重要;
动画在应用程序中具有许多用途,例如引导用户注意力、增强视觉效果、提供更好的交互体验等,通过精心设计的动画,应用程序可以变得更加有趣、吸引人,并提高用户的参与度和满意度;
Flutter开发进阶之动画
Flutter开发中动画可以有多种分类,其中可以通过是否模拟现实世界中物理行为物理动画和补间动画。

物理动画

物理动画模拟现实世界中的物理行为,如重力、弹性、摩擦力等;
Flutter的AnimatedPhysics组件提供了这种能力,可以为AnimatedPhysics组件提供一个AccelerationSpec和DecelerationSpec,以定义动画的加速和减速行为。

  Widget build(BuildContext context) {  return AnimatedPhysics(  mass: 100.0,  acceleration: 9.8, // 重力加速度  deceleration: 0.5, // 阻尼系数,决定物体下落后的减速程度  child: Container(  width: 200.0,  height: 200.0,  color: Colors.red,  ),  animationId: _animationId, // 动画ID,用于标识这个动画,用于后续的动画控制操作,例如暂停、恢复等。  );  }  

补间动画

补间动画是更传统的动画方式,它允许你定义动画的开始和结束状态,以及中间过渡状态;
Flutter的AnimatedBuilder和AnimatedCrossFade组件提供了这种能力,还可以通过Tween来构建。

  Widget build(BuildContext context) {  return AnimatedBuilder(  animation: _animationController,  builder: (context, child) {  if (_isFirst) {  return Container(color: Colors.red, child: child);  } else {  return Container(color: Colors.green, child: child);  }  },  child: AnimatedCrossFade(  duration: Duration(seconds: 1),  firstChild: Container(color: Colors.red), // 第一个显示的子组件,初始为红色容器  secondChild: Container(color: Colors.green), // 第二个显示的子组件,初始为绿色容器  crossfadeState: _isFirst ? 1 : 0, // 控制动画状态,true表示第二个子组件显示,false表示第一个子组件显示  firstChildBuilder: (BuildContext context, child, bool visible) { // 可选,用于定制第一个子组件的动画效果,这里我们保持默认行为,不进行定制。  return visible ? child : null;  },  secondChildBuilder: (BuildContext context, child, bool visible) { // 可选,用于定制第二个子组件的动画效果,这里我们保持默认行为,不进行定制。  return visible ? child : null;  },  ),  animationId: _animationController.id, // 动画ID,用于标识这个动画,用于后续的动画控制操作,例如暂停、恢复等。这里我们使用了和之前示例中一样的ID。  );  }  

还可以根据动画的触发方式(隐式和显式)分为隐式动画和显式动画。

隐式动画

隐式动画是Flutter框架自动处理的动画,通常可以使用对应的组件(AnimatedOpacity、AnimatedContainer、AnimatedPadding、AnimatedPositioned、AnimatedSwitcher等)来实现,通过简单的属性设置来实现动画效果,而不需要手动创建和管理动画控制器。

  
Widget build(BuildContext context) {  return Column(  mainAxisAlignment: MainAxisAlignment.center,  children: [  // 示例1:使用AnimatedOpacity创建动画  AnimatedOpacity(  value: _animationValue, // 动画值范围:0.0(完全透明)到1.0(完全不透明)  duration: _controller.duration, // 使用AnimationController的持续时间,确保动画同步。  child: Container(  width: 100,  height: 100,  color: Colors.red,  ),  ),  // 示例2:使用AnimatedContainer创建动画  AnimatedContainer(  duration: _controller.duration, // 使用AnimationController的持续时间,确保动画同步。  curve: Curves.bounceInOut, // 动画曲线  child: Container(  width: 100,  height: 50,  color: Colors.green,  ),  ),  // 其他使用动画效果的子项...  ],  );  
}

显式动画

显式动画是开发者需要手动创建和管理的动画,在Flutter中,这通常通过使用Animated组件(例如AnimatedPositioned,AnimatedBuilder等)以及AnimationController类来完成;
需要明确指定动画的起始状态、结束状态以及过渡的时间和曲线。

class MyAnimatedWidget extends StatefulWidget {    _MyAnimatedWidgetState createState() => _MyAnimatedWidgetState();  
}  class _MyAnimatedWidgetState extends State<MyAnimatedWidget> with SingleTickerProviderStateMixin {  AnimationController _controller;  Tween<double> _tween = Tween<double>(  begin: 0.0,  end: 200.0,  ).animate(_controller);    void initState() {  super.initState();  _controller = AnimationController(  vsync: this,  duration: Duration(seconds: 2), // 动画持续时间  );  }    Widget build(BuildContext context) {  return Column(  mainAxisAlignment: MainAxisAlignment.center,  children: [  AnimatedPositioned(  animation: _controller, // 将动画控制器绑定到AnimatedPositioned上  duration: _controller.duration, // 使用AnimationController的持续时间,确保动画同步。  x: 0, // x位置保持不变,仅y位置进行动画处理。  y: _tween.value, // 使用Tween的值作为y位置。随着动画的进行,y位置会从0变化到200。  child: Container(  width: 100,  height: 100,  color: Colors.red,  ),  ),  ],  );  }  
}

这些动画都是一些动画元素,可以将多个动画元素组装成复杂动画(Complex Animations),通常指交错动画、序列动画、交错动画、组合动画等;
除了使用系统提供的组件去构建动画外,我们还可以使用CustomPainter构建自定义路径动画。

自定义路径动画

自定义路径动画可以根据需要绘制任何形状的路径,并在动画中改变路径的形状、位置和颜色等属性实现相应的效果。

class MyPainter extends CustomPainter {    bool shouldRepaint(CustomPainter oldDelegate) {  // 返回true表示需要重新绘制,这里我们简单地将返回true,以便每次重新渲染时都会重新绘制路径。  return true;  }    void paint(Canvas canvas, Size size) {  // 获取屏幕的宽高,以便根据需要调整绘制的比例。  var paint = Paint()..strokeCap = StrokeCap.round;  paint.strokeWidth = 10.0;  paint.color = Colors.red; // 设置画笔颜色为红色。  // 创建路径并绘制。这里我们只是简单地将一个圆形的路径绘制在屏幕上。您可以根据需要创建复杂的路径。  var path = Path()..moveTo(0, 0)..circle(size.width / 2, size.height / 2, size.width / 4); // 在屏幕中心绘制一个圆形的路径。您可以根据需要修改路径的形状和位置。  canvas.drawPath(path, paint);  }  
}

除此之外,还可以通过使用Flutter对应的动画库来创建动画。

spring_simulation

spring_simulation是一种物理模拟器,用于创建具有弹簧效果的动画;
它基于弹簧动力学原理,通过模拟弹簧的伸展、压缩和反弹等行为,实现动画的逼真效果。

class MyApp extends StatelessWidget {    Widget build(BuildContext context) {  return MaterialApp(  title: 'SpringSimulation Example',  theme: ThemeData(primarySwatch: Colors.blue),  home: Scaffold(  appBar: AppBar(title: Text('SpringSimulation Example')),  body: Center(  child: SpringSimulation(  // 定义弹簧模拟器的参数  mass: 1.0, // 质量(kg)  stiffness: 400.0, // 刚度(N/m)  damping: 5.0, // 阻尼(N·s/m)  target: 0.0, // 目标位置(m)  // 定义弹簧的初始位置和速度  initialPosition: -1.0, // 初始位置(m)  initialVelocity: 2.0, // 初始速度(m/s)  // 定义弹簧的约束条件(可选)  constraints: SpringConstraints(min: -1.0, max: 1.0), // 限制弹簧的最小和最大位置范围  // 将弹簧附加到屏幕上,这里使用了一个简单的圆点作为示例  onUpdate: (mass, stiffness, damping, position, velocity) {  Positioned(  left: position * 300, // 将弹簧的位置转换为屏幕坐标  child: Container(  width: 30,  height: 30,  decoration: BoxDecoration(shape: BoxShape.circle), // 创建一个圆点作为弹簧的表示  child: Icon(Icons.play_arrow), // 使用播放箭头图标作为示例,您可以根据需要自定义图标或文本等UI元素。  ),  );  },  ),  ),  ),  );  }  
}

还有很多,就不一一列举了。
在Flutter开发中构建动画,需要注意的点是:
选择合适的动画库
根据项目需求选择合适的动画库,如Tween、Dart-Animation-Lib、flutter_animator等。
理解动画生命周期
Flutter中的动画有一个开始、执行和结束的过程。你需要确保在动画开始时设置动画的初始状态,并在动画结束时设置动画的结束状态。
使用状态管理
Flutter中的动画通常与状态变化相关联,因此,建议使用Flutter的状态管理解决方案,如Redux或Bloc,来管理动画状态。
优化性能
在构建动画时,需要注意性能问题,避免在动画执行期间进行不必要的渲染或计算操作,这可能导致动画卡顿。
测试不同设备和场景
在不同设备和不同场景下测试动画效果,以确保动画在不同环境下都能正常工作。
考虑用户反馈
根据用户反馈调整动画效果,以提供更好的用户体验。
遵循Material Design规范
如果项目需要遵循Material Design规范,确保动画效果与规范保持一致。

如何对动画优化性能

对于Flutter开发中的性能优化,有以下几个方向。
避免不必要的渲染和计算
在动画执行期间,应避免进行不必要的渲染或计算操作,这可以减少GPU和CPU的工作负担,提高动画性能;
比如使用Reusable或MemoizedListTile等复用组件,这些组件可以在列表中使用,避免在列表滚动时重复创建和销毁组件,从而提高性能;
使用const关键字,在可能的情况下,尽量使用const关键字来创建不可变对象,这样可以避免在动画执行期间进行不必要的重新构建;
避免在动画执行期间进行复杂的布局和计算,在动画执行期间,应避免进行复杂的布局和计算操作,这些操作可能会导致不必要的渲染和计算;
使用ListView的physics属性,通过设置physics属性为NeverScrollableScrollPhysics(),可以避免在滚动时触发不必要的布局和渲染;
使用WidgetsBinding.instance.addPostFrameCallback,这个方法可以用来在每一帧渲染后执行回调函数,可以在这里进行一些必要的计算和更新操作,避免在动画执行期间进行不必要的渲染和计算;
使用Binding.instance.addPostFrameCallback,这个方法与WidgetsBinding.instance.addPostFrameCallback类似,但作用范围更广,可以在整个应用中避免不必要的渲染和计算。
使用列表复用
在处理大量数据时,使用列表复用技术可以避免频繁创建和销毁列表项,从而提高性能;
比如通过使用ListTile的key属性,可以确保列表项的唯一性,从而实现列表复用;
另外,也可以使用Reusable组件或MemoizedListTile组件来实现列表复用。
使用GPU加速
Flutter的渲染引擎支持GPU加速,通过将部分或全部渲染工作交给GPU处理,可以提高动画性能。
优化Widget树
在构建动画时,应尽量减少Widget树的深度和复杂度,避免不必要的布局和重绘操作,从而提高性能。
使用帧调度器
通过使用Flutter的帧调度器,可以更好地控制动画的执行时间和帧率,从而提高性能。
使用Profile工具
Flutter提供了Profile工具,可以用来分析应用的性能瓶颈,通过使用Profile工具,可以找到性能问题并进行针对性的优化。
使用高性能的动画库
如Tween等高性能的动画库,可以提供更好的动画性能。

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

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

相关文章

架设一台NFS服务器,并按照以下要求配置

1、开放/nfs/shared目录&#xff0c;供所有用户查询资料 2、开放/nfs/upload目录&#xff0c;为192.168.xxx.0/24网段主机可以上传目录&#xff0c; 并将所有用户及所属的组映射为nfs-upload,其UID和GID均为210 3、将/home/tom目录仅共享给192.168.xxx.xxx这台主机&#xff0c;…

Linux系统:yum仓库

目录 一、yum 1、yum概述 2、yum仓库 3、yum实现过程原理 二、yum配置文件详解 1、主配置文件 2、yum仓库设置文件 3、yum日志文件 三、yum命令详解 1、查询 1.1 yum list [软件名] 1.2 yum info [软件名] 1.3 yum search <关键词> 1.4 yum provides <关…

【Web】CTFSHOW PHP特性刷题记录(全)

知其然知其所以然&#xff0c;尽量把每种特性都详细讲明白。 目录 web89 web90 web91 web92 web93 web94 web95 web96 web97 web98 web99 web100 web101 web102 web103 web104 web105 web106 web107 web108 web109 web110 web111 web112 web113 web…

Clickhouse表引擎之CollapsingMergeTree引擎的原理与使用

前言 继续上次关于clickhouse的一些踩坑点&#xff0c;今天讲讲另外一个表引擎——CollapsingMergeTree。这个对于引擎对于数据量较大的场景是个不错的选择。注意&#xff0c;选择clickhouse的一般原因都是为了高效率查询&#xff0c;提高用户体验感&#xff0c;说白了就是以空…

zabbix实验

目录 一、zabbix 自动发现与自动注册 1、zabbix 自动发现 ①关闭防火墙和安全机制 ②在服务端和客户端上配置 hosts 解析 ③在 Web 页面配置自动发现 2、zabbix 自动注册 ①环境准备 ②在服务端和客户端上配置 hosts 解析 ③修改 zabbix-agent2 配置文件 ④在 Web 页…

查看centos的CPU、内存、磁盘空间等配置信息

目录 查看CPU/proc/cpuinfo中的信息 查看内存/proc/meminfo中的信息 查看磁盘空间df 命令du命令使用fdisk命令 查看CPU /proc/cpuinfo中的信息 前置&#xff1a; [ltkjltkj front]$ cat /proc/cpuinfo| grep "physical id" physical id : 0 physical id : 0 physi…

7.5 MySQL对数据的增改删操作(❤❤❤)

7.5 MySQL对数据的基本操作 1. 提要2. 数据添加2.1 insert语法2.2 insert 子查询2.3 ignore关键字 3. 数据修改3.1 update语句3.2 update表连接 4. 数据删除4.1 delete语句4.2 delete表连接4.3 快速删除数据表全部数据 1. 提要 2. 数据添加 2.1 insert语法 2.2 insert 子查询 …

【白皮书下载】GPU计算在汽车中的应用

驾驶舱域控制器 (CDC) 是汽车 GPU 的传统应用领域。在这里&#xff0c;它可以驱动仪表板上的图形&#xff0c;与车辆保持高度响应和直观的用户界面&#xff0c;甚至为乘客提供游戏体验。随着车辆屏幕数量的增加和分辨率的提高&#xff0c;对汽车 GPU 在 CDC 中进行图形处理的需…

Hive数据定义(1)

hive数据定义是hive的基础知识&#xff0c;所包含的知识点有&#xff1a;数据仓库的创建、数据仓库的查询、数据仓库的修改、数据仓库的删除、表的创建、表的删除、内部表、外部表、分区表、桶表、表的修改、视图。本篇文章先介绍&#xff1a;数据仓库的创建、数据仓库的查询、…

HttpServletRequest getServerPort()、getLocalPort() 、getRemotePort() 区别

getRemotePort() 、getServerPort()、getLocalPort() request.getServerPort()、request.getLocalPort() 和 request.getRemotePort() 这三个方法都是获取与HTTP请求相关的端口信息的 客户端(如浏览器)通过某个随机分配的网络连接端口(7070) 向服务器发送HTTP请求( http://exam…

更适合3D项目的UI、事件交互!纯国产数字孪生引擎持续升级中!!!

UI和事件交互是3D可视化项目中最常见的模块&#xff0c;主要用于信息添加、展示&#xff0c;用来确保按照用户需求呈现内容并完成交互。 平时工作在进行UI和交互设计时&#xff0c;经常出现以下问题&#xff1a;UI过于复杂导致3D项目内交互效率低下&#xff0c;或者是结合3D项目…

【iOS】UIColor、CGColor、CIColor的区别和联系

编者在实验室小组的指导下&#xff0c;仿写了许多App&#xff0c;其中UI的颜色模仿也是令人头痛的点。设计颜色一般使用UIColor类方法直接获取颜色&#xff1a; 有时会使用 (UIColor *)colorWithRed:(CGFloat)red green:(CGFloat)green blue:(CGFloat)blue alpha:(CGFloat)alph…

【数据结构】二叉树-堆(top-k问题,堆排序,时间复杂度)

&#x1f308;个人主页&#xff1a;秦jh__https://blog.csdn.net/qinjh_?spm1010.2135.3001.5343&#x1f525; 系列专栏&#xff1a;《数据结构》https://blog.csdn.net/qinjh_/category_12536791.html?spm1001.2014.3001.5482 ​​ 目录 堆排序 第一种 ​编辑 第二种 …

【不用找素材】ECS 游戏Demo制作教程(3) 1.17

一、生成墓碑 新建脚本如下&#xff1a; using Unity.Entities; using Unity.Mathematics;namespace ECSdemo {public struct GraveyardRandom : IComponentData{public Random Value;}}扩充GraveyardMono如下&#xff1a; using Unity.Entities; using Unity.Mathematics; …

使用arcgis pro是类似的控件样式 WPF

1.资源加载 <controls:ProWindow.Resources><ResourceDictionary><ResourceDictionary.MergedDictionaries><extensions:DesignOnlyResourceDictionary Source"pack://application:,,,/ArcGIS.Desktop.Framework;component\Themes\Default.xaml&quo…

Unreal Engine(UE5)中构建离线地图服务

1. 首先需要用到3个软件&#xff0c;Unreal Engine&#xff0c;gis office 和 bigemap离线服务器 Unreal Engine下载地址:点击前往下载页面 Gis office下载地址:点击前往下载页面 Bigemap离线服务器 下载地址: 点击前往下载页面 Unreal Engine用于数字孪生项目开发&#x…

Qt绘画的使用

1. 绘图 绘图组件&#xff1a; 1、绘画对象 2、绘画位置 3、绘画工具 4、绘画时机 绘画时机&#xff1a; 当整个窗口或窗口的一部分需要重新绘制时&#xff0c;会调用绘制事件处理函数 void QWidget::paintEvent(QPaintEvent *event) 绘画对象&#xff1a; QPainter类&#xff…

axios的原理及源码解析

面试官&#xff1a;你了解axios的原理吗&#xff1f;有看过它的源码吗&#xff1f; 一、axios的使用 关于axios的基本使用&#xff0c;上篇文章已经有所涉及&#xff0c;这里再稍微回顾下&#xff1a; 发送请求 import axios from axios;axios(config) // 直接传入配置 axio…

龙腾荆楚 | 软件供应链安全检测中心落地襄阳

1月16日&#xff0c;襄阳市东津新区“园区提质、企业满园”行动暨2024年东津云谷首月重大项目集中签约活动圆满完成&#xff0c;开源网安城市级项目再下一城&#xff0c;分别与襄阳市政府、高校、国投签订战略合作协议&#xff0c;推动荆楚地区数字政府、数字经济、数字社会、数…

阿里云 linux Centos7 安装 Miniconda3 + 创建Python环境

1.下载miniconda &#xff08;1&#xff09;法一&#xff1a;可以去下载清华源的miniconda镜像源&#xff0c;选择自己需要的版本&#xff0c;然后上传到Linux服务器上&#xff0c;linux上使用请选择linux版本&#xff0c;如下&#xff1a; &#xff08;2&#xff09;法二&…