极简实现酷炫动效:Flutter隐式动画指南第三篇自定义Flutter隐式动画

news2025/1/4 19:37:34

目录

前言

一、TweenAnimationBuilder

二、使用TweenAnimationBuilder实现的一些动画效果

1.调整透明度的动画

2.稍微复杂点的组合动画

3.数字跳动的动画效果


前言

        上两节博客分别介绍了Flutter中的隐式动画的基础知识以及使用隐式动画实现的一些动画效果。当系统提供的隐式动画不能满足我们的需求的时候,我们还可以通过TweenAnimationBuilder 自定义我们的隐式动画。

        下面进入正题,我们了解下TweenAnimationBuilder的一些知识。

一、TweenAnimationBuilder

        TweenAnimationBuilder 是 Flutter 中用于创建动画的便捷小部件,它提供了一种简单的方法来使用 Tween 定义的值在一段时间内生成平滑的过渡效果。适用于需要根据数值变化进行动画的场景,例如颜色、大小、透明度或位置等过渡效果。

        TweenAnimationBuilder 的主要属性包括:

  1. tween:用于描述动画的开始和结束值。例如,可以是 Tween<double>(begin: 0, end: 100)

  2. duration:动画的持续时间。

  3. builder:每一帧调用一次,构建动画效果的 UI,传入当前的动画值。

  4. onEnd:动画结束时的回调,可用于触发其他动。

二、使用TweenAnimationBuilder实现的一些动画效果

1.调整透明度的动画

        以下代码演示了如何使用 TweenAnimationBuilder 在两种透明度之间创建渐变动画效果:

        图1.透明度渐变的动画效果

代码如下:

import 'package:flutter/material.dart';

class TweenAnimationExample extends StatelessWidget {
  const TweenAnimationExample({super.key});

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        backgroundColor: Colors.purple,
        title: const Text("TweenAnimationBuilder ",style: TextStyle(color: Colors.white,fontWeight: FontWeight.bold,fontSize: 16),),
      ),
      body: Center(
        child: TweenAnimationBuilder(
          tween: Tween<double>(begin: 0.0, end: 1.0),
          duration: const Duration(seconds: 2),
          builder: (context, double opacity, child) {
            return Opacity(
              opacity: opacity,
              child: child,
            );
          },
          child: Container(
            width: 100,
            height: 100,
            color: Colors.blue,
          ),
        ),
      ),
    );
  }
}

在这个示例中,我们定义了一下属性:

  1. tween 定义了从 0.0 到 1.0 的透明度过渡。
  2. duration 设置动画持续时间为 2 秒。
  3. builder 函数在每一帧构建一个带有当前透明度的 Opacity 小部件,从而形成透明度渐变的动画效果。

2.稍微复杂点的组合动画

        我们还可以结合隐式动画的渐变、缩放、圆角、旋转等属性实现稍微复杂的动画效果。

        例如我们要实现图2所示的动画效果:

图2.复杂的动画效果

代码如下:

import 'package:flutter/material.dart';
import 'dart:math';

class ComplexAnimationExample extends StatefulWidget {
  const ComplexAnimationExample({super.key});

  @override
  State<ComplexAnimationExample> createState() => _ComplexAnimationExampleState();
}

class _ComplexAnimationExampleState extends State<ComplexAnimationExample> {
  bool _isAnimated = false;

  void _toggleAnimation() {
    setState(() {
      _isAnimated = !_isAnimated;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        backgroundColor: Colors.purple,
        title: const Text('TweenAnimationBuilder',style: TextStyle(color: Colors.white,fontWeight: FontWeight.bold,fontSize: 16),),      ),
      body: Center(
        child: GestureDetector(
          onTap: _toggleAnimation,
          child: TweenAnimationBuilder(
            duration: const Duration(seconds: 2),
            tween: Tween<double>(begin: 0, end: _isAnimated ? 2 * pi : 0),
            builder: (context, double rotation, child) {
              return Transform.rotate(
                angle: rotation,
                child: TweenAnimationBuilder(
                  duration: const Duration(seconds: 2),
                  tween: ColorTween(
                      begin: Colors.purple, end: _isAnimated ? Colors.orange : Colors.purple),
                  builder: (context, Color? color, child) {
                    return TweenAnimationBuilder(
                      duration: const Duration(seconds: 2),
                      tween: BorderRadiusTween(
                          begin: BorderRadius.circular(10),
                          end: BorderRadius.circular(_isAnimated ? 75 : 10)),
                      builder: (context, BorderRadius? borderRadius, child) {
                        return TweenAnimationBuilder(
                          duration: const Duration(seconds: 2),
                          tween: Tween<double>(
                              begin: 100, end: _isAnimated ? 200 : 100),
                          builder: (context, double size, child) {
                            return Container(
                              width: size,
                              height: size,
                              decoration: BoxDecoration(
                                color: color,
                                borderRadius: borderRadius,
                              ),
                              child: child,
                            );
                          },
                          child: const Center(
                            child: Text(
                              '点击看动画!',
                              style: TextStyle(
                                  fontSize: 14, color: Colors.white),
                            ),
                          ),
                        );
                      },
                    );
                  },
                ),
              );
            },
          ),
        ),
      ),
    );
  }
}

    在我们的例子中:

1. 旋转:最外层的 TweenAnimationBuilder 使用 Tween<double> 定义旋转角度的动画,从 0 到 2π,即旋转一整圈。

2. 颜色渐变:在旋转的基础上,嵌套了一个 TweenAnimationBuilder,使用 ColorTween 从紫色渐变到橙色。

3. 边框圆角:进一步嵌套了一个 BorderRadiusTween,实现容器边框从圆角半径 10 变化到 75。

4. 尺寸变化:最内层的 TweenAnimationBuilder 使用 Tween<double> 控制容器大小在 100 到 200 之间平滑过渡。

5. 点击切换动画:通过 GestureDetector 包裹,点击容器触发 _toggleAnimation() 方法,使 _isAnimated 状态切换并重建 Widget,从而实现动画的来回切换。

3.数字跳动的动画效果

        我们在使用一些支付app的时候,经常会看到图3所示的动画效果。

        我们看看如何使用TweenAnimationBuilder实现数字的渐变动画。

        大致的思路如下:使用 TweenAnimationBuilder 实现一个数字从 1000 到 2000 的渐变动画。TweenAnimationBuilder 能够逐步生成中间值,从而呈现数字的平滑渐变效果。下面是一个完整的示例代码,通过 Tween<int> 来定义数字范围并逐步更新显示的数值。

        图3.数字渐变动画

代码如下:

import 'package:flutter/material.dart';

class NumberTransitionExample extends StatefulWidget {
  const NumberTransitionExample({super.key});

  @override
  State<NumberTransitionExample> createState() => _NumberTransitionExampleState();
}
class _NumberTransitionExampleState extends State<NumberTransitionExample> {
  bool _isAnimating = false;

  void _toggleAnimation() {
    setState(() {
      _isAnimating = !_isAnimating;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        backgroundColor: Colors.purple,
        title: const Text('TweenAnimationBuilder',style: TextStyle(color: Colors.white,fontWeight: FontWeight.bold,fontSize: 16),),
      ),
      body: Center(
        child: TweenAnimationBuilder(
          tween: IntTween(begin: 1000, end: _isAnimating ? 2000 : 1000),
          duration: const Duration(seconds: 3),
          onEnd: () => debugPrint("Animation Complete"),
          builder: (context, int value, child) {
            return Text(
              '$value',
              style: const TextStyle(fontSize: 48, fontWeight: FontWeight.bold),
            );
          },
        ),
      ),
      floatingActionButton: FloatingActionButton(
        onPressed: _toggleAnimation,
        child: const Icon(Icons.play_arrow),
      ),
    );
  }
}

        在上面的例子中:Tween<int>:使用 IntTween 将动画的整数起点设为 1000,终点为 2000。点击按钮后,起点和终点之间的值会逐步更新。

        点击“播放”按钮后,屏幕上数字会从 1000 平滑过渡到 2000,直到动画完成。

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

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

相关文章

怎么能监控电脑屏幕?四个真心好用的电脑屏幕监控小妙招,一分钟看完!

怎么能监控电脑屏幕&#xff1f;这或许是许多家长、企业管理者和IT安全人员心中的疑问。 有人说&#xff1a;用魔法水晶球&#xff01; 当然&#xff0c;这个方法些许梦幻&#xff0c;现实中我们还是要依靠科技手段来实现电脑屏幕的监控。 接下来&#xff0c;我将为大家介绍四…

中仕公考:25年浙江省公务员考试今日开始报名

2025年浙江省公务员考试于今日开始报名&#xff0c;准备参加考试的各位考生不要错过报名时间! 报名时间&#xff1a;2024年11月6日9时—11月11日17时。 资格初审时间&#xff1a;2024年11月6日9时—11月13日17时。 准考证下载时间&#xff1a;2024年12月3日9时—12月8日17时…

【IEEE出版】第六届国际科技创新学术交流大会暨信息技术与计算机应用学术会议(ITCA 2024,12月06-08)

第六届国际科技创新学术交流大会暨信息技术与计算机应用学术会议&#xff08;ITCA 2024) 2024 6th International Conference on Information Technology and Computer Application 会议官网&#xff1a;itca2024.iaecst.org 会议时间&#xff1a;2024年12月06-08日 截稿时…

SpringMVC总结 我的学习笔记

SpringMVC总结 我的学习笔记 一、SpringMVC简介1.MVC2.SpringMVC概述3. SpringMVC中的核心组件4.SpringMVC核心架构流程 二、SpringMVC框架实例具体实现使用注解实现 四、数据处理及跳转1.结果跳转方式2.处理器方法的参数与返回值处理提交数据数据显示到前端 五、RestFul风格1.…

基于SpringBoot的免税商品优选购物商城的设计与实现

一、项目背景 从古至今&#xff0c;通过书本获取知识信息的方式完全被互联网络信息化&#xff0c;但是免税商品优选购物商城&#xff0c;对于购物商城工作来说&#xff0c;仍然是一项非常重要的工作。尤其是免税商品优选购物商城&#xff0c;传统人工记录模式已不符合当前社会…

【Python】计算机视觉应用:OpenCV库图像处理入门

计算机视觉应用&#xff1a;OpenCV库图像处理入门 在当今的数字化时代&#xff0c;计算机视觉&#xff08;Computer Vision&#xff09;已经渗透到各行各业&#xff0c;比如自动驾驶、智能监控、医疗影像分析等。而 Python 的 OpenCV 库&#xff08;Open Source Computer Visi…

Spring Boot开发入门教程

简介 Spring Boot是一个开源的Java基础框架&#xff0c;用于创建独立、生产级的基于Spring框架的应用程序。通过Spring Boot&#xff0c;你可以轻松地创建独立的、生产级的Spring应用程序。 环境准备 Java开发环境&#xff1a;确保你的机器上安装了Java 8或更高版本。Maven…

vue3入门知识(一)

vue3简介 性能的提升 打包大小减少41%初次渲染快55%&#xff0c;更新渲染快133%内存减少54% 源码的升级 使用Proxy代替defineProperty实现响应式重写虚拟DOM的实现和Tree-Shaking 新的特性 1. Composition API&#xff08;组合API&#xff09; setupref与reactivecomput…

【产品经理】工业互联网企业上市之路

树根互联2022年6月2日提交招股书之后&#xff0c;因财务资料超过六个月有效期加三个月延长期&#xff0c;2022年9月30日上市审核中止&#xff1b;2022年12月26日树根互联更新了2022年半年度财务资料&#xff0c;又九个月过去了&#xff0c;其上市进程将面临再一次中止。 处于上…

Centos 7系统一键安装宝塔教程

服务器推荐青鸟云服务器&#xff0c;2H2G低至16元/月 官网地址&#xff1a; 所有产品_香港轻量云 2核 2G-A型_青鸟云 推荐Finalshell软件连接至服务器&#xff0c;下载地址&#xff1a; https://dl.hostbuf.com/finalshell3/finalshell_windows_x64.exe 下载完成后连接服务…

Vue3-实现父子组件通信

Vue3-实现父子组件通信 父组件向子组件传值defineProps()代码示例 子组件给父组件传值defineEmits()示例代码 结语 &#x1f600;大家好&#xff01;我是向阳&#x1f31e;&#xff0c;一个想成为优秀全栈开发工程师的有志青年&#xff01; &#x1f4d4;今天来讨论前端中Vue3父…

Imperva 数据库与安全解决方案

Imperva是网络安全解决方案的专业提供商&#xff0c;能够在云端和本地对业务关键数据和应用程序提供保护。公司成立于 2002 年&#xff0c;拥有稳定的发展和成功历史并于 2014 年实现产值1.64亿美元&#xff0c;公司的3700多位客户及300个合作伙伴分布于全球各地的90多个国家。…

【TabBar嵌套Navigation案例-常见问题按钮-WebView-加载本地html文件 Objective-C语言】

一、接下来,我们来说,webView如何加载本地的html文件 1.把这里的http://www.baidu.com/ 如何替换成本地的html文件,实际上,我们只需要把URL替换一下就可以了, 然后呢,先给大家看一眼素材,在我们的help.json里边,比如说,第一个按钮, 如何领奖,这块儿有一个叫做htm…

搜维尔科技:【煤矿虚拟仿真】煤矿企业、高校、科研单位-多语言支持、数字孪生、交互式学习体验

品牌&#xff1a;SouVR 发票&#xff1a;支持专票、普票 单位&#xff1a;套 版本号&#xff1a;1.0 包装清单&#xff1a;软件1套 软件形式&#xff1a;U盘、光盘 运行环境&#xff1a;windows 应用对象&#xff1a;煤矿企业、高校、科研单位 系统配置&#xff1a;…

【CSS】标准怪异盒模型

概念 CSS 盒模型本质上是一个盒子&#xff0c;盒子包裹着HTML 元素&#xff0c;盒子由四个属性组成&#xff0c;从内到外分别是&#xff1a;content 内容、padding 内填充、border 边框、外边距 margin 盒模型的分类 W3C 盒子模型(标准盒模型) IE 盒子模型(怪异盒模型) 两种…

国标GB28181视频平台EasyCVR私有化视频平台工地防盗视频监控系统方案

一、方案背景 在当代建筑施工领域&#xff0c;安全监管和防盗监控是保障工程顺利进行和资产安全的关键措施。随着科技进步&#xff0c;传统的监控系统已不足以应对现代工地的安全挑战。因此&#xff0c;基于国标GB28181视频平台EasyCVR的工地防盗视频监控系统应运而生&#xf…

动态规划 —— dp 问题-打家劫舍II

1.打家劫舍II 题目链接&#xff1a; 213. 打家劫舍 II - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/house-robber-ii/ 2. 题目解析 通过分类讨论&#xff0c;将环形问题转换为两个线性的“打家劫舍|” 当偷第一个位置的时候&#xff0c;rob1在&#…

配置elk插件安全访问elk前台页面

编辑els配置文件vim elasticsearch.yml,添加以下配置文件 用elk用户&#xff0c;启动els服务 关闭防火墙&#xff0c;查看els启动是否成功&#xff0c;通过是否启动java进程来判断 或者通过查看是否启动9200和9300端口来判断是否启动 交互模式启动密码配置文件interactive表示交…

通过mybatis和mybatis plus 实现用户注册功能和基础的增删改查

1,切分application的环境并引入依赖 通过将application.yml进行切分&#xff0c;切分成三个不同的环境&#xff0c;生产环境和开发环境&#xff0c;我们可以在不同情况下所需要的不同环境上进行相关的配置 我们对代码进行一次切分 application.yml spring:application:name…

TypeError: Cannot read properties of undefined (reading ‘__asyncLoader‘)

项目场景&#xff1a; vue3element-plus 项目场景&#xff1a;vue3element-plustsvite的技术栈开发的后台&#xff0c;一个后台列表页面&#xff0c;使用了ElTable组件 问题描述 页面提示报一个好像是异步的问题 runtime-core.esm-bundler.js:2261 Uncaught (in promise) Ty…