Commit 6d4fd850 authored by Adam Barth's avatar Adam Barth

Improve float action button transitions (#4309)

Previously the floating action button would disappear during transitions. Now
we animate properly. I've also centralized the animation in Scaffold instead of
spreading it across the Scaffold and the FloatingActionButton.

Fixes #1718
parent 0618da7c
...@@ -92,39 +92,6 @@ class FloatingActionButton extends StatefulWidget { ...@@ -92,39 +92,6 @@ class FloatingActionButton extends StatefulWidget {
} }
class _FloatingActionButtonState extends State<FloatingActionButton> { class _FloatingActionButtonState extends State<FloatingActionButton> {
Animation<double> _childSegue;
AnimationController _childSegueController;
@override
void initState() {
super.initState();
_childSegueController = new AnimationController(duration: _kChildSegue)
..forward();
_childSegue = new Tween<double>(
begin: -0.125,
end: 0.0
).animate(new CurvedAnimation(
parent: _childSegueController,
curve: _kChildSegueInterval
));
}
@override
void dispose() {
_childSegueController.dispose();
super.dispose();
}
@override
void didUpdateConfig(FloatingActionButton oldConfig) {
super.didUpdateConfig(oldConfig);
if (Widget.canUpdate(oldConfig.child, config.child) && config.backgroundColor == oldConfig.backgroundColor)
return;
_childSegueController
..value = 0.0
..forward();
}
bool _highlight = false; bool _highlight = false;
void _handleHighlightChanged(bool value) { void _handleHighlightChanged(bool value) {
...@@ -146,10 +113,7 @@ class _FloatingActionButtonState extends State<FloatingActionButton> { ...@@ -146,10 +113,7 @@ class _FloatingActionButtonState extends State<FloatingActionButton> {
Widget result = new Center( Widget result = new Center(
child: new IconTheme( child: new IconTheme(
data: new IconThemeData(color: iconColor), data: new IconThemeData(color: iconColor),
child: new RotationTransition( child: config.child
turns: _childSegue,
child: config.child
)
) )
); );
......
...@@ -17,7 +17,8 @@ import 'material.dart'; ...@@ -17,7 +17,8 @@ import 'material.dart';
import 'snack_bar.dart'; import 'snack_bar.dart';
const double _kFloatingActionButtonMargin = 16.0; // TODO(hmuller): should be device dependent const double _kFloatingActionButtonMargin = 16.0; // TODO(hmuller): should be device dependent
const Duration _kFloatingActionButtonSegue = const Duration(milliseconds: 400); const Duration _kFloatingActionButtonSegue = const Duration(milliseconds: 200);
final Tween<double> _kFloatingActionButtonTurnTween = new Tween<double>(begin: -0.125, end: 0.0);
/// The Scaffold's appbar is the toolbar, tabbar, and the "flexible space" that's /// The Scaffold's appbar is the toolbar, tabbar, and the "flexible space" that's
/// stacked behind them. The Scaffold's appBarBehavior defines how the appbar /// stacked behind them. The Scaffold's appBarBehavior defines how the appbar
...@@ -131,9 +132,7 @@ class _FloatingActionButtonTransition extends StatefulWidget { ...@@ -131,9 +132,7 @@ class _FloatingActionButtonTransition extends StatefulWidget {
_FloatingActionButtonTransition({ _FloatingActionButtonTransition({
Key key, Key key,
this.child this.child
}) : super(key: key) { }) : super(key: key);
assert(child != null);
}
final Widget child; final Widget child;
...@@ -142,60 +141,91 @@ class _FloatingActionButtonTransition extends StatefulWidget { ...@@ -142,60 +141,91 @@ class _FloatingActionButtonTransition extends StatefulWidget {
} }
class _FloatingActionButtonTransitionState extends State<_FloatingActionButtonTransition> { class _FloatingActionButtonTransitionState extends State<_FloatingActionButtonTransition> {
final AnimationController controller = new AnimationController(duration: _kFloatingActionButtonSegue); final AnimationController _previousController = new AnimationController(duration: _kFloatingActionButtonSegue);
Widget oldChild; final AnimationController _currentController = new AnimationController(duration: _kFloatingActionButtonSegue);
CurvedAnimation _previousAnimation;
CurvedAnimation _currentAnimation;
Widget _previousChild;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
controller.forward().then((_) { _previousAnimation = new CurvedAnimation(
oldChild = null; parent: _previousController,
}); curve: Curves.easeIn
);
_currentAnimation = new CurvedAnimation(
parent: _currentController,
curve: Curves.easeIn
);
_previousController.addStatusListener(_handleAnimationStatusChanged);
_currentController.forward();
} }
@override @override
void dispose() { void dispose() {
controller.stop(); _previousController.stop();
_currentController.stop();
super.dispose(); super.dispose();
} }
@override @override
void didUpdateConfig(_FloatingActionButtonTransition oldConfig) { void didUpdateConfig(_FloatingActionButtonTransition oldConfig) {
if (Widget.canUpdate(oldConfig.child, config.child)) final bool oldChildIsNull = oldConfig.child == null;
final bool newChildIsNull = config.child == null;
if (oldChildIsNull == newChildIsNull && oldConfig.child?.key == config.child?.key)
return; return;
oldChild = oldConfig.child; if (_previousController.status == AnimationStatus.dismissed) {
controller final double currentValue = _currentController.value;
..value = 0.0 if (currentValue == 0.0 || oldConfig.child == null) {
..forward().then((_) { // The current child hasn't started its entrance animation yet. We can
oldChild = null; // just skip directly to the new child's entrance.
}); _previousChild = null;
if (config.child != null)
_currentController.forward();
} else {
// Otherwise, we need to copy the state from the current controller to
// the previous controller and run an exit animation for the previous
// widget before running the entrance animation for the new child.
_previousChild = oldConfig.child;
_previousController
..value = currentValue
..reverse();
_currentController.value = 0.0;
}
}
}
void _handleAnimationStatusChanged(AnimationStatus status) {
setState(() {
if (status == AnimationStatus.dismissed) {
assert(_currentController.status == AnimationStatus.dismissed);
if (config.child != null)
_currentController.forward();
}
});
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final List<Widget> children = new List<Widget>(); final List<Widget> children = new List<Widget>();
if (oldChild != null) { if (_previousAnimation.status != AnimationStatus.dismissed) {
children.add(new ScaleTransition( children.add(new ScaleTransition(
// TODO(abarth): We should use ReversedAnimation here. scale: _previousAnimation,
scale: new Tween<double>( child: _previousChild
begin: 1.0, ));
end: 0.0 }
).animate(new CurvedAnimation( if (_currentAnimation.status != AnimationStatus.dismissed) {
parent: controller, children.add(new ScaleTransition(
curve: const Interval(0.0, 0.5, curve: Curves.easeIn) scale: _currentAnimation,
)), child: new RotationTransition(
child: oldChild turns: _kFloatingActionButtonTurnTween.animate(_currentAnimation),
child: config.child
)
)); ));
} }
children.add(new ScaleTransition(
scale: new CurvedAnimation(
parent: controller,
curve: const Interval(0.5, 1.0, curve: Curves.easeIn)
),
child: config.child
));
return new Stack(children: children); return new Stack(children: children);
} }
} }
...@@ -664,13 +694,12 @@ class ScaffoldState extends State<Scaffold> { ...@@ -664,13 +694,12 @@ class ScaffoldState extends State<Scaffold> {
if (_snackBars.isNotEmpty) if (_snackBars.isNotEmpty)
_addIfNonNull(children, _snackBars.first._widget, _ScaffoldSlot.snackBar); _addIfNonNull(children, _snackBars.first._widget, _ScaffoldSlot.snackBar);
if (config.floatingActionButton != null) { children.add(new LayoutId(
final Widget fab = new _FloatingActionButtonTransition( id: _ScaffoldSlot.floatingActionButton,
key: new ValueKey<Key>(config.floatingActionButton.key), child: new _FloatingActionButtonTransition(
child: config.floatingActionButton child: config.floatingActionButton
); )
children.add(new LayoutId(child: fab, id: _ScaffoldSlot.floatingActionButton)); ));
}
if (config.drawer != null) { if (config.drawer != null) {
children.add(new LayoutId( children.add(new LayoutId(
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment