mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-03-22 23:28:53 +00:00
* init commit * onramper * moonPay * dfx provider * meld * dfx payment methods * fiat buy credentials * moonpay payment method * payment loading state * dfx sell quote * onramper launch trade * meld launch trade * country picker * update option tile * buy/sell action * meld refactor * update pr_test_build.yml * ui fixes * revert country picker commit * update the ui * recommended providers * payment method [skip ci] * provider option tile * remove buy action * minor fixes * update the best rate when the amount is changed * fixes for currency title * fix icons * code refactoring * null issue * code review fixes * Update pr_test_build_linux.yml * Update meld_buy_provider.dart * Update meld_buy_provider.dart * add show wallets action * remove default sell / buy provider setting * localisation * providerTypes * icons * remove duplicate file [skip ci] * minor changes [skip ci] * fixes from review * disable dfx for non eur/chf currencies fix providers to be fetched with the selected currency * fix breaking from loop if one element failed * fix minor naming issue from merging conflicts * add fiat check for moonpay * fix address validation * merge conflict * fix destination and source currency * minor fix * minor fix * update the flow * fix bch address format * fix wallet addresses * fix initial fetching amount. * Update address_validator.dart * review comments * revert switch case to return null * minor fix --------- Co-authored-by: OmarHatem <omarh.ismail1@gmail.com>
279 lines
11 KiB
Dart
279 lines
11 KiB
Dart
import 'package:cake_wallet/utils/device_info.dart';
|
|
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
|
import 'package:cake_wallet/utils/responsive_layout_util.dart';
|
|
import 'package:cw_core/currency.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:cake_wallet/routes.dart';
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
import 'package:cake_wallet/entities/qr_scanner.dart';
|
|
import 'package:cake_wallet/entities/contact_base.dart';
|
|
import 'package:cake_wallet/themes/extensions/send_page_theme.dart';
|
|
import 'package:cake_wallet/utils/permission_handler.dart';
|
|
import 'package:permission_handler/permission_handler.dart';
|
|
|
|
enum AddressTextFieldOption { paste, qrCode, addressBook, walletAddresses }
|
|
|
|
|
|
class AddressTextField<T extends Currency> extends StatelessWidget{
|
|
AddressTextField({
|
|
required this.controller,
|
|
this.isActive = true,
|
|
this.placeholder,
|
|
this.options = const [AddressTextFieldOption.qrCode, AddressTextFieldOption.addressBook],
|
|
this.onURIScanned,
|
|
this.focusNode,
|
|
this.isBorderExist = true,
|
|
this.buttonColor,
|
|
this.borderColor,
|
|
this.iconColor,
|
|
this.textStyle,
|
|
this.hintStyle,
|
|
this.validator,
|
|
this.onPushPasteButton,
|
|
this.onPushAddressBookButton,
|
|
this.onPushAddressPickerButton,
|
|
this.onSelectedContact,
|
|
this.selectedCurrency,
|
|
this.addressKey,
|
|
});
|
|
|
|
static const prefixIconWidth = 34.0;
|
|
static const prefixIconHeight = 34.0;
|
|
static const spaceBetweenPrefixIcons = 10.0;
|
|
|
|
final TextEditingController? controller;
|
|
final bool isActive;
|
|
final String? placeholder;
|
|
final Function(Uri)? onURIScanned;
|
|
final List<AddressTextFieldOption> options;
|
|
final FormFieldValidator<String>? validator;
|
|
final bool isBorderExist;
|
|
final Color? buttonColor;
|
|
final Color? borderColor;
|
|
final Color? iconColor;
|
|
final TextStyle? textStyle;
|
|
final TextStyle? hintStyle;
|
|
final FocusNode? focusNode;
|
|
final Function(BuildContext context)? onPushPasteButton;
|
|
final Function(BuildContext context)? onPushAddressBookButton;
|
|
final Function(BuildContext context)? onPushAddressPickerButton;
|
|
final Function(ContactBase contact)? onSelectedContact;
|
|
final T? selectedCurrency;
|
|
final Key? addressKey;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Stack(
|
|
children: <Widget>[
|
|
TextFormField(
|
|
key: addressKey,
|
|
enableIMEPersonalizedLearning: false,
|
|
keyboardType: TextInputType.visiblePassword,
|
|
onFieldSubmitted: (_) => FocusScope.of(context).unfocus(),
|
|
enabled: isActive,
|
|
controller: controller,
|
|
focusNode: focusNode,
|
|
style: textStyle ??
|
|
TextStyle(
|
|
fontSize: 16, color: Theme.of(context).extension<CakeTextTheme>()!.titleColor),
|
|
decoration: InputDecoration(
|
|
suffixIcon: SizedBox(
|
|
width: prefixIconWidth * options.length + (spaceBetweenPrefixIcons * options.length),
|
|
),
|
|
hintStyle: hintStyle ?? TextStyle(fontSize: 16, color: Theme.of(context).hintColor),
|
|
hintText: placeholder ?? S.current.widgets_address,
|
|
focusedBorder: isBorderExist
|
|
? UnderlineInputBorder(
|
|
borderSide: BorderSide(
|
|
color: borderColor ?? Theme.of(context).dividerColor, width: 1.0))
|
|
: InputBorder.none,
|
|
disabledBorder: isBorderExist
|
|
? UnderlineInputBorder(
|
|
borderSide: BorderSide(
|
|
color: borderColor ?? Theme.of(context).dividerColor, width: 1.0))
|
|
: InputBorder.none,
|
|
enabledBorder: isBorderExist
|
|
? UnderlineInputBorder(
|
|
borderSide: BorderSide(
|
|
color: borderColor ?? Theme.of(context).dividerColor, width: 1.0))
|
|
: InputBorder.none,
|
|
),
|
|
validator: validator,
|
|
),
|
|
Positioned(
|
|
top: 2,
|
|
right: 0,
|
|
child: SizedBox(
|
|
width:
|
|
(prefixIconWidth * options.length) + (spaceBetweenPrefixIcons * options.length),
|
|
child: Row(
|
|
mainAxisAlignment: responsiveLayoutUtil.shouldRenderMobileUI
|
|
? MainAxisAlignment.spaceBetween
|
|
: MainAxisAlignment.end,
|
|
children: [
|
|
if (this.options.contains(AddressTextFieldOption.paste)) ...[
|
|
SizedBox(width: 5),
|
|
Container(
|
|
width: prefixIconWidth,
|
|
height: prefixIconHeight,
|
|
padding: EdgeInsets.only(top: 0),
|
|
child: Semantics(
|
|
label: S.of(context).paste,
|
|
child: InkWell(
|
|
onTap: () async => _pasteAddress(context),
|
|
child: Container(
|
|
padding: EdgeInsets.all(8),
|
|
decoration: BoxDecoration(
|
|
color:
|
|
buttonColor ?? Theme.of(context).dialogTheme.backgroundColor,
|
|
borderRadius: BorderRadius.all(Radius.circular(6))),
|
|
child: Image.asset(
|
|
'assets/images/paste_ios.png',
|
|
color: iconColor ??
|
|
Theme.of(context)
|
|
.extension<SendPageTheme>()!
|
|
.textFieldButtonIconColor,
|
|
)),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
if (this.options.contains(AddressTextFieldOption.qrCode) &&
|
|
DeviceInfo.instance.isMobile) ...[
|
|
SizedBox(width: 5),
|
|
Container(
|
|
width: prefixIconWidth,
|
|
height: prefixIconHeight,
|
|
padding: EdgeInsets.only(top: 0),
|
|
child: Semantics(
|
|
label: S.of(context).scan_qr_code,
|
|
child: InkWell(
|
|
onTap: () async => _presentQRScanner(context),
|
|
child: Container(
|
|
padding: EdgeInsets.all(8),
|
|
decoration: BoxDecoration(
|
|
color:
|
|
buttonColor ?? Theme.of(context).dialogTheme.backgroundColor,
|
|
borderRadius: BorderRadius.all(Radius.circular(6))),
|
|
child: Image.asset(
|
|
'assets/images/qr_code_icon.png',
|
|
color: iconColor ??
|
|
Theme.of(context)
|
|
.extension<SendPageTheme>()!
|
|
.textFieldButtonIconColor,
|
|
)),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
if (this.options.contains(AddressTextFieldOption.addressBook)) ...[
|
|
SizedBox(width: 5),
|
|
Container(
|
|
width: prefixIconWidth,
|
|
height: prefixIconHeight,
|
|
padding: EdgeInsets.only(top: 0),
|
|
child: Semantics(
|
|
label: S.of(context).address_book,
|
|
child: InkWell(
|
|
onTap: () async => _presetAddressBookPicker(context),
|
|
child: Container(
|
|
padding: EdgeInsets.all(8),
|
|
decoration: BoxDecoration(
|
|
color:
|
|
buttonColor ?? Theme.of(context).dialogTheme.backgroundColor,
|
|
borderRadius: BorderRadius.all(Radius.circular(6))),
|
|
child: Image.asset(
|
|
'assets/images/open_book.png',
|
|
color: iconColor ??
|
|
Theme.of(context)
|
|
.extension<SendPageTheme>()!
|
|
.textFieldButtonIconColor,
|
|
)),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
if (this.options.contains(AddressTextFieldOption.walletAddresses)) ...[
|
|
SizedBox(width: 5),
|
|
Container(
|
|
width: prefixIconWidth,
|
|
height: prefixIconHeight,
|
|
padding: EdgeInsets.only(top: 0),
|
|
child: Semantics(
|
|
label: S.of(context).address_book,
|
|
child: InkWell(
|
|
onTap: () async => _presetWalletAddressPicker(context),
|
|
child: Container(
|
|
padding: EdgeInsets.all(8),
|
|
decoration: BoxDecoration(
|
|
color:
|
|
buttonColor ?? Theme.of(context).dialogTheme.backgroundColor,
|
|
borderRadius: BorderRadius.all(Radius.circular(6))),
|
|
child: Image.asset(
|
|
'assets/images/open_book.png',
|
|
color: iconColor ??
|
|
Theme.of(context)
|
|
.extension<SendPageTheme>()!
|
|
.textFieldButtonIconColor,
|
|
)),
|
|
),
|
|
),
|
|
)
|
|
],
|
|
],
|
|
),
|
|
))
|
|
],
|
|
);
|
|
}
|
|
|
|
Future<void> _presentQRScanner(BuildContext context) async {
|
|
bool isCameraPermissionGranted =
|
|
await PermissionHandler.checkPermission(Permission.camera, context);
|
|
if (!isCameraPermissionGranted) return;
|
|
final code = await presentQRScanner(context);
|
|
if (code.isEmpty) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
final uri = Uri.parse(code);
|
|
controller?.text = uri.path;
|
|
onURIScanned?.call(uri);
|
|
} catch (_) {
|
|
controller?.text = code;
|
|
}
|
|
}
|
|
|
|
Future<void> _presetAddressBookPicker(BuildContext context) async {
|
|
final contact = await Navigator.of(context)
|
|
.pushNamed(Routes.pickerAddressBook, arguments: selectedCurrency);
|
|
|
|
if (contact is ContactBase) {
|
|
controller?.text = contact.address;
|
|
onPushAddressBookButton?.call(context);
|
|
onSelectedContact?.call(contact);
|
|
}
|
|
}
|
|
|
|
Future<void> _presetWalletAddressPicker(BuildContext context) async {
|
|
final address = await Navigator.of(context).pushNamed(Routes.pickerWalletAddress);
|
|
|
|
if (address is String) {
|
|
controller?.text = address;
|
|
onPushAddressPickerButton?.call(context);
|
|
}
|
|
}
|
|
|
|
Future<void> _pasteAddress(BuildContext context) async {
|
|
final clipboard = await Clipboard.getData('text/plain');
|
|
final address = clipboard?.text ?? '';
|
|
|
|
if (address.isNotEmpty) {
|
|
controller?.text = address;
|
|
}
|
|
|
|
onPushPasteButton?.call(context);
|
|
}
|
|
}
|