mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-01-23 19:05:51 +00:00
add custom icon rotate widget
This commit is contained in:
parent
e3dbc64f17
commit
658708da95
3 changed files with 115 additions and 18 deletions
|
@ -6,6 +6,7 @@ import 'package:stackwallet/utilities/assets.dart';
|
||||||
import 'package:stackwallet/utilities/text_styles.dart';
|
import 'package:stackwallet/utilities/text_styles.dart';
|
||||||
import 'package:stackwallet/utilities/theme/stack_colors.dart';
|
import 'package:stackwallet/utilities/theme/stack_colors.dart';
|
||||||
import 'package:stackwallet/utilities/util.dart';
|
import 'package:stackwallet/utilities/util.dart';
|
||||||
|
import 'package:stackwallet/widgets/animated_widgets/rotate_icon.dart';
|
||||||
import 'package:stackwallet/widgets/expandable.dart';
|
import 'package:stackwallet/widgets/expandable.dart';
|
||||||
|
|
||||||
class ExpandingSubListItem extends StatefulWidget {
|
class ExpandingSubListItem extends StatefulWidget {
|
||||||
|
@ -14,11 +15,17 @@ class ExpandingSubListItem extends StatefulWidget {
|
||||||
required this.title,
|
required this.title,
|
||||||
required this.entities,
|
required this.entities,
|
||||||
required this.initialState,
|
required this.initialState,
|
||||||
}) : super(key: key);
|
double? animationDurationMultiplier,
|
||||||
|
this.curve = Curves.easeInOutCubicEmphasized,
|
||||||
|
}) : animationDurationMultiplier =
|
||||||
|
animationDurationMultiplier ?? entities.length * 0.11,
|
||||||
|
super(key: key);
|
||||||
|
|
||||||
final String title;
|
final String title;
|
||||||
final List<AddWalletListEntity> entities;
|
final List<AddWalletListEntity> entities;
|
||||||
final ExpandableState initialState;
|
final ExpandableState initialState;
|
||||||
|
final double animationDurationMultiplier;
|
||||||
|
final Curve curve;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<ExpandingSubListItem> createState() => _ExpandingSubListItemState();
|
State<ExpandingSubListItem> createState() => _ExpandingSubListItemState();
|
||||||
|
@ -28,31 +35,40 @@ class _ExpandingSubListItemState extends State<ExpandingSubListItem> {
|
||||||
final isDesktop = Util.isDesktop;
|
final isDesktop = Util.isDesktop;
|
||||||
|
|
||||||
late final ExpandableController _controller;
|
late final ExpandableController _controller;
|
||||||
|
late final RotateIconController _rotateIconController;
|
||||||
late bool _expandedState;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
_expandedState = widget.initialState == ExpandableState.expanded;
|
|
||||||
_controller = ExpandableController();
|
_controller = ExpandableController();
|
||||||
|
_rotateIconController = RotateIconController();
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
if (_expandedState) {
|
if (widget.initialState == ExpandableState.expanded) {
|
||||||
_controller.toggle?.call();
|
_controller.toggle?.call();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
super.initState();
|
super.initState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_controller.toggle = null;
|
||||||
|
_rotateIconController.forward = null;
|
||||||
|
_rotateIconController.reverse = null;
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Expandable(
|
return Expandable(
|
||||||
animationDurationMultiplier: 0.1 * widget.entities.length,
|
animationDurationMultiplier: widget.animationDurationMultiplier,
|
||||||
curve: Curves.easeInOutCubicEmphasized,
|
curve: widget.curve,
|
||||||
controller: _controller,
|
controller: _controller,
|
||||||
onExpandChanged: (state) {
|
onExpandWillChange: (state) {
|
||||||
setState(() {
|
if (state == ExpandableState.expanded) {
|
||||||
_expandedState = state == ExpandableState.expanded;
|
_rotateIconController.forward?.call();
|
||||||
});
|
} else {
|
||||||
|
_rotateIconController.reverse?.call();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
header: Container(
|
header: Container(
|
||||||
color: Colors.transparent,
|
color: Colors.transparent,
|
||||||
|
@ -76,14 +92,19 @@ class _ExpandingSubListItemState extends State<ExpandingSubListItem> {
|
||||||
: STextStyles.smallMed12(context),
|
: STextStyles.smallMed12(context),
|
||||||
textAlign: TextAlign.left,
|
textAlign: TextAlign.left,
|
||||||
),
|
),
|
||||||
SvgPicture.asset(
|
RotateIcon(
|
||||||
_expandedState ? Assets.svg.chevronUp : Assets.svg.chevronDown,
|
icon: SvgPicture.asset(
|
||||||
|
Assets.svg.chevronDown,
|
||||||
width: 12,
|
width: 12,
|
||||||
height: 6,
|
height: 6,
|
||||||
color: Theme.of(context)
|
color: Theme.of(context)
|
||||||
.extension<StackColors>()!
|
.extension<StackColors>()!
|
||||||
.textFieldActiveSearchIconRight,
|
.textFieldActiveSearchIconRight,
|
||||||
),
|
),
|
||||||
|
curve: widget.curve,
|
||||||
|
animationDurationMultiplier: widget.animationDurationMultiplier,
|
||||||
|
controller: _rotateIconController,
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
72
lib/widgets/animated_widgets/rotate_icon.dart
Normal file
72
lib/widgets/animated_widgets/rotate_icon.dart
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
import 'package:flutter/widgets.dart';
|
||||||
|
|
||||||
|
class RotateIconController {
|
||||||
|
VoidCallback? forward;
|
||||||
|
VoidCallback? reverse;
|
||||||
|
}
|
||||||
|
|
||||||
|
class RotateIcon extends StatefulWidget {
|
||||||
|
const RotateIcon({
|
||||||
|
Key? key,
|
||||||
|
required this.icon,
|
||||||
|
required this.curve,
|
||||||
|
this.controller,
|
||||||
|
this.animationDurationMultiplier = 1.0,
|
||||||
|
this.rotationPercent = 0.5,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
final Widget icon;
|
||||||
|
final Curve curve;
|
||||||
|
final RotateIconController? controller;
|
||||||
|
final double animationDurationMultiplier;
|
||||||
|
final double rotationPercent;
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<RotateIcon> createState() => _RotateIconState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _RotateIconState extends State<RotateIcon>
|
||||||
|
with SingleTickerProviderStateMixin {
|
||||||
|
late final AnimationController animationController;
|
||||||
|
late final Animation<double> animation;
|
||||||
|
late final Duration duration;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
duration = Duration(
|
||||||
|
milliseconds: (500 * widget.animationDurationMultiplier).toInt(),
|
||||||
|
);
|
||||||
|
animationController = AnimationController(
|
||||||
|
vsync: this,
|
||||||
|
duration: duration,
|
||||||
|
);
|
||||||
|
animation = Tween<double>(
|
||||||
|
begin: 0.0,
|
||||||
|
end: widget.rotationPercent,
|
||||||
|
).animate(
|
||||||
|
CurvedAnimation(
|
||||||
|
curve: widget.curve,
|
||||||
|
parent: animationController,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
widget.controller?.forward = animationController.forward;
|
||||||
|
widget.controller?.reverse = animationController.reverse;
|
||||||
|
|
||||||
|
super.initState();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
animationController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return RotationTransition(
|
||||||
|
turns: animation,
|
||||||
|
child: widget.icon,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,6 +19,7 @@ class Expandable extends StatefulWidget {
|
||||||
this.animation,
|
this.animation,
|
||||||
this.animationDurationMultiplier = 1.0,
|
this.animationDurationMultiplier = 1.0,
|
||||||
this.onExpandChanged,
|
this.onExpandChanged,
|
||||||
|
this.onExpandWillChange,
|
||||||
this.controller,
|
this.controller,
|
||||||
this.expandOverride,
|
this.expandOverride,
|
||||||
this.curve = Curves.easeInOut,
|
this.curve = Curves.easeInOut,
|
||||||
|
@ -30,6 +31,7 @@ class Expandable extends StatefulWidget {
|
||||||
final Animation<double>? animation;
|
final Animation<double>? animation;
|
||||||
final double animationDurationMultiplier;
|
final double animationDurationMultiplier;
|
||||||
final void Function(ExpandableState)? onExpandChanged;
|
final void Function(ExpandableState)? onExpandChanged;
|
||||||
|
final void Function(ExpandableState)? onExpandWillChange;
|
||||||
final ExpandableController? controller;
|
final ExpandableController? controller;
|
||||||
final VoidCallback? expandOverride;
|
final VoidCallback? expandOverride;
|
||||||
final Curve curve;
|
final Curve curve;
|
||||||
|
@ -48,10 +50,12 @@ class _ExpandableState extends State<Expandable> with TickerProviderStateMixin {
|
||||||
|
|
||||||
Future<void> toggle() async {
|
Future<void> toggle() async {
|
||||||
if (animation.isDismissed) {
|
if (animation.isDismissed) {
|
||||||
|
widget.onExpandWillChange?.call(ExpandableState.expanded);
|
||||||
await animationController.forward();
|
await animationController.forward();
|
||||||
_toggleState = ExpandableState.expanded;
|
_toggleState = ExpandableState.expanded;
|
||||||
widget.onExpandChanged?.call(_toggleState);
|
widget.onExpandChanged?.call(_toggleState);
|
||||||
} else if (animation.isCompleted) {
|
} else if (animation.isCompleted) {
|
||||||
|
widget.onExpandWillChange?.call(ExpandableState.collapsed);
|
||||||
await animationController.reverse();
|
await animationController.reverse();
|
||||||
_toggleState = ExpandableState.collapsed;
|
_toggleState = ExpandableState.collapsed;
|
||||||
widget.onExpandChanged?.call(_toggleState);
|
widget.onExpandChanged?.call(_toggleState);
|
||||||
|
|
Loading…
Reference in a new issue