animated_to 0.0.4 animated_to: ^0.0.4 copied to clipboard
AnimateTo animates its child to a new position requiring no calculation.
import 'package:animated_to/animated_to.dart';
import 'package:flutter/material.dart';
void main() => runApp(const MyApp());
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(home: const AnimatedToSamplePage());
}
}
class AnimatedToSamplePage extends StatefulWidget {
const AnimatedToSamplePage({super.key});
@override
State<AnimatedToSamplePage> createState() => _AnimatedToSamplePageState();
}
class _AnimatedToSamplePageState extends State<AnimatedToSamplePage>
with TickerProviderStateMixin {
var _isExpanded = true;
/// Some item objects. In this demo, simply a list of [String].
final _items = List.generate(
5,
(index) => index.toString(),
);
bool _enabled = true;
@override
Widget build(BuildContext context) {
final children = _items
.map(
(item) => _Card(
item: item,
index: _items.indexOf(item),
opacity: 1.0,
isExpanded: _isExpanded,
vsync: this,
enabled: _enabled,
),
)
.toList();
return Scaffold(
appBar: AppBar(title: const Text('Animated Sample Page')),
// TODO(chooyan-eng): note that [AnimatedTo] doesn't work on scrollable widgets
body: NotificationListener<ScrollNotification>(
// workaround to fix scrolling issue by disabling animation when scrolling
onNotification: (notification) {
if (notification is ScrollStartNotification) {
WidgetsBinding.instance.addPostFrameCallback((_) {
setState(() => _enabled = false);
});
}
if (notification is ScrollEndNotification) {
WidgetsBinding.instance.addPostFrameCallback((_) {
setState(() => _enabled = true);
});
}
return true;
},
child: SingleChildScrollView(
child: Center(
child: Padding(
padding: const EdgeInsets.all(4),
child: switch (_isExpanded) {
// toggle [Wrap] and [Column] with animation
true => Wrap(
spacing: 4,
runSpacing: 4,
children: children,
),
false => Column(
mainAxisSize: MainAxisSize.min,
children: children,
),
},
),
),
),
),
floatingActionButton: FloatingActionButton.extended(
onPressed: () {
setState(() => _isExpanded = !_isExpanded);
},
icon: Icon(_isExpanded ? Icons.collections : Icons.expand),
label: Text(_isExpanded ? 'Collapse' : 'Expand'),
),
);
}
}
/// A circle or a rectangle shaped widget to be animated with [AnimatedTo].
class _Card extends StatelessWidget {
const _Card({
required this.item,
required this.index,
required this.opacity,
required this.isExpanded,
required this.vsync,
required this.enabled,
});
final String item;
final int index;
final double opacity;
final bool isExpanded;
final TickerProvider vsync;
final bool enabled;
@override
Widget build(BuildContext context) {
final size = 60.0;
return AnimatedTo(
// try either of [appearingFrom] or [slidingFrom]
appearingFrom: const Offset(100, 0),
// slidingFrom: const Offset(0, 100),
// [TickerProviderStateMixin] is currently required to use [AnimatedTo]
vsync: vsync,
// [GlobalObjectKey] is required to identify the widget
key: GlobalObjectKey(item),
duration: Duration(milliseconds: 300 + (10 * index)),
curve: Curves.easeOutQuad,
enabled: enabled,
onEnd: (cause) {
switch (cause) {
case AnimationEndCause.interrupted:
// called when the animation is interrupted by another animation
break;
case AnimationEndCause.completed:
// called when the animation is completed
break;
}
},
// [AnimatedTo] can be combined with some Animated widgets
child: Container(
// duration: Duration(milliseconds: 300 + (10 * index)),
// curve: Curves.easeIn,
decoration: BoxDecoration(
color: Colors.primaries[int.parse(item) % Colors.primaries.length]
.withAlpha(isExpanded ? 255 : 128),
borderRadius: isExpanded
? BorderRadius.circular(100)
: BorderRadius.circular(4),
),
margin: const EdgeInsets.all(2),
width: size,
height: size,
child: Center(
child: Text(
item,
style: TextStyle(
color: Colors.white.withAlpha(isExpanded ? 255 : 128),
fontSize: 24,
fontWeight: FontWeight.bold,
),
),
),
),
);
}
}