You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							215 lines
						
					
					
						
							5.0 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							215 lines
						
					
					
						
							5.0 KiB
						
					
					
				| import 'dart:math' as math; | |
| import 'package:flutter/material.dart'; | |
| import 'package:teso/util/SizeConfig.dart'; | |
| import 'package:teso/util/consts.dart'; | |
| 
 | |
| @immutable | |
| class TesoFAB extends StatefulWidget { | |
|   const TesoFAB({ | |
|     super.key, | |
|     this.initialOpen, | |
|     required this.distance, | |
|     required this.children, | |
|   }); | |
| 
 | |
|   final bool? initialOpen; | |
|   final double distance; | |
|   final List<Widget> children; | |
| 
 | |
|   @override | |
|   _TesoFABState createState() => _TesoFABState(); | |
| } | |
| 
 | |
| class _TesoFABState extends State<TesoFAB> | |
|     with SingleTickerProviderStateMixin { | |
|   late final AnimationController _controller; | |
|   late final Animation<double> _expandAnimation; | |
|   bool _open = false; | |
| 
 | |
|   @override | |
|   void initState() { | |
|     super.initState(); | |
|     _open = widget.initialOpen ?? false; | |
|     _controller = AnimationController( | |
|       value: _open ? 1.0 : 0.0, | |
|       duration: const Duration(milliseconds: 250), | |
|       vsync: this, | |
|     ); | |
|     _expandAnimation = CurvedAnimation( | |
|       curve: Curves.fastOutSlowIn, | |
|       reverseCurve: Curves.easeOutQuad, | |
|       parent: _controller, | |
|     ); | |
|   } | |
| 
 | |
|   @override | |
|   void dispose() { | |
|     _controller.dispose(); | |
|     super.dispose(); | |
|   } | |
| 
 | |
|   void _toggle() { | |
|     setState(() { | |
|       _open = !_open; | |
|       if (_open) { | |
|         _controller.forward(); | |
|       } else { | |
|         _controller.reverse(); | |
|       } | |
|     }); | |
|   } | |
| 
 | |
|   @override | |
|   Widget build(BuildContext context) { | |
|     return SizedBox.expand( | |
|       child: Stack( | |
|         alignment: Alignment.bottomRight, | |
|         clipBehavior: Clip.none, | |
|         children: [ | |
|           _buildTapToCloseFab(), | |
|           ..._buildExpandingActionButtons(), | |
|           _buildTapToOpenFab(), | |
|         ], | |
|       ), | |
|     ); | |
|   } | |
| 
 | |
|   Widget _buildTapToCloseFab() { | |
|     return SizedBox( | |
|       width: 56.0, | |
|       height: 56.0, | |
|       child: Center( | |
|         child: Material(color: Theme.of(context).primaryColor, | |
|           shape: const CircleBorder(), | |
|           clipBehavior: Clip.antiAlias, | |
|           elevation: 4.0, | |
|           child: InkWell( | |
|             onTap: _toggle, | |
|             child: Padding( | |
|               padding: const EdgeInsets.all(8.0), | |
|               child: Icon( | |
|                 Icons.close, | |
|                 color: Colors.grey, | |
|               ), | |
|             ), | |
|           ), | |
|         ), | |
|       ), | |
|     ); | |
|   } | |
| 
 | |
|   List<Widget> _buildExpandingActionButtons() { | |
|     final children = <Widget>[]; | |
|     final count = widget.children.length; | |
|     final step = 90.0 / (count - 1); | |
|     for (var i = 0, angleInDegrees = 0.0; | |
|         i < count; | |
|         i++, angleInDegrees += step) { | |
|       children.add( | |
|         _ExpandingActionButton( | |
|           directionInDegrees: angleInDegrees, | |
|           maxDistance: widget.distance, | |
|           progress: _expandAnimation, | |
|           child: widget.children[i], | |
|         ), | |
|       ); | |
|     } | |
|     return children; | |
|   } | |
| 
 | |
|   Widget _buildTapToOpenFab() { | |
|     return IgnorePointer( | |
|       ignoring: _open, | |
|       child: AnimatedContainer( | |
|         transformAlignment: Alignment.center, | |
|         transform: Matrix4.diagonal3Values( | |
|           _open ? 0.7 : 1.0, | |
|           _open ? 0.7 : 1.0, | |
|           1.0, | |
|         ), | |
|         duration: const Duration(milliseconds: 250), | |
|         curve: const Interval(0.0, 0.5, curve: Curves.easeOut), | |
|         child: AnimatedOpacity( | |
|           opacity: _open ? 0.0 : 1.0, | |
|           curve: const Interval(0.25, 1.0, curve: Curves.easeInOut), | |
|           duration: const Duration(milliseconds: 250), | |
|           child: FloatingActionButton( | |
|         backgroundColor: tesoGold, | |
|         onPressed: _toggle, | |
|         child: ImageIcon( | |
|           AssetImage("assets/images/rawLogo.png"), | |
|           size: SizeConfig.safeBlockHorizontal * 6.7, | |
|         ), | |
|       ), | |
|         ), | |
|       ), | |
|     ); | |
|   } | |
| } | |
| 
 | |
| @immutable | |
| class _ExpandingActionButton extends StatelessWidget { | |
|   const _ExpandingActionButton({ | |
|     required this.directionInDegrees, | |
|     required this.maxDistance, | |
|     required this.progress, | |
|     required this.child, | |
|   }); | |
| 
 | |
|   final double directionInDegrees; | |
|   final double maxDistance; | |
|   final Animation<double> progress; | |
|   final Widget child; | |
| 
 | |
|   @override | |
|   Widget build(BuildContext context) { | |
|     return AnimatedBuilder( | |
|       animation: progress, | |
|       builder: (context, child) { | |
|         final offset = Offset.fromDirection( | |
|           directionInDegrees * (math.pi / 180.0), | |
|           progress.value * maxDistance, | |
|         ); | |
|         return Positioned( | |
|           right: 4.0 + offset.dx, | |
|           bottom: 4.0 + offset.dy, | |
|           child: Transform.rotate( | |
|             angle: (1.0 - progress.value) * math.pi / 2, | |
|             child: child!, | |
|           ), | |
|         ); | |
|       }, | |
|       child: FadeTransition( | |
|         opacity: progress, | |
|         child: child, | |
|       ), | |
|     ); | |
|   } | |
| } | |
| 
 | |
| @immutable | |
| class ActionButton extends StatelessWidget { | |
|   const ActionButton({ | |
|     super.key, | |
|     this.onPressed, | |
|     required this.icon, | |
|   }); | |
| 
 | |
|   final VoidCallback? onPressed; | |
|   final Widget icon; | |
| 
 | |
|   @override | |
|   Widget build(BuildContext context) { | |
|     return Material( | |
|       shape: const CircleBorder(), | |
|       clipBehavior: Clip.antiAlias, | |
|       color: tesoGold, | |
|       elevation: 4.0, | |
|       child: IconButton( | |
|         onPressed: onPressed, | |
|         icon: icon, | |
|         color: Colors.black, | |
|       ), | |
|     ); | |
|   } | |
| } |