mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-06-28 04:19:50 +00:00
* feat: Integration tests setup and tests for Disclaimer, Welcome and Setup Pin Code pages * feat: Integration test flow from start to restoring a wallet successfully done * test: Dashboard view test and linking to flow * feat: Testing the Exchange flow section, selecting sending and receiving currencies * test: Successfully create an exchange section * feat: Implement flow up to sending section * test: Complete Exchange flow * fix dependency issue * test: Final cleanups * feat: Add CI to run automated integration tests withan android emulator * feat: Adjust Automated integration test CI to run on ubuntu 20.04-a * fix: Move integration test CI into PR test build CI * ci: Add automated test ci which is a streamlined replica of pr test build ci * ci: Re-add step to access branch name * ci: Add KVM * ci: Add filepath to trigger the test run from * ci: Add required key * ci: Add required key * ci: Add missing secret key * ci: Add missing secret key * ci: Add nano secrets to workflow * ci: Switch step to free space on runner * ci: Remove timeout from workflow * ci: Confirm impact that removing copy_monero_deps would have on entire workflow time * ci: Update CI and temporarily remove cache related to emulator * ci: Remove dynamic java version * ci: Temporarily switch CI * ci: Switch to 11.x jdk * ci: Temporarily switch CI * ci: Revert ubuntu version * ci: Add more api levels * ci: Add more target options * ci: Settled on stable emulator matrix options * ci: Add more target options * ci: Modify flow * ci: Streamline api levels to 28 and 29 * ci: One more trial * ci: Switch to flutter drive * ci: Reduce options * ci: Remove haven from test * ci: Check for solana in list * ci: Adjust amounts and currencies for exchange flow * ci: Set write response on failure to true * ci: Split ci to funds and non funds related tests * test: Test for Send flow scenario and minor restructuring for test folders and files * chore: cleanup * ci: Pause CI for now * ci: Pause CI for now * ci: Pause CI for now * Fix: Add keys back to currency amount textfield widget * fix: Switch variable name * fix: remove automation for now * test: Updating send page robot and also syncing branch with main --------- Co-authored-by: OmarHatem <omarh.ismail1@gmail.com>
306 lines
10 KiB
Dart
306 lines
10 KiB
Dart
import 'package:cake_wallet/themes/extensions/pin_code_theme.dart';
|
|
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
|
import 'package:cake_wallet/utils/responsive_layout_util.dart';
|
|
import 'package:cake_wallet/utils/show_bar.dart';
|
|
import 'package:another_flushbar/flushbar.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
import 'package:flutter/services.dart';
|
|
|
|
class PinCodeWidget extends StatefulWidget {
|
|
PinCodeWidget({
|
|
required Key key,
|
|
required this.onFullPin,
|
|
required this.initialPinLength,
|
|
required this.onChangedPin,
|
|
required this.hasLengthSwitcher,
|
|
this.onChangedPinLength,
|
|
}) : super(key: key);
|
|
|
|
final void Function(String pin, PinCodeState state) onFullPin;
|
|
final void Function(String pin) onChangedPin;
|
|
final void Function(int length)? onChangedPinLength;
|
|
final bool hasLengthSwitcher;
|
|
final int initialPinLength;
|
|
|
|
@override
|
|
State<StatefulWidget> createState() => PinCodeState();
|
|
}
|
|
|
|
class PinCodeState<T extends PinCodeWidget> extends State<T> {
|
|
PinCodeState()
|
|
: _aspectRatio = 0,
|
|
pinLength = 0,
|
|
pin = '',
|
|
title = '';
|
|
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;
|
|
Flushbar<void>? _progressBar;
|
|
|
|
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() {
|
|
final renderBox = _gridViewKey.currentContext!.findRenderObject() as RenderBox;
|
|
final cellWidth = renderBox.size.width / 3;
|
|
final cellHeight = renderBox.size.height / 4;
|
|
|
|
if (cellWidth > 0 && cellHeight > 0) {
|
|
_aspectRatio = cellWidth / cellHeight;
|
|
}
|
|
|
|
setState(() {});
|
|
}
|
|
|
|
void changeProcessText(String text) {
|
|
hideProgressText();
|
|
_progressBar = createBar<void>(text, duration: null)..show(_key.currentContext!);
|
|
}
|
|
|
|
void close() {
|
|
_progressBar?.dismiss();
|
|
Navigator.of(_key.currentContext!).pop();
|
|
}
|
|
|
|
void hideProgressText() {
|
|
_progressBar?.dismiss();
|
|
_progressBar = null;
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) =>
|
|
Scaffold(key: _key, body: body(context), resizeToAvoidBottomInset: false);
|
|
|
|
Widget body(BuildContext context) {
|
|
final deleteIconImage = Image.asset(
|
|
'assets/images/delete_icon.png',
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
|
);
|
|
final faceImage = Image.asset(
|
|
'assets/images/face.png',
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
|
);
|
|
|
|
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(
|
|
color: Theme.of(context).colorScheme.background,
|
|
padding: EdgeInsets.only(left: 40.0, right: 40.0, bottom: 40.0),
|
|
child: Column(
|
|
children: <Widget>[
|
|
Spacer(flex: 2),
|
|
Text(title,
|
|
style: TextStyle(
|
|
fontSize: 20,
|
|
fontWeight: FontWeight.w500,
|
|
color:
|
|
Theme.of(context).extension<CakeTextTheme>()!.titleColor)),
|
|
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
|
|
? Theme.of(context).extension<CakeTextTheme>()!.titleColor
|
|
: Theme.of(context).extension<PinCodeTheme>()!.indicatorsColor
|
|
.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,
|
|
color: Theme.of(context).extension<PinCodeTheme>()!.switchColor),
|
|
),
|
|
)
|
|
],
|
|
Spacer(flex: 1),
|
|
Flexible(
|
|
flex: 24,
|
|
child: Center(
|
|
child: ConstrainedBox(
|
|
constraints: BoxConstraints(
|
|
maxWidth: ResponsiveLayoutUtilBase.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) {
|
|
// Empty container
|
|
return Container(
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
);
|
|
} else if (index == 10) {
|
|
index = 0;
|
|
} else if (index == 11) {
|
|
return MergeSemantics(
|
|
child: Container(
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
child: Semantics(
|
|
label: S.of(context).delete,
|
|
button: true,
|
|
onTap: () => _pop(),
|
|
child: TextButton(
|
|
onPressed: () => _pop(),
|
|
style: TextButton.styleFrom(
|
|
backgroundColor: Theme.of(context).colorScheme.background,
|
|
shape: CircleBorder(),
|
|
),
|
|
child: deleteIconImage,
|
|
),
|
|
),
|
|
),
|
|
);
|
|
} else {
|
|
index++;
|
|
}
|
|
|
|
return Container(
|
|
margin: EdgeInsets.only(left: marginLeft, right: marginRight),
|
|
child: TextButton(
|
|
key: ValueKey('pin_code_button_${index}_key'),
|
|
onPressed: () => _push(index),
|
|
style: TextButton.styleFrom(
|
|
backgroundColor: Theme.of(context).colorScheme.background,
|
|
shape: CircleBorder(),
|
|
),
|
|
child: Text('$index',
|
|
style: TextStyle(
|
|
fontSize: 30.0,
|
|
fontWeight: FontWeight.w600,
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor)),
|
|
),
|
|
);
|
|
}),
|
|
),
|
|
)
|
|
: null,
|
|
),
|
|
),
|
|
),
|
|
)
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
setState(() => pin = pin.substring(0, pin.length - 1));
|
|
}
|
|
|
|
String _changePinLengthText() {
|
|
return S.current.use +
|
|
(pinLength == PinCodeState.fourPinLength
|
|
? '${PinCodeState.sixPinLength}'
|
|
: '${PinCodeState.fourPinLength}') +
|
|
S.current.digit_pin;
|
|
}
|
|
|
|
void _afterLayout(dynamic _) {
|
|
setDefaultPinLength();
|
|
calculateAspectRatio();
|
|
}
|
|
}
|