2023-04-14 06:39:08 +02:00
|
|
|
import 'package:cake_wallet/utils/responsive_layout_util.dart';
|
2020-10-22 21:24:24 +03:00
|
|
|
import 'package:cake_wallet/utils/show_bar.dart';
|
2022-10-26 20:13:44 +02:00
|
|
|
import 'package:another_flushbar/flushbar.dart';
|
2020-09-21 14:50:26 +03:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
2023-04-14 06:39:08 +02:00
|
|
|
import 'package:flutter/services.dart';
|
2020-09-21 14:50:26 +03:00
|
|
|
|
|
|
|
class PinCodeWidget extends StatefulWidget {
|
2023-04-14 06:39:08 +02:00
|
|
|
PinCodeWidget({
|
|
|
|
required Key key,
|
|
|
|
required this.onFullPin,
|
|
|
|
required this.initialPinLength,
|
|
|
|
required this.onChangedPin,
|
|
|
|
required this.hasLengthSwitcher,
|
|
|
|
this.onChangedPinLength,
|
|
|
|
}) : super(key: key);
|
2020-09-21 14:50:26 +03:00
|
|
|
|
|
|
|
final void Function(String pin, PinCodeState state) onFullPin;
|
|
|
|
final void Function(String pin) onChangedPin;
|
2022-10-12 13:09:57 -04:00
|
|
|
final void Function(int length)? onChangedPinLength;
|
2020-09-21 14:50:26 +03:00
|
|
|
final bool hasLengthSwitcher;
|
|
|
|
final int initialPinLength;
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => PinCodeState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class PinCodeState<T extends PinCodeWidget> extends State<T> {
|
2022-10-12 13:09:57 -04:00
|
|
|
PinCodeState()
|
2023-04-14 06:39:08 +02:00
|
|
|
: _aspectRatio = 0,
|
|
|
|
pinLength = 0,
|
|
|
|
pin = '',
|
|
|
|
title = '';
|
2020-09-21 14:50:26 +03:00
|
|
|
static const defaultPinLength = fourPinLength;
|
|
|
|
static const sixPinLength = 6;
|
|
|
|
static const fourPinLength = 4;
|
|
|
|
final _gridViewKey = GlobalKey();
|
|
|
|
final _key = GlobalKey<ScaffoldState>();
|
|
|
|
|
|
|
|
int pinLength;
|
|
|
|
String pin;
|
|
|
|
String title;
|
|
|
|
double _aspectRatio;
|
2022-10-26 20:13:44 +02:00
|
|
|
Flushbar<void>? _progressBar;
|
2020-09-21 14:50:26 +03:00
|
|
|
|
|
|
|
int currentPinLength() => pin.length;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
pinLength = widget.initialPinLength;
|
|
|
|
pin = '';
|
|
|
|
title = S.current.enter_your_pin;
|
|
|
|
_aspectRatio = 0;
|
|
|
|
WidgetsBinding.instance.addPostFrameCallback(_afterLayout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setTitle(String title) => setState(() => this.title = title);
|
|
|
|
|
|
|
|
void clear() => setState(() => pin = '');
|
|
|
|
|
|
|
|
void reset() => setState(() {
|
|
|
|
pin = '';
|
|
|
|
pinLength = widget.initialPinLength;
|
|
|
|
title = S.current.enter_your_pin;
|
|
|
|
});
|
|
|
|
|
|
|
|
void changePinLength(int length) {
|
|
|
|
setState(() {
|
|
|
|
pinLength = length;
|
|
|
|
pin = '';
|
|
|
|
});
|
|
|
|
|
|
|
|
widget.onChangedPinLength?.call(length);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setDefaultPinLength() => changePinLength(widget.initialPinLength);
|
|
|
|
|
|
|
|
void calculateAspectRatio() {
|
2023-04-14 06:39:08 +02:00
|
|
|
final renderBox = _gridViewKey.currentContext!.findRenderObject() as RenderBox;
|
2020-09-21 14:50:26 +03:00
|
|
|
final cellWidth = renderBox.size.width / 3;
|
|
|
|
final cellHeight = renderBox.size.height / 4;
|
|
|
|
|
|
|
|
if (cellWidth > 0 && cellHeight > 0) {
|
|
|
|
_aspectRatio = cellWidth / cellHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
setState(() {});
|
|
|
|
}
|
|
|
|
|
2020-10-22 21:24:24 +03:00
|
|
|
void changeProcessText(String text) {
|
|
|
|
hideProgressText();
|
2023-04-14 06:39:08 +02:00
|
|
|
_progressBar = createBar<void>(text, duration: null)..show(_key.currentContext!);
|
2020-10-22 21:24:24 +03:00
|
|
|
}
|
|
|
|
|
2020-10-24 17:35:55 +03:00
|
|
|
void close() {
|
2022-10-26 20:13:44 +02:00
|
|
|
_progressBar?.dismiss();
|
2022-10-12 13:09:57 -04:00
|
|
|
Navigator.of(_key.currentContext!).pop();
|
2020-10-24 17:35:55 +03:00
|
|
|
}
|
|
|
|
|
2020-10-22 21:24:24 +03:00
|
|
|
void hideProgressText() {
|
2022-10-26 20:13:44 +02:00
|
|
|
_progressBar?.dismiss();
|
|
|
|
_progressBar = null;
|
2020-10-22 21:24:24 +03:00
|
|
|
}
|
|
|
|
|
2020-09-21 14:50:26 +03:00
|
|
|
@override
|
2023-04-14 06:39:08 +02:00
|
|
|
Widget build(BuildContext context) =>
|
|
|
|
Scaffold(key: _key, body: body(context), resizeToAvoidBottomInset: false);
|
2020-09-21 14:50:26 +03:00
|
|
|
|
|
|
|
Widget body(BuildContext context) {
|
|
|
|
final deleteIconImage = Image.asset(
|
|
|
|
'assets/images/delete_icon.png',
|
2023-05-24 20:19:51 -03:00
|
|
|
color: Theme.of(context).primaryTextTheme!.titleLarge!.color!,
|
2020-09-21 14:50:26 +03:00
|
|
|
);
|
|
|
|
final faceImage = Image.asset(
|
|
|
|
'assets/images/face.png',
|
2023-05-24 20:19:51 -03:00
|
|
|
color: Theme.of(context).primaryTextTheme!.titleLarge!.color!,
|
2020-09-21 14:50:26 +03:00
|
|
|
);
|
|
|
|
|
2023-04-14 06:39:08 +02:00
|
|
|
return RawKeyboardListener(
|
|
|
|
focusNode: FocusNode(),
|
|
|
|
autofocus: true,
|
|
|
|
onKey: (keyEvent) {
|
|
|
|
if (keyEvent is RawKeyDownEvent) {
|
|
|
|
if (keyEvent.logicalKey.keyLabel == "Backspace") {
|
|
|
|
_pop();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int? number = int.tryParse(keyEvent.character ?? '');
|
|
|
|
if (number != null) {
|
|
|
|
_push(number);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
child: Container(
|
2023-05-24 20:19:51 -03:00
|
|
|
color: Theme.of(context).colorScheme.background,
|
2023-04-14 06:39:08 +02:00
|
|
|
padding: EdgeInsets.only(left: 40.0, right: 40.0, bottom: 40.0),
|
|
|
|
child: Column(
|
|
|
|
children: <Widget>[
|
|
|
|
Spacer(flex: 2),
|
|
|
|
Text(title,
|
2020-09-21 14:50:26 +03:00
|
|
|
style: TextStyle(
|
2023-04-14 06:39:08 +02:00
|
|
|
fontSize: 20,
|
|
|
|
fontWeight: FontWeight.w500,
|
2023-05-24 20:19:51 -03:00
|
|
|
color:
|
|
|
|
Theme.of(context).primaryTextTheme!.titleLarge!.color!)),
|
2023-04-14 06:39:08 +02:00
|
|
|
Spacer(flex: 3),
|
|
|
|
Container(
|
|
|
|
width: 180,
|
|
|
|
child: Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
|
|
children: List.generate(pinLength, (index) {
|
|
|
|
const size = 10.0;
|
|
|
|
final isFilled = pin.length > index ? pin[index] != null : false;
|
|
|
|
|
|
|
|
return Container(
|
|
|
|
width: size,
|
|
|
|
height: size,
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
shape: BoxShape.circle,
|
|
|
|
color: isFilled
|
2023-05-24 20:19:51 -03:00
|
|
|
? Theme.of(context)
|
|
|
|
.primaryTextTheme!
|
|
|
|
.titleLarge!
|
|
|
|
.color!
|
2023-04-14 06:39:08 +02:00
|
|
|
: Theme.of(context)
|
|
|
|
.accentTextTheme!
|
2023-05-24 20:19:51 -03:00
|
|
|
.bodyMedium!
|
2023-04-14 06:39:08 +02:00
|
|
|
.color!
|
|
|
|
.withOpacity(0.25),
|
|
|
|
));
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
Spacer(flex: 2),
|
|
|
|
if (widget.hasLengthSwitcher) ...[
|
|
|
|
TextButton(
|
|
|
|
onPressed: () {
|
|
|
|
changePinLength(pinLength == PinCodeState.fourPinLength
|
|
|
|
? PinCodeState.sixPinLength
|
|
|
|
: PinCodeState.fourPinLength);
|
|
|
|
},
|
|
|
|
child: Text(
|
|
|
|
_changePinLengthText(),
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 14.0,
|
|
|
|
fontWeight: FontWeight.normal,
|
2023-05-24 20:19:51 -03:00
|
|
|
color: Theme.of(context)
|
|
|
|
.accentTextTheme!
|
|
|
|
.bodyMedium!
|
|
|
|
.decorationColor!),
|
2023-04-14 06:39:08 +02:00
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
Spacer(flex: 1),
|
|
|
|
Flexible(
|
|
|
|
flex: 24,
|
|
|
|
child: Center(
|
|
|
|
child: ConstrainedBox(
|
|
|
|
constraints: BoxConstraints(
|
|
|
|
maxWidth: ResponsiveLayoutUtil.kDesktopMaxWidthConstraint,
|
|
|
|
),
|
|
|
|
child: Container(
|
|
|
|
key: _gridViewKey,
|
|
|
|
child: _aspectRatio > 0
|
|
|
|
? ScrollConfiguration(
|
|
|
|
behavior: ScrollConfiguration.of(context).copyWith(scrollbars: false),
|
|
|
|
child: GridView.count(
|
|
|
|
shrinkWrap: true,
|
|
|
|
crossAxisCount: 3,
|
|
|
|
childAspectRatio: _aspectRatio,
|
|
|
|
physics: const NeverScrollableScrollPhysics(),
|
|
|
|
children: List.generate(12, (index) {
|
|
|
|
const double marginRight = 15;
|
|
|
|
const double marginLeft = 15;
|
|
|
|
|
|
|
|
if (index == 9) {
|
2023-04-18 17:36:56 +00:00
|
|
|
// Empty container
|
2023-04-14 06:39:08 +02:00
|
|
|
return Container(
|
|
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
|
|
);
|
|
|
|
} else if (index == 10) {
|
|
|
|
index = 0;
|
|
|
|
} else if (index == 11) {
|
2023-04-18 17:36:56 +00:00
|
|
|
return MergeSemantics(
|
|
|
|
child: Container(
|
|
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
|
|
child: Semantics(
|
2023-07-11 18:21:59 -03:00
|
|
|
label: S.of(context).delete,
|
2023-04-18 17:36:56 +00:00
|
|
|
button: true,
|
|
|
|
onTap: () => _pop(),
|
|
|
|
child: TextButton(
|
|
|
|
onPressed: () => _pop(),
|
|
|
|
style: TextButton.styleFrom(
|
2023-05-24 20:19:51 -03:00
|
|
|
backgroundColor: Theme.of(context).colorScheme.background,
|
2023-04-18 17:36:56 +00:00
|
|
|
shape: CircleBorder(),
|
|
|
|
),
|
|
|
|
child: deleteIconImage,
|
|
|
|
),
|
2023-04-14 06:39:08 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Container(
|
|
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
|
|
child: TextButton(
|
|
|
|
onPressed: () => _push(index),
|
|
|
|
style: TextButton.styleFrom(
|
2023-05-24 20:19:51 -03:00
|
|
|
backgroundColor: Theme.of(context).colorScheme.background,
|
2023-04-14 06:39:08 +02:00
|
|
|
shape: CircleBorder(),
|
|
|
|
),
|
|
|
|
child: Text('$index',
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 30.0,
|
|
|
|
fontWeight: FontWeight.w600,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.primaryTextTheme!
|
2023-05-24 20:19:51 -03:00
|
|
|
.titleLarge!
|
2023-04-14 06:39:08 +02:00
|
|
|
.color!)),
|
2020-09-21 14:50:26 +03:00
|
|
|
),
|
2023-04-14 06:39:08 +02:00
|
|
|
);
|
|
|
|
}),
|
2020-09-21 14:50:26 +03:00
|
|
|
),
|
2023-04-14 06:39:08 +02:00
|
|
|
)
|
|
|
|
: null,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
2020-09-21 14:50:26 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _push(int num) {
|
|
|
|
setState(() {
|
|
|
|
if (currentPinLength() >= pinLength) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pin += num.toString();
|
|
|
|
|
|
|
|
widget.onChangedPin(pin);
|
|
|
|
|
|
|
|
if (pin.length == pinLength) {
|
|
|
|
widget.onFullPin(pin, this);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void _pop() {
|
|
|
|
if (currentPinLength() == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-09-25 18:32:44 +03:00
|
|
|
setState(() => pin = pin.substring(0, pin.length - 1));
|
2020-09-21 14:50:26 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
String _changePinLengthText() {
|
|
|
|
return S.current.use +
|
|
|
|
(pinLength == PinCodeState.fourPinLength
|
|
|
|
? '${PinCodeState.sixPinLength}'
|
|
|
|
: '${PinCodeState.fourPinLength}') +
|
|
|
|
S.current.digit_pin;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _afterLayout(dynamic _) {
|
|
|
|
setDefaultPinLength();
|
|
|
|
calculateAspectRatio();
|
|
|
|
}
|
|
|
|
}
|