Flutter-動畫-實踐篇

1、瞭解AnimatedWidget

  • 一般咱們給一個Widget添加動畫的時候都須要監聽Animation的addListener()方法,並在這個方法裏面不停的調用setState()方法通知Weight進行重繪。
  • AnimatedWidget是Flutter封裝的用於執行動畫的助手類。使用它可使咱們建立一個可重用動畫的Widget。並且咱們也沒必要關心Weight在何時須要重繪,由於AnimatedWidget中會自動調用addListener()和setState()。
  • AnimatedWidget其實是一個StatefulWidget,它裏面是定義了一套Widget並由外部將執行的動畫傳進來,而後根據Animation的value使各個Widget作出相應的改變。
  • Flutter封裝了不少AnimatedWidget的助手類。SlideTransition、AlignTransitionPositionedTransitionRelativePositionedTransition等等。

 

// 自定義AnimatedWidget
class CustomAnimatedWidget extends AnimatedWidget {
  CustomAnimatedWidget({Key key, Animation<double> animation})
                : super(key: key, listenable: animation);

  Widget build(BuildContext context) {
    final Animation<double> custom = animation;
    return new Center(
      child: new Container(
        height: animation.value,
        width: animation.value,
        child: new Container(),
      ),
    );
  }
}

// 使用CustomAnimatedWidget
class CustomApp extends StatefulWidget {
  _CustomAppState createState() => new _CustomAppState();
}

class _CustomAppState extends State<LogoApp> with SingleTickerProviderStateMixin {
  AnimationController controller;
  Animation<double> animation;

  initState() {
    super.initState();
    controller = new AnimationController(
        duration: const Duration(milliseconds: 5000), vsync: this);
    animation = new Tween(begin: 0.0, end: 100.0).animate(controller);
    controller.forward();
  }

  Widget build(BuildContext context) {
    return new CustomAnimatedWidget(animation: animation);
  }

  dispose() {
    controller.dispose();
    super.dispose();
  }
}
View Code

 

 

2、瞭解AnimatedBuilder

  • AnimatedBuilder相比較AnimatedWidget來講更加純粹。它不知道如何渲染Widget,也不會管理Animatiion。我的感受AnimatedWidget是一個執行具體動畫的控件,而AnimatedBuilder則是一個執行特定動畫的容器。
  • Flutter中也建立了不少基於AnimatedBuilder的控件。例如:BottomSheet、 PopupMenu、ProgressIndicator、RefreshIndicator、Scaffold、SnackBar、TabBar、TextField。

 

// 自定義CustomAnimatedBuilder
class CustomAnimatedBuilder extends StatelessWidget {
  GrowTransition({this.child, this.animation});

  final Widget child;
  final Animation<double> animation;

  Widget build(BuildContext context) {
    return new Center(
      child: new AnimatedBuilder(
          animation: animation,
          builder: (BuildContext context, Widget child) {
            return new Container(
                height: animation.value, width: animation.value, child: child);
          },
          child: child),
    );
  }
}

// 使用CustomAnimatedBuilder
class CustomApp extends StatefulWidget {
  _CustomAppState createState() => new _CustomAppState();
}

class _CustomAppState extends State<LogoApp> with TickerProviderStateMixin {
  Animation animation;
  AnimationController controller;

  initState() {
    super.initState();
    controller = new AnimationController(
        duration: const Duration(milliseconds: 1000), vsync: this);
    final CurvedAnimation curve =
        new CurvedAnimation(parent: controller, curve: Curves.easeIn);
    animation = new Tween(begin: 0.0, end: 100.0).animate(curve);
    controller.forward();
  }

  Widget build(BuildContext context) {
    return new CustomAnimatedBuilder(child: ‘本身的view’, animation: animation);
  }

  dispose() {
    controller.dispose();
    super.dispose();
  }
}
View Code

 

 

3、動畫示例 

一、位移動畫

只要Widget能在必定的時間內按照必定的規則位移必定的距離,那邊是產生了位移動畫。能夠經過改變Widget自己的margin,也能夠經過改變父容器的padding,也能夠經過SlideTransition的Offset產生位移,也可以使用Matrix4的transform產生移動(Matrix4解釋和使用)。下面看示例:html

 

// 位移動畫 copy 代碼能夠直接使用
import 'package:flutter/material.dart';

class TransferAnim extends StatefulWidget {
  @override
  _TransferAnimState createState() => _TransferAnimState();
}

// ignore: slash_for_doc_comments
/**
 * 這個實現 其實是改變 父容器的padding/margin完成的
 */
class _TransferAnimState extends State<TransferAnim>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;
  Animation<EdgeInsets> anim;

  Animation<Offset> slideTransition;

  @override
  void initState() {
    super.initState();
    _initAnim();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("位移動畫"),
      ),
      body: Column(
        children: <Widget>[
          Expanded(
            child: Container(
              padding: anim.value,
              child: Center(
                child: Container(
                  width: 100,
                  height: 50,
                  margin: ,
                  color: Colors.amber,
                  child: Center(
                    child: Text("位移動畫"),
                  ),
                ),
              ),
            ),
          ),
          Expanded(
            child: Container(
              child: Center(
                child: SlideTransition(
                  position: slideTransition,
                  child: Container(
                    width: 100,
                    height: 50,
                    color: Colors.amber,
                    child: Center(
                      child: Text("位移動畫"),
                    ),
                  ),
                ),
              ),
            ),
          ),
        ],
      ),
    );
  }

  void _initAnim() {
    _controller = AnimationController(
      vsync: this,
      duration: Duration(
        seconds: 3,
      ),
    );
    anim = new EdgeInsetsTween(
      begin: EdgeInsets.only(left: 0, top: 0),
      end: EdgeInsets.only(left: 100, top: 150),
    ).animate(_controller);

    //Offset  這裏解釋一下,是相對於本身移動的比例倍數
    slideTransition = Tween<Offset>(
      begin: Offset(0, 0),
      end: Offset(0, 2),
    ).animate(_controller);

    anim.addListener(() {
      setState(() {});
    });

    anim.addStatusListener((status) {
      debugPrint('fanlei => $status');
      switch (status) {
        case AnimationStatus.dismissed:
          _controller?.forward();
          break;
        case AnimationStatus.forward:
          break;
        case AnimationStatus.reverse:
          break;
        case AnimationStatus.completed:
          _controller?.reverse();
          break;
      }
    });

    _controller?.forward();
  }

  @override
  void dispose() {
    _controller?.dispose();
    super.dispose();
  }
}
View Code

 

 

二、旋轉動畫

旋轉動畫就是一個Weight以某個點或者某個座標軸旋轉。可使用Container()的transform(接受Matrix4)屬性,也可使用RotationTransition()執行旋轉動畫。下面看示例:android

 

import 'package:flutter/material.dart';

class RotateAnim extends StatefulWidget {
  @override
  _RotateAnimState createState() => _RotateAnimState();
}
// ignore: slash_for_doc_comments

class _RotateAnimState extends State<RotateAnim>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;
  Animation<Matrix4> anim;

  Animation<double> doubleAnim;

  @override
  void initState() {
    super.initState();
    _initAnim();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("旋轉動畫"),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Expanded(
              child: Center(
                child: Container(
                  transform: anim.value,
                  width: 200,
                  height: 50,
                  color: Colors.amber,
                  child: Center(
                    child: Text("旋轉動畫矩陣變換"),
                  ),
                ),
              ),
            ),
            Expanded(
              child: Center(
                child: RotationTransition(
                  turns: doubleAnim,
                  child: Container(
                    width: 200,
                    height: 50,
                    color: Colors.greenAccent,
                    child: Center(
                      child: Text("旋轉動畫Rotation"),
                    ),
                  ),
                ),
              ),
            ),
          ],
        ),
      ),
    );
  }

  void _initAnim() {
    _controller = AnimationController(
      vsync: this,
      duration: Duration(
        seconds: 3,
      ),
    );
    anim = new Matrix4Tween(
      begin: Matrix4.rotationZ(0),
      end: Matrix4.rotationZ(2.0),
    ).animate(_controller);
    anim.addListener(() {
      setState(() {});
    });

    anim.addStatusListener((status) {
      debugPrint('fanlei => $status');
      switch (status) {
        case AnimationStatus.dismissed:
          _controller?.forward();
          break;
        case AnimationStatus.forward:
          break;
        case AnimationStatus.reverse:
          break;
        case AnimationStatus.completed:
          _controller?.reverse();
          break;
      }
    });

    doubleAnim = Tween<double>(
      begin: 0,
      end: 1.0,
    ).animate(_controller);

    _controller?.forward();
  }

  @override
  void dispose() {
    _controller?.dispose();
    super.dispose();
  }
}
View Code

 

 

三、縮放動畫

本文檔使用了兩種方式來達到縮放的效果。一種是直接使用Animation改變Container()的width和height的值。另外一種則是使用ScaleTransition(),其原理也是使用了Matrix4。下面看代碼示例:git

 

import 'package:flutter/material.dart';

class ScaleAnim extends StatefulWidget {
  @override
  _ScaleAnimState createState() => _ScaleAnimState();
}
// ignore: slash_for_doc_comments

class _ScaleAnimState extends State<ScaleAnim>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;
  Animation<double> animWidth;
  Animation<double> animHeight;

  Animation<double> animScale;

  @override
  void initState() {
    super.initState();
    _initAnim();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("縮放動畫"),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Expanded(
              child: Center(
                child: Container(
                  width: animWidth.value,
                  height: animHeight.value,
                  color: Colors.amber,
                  child: Center(
                    child: Text("縮放動畫"),
                  ),
                ),
              ),
            ),
            Expanded(
              child: ScaleTransition(
                scale: animScale,
                child: Center(
                  child: Container(
                    width: 200,
                    height: 50,
                    color: Colors.greenAccent,
                    child: Center(
                      child: Text("縮放動畫"),
                    ),
                  ),
                ),
              ),
            ),
          ],
        ),
      ),
    );
  }

  void _initAnim() {
    _controller = AnimationController(
      vsync: this,
      duration: Duration(
        seconds: 3,
      ),
    );
    animWidth = new Tween<double>(
      begin: 100,
      end: 300,
    ).animate(_controller);
    animWidth.addListener(() {
      setState(() {});
    });

    animHeight = new Tween<double>(
      begin: 50,
      end: 150,
    ).animate(_controller);

    animScale = new Tween<double>(
      begin: 1,
      end: 1.5,
    ).animate(_controller);

    animWidth.addStatusListener((status) {
      debugPrint('fanlei => $status');
      switch (status) {
        case AnimationStatus.dismissed:
          _controller?.forward();
          break;
        case AnimationStatus.forward:
          break;
        case AnimationStatus.reverse:
          break;
        case AnimationStatus.completed:
          _controller?.reverse();
          break;
      }
    });

    _controller?.forward();
  }

  @override
  void dispose() {
    _controller?.dispose();
    super.dispose();
  }
}
View Code

 

 

四、透明度動畫(漸隱漸現)

本文檔使用了兩種方式來達到漸隱漸現的效果。第一種使用了Opacity(),經過Animation改變其opacity屬性。第二種則是使用FadeTransition()。下面看代碼示例:github

 

import 'package:flutter/material.dart';

class FadeAnim extends StatefulWidget {
  @override
  _FadeAnimState createState() => _FadeAnimState();
}
// ignore: slash_for_doc_comments

class _FadeAnimState extends State<FadeAnim>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;
  Animation<double> anim;

  Animation<double> fadeTransition;

  @override
  void initState() {
    super.initState();
    _initAnim();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("透明度動畫"),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Expanded(
              child: Center(
                child: Opacity(
                  opacity: anim.value,
                  child: Container(
                    width: 200,
                    height: 50,
                    color: Colors.amber,
                    child: Center(
                      child: Text("透明度動畫"),
                    ),
                  ),
                ),
              ),
            ),
            Expanded(
              child: Center(
                child: FadeTransition(
                  opacity: fadeTransition,
                  child: Container(
                    width: 200,
                    height: 50,
                    color: Colors.greenAccent,
                    child: Center(
                      child: Text("透明度動畫"),
                    ),
                  ),
                ),
              ),
            ),
          ],
        ),
      ),
    );
  }

  void _initAnim() {
    _controller = AnimationController(
      vsync: this,
      duration: Duration(
        seconds: 3,
      ),
    );
    anim = new Tween<double>(
      begin: 1,
      end: 0.2,
    ).animate(_controller);
    anim.addListener(() {
      setState(() {});
    });

    anim.addStatusListener((status) {
      debugPrint('fanlei => $status');
      switch (status) {
        case AnimationStatus.dismissed:
          _controller?.forward();
          break;
        case AnimationStatus.forward:
          break;
        case AnimationStatus.reverse:
          break;
        case AnimationStatus.completed:
          _controller?.reverse();
          break;
      }
    });

    fadeTransition = Tween<double>(
      begin: 0,
      end: 1,
    ).animate(_controller);

    _controller?.forward();
  }

  @override
  void dispose() {
    _controller?.dispose();
    super.dispose();
  }
}
View Code

 

 

五、非線性曲線動畫

非線性曲線動畫其主要類是使用CurvedAnimation建立一個非線性的Animation。CurvedAnimation的curve屬性接受一個Curve類。Flutter SDK API中的Curves類是Flutter已經爲咱們寫好了的各類非線性曲線。Curves非線性gif示例。下面看代碼示例:json

 

import 'package:flutter/material.dart';
// 本示例融合前面四種動畫
class NonlinearAnim extends StatefulWidget {
  @override
  _NonlinearAnimState createState() => _NonlinearAnimState();
}

class _NonlinearAnimState extends State<NonlinearAnim>
    with SingleTickerProviderStateMixin {
  // 位移
  AnimationController _animController;
  CurvedAnimation _translateCurved;
  Animation<double> _translateAnim;

  CurvedAnimation _scaleCurved;
  Animation<double> _scaleAnim;

  // 旋轉

  CurvedAnimation _rotationCurved;
  Animation<double> _rotationAnim;

  // 透明度
  CurvedAnimation _opacityCurved;
  Animation<double> _opacityAnim;

  @override
  void initState() {
    _initTransfer();
    _initScale();
    _initRotation();
    _initOpacity();
    _initListener();
    super.initState();
  }

  @override
  void dispose() {
    _animController?.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("非線性曲線動畫"),
      ),
      body: Container(
        child: GridView.count(
          crossAxisCount: 2,
          padding: EdgeInsets.only(left: 10, right: 10, top: 10),
          mainAxisSpacing: 10,
          crossAxisSpacing: 10,
          children: <Widget>[
            GestureDetector(
              onTap: () {
                if (!_animController.isAnimating) {
                  _animController.forward();
                }
              },
              child: Container(
                color: Colors.amber,
                child: Stack(
                  children: <Widget>[
                    Align(
                      alignment: Alignment.topCenter,
                      child: Text("bounceOut"),
                    ),
                    Align(
                      alignment: Alignment.bottomCenter,
                      child: Transform.translate(
                        offset: Offset(0, _translateAnim.value),
                        child: Container(
                          height: 40,
                          width: 40,
                          color: Colors.white,
                        ),
                      ),
                    )
                  ],
                ),
              ),
            ),

//            GestureDetector(
//              onTap: (){
//                _animController.forward();
//              },
//              child: ,
//            ),
            GestureDetector(
              onTap: () {
                _animController.forward();
              },
              child: Container(
                color: Colors.cyan,
                child: Center(
                  child: Container(
                    width: _scaleAnim.value,
                    height: _scaleAnim.value,
                    color: Colors.white,
                  ),
                ),
              ),
            ),
            GestureDetector(
              onTap: () {
                _animController.forward();
              },
              child: Container(
                color: Colors.green,
                child: Center(
                  child: RotationTransition(
                    turns: _rotationAnim,
                    child: Container(
                      height: 40,
                      width: 40,
                      color: Colors.white,
                    ),
                  ),
                ),
              ),
            ),
//            AnimatedOpacity(opacity: null, duration: null)

            Container(
              color: Colors.indigoAccent,
              child: Center(
                child: Opacity(
                  opacity:_getOpacityValue(_opacityAnim.value),
                  child: Container(
                    height: 40,
                    width: 40,
                    color: Colors.white,
                  ),
                ),
              ),
            ),
          ],
        ),
      ),
    );
  }

  void _initTransfer() {
    _animController =
        AnimationController(vsync: this, duration: Duration(seconds: 2));
    _translateCurved =
        CurvedAnimation(parent: _animController, curve: Curves.bounceOut);
    _translateAnim = Tween<double>(
      begin: 0,
      end: -100,
    ).animate(_translateCurved);

    _translateAnim.addListener(() {
      setState(() {});
    });
  }

  void _initScale() {
    _scaleCurved =
        CurvedAnimation(parent: _animController, curve: Curves.easeInOutBack);
    _scaleAnim = Tween<double>(
      begin: 40,
      end: 140,
    ).animate(_scaleCurved);
  }

  void _initRotation() {
    _rotationCurved =
        CurvedAnimation(parent: _animController, curve: Curves.easeInOutBack);
    _rotationAnim = Tween<double>(
      begin: 0,
      end: 1,
    ).animate(_rotationCurved);
  }

  void _initOpacity() {
    _opacityCurved =
        CurvedAnimation(parent: _animController, curve: Curves.elasticInOut);
    _opacityAnim = Tween<double>(
      begin: 0,
      end: 1,
    ).animate(_opacityCurved);
  }

  double _getOpacityValue(double opacity) {
    double temp = 0;
    if (opacity < 0) {
      temp = 0;
      return temp;
    }
    if (opacity > 1) {
      temp = 1;
      return temp;
    }
    temp = opacity;
    return temp;
  }

  void _initListener() {
    _animController.addStatusListener((status) {
      switch (status) {
        case AnimationStatus.dismissed:
          _animController.forward();
          break;
        case AnimationStatus.forward:
        // TODO: Handle this case.
          break;
        case AnimationStatus.reverse:
        // TODO: Handle this case.
          break;
        case AnimationStatus.completed:
          _animController.reverse();
          break;
      }
    });

    _animController.forward();
  }
}
View Code

 

 

六、使用AnimatedWidget構建可重用動畫

編寫一個繼承AnimatedWidget的Widget,在此Widget裏咱們能夠編寫本身想要的樣式並執行相應的動畫。下面是代碼示例:api

 

import 'package:flutter/material.dart';

class AnimWidgetPage extends StatefulWidget {
  @override
  _AnimWidgetPageState createState() => _AnimWidgetPageState();
}

class _AnimWidgetPageState extends State<AnimWidgetPage>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;

  CurvedAnimation _curved;
  Animation<double> _anim;

  @override
  void initState() {
    _controller = AnimationController(vsync: this,duration: Duration(milliseconds: 1500));
    _curved = CurvedAnimation(parent: _controller, curve: Curves.bounceInOut);
    _anim = Tween<double>(begin: 1, end: 5).animate(_curved);
    _controller.forward();
    super.initState();
  }

  @override
  void dispose() {
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("AnimatedWidget"),
      ),
      body: Container(
        child: Center(
          child: CustomAnimWidget(
            animation: _anim,
          ),
        ),
      ),
    );
  }
}

class CustomAnimWidget extends AnimatedWidget {
  CustomAnimWidget({Key key, Animation<double> animation})
      : super(key: key, listenable: animation);

  Widget build(BuildContext context) {
    final Animation<double> animation = listenable;
    return Center(
      child: Container(
        height: 200,
        width: 200,
        child: ScaleTransition(
          scale: animation,
          child: Icon(
            Icons.android,
            color: Colors.green,
          ),
        ),
      ),
    );
  }
}
View Code

 

 

七、組合動畫

組合動畫顧名思義就是一個或多個Widget被幾個動畫同時做用。app

 

import 'package:flutter/material.dart';

class CombinationAnimPage extends StatefulWidget {
  @override
  _CombinationAnimPageState createState() => _CombinationAnimPageState();
}

class _CombinationAnimPageState extends State<CombinationAnimPage>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;

  // 曲線
  CurvedAnimation _curvedAnimation;

  // 縮放
  Animation<double> _scaleAnim;

  // 旋轉
  Animation<double> _rotationAnim;

  @override
  void initState() {
    _controller = AnimationController(vsync: this,duration: Duration(milliseconds: 2000));
    _curvedAnimation =
        CurvedAnimation(parent: _controller, curve: Curves.bounceInOut);
    _scaleAnim = Tween<double>(
      begin: 1,
      end: 5,
    ).animate(_curvedAnimation);
    _rotationAnim = Tween<double>(
      begin: 0,
      end: 1,
    ).animate(_curvedAnimation);

    _controller.addStatusListener((status) {
      switch (status) {
        case AnimationStatus.dismissed:
          _controller.forward();
          break;
        case AnimationStatus.forward:
          // TODO: Handle this case.
          break;
        case AnimationStatus.reverse:
          // TODO: Handle this case.
          break;
        case AnimationStatus.completed:
          _controller.reverse();
          break;
      }
    });

    _controller.forward();

    super.initState();
  }

  @override
  void dispose() {
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("組合動畫"),
        centerTitle: true,
      ),
      body: Container(
        child: Center(
          child: ScaleTransition(
            scale: _scaleAnim,
            child: RotationTransition(
              turns: _rotationAnim,
              child: Icon(
                Icons.android,
                color: Colors.green,
              ),
            ),
          ),
        ),
      ),
    );
  }
}
View Code

 

   

八、使用AnimatedBuilder構建動畫

AnimatedBuilder將動畫和視圖分離。它接受一個Animation和一個Widget。下面是代碼示例:框架

 

import 'package:flutter/material.dart';

class CustomAnimBuildPage extends StatefulWidget {
  @override
  _CustomAnimBuildPageState createState() => _CustomAnimBuildPageState();
}

class _CustomAnimBuildPageState extends State<CustomAnimBuildPage>
    with SingleTickerProviderStateMixin {
  AnimationController _controller;
  Animation<double> animation;
  Animation<Color> colorAnim;

  @override
  void initState() {
    _controller =
        AnimationController(vsync: this, duration: Duration(seconds: 1));
    animation = Tween<double>(
      begin: 50,
      end: 200,
    ).animate(_controller);

    colorAnim = ColorTween(begin: Colors.amber, end: Colors.deepPurple)
        .animate(_controller);

    _controller.addStatusListener((status) {
      switch (status) {
        case AnimationStatus.dismissed:
          _controller.forward();
          break;
        case AnimationStatus.forward:
          // TODO: Handle this case.
          break;
        case AnimationStatus.reverse:
          // TODO: Handle this case.
          break;
        case AnimationStatus.completed:
          _controller.reverse();
          break;
      }
    });

    _controller.forward();
    super.initState();
  }

  @override
  void dispose() {
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("AnimatedBuilder"),
        centerTitle: true,
      ),
      body: Container(
        child: Center(
          child: _CustomTransition(
            child: Icon(
              Icons.android,
              size: 50,
              color: Colors.green,
            ),
            animation: animation,
            colorAnim: colorAnim,
          ),
        ),
      ),
    );
  }
}

class _CustomTransition extends StatelessWidget {
  Widget child;
  Animation<double> animation;
  Animation<Color> colorAnim;

  _CustomTransition({this.child, this.animation, this.colorAnim});

  @override
  Widget build(BuildContext context) {
    return Container(
      child: AnimatedBuilder(
        animation: animation,
        builder: (BuildContext context, Widget widget) {
          return Container(
            color: colorAnim.value,
            width: animation.value,
            height: animation.value,
            child: Transform.translate(
              offset: Offset(animation.value-50, 0),
              child: child,
            ),
          );
        },
      ),
    );
  }
}
View Code

 

 

九、列表動畫

列表動畫本質上也是每一個Item作相應的動畫。下面是官方代碼示例:less

 

import 'package:flutter/material.dart';

class AnimatedListSample extends StatefulWidget {
  @override
  _AnimatedListSampleState createState() => new _AnimatedListSampleState();
}

class _AnimatedListSampleState extends State<AnimatedListSample> {
  final GlobalKey<AnimatedListState> _listKey =
      new GlobalKey<AnimatedListState>();
  ListModel<int> _list;
  int _selectedItem;
  int _nextItem; // The next item inserted when the user presses the '+' button.

  @override
  void initState() {
    super.initState();
    _list = new ListModel<int>(
      listKey: _listKey,
      initialItems: <int>[0, 1, 2],
      removedItemBuilder: _buildRemovedItem,
    );
    _nextItem = 3;
  }

  // Used to build list items that haven't been removed.
  Widget _buildItem(
      BuildContext context, int index, Animation<double> animation) {
    return new CardItem(
      animation: animation,
      item: _list[index],
      selected: _selectedItem == _list[index],
      onTap: () {
        setState(() {
          _selectedItem = _selectedItem == _list[index] ? null : _list[index];
        });
      },
    );
  }

  // Used to build an item after it has been removed from the list. This method is
  // needed because a removed item remains  visible until its animation has
  // completed (even though it's gone as far this ListModel is concerned).
  // The widget will be used by the [AnimatedListState.removeItem] method's
  // [AnimatedListRemovedItemBuilder] parameter.
  Widget _buildRemovedItem(
      int item, BuildContext context, Animation<double> animation) {
    return new CardItem(
      animation: animation,
      item: item,
      selected: false,
      // No gesture detector here: we don't want removed items to be interactive.
    );
  }

  // Insert the "next item" into the list model.
  void _insert() {
    final int index =
        _selectedItem == null ? _list.length : _list.indexOf(_selectedItem);
    _list.insert(index, _nextItem++);
  }

  // Remove the selected item from the list model.
  void _remove() {
    if (_selectedItem != null) {
      _list.removeAt(_list.indexOf(_selectedItem));
      setState(() {
        _selectedItem = null;
      });
    }
  }

  @override
  Widget build(BuildContext context) {
    return new MaterialApp(
      home: new Scaffold(
        appBar: new AppBar(
          title: const Text('AnimatedList'),
          actions: <Widget>[
            new IconButton(
              icon: const Icon(Icons.add_circle),
              onPressed: _insert,
              tooltip: 'insert a new item',
            ),
            new IconButton(
              icon: const Icon(Icons.remove_circle),
              onPressed: _remove,
              tooltip: 'remove the selected item',
            ),
          ],
        ),
        body: new Padding(
          padding: const EdgeInsets.all(16.0),
          child: new AnimatedList(
            key: _listKey,
            initialItemCount: _list.length,
            itemBuilder: _buildItem,
          ),
        ),
      ),
    );
  }
}

/// Keeps a Dart List in sync with an AnimatedList.
///
/// The [insert] and [removeAt] methods apply to both the internal list and the
/// animated list that belongs to [listKey].
///
/// This class only exposes as much of the Dart List API as is needed by the
/// sample app. More list methods are easily added, however methods that mutate the
/// list must make the same changes to the animated list in terms of
/// [AnimatedListState.insertItem] and [AnimatedList.removeItem].
class ListModel<E> {
  ListModel({
    @required this.listKey,
    @required this.removedItemBuilder,
    Iterable<E> initialItems,
  })  : assert(listKey != null),
        assert(removedItemBuilder != null),
        _items = new List<E>.from(initialItems ?? <E>[]);

  final GlobalKey<AnimatedListState> listKey;
  final dynamic removedItemBuilder;
  final List<E> _items;

  AnimatedListState get _animatedList => listKey.currentState;

  void insert(int index, E item) {
    _items.insert(index, item);
    _animatedList.insertItem(index);
  }

  E removeAt(int index) {
    final E removedItem = _items.removeAt(index);
    if (removedItem != null) {
      _animatedList.removeItem(index,
          (BuildContext context, Animation<double> animation) {
        return removedItemBuilder(removedItem, context, animation);
      });
    }
    return removedItem;
  }

  int get length => _items.length;

  E operator [](int index) => _items[index];

  int indexOf(E item) => _items.indexOf(item);
}

/// Displays its integer item as 'item N' on a Card whose color is based on
/// the item's value. The text is displayed in bright green if selected is true.
/// This widget's height is based on the animation parameter, it varies
/// from 0 to 128 as the animation varies from 0.0 to 1.0.
class CardItem extends StatelessWidget {
  const CardItem(
      {Key key,
      @required this.animation,
      this.onTap,
      @required this.item,
      this.selected: false})
      : assert(animation != null),
        assert(item != null && item >= 0),
        assert(selected != null),
        super(key: key);

  final Animation<double> animation;
  final VoidCallback onTap;
  final int item;
  final bool selected;

  @override
  Widget build(BuildContext context) {
    TextStyle textStyle = Theme.of(context).textTheme.display1;
    if (selected)
      textStyle = textStyle.copyWith(color: Colors.lightGreenAccent[400]);
    return new Padding(
      padding: const EdgeInsets.all(2.0),
      child: new SizeTransition(
        axis: Axis.vertical,
        sizeFactor: animation,
        child: new GestureDetector(
          behavior: HitTestBehavior.opaque,
          onTap: onTap,
          child: new SizedBox(
            height: 128.0,
            child: new Card(
              color: Colors.primaries[item % Colors.primaries.length],
              child: new Center(
                child: new Text('Item $item', style: textStyle),
              ),
            ),
          ),
        ),
      ),
    );
  }
}
View Code

 

 

十、共享元素動畫

所謂共享元素動畫能夠簡單的理解爲兩個頁面共用同一個元素。可是實際上是兩個頁面的的兩個元素被相同的Tag所標記,再進行頁面跳轉的時候被框架識別,從而執行相應的動畫。Flutter中使用共享元素動畫須要使用Hero這個StatefulWidget。Hero的tag屬性標記兩個元素。下面是代碼示例:ide

 

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

class HeroAnimation extends StatelessWidget {
  Widget build(BuildContext context) {
//    timeDilation = 5.0; // 1.0 means normal animation speed.

    return Scaffold(
      appBar: AppBar(
        title: Text('Basic Hero Animation'),
        centerTitle: true,
      ),
      body: GridView.count(
        crossAxisCount: 2,
        children: <Widget>[
          ItemView(myData[0], 150),
          ItemView(myData[1], 150),
          ItemView(myData[2], 150),
          ItemView(myData[3], 150),
          ItemView(myData[4], 150),
          ItemView(myData[5], 150),
          ItemView(myData[6], 150),
          ItemView(myData[7], 150),
          ItemView(myData[8], 150),
          ItemView(myData[9], 150),
        ],
      ),
    );
  }
}

Widget getHeroAnim2(ItemModel itemModel) {
  return Scaffold(
    appBar: AppBar(
      title: Text("共享元素"),
      centerTitle: true,
    ),
    body: Container(
      alignment: Alignment.topLeft,
      child: ItemView(itemModel, 400),
    ),
  );
}

List<ItemModel> myData = <ItemModel>[
  ItemModel(
      title: '啦啦啦1111',
      imgUrl:
          'https://ss1.bdstatic.com/70cFuXSh_Q1YnxGkpoWK1HF6hhy/it/u=2717595227,1512397782&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦2222',
      imgUrl:
          'https://ss0.bdstatic.com/70cFvHSh_Q1YnxGkpoWK1HF6hhy/it/u=3454574876,1377139334&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦3333',
      imgUrl:
          'https://ss2.bdstatic.com/70cFvnSh_Q1YnxGkpoWK1HF6hhy/it/u=1499844476,2082399552&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦4444',
      imgUrl:
          'https://ss2.bdstatic.com/70cFvnSh_Q1YnxGkpoWK1HF6hhy/it/u=1938482571,2420691429&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦5555',
      imgUrl:
          'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=3548575507,3156953806&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦6666',
      imgUrl:
          'https://ss2.bdstatic.com/70cFvnSh_Q1YnxGkpoWK1HF6hhy/it/u=3484495061,2102329231&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦7777',
      imgUrl:
          'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=3562330430,950864085&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦8888',
      imgUrl:
          'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=2985783351,2052499916&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦9999',
      imgUrl:
          'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=311914474,2668302507&fm=26&gp=0.jpg'),
  ItemModel(
      title: '啦啦啦0000',
      imgUrl:
          'https://ss0.bdstatic.com/70cFvHSh_Q1YnxGkpoWK1HF6hhy/it/u=2471845590,913308006&fm=26&gp=0.jpg'),
];

// 數據類型
class ItemModel {
  String title;
  String imgUrl;

  ItemModel({this.title, this.imgUrl});
}

class ItemView extends StatelessWidget {
  ItemModel model;

  double height;

  ItemView(this.model, this.height);

  @override
  Widget build(BuildContext context) {
    return GestureDetector(
      onTap: () {
        Navigator.of(context).push(
          new MaterialPageRoute<Null>(
            builder: (BuildContext context) {
              return getHeroAnim2(model);
            },
          ),
        );
      },
      child: Container(
        alignment: Alignment.center,
        child: SizedBox(
          width: height,
          height: height,
          child: Container(
            decoration: BoxDecoration(
                borderRadius: BorderRadius.all(Radius.circular(6)),
                color: Colors.white),
            child: Hero(
              // 一個viewTree下面不能有相同的
              tag: model.imgUrl,
              child: Material(
                color: Colors.transparent,
                child: Column(
                  children: <Widget>[
                    Expanded(
                        child: Image.network(
                      model.imgUrl,
                      fit: BoxFit.cover,
                    )),
                    Text(model.title),
                  ],
                ),
              ),
            ),
          ),
        ),
      ),
    );
  }
}
View Code

 

 

十一、拖拽動畫

拖拽須要使用GestureDetector()監聽用戶手勢,其函數會返回一個DragUpdateDetails對象,這個對象能夠獲取當前手指位移的座標,而後經過Offset()給Widget設置偏移量。下面是代碼示例:

 

import 'package:flutter/material.dart';

class DragAnimPage extends StatefulWidget {
  @override
  _DragAnimPageState createState() => _DragAnimPageState();
}

class _DragAnimPageState extends State<DragAnimPage> {

  double mDx = 0;
  double mDy = 0;

  GlobalKey _globalKey = new GlobalKey();

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("拖拽動畫"),
        centerTitle: true,
      ),
      body: Container(
        child: Transform.translate(
          offset: Offset(mDx, mDy),
          child: GestureDetector(
            onPanUpdate: (dragUpdateDetails) {
              mDx = dragUpdateDetails.globalPosition.dx;
              mDy = dragUpdateDetails.globalPosition.dy;
              setState(() {});
            },
            child: Container(
              width: 100,
              height: 50,
              alignment: Alignment.center,
              color: Colors.indigoAccent,
              key: _globalKey,
              child: Text("拖拽"),
            ),
          ),
        ),
      ),
    );
  }
}
View Code

 

 

十二、第三方動畫

①、Lottie

Lottie動畫是Airbnb公司出的一款跨平臺的動畫框架(基礎篇有介紹連接)。下面是代碼示例:

 

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

class LottieAnimPage extends StatefulWidget {
  @override
  _LottieAnimPageState createState() => _LottieAnimPageState();
}

class _LottieAnimPageState extends State<LottieAnimPage> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("Lottie動畫"),
        centerTitle: true,
      ),
      body: Container(
        padding: EdgeInsets.all(20),
        child: Center(
          child: LottieView.fromFile(
            filePath: "assets/anim/8075-the-frog-to-drive.json",
            autoPlay: true,
            loop: true,
            reverse: true,
            onViewCreated: (lottieController) {

            },
          ),
        ),
      ),
    );
  }
}
View Code

 

 

 

②、Flare

Flare動畫框架是Flutter官方推薦的一個動畫框架(詳細介紹請看基礎篇)。下面是代碼示例:

 

import 'package:flare_flutter/flare_actor.dart';
import 'package:flutter/material.dart';

class FlareAnimPage extends StatefulWidget {
  @override
  _FlareAnimPageState createState() => _FlareAnimPageState();
}

class _FlareAnimPageState extends State<FlareAnimPage> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text("Flare動畫(官方推薦)"),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Expanded(
              child: FlareActor(
                "assets/anim/Filip.flr",
                alignment: Alignment.center,
                fit: BoxFit.contain,
                animation: 'idle',
              ),
            ),
          ],
        ),
      ),
    );
  }
}
View Code

 

 

 

4、Demo代碼地址

點擊進入github項目地址

 

 

參考文獻:

相關文章
相關標籤/搜索