mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-01-12 13:44:31 +00:00
312 lines
12 KiB
Dart
312 lines
12 KiB
Dart
|
import 'package:flutter/material.dart';
|
||
|
import 'package:flutter_svg/svg.dart';
|
||
|
import 'package:stackwallet/utilities/text_styles.dart';
|
||
|
import 'package:stackwallet/utilities/theme/stack_colors.dart';
|
||
|
import 'package:stackwallet/utilities/util.dart';
|
||
|
|
||
|
class Toggle extends StatefulWidget {
|
||
|
const Toggle({
|
||
|
Key? key,
|
||
|
this.onIcon,
|
||
|
this.onText,
|
||
|
this.offIcon,
|
||
|
this.offText,
|
||
|
this.onValueChanged,
|
||
|
required this.isOn,
|
||
|
// this.enabled = true,
|
||
|
this.controller,
|
||
|
required this.onColor,
|
||
|
required this.offColor,
|
||
|
this.decoration,
|
||
|
}) : super(key: key);
|
||
|
|
||
|
final String? onIcon;
|
||
|
final String? onText;
|
||
|
final String? offIcon;
|
||
|
final String? offText;
|
||
|
final void Function(bool)? onValueChanged;
|
||
|
final bool isOn;
|
||
|
// final bool enabled;
|
||
|
final DSBController? controller;
|
||
|
|
||
|
final Color onColor;
|
||
|
final Color offColor;
|
||
|
final BoxDecoration? decoration;
|
||
|
|
||
|
@override
|
||
|
ToggleState createState() => ToggleState();
|
||
|
}
|
||
|
|
||
|
class ToggleState extends State<Toggle> {
|
||
|
late final BoxDecoration? decoration;
|
||
|
late final Color onColor;
|
||
|
late final Color offColor;
|
||
|
late final DSBController? controller;
|
||
|
|
||
|
final bool isDesktop = Util.isDesktop;
|
||
|
|
||
|
late bool _isOn;
|
||
|
bool get isOn => _isOn;
|
||
|
|
||
|
// late bool _enabled;
|
||
|
|
||
|
late ValueNotifier<double> valueListener;
|
||
|
|
||
|
final tapAnimationDuration = const Duration(milliseconds: 150);
|
||
|
bool _isDragging = false;
|
||
|
|
||
|
// Color _colorBG(bool isOn, double alpha) {
|
||
|
// return Color.alphaBlend(
|
||
|
// onColor.withOpacity(alpha),
|
||
|
// offColor,
|
||
|
// );
|
||
|
// }
|
||
|
|
||
|
// Color _colorFG(bool isOn, double alpha) {
|
||
|
// return Color.alphaBlend(
|
||
|
// onColor.withOpacity(alpha),
|
||
|
// offColor,
|
||
|
// );
|
||
|
// }
|
||
|
|
||
|
@override
|
||
|
initState() {
|
||
|
onColor = widget.onColor;
|
||
|
offColor = widget.offColor;
|
||
|
decoration = widget.decoration;
|
||
|
controller = widget.controller;
|
||
|
_isOn = widget.isOn;
|
||
|
// _enabled = widget.enabled;
|
||
|
valueListener = _isOn ? ValueNotifier(1.0) : ValueNotifier(0.0);
|
||
|
|
||
|
widget.controller?.activate = () {
|
||
|
_isOn = !_isOn;
|
||
|
// widget.onValueChanged?.call(_isOn);
|
||
|
valueListener.value = _isOn ? 1.0 : 0.0;
|
||
|
};
|
||
|
super.initState();
|
||
|
}
|
||
|
|
||
|
@override
|
||
|
void dispose() {
|
||
|
valueListener.dispose();
|
||
|
super.dispose();
|
||
|
}
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
debugPrint("BUILD: $runtimeType");
|
||
|
|
||
|
return GestureDetector(
|
||
|
onTap: () {
|
||
|
_isOn = !_isOn;
|
||
|
widget.onValueChanged?.call(_isOn);
|
||
|
valueListener.value = _isOn ? 1.0 : 0.0;
|
||
|
setState(() {});
|
||
|
},
|
||
|
child: LayoutBuilder(
|
||
|
builder: (context, constraint) {
|
||
|
return Stack(
|
||
|
children: [
|
||
|
AnimatedBuilder(
|
||
|
animation: valueListener,
|
||
|
builder: (context, child) {
|
||
|
return AnimatedContainer(
|
||
|
duration: tapAnimationDuration,
|
||
|
height: constraint.maxHeight,
|
||
|
width: constraint.maxWidth,
|
||
|
decoration: decoration?.copyWith(
|
||
|
color: offColor,
|
||
|
),
|
||
|
);
|
||
|
},
|
||
|
),
|
||
|
Builder(
|
||
|
builder: (context) {
|
||
|
final handle = GestureDetector(
|
||
|
key: const Key("draggableSwitchButtonSwitch"),
|
||
|
onHorizontalDragStart: (_) => _isDragging = true,
|
||
|
onHorizontalDragUpdate: (details) {
|
||
|
valueListener.value = (valueListener.value +
|
||
|
details.delta.dx / constraint.maxWidth)
|
||
|
.clamp(0.0, 1.0);
|
||
|
},
|
||
|
onHorizontalDragEnd: (details) {
|
||
|
bool oldValue = _isOn;
|
||
|
if (valueListener.value > 0.5) {
|
||
|
valueListener.value = 1.0;
|
||
|
_isOn = true;
|
||
|
} else {
|
||
|
valueListener.value = 0.0;
|
||
|
_isOn = false;
|
||
|
}
|
||
|
if (_isOn != oldValue) {
|
||
|
widget.onValueChanged?.call(_isOn);
|
||
|
controller?.isOn = _isOn;
|
||
|
setState(() {});
|
||
|
}
|
||
|
_isDragging = false;
|
||
|
},
|
||
|
child: AnimatedBuilder(
|
||
|
animation: valueListener,
|
||
|
builder: (context, child) {
|
||
|
return AnimatedContainer(
|
||
|
duration: tapAnimationDuration,
|
||
|
height: constraint.maxHeight,
|
||
|
width: constraint.maxWidth / 2,
|
||
|
decoration: decoration?.copyWith(
|
||
|
color:
|
||
|
onColor, //_colorFG(_isOn, valueListener.value),
|
||
|
),
|
||
|
);
|
||
|
},
|
||
|
),
|
||
|
);
|
||
|
return AnimatedBuilder(
|
||
|
animation: valueListener,
|
||
|
builder: (context, child) {
|
||
|
return AnimatedAlign(
|
||
|
duration:
|
||
|
_isDragging ? Duration.zero : tapAnimationDuration,
|
||
|
alignment: Alignment(valueListener.value * 2 - 1, 0.5),
|
||
|
child: child,
|
||
|
);
|
||
|
},
|
||
|
child: handle,
|
||
|
);
|
||
|
},
|
||
|
),
|
||
|
IgnorePointer(
|
||
|
child: Row(
|
||
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||
|
children: [
|
||
|
SizedBox(
|
||
|
width: constraint.maxWidth / 2,
|
||
|
child: Center(
|
||
|
child: Row(
|
||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||
|
children: [
|
||
|
SvgPicture.asset(
|
||
|
widget.onIcon ?? "",
|
||
|
width: 12,
|
||
|
height: 14,
|
||
|
color: isDesktop
|
||
|
? !_isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.accentColorBlue
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.buttonTextSecondary
|
||
|
: !_isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textDark
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textSubtitle1,
|
||
|
),
|
||
|
const SizedBox(
|
||
|
width: 5,
|
||
|
),
|
||
|
Text(
|
||
|
widget.onText ?? "",
|
||
|
style: isDesktop
|
||
|
? STextStyles.desktopTextExtraExtraSmall(
|
||
|
context)
|
||
|
.copyWith(
|
||
|
color: !_isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.accentColorBlue
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.buttonTextSecondary,
|
||
|
)
|
||
|
: STextStyles.smallMed12(context).copyWith(
|
||
|
color: !_isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textDark
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textSubtitle1,
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
SizedBox(
|
||
|
width: constraint.maxWidth / 2,
|
||
|
child: Center(
|
||
|
child: Row(
|
||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||
|
children: [
|
||
|
SvgPicture.asset(
|
||
|
widget.offIcon ?? "",
|
||
|
width: 12,
|
||
|
height: 14,
|
||
|
color: isDesktop
|
||
|
? _isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.accentColorBlue
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.buttonTextSecondary
|
||
|
: _isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textDark
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textSubtitle1,
|
||
|
),
|
||
|
const SizedBox(
|
||
|
width: 5,
|
||
|
),
|
||
|
Text(
|
||
|
widget.offText ?? "",
|
||
|
style: isDesktop
|
||
|
? STextStyles.desktopTextExtraExtraSmall(
|
||
|
context)
|
||
|
.copyWith(
|
||
|
color: _isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.accentColorBlue
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.buttonTextSecondary,
|
||
|
)
|
||
|
: STextStyles.smallMed12(context).copyWith(
|
||
|
color: _isOn
|
||
|
? Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textDark
|
||
|
: Theme.of(context)
|
||
|
.extension<StackColors>()!
|
||
|
.textSubtitle1,
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
);
|
||
|
},
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class DSBController {
|
||
|
VoidCallback? activate;
|
||
|
bool? isOn;
|
||
|
}
|