// 自定義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(); } }
// 自定義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(); } }
只要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(); } }
旋轉動畫就是一個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(); } }
本文檔使用了兩種方式來達到縮放的效果。一種是直接使用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(); } }
本文檔使用了兩種方式來達到漸隱漸現的效果。第一種使用了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(); } }
非線性曲線動畫其主要類是使用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(); } }
編寫一個繼承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, ), ), ), ); } }
組合動畫顧名思義就是一個或多個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, ), ), ), ), ), ); } }
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, ), ); }, ), ); } }
列表動畫本質上也是每一個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), ), ), ), ), ), ); } }
所謂共享元素動畫能夠簡單的理解爲兩個頁面共用同一個元素。可是實際上是兩個頁面的的兩個元素被相同的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), ], ), ), ), ), ), ), ); } }
拖拽須要使用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("拖拽"), ), ), ), ), ); } }
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) { }, ), ), ), ); } }
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', ), ), ], ), ), ); } }
參考文獻: