首页/移动开发/flutter-animating-apps
F

flutter-animating-apps

by @flutterv1.0.0
0.0(0)

讲解如何在Flutter应用中实现动画效果、过渡和运动,以提升用户体验和界面互动性。

Flutter AnimationsImplicit AnimationsExplicit AnimationsHero AnimationsUI/UX DesignGitHub
安装方式
npx skills add flutter/skills --skill flutter-animating-apps
compare_arrows

Before / After 效果对比

1
使用前

在Flutter应用中实现复杂动画效果时,手动编写动画逻辑繁琐,性能优化困难。动画效果不流畅,用户体验受损,开发耗时。

使用后

高效实现Flutter应用中的动画效果、过渡和手势动效,提供流畅自然的用户体验。简化动画开发,优化性能,提升应用吸引力。

description SKILL.md


name: "flutter-animating-apps" description: "Implements animated effects, transitions, and motion in a Flutter app. Use when adding visual feedback, shared element transitions, or physics-based animations." metadata: model: "models/gemini-3.1-pro-preview" last_modified: "Thu, 12 Mar 2026 22:16:34 GMT"


Implementing Flutter Animations

Contents

Core Concepts

Manage Flutter animations using the core typed Animation system. Do not manually calculate frames; rely on the framework's ticker and interpolation classes.

  • Animation<T>: Treat this as an abstract representation of a value that changes over time. It holds state (completed, dismissed) and notifies listeners, but knows nothing about the UI.
  • AnimationController: Instantiate this to drive the animation. It generates values (typically 0.0 to 1.0) tied to the screen refresh rate. Always provide a vsync (usually via SingleTickerProviderStateMixin) to prevent offscreen resource consumption. Always dispose() controllers to prevent memory leaks.
  • Tween<T>: Define a stateless mapping from an input range (usually 0.0-1.0) to an output type (e.g., Color, Offset, double). Chain tweens with curves using .animate().
  • Curve: Apply non-linear timing (e.g., Curves.easeIn, Curves.bounceOut) to an animation using a CurvedAnimation or CurveTween.

Animation Strategies

Apply conditional logic to select the correct animation approach:

  • If animating simple property changes (size, color, opacity) without playback control: Use Implicit Animations (e.g., AnimatedContainer, AnimatedOpacity, TweenAnimationBuilder).
  • If requiring playback control (play, pause, reverse, loop) or coordinating multiple properties: Use Explicit Animations (e.g., AnimationController with AnimatedBuilder or AnimatedWidget).
  • If animating elements between two distinct routes: Use Hero Animations (Shared Element Transitions).
  • If modeling real-world motion (e.g., snapping back after a drag): Use Physics-Based Animations (e.g., SpringSimulation).
  • If animating a sequence of overlapping or delayed motions: Use Staggered Animations (multiple Tweens driven by a single AnimationController using Interval curves).

Workflows

Implementing Implicit Animations

Use this workflow for "fire-and-forget" state-driven animations.

  • Task Progress:
    • Identify the target properties to animate (e.g., width, color).
    • Replace the static widget (e.g., Container) with its animated counterpart (e.g., AnimatedContainer).
    • Define the duration property.
    • (Optional) Define the curve property for non-linear motion.
    • Trigger the animation by updating the properties inside a setState() call.
    • Run validator -> review UI for jank -> adjust duration/curve if necessary.

Implementing Explicit Animations

Use this workflow when you need granular control over the animation lifecycle.

  • Task Progress:
    • Add SingleTickerProviderStateMixin (or TickerProviderStateMixin for multiple controllers) to the State class.
    • Initialize an AnimationController in initState(), providing vsync: this and a duration.
    • Define a Tween and chain it to the controller using .animate().
    • Wrap the target UI in an AnimatedBuilder (preferred for complex trees) or subclass AnimatedWidget.
    • Pass the Animation object to the AnimatedBuilder's animation property.
    • Control playback using controller.forward(), controller.reverse(), or controller.repeat().
    • Call controller.dispose() in the dispose() method.
    • Run validator -> check for memory leaks -> ensure dispose() is called.

Implementing Hero Transitions

Use this workflow to fly a widget between two routes.

  • Task Progress:
    • Wrap the source widget in a Hero widget.
    • Assign a unique, data-driven tag to the source Hero.
    • Wrap the destination widget in a Hero widget.
    • Assign the exact same tag to the destination Hero.
    • Ensure the widget trees inside both Hero widgets are visually similar to prevent jarring jumps.
    • Trigger the transition by pushing the destination route via Navigator.

Implementing Physics-Based Animations

Use this workflow for gesture-driven, natural motion.

  • Task Progress:
    • Set up an AnimationController (do not set a fixed duration).
    • Capture gesture velocity using a GestureDetector (e.g., onPanEnd providing DragEndDetails).
    • Convert the pixel velocity to the coordinate space of the animating property.
    • Instantiate a SpringSimulation with mass, stiffness, damping, and the calculated velocity.
    • Drive the controller using controller.animateWith(simulation).

Examples

class StaggeredAnimationDemo extends StatefulWidget {
  @override
  State<StaggeredAnimationDemo> createState() => _StaggeredAnimationDemoState();
}

class _StaggeredAnimationDemoState extends State<StaggeredAnimationDemo> with SingleTickerProviderStateMixin {
  late AnimationController _controller;
  late Animation<double> _widthAnimation;
  late Animation<Color?> _colorAnimation;

  @override
  void initState() {
    super.initState();
    _controller = AnimationController(
      duration: const Duration(seconds: 2),
      vsync: this,
    );

    // Staggered width animation (0.0 to 0.5 interval)
    _widthAnimation = Tween<double>(begin: 50.0, end: 200.0).animate(
      CurvedAnimation(
        parent: _controller,
        curve: const Interval(0.0, 0.5, curve: Curves.easeIn),
      ),
    );

    // Staggered color animation (0.5 to 1.0 interval)
    _colorAnimation = ColorTween(begin: Colors.blue, end: Colors.red).animate(
      CurvedAnimation(
        parent: _controller,
        curve: const Interval(0.5, 1.0, curve: Curves.easeOut),
      ),
    );

    _controller.forward();
  }

  @override
  void dispose() {
    _controller.dispose(); // CRITICAL: Prevent memory leaks
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return AnimatedBuilder(
      animation: _controller,
      builder: (context, child) {
        return Container(
          width: _widthAnimation.value,
          height: 50.0,
          color: _colorAnimation.value,
        );
      },
    );
  }
}
Route createCustomRoute(Widget destination) {
  return PageRouteBuilder(
    pageBuilder: (context, animation, secondaryAnimation) => destination,
    transitionsBuilder: (context, animation, secondaryAnimation, child) {
      const begin = Offset(0.0, 1.0); // Start from bottom
      const end = Offset.zero;
      const curve = Curves.easeOut;

      final tween = Tween(begin: begin, end: end).chain(CurveTween(curve: curve));
      final offsetAnimation = animation.drive(tween);

      return SlideTransition(
        position: offsetAnimation,
        child: child,
      );
    },
  );
}

// Usage: Navigator.of(context).push(createCustomRoute(const NextPage()));

forum用户评价 (0)

发表评价

效果
易用性
文档
兼容性

暂无评价,来写第一条吧

统计数据

安装量0
评分0.0 / 5.0
版本1.0.0
更新日期2026年3月16日
对比案例1 组

用户评分

0.0(0)
5
0%
4
0%
3
0%
2
0%
1
0%

为此 Skill 评分

0.0

兼容平台

🔧Claude Code
🔧OpenClaw
🔧OpenCode
🔧Codex
🔧Gemini CLI
🔧GitHub Copilot
🔧Amp
🔧Kimi CLI

时间线

创建2026年3月16日
最后更新2026年3月16日