mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2024-12-23 11:59:30 +00:00
WIP desktop send flow address book address chooser
This commit is contained in:
parent
60b332ad8a
commit
4d8804f78b
4 changed files with 385 additions and 84 deletions
|
@ -0,0 +1,143 @@
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
|
import 'package:stackwallet/models/contact.dart';
|
||||||
|
import 'package:stackwallet/pages_desktop_specific/home/my_stack_view/wallet_view/sub_widgets/address_book_address_chooser/sub_widgets/contact_list_item.dart';
|
||||||
|
import 'package:stackwallet/providers/global/address_book_service_provider.dart';
|
||||||
|
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
||||||
|
import 'package:stackwallet/utilities/text_styles.dart';
|
||||||
|
|
||||||
|
class AddressBookAddressChooser extends StatefulWidget {
|
||||||
|
const AddressBookAddressChooser({
|
||||||
|
Key? key,
|
||||||
|
this.coin,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
final Coin? coin;
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<AddressBookAddressChooser> createState() =>
|
||||||
|
_AddressBookAddressChooserState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _AddressBookAddressChooserState extends State<AddressBookAddressChooser> {
|
||||||
|
int _compareContactFavorite(Contact a, Contact b) {
|
||||||
|
if (a.isFavorite && b.isFavorite) {
|
||||||
|
return 0;
|
||||||
|
} else if (a.isFavorite) {
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Contact> pullOutFavorites(List<Contact> contacts) {
|
||||||
|
final List<Contact> favorites = [];
|
||||||
|
contacts.removeWhere((contact) {
|
||||||
|
if (contact.isFavorite) {
|
||||||
|
favorites.add(contact);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
return favorites;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Contact> filter(List<Contact> contacts) {
|
||||||
|
if (widget.coin != null) {
|
||||||
|
contacts.removeWhere(
|
||||||
|
(e) => e.addresses.where((a) => a.coin == widget.coin!).isEmpty);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (contacts.length < 2) {
|
||||||
|
return contacts;
|
||||||
|
}
|
||||||
|
|
||||||
|
contacts.sort(_compareContactFavorite);
|
||||||
|
|
||||||
|
// TODO: other filtering?
|
||||||
|
|
||||||
|
return contacts;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Column(
|
||||||
|
children: [
|
||||||
|
// search field
|
||||||
|
const TextField(),
|
||||||
|
const SizedBox(
|
||||||
|
height: 16,
|
||||||
|
),
|
||||||
|
Expanded(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 32,
|
||||||
|
right: 32,
|
||||||
|
bottom: 32,
|
||||||
|
),
|
||||||
|
child: Consumer(
|
||||||
|
builder: (context, ref, _) {
|
||||||
|
List<Contact> contacts = ref
|
||||||
|
.watch(addressBookServiceProvider
|
||||||
|
.select((value) => value.contacts))
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
contacts = filter(contacts);
|
||||||
|
|
||||||
|
final favorites = pullOutFavorites(contacts);
|
||||||
|
|
||||||
|
return ListView.builder(
|
||||||
|
primary: false,
|
||||||
|
shrinkWrap: true,
|
||||||
|
itemCount: favorites.length +
|
||||||
|
contacts.length +
|
||||||
|
2, // +2 for "fav" and "all" headers
|
||||||
|
itemBuilder: (context, index) {
|
||||||
|
if (index == 0) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
bottom: 10,
|
||||||
|
),
|
||||||
|
child: Text(
|
||||||
|
"Favorites",
|
||||||
|
style:
|
||||||
|
STextStyles.desktopTextExtraExtraSmall(context),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else if (index <= favorites.length) {
|
||||||
|
final id = favorites[index - 1].id;
|
||||||
|
return ContactListItem(
|
||||||
|
key: Key("contactCard_${id}_key"),
|
||||||
|
contactId: id,
|
||||||
|
filterByCoin: widget.coin,
|
||||||
|
);
|
||||||
|
} else if (index == favorites.length + 1) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
vertical: 10,
|
||||||
|
),
|
||||||
|
child: Text(
|
||||||
|
"All contacts",
|
||||||
|
style:
|
||||||
|
STextStyles.desktopTextExtraExtraSmall(context),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
final id = contacts[index - favorites.length - 1].id;
|
||||||
|
return ContactListItem(
|
||||||
|
key: Key("contactCard_${id}_key"),
|
||||||
|
contactId: id,
|
||||||
|
filterByCoin: widget.coin,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,114 @@
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
|
import 'package:stackwallet/providers/global/address_book_service_provider.dart';
|
||||||
|
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
||||||
|
import 'package:stackwallet/utilities/text_styles.dart';
|
||||||
|
import 'package:stackwallet/utilities/theme/stack_colors.dart';
|
||||||
|
import 'package:stackwallet/widgets/address_book_card.dart';
|
||||||
|
import 'package:stackwallet/widgets/custom_buttons/blue_text_button.dart';
|
||||||
|
import 'package:stackwallet/widgets/expandable.dart';
|
||||||
|
import 'package:stackwallet/widgets/rounded_white_container.dart';
|
||||||
|
import 'package:stackwallet/widgets/wallet_info_row/sub_widgets/wallet_info_row_coin_icon.dart';
|
||||||
|
|
||||||
|
class ContactListItem extends ConsumerWidget {
|
||||||
|
const ContactListItem({
|
||||||
|
Key? key,
|
||||||
|
required this.contactId,
|
||||||
|
this.filterByCoin,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
final String contactId;
|
||||||
|
final Coin? filterByCoin;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
|
final contact = ref.watch(addressBookServiceProvider
|
||||||
|
.select((value) => value.getContactById(contactId)));
|
||||||
|
|
||||||
|
return RoundedWhiteContainer(
|
||||||
|
padding: const EdgeInsets.all(0),
|
||||||
|
borderColor: Theme.of(context).extension<StackColors>()!.background,
|
||||||
|
child: Expandable(
|
||||||
|
header: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 20,
|
||||||
|
vertical: 14,
|
||||||
|
),
|
||||||
|
child: AddressBookCard(
|
||||||
|
contactId: contactId,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
// filter addresses by coin is provided before building address list
|
||||||
|
...contact.addresses
|
||||||
|
.where((e) =>
|
||||||
|
filterByCoin != null ? e.coin == filterByCoin! : true)
|
||||||
|
.map(
|
||||||
|
(e) => Column(
|
||||||
|
key: Key("contactAddress_${e.address}_${e.label}_key"),
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
height: 1,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.extension<StackColors>()!
|
||||||
|
.background,
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 20,
|
||||||
|
vertical: 14,
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
WalletInfoCoinIcon(coin: e.coin),
|
||||||
|
const SizedBox(
|
||||||
|
width: 12,
|
||||||
|
),
|
||||||
|
Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
"${contactId == "default" ? e.other! : e.label} (${e.coin.ticker})",
|
||||||
|
style: STextStyles
|
||||||
|
.desktopTextExtraExtraSmall(
|
||||||
|
context)
|
||||||
|
.copyWith(
|
||||||
|
color: Theme.of(context)
|
||||||
|
.extension<StackColors>()!
|
||||||
|
.textDark,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
e.address,
|
||||||
|
style: STextStyles
|
||||||
|
.desktopTextExtraExtraSmall(context),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
BlueTextButton(
|
||||||
|
text: "Select wallet",
|
||||||
|
onTap: () {
|
||||||
|
Navigator.of(context).pop(e);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,17 +5,17 @@ import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:flutter_svg/flutter_svg.dart';
|
import 'package:flutter_svg/flutter_svg.dart';
|
||||||
|
import 'package:stackwallet/models/contact_address_entry.dart';
|
||||||
import 'package:stackwallet/models/send_view_auto_fill_data.dart';
|
import 'package:stackwallet/models/send_view_auto_fill_data.dart';
|
||||||
import 'package:stackwallet/pages/address_book_views/address_book_view.dart';
|
|
||||||
import 'package:stackwallet/pages/send_view/confirm_transaction_view.dart';
|
import 'package:stackwallet/pages/send_view/confirm_transaction_view.dart';
|
||||||
import 'package:stackwallet/pages/send_view/sub_widgets/building_transaction_dialog.dart';
|
import 'package:stackwallet/pages/send_view/sub_widgets/building_transaction_dialog.dart';
|
||||||
import 'package:stackwallet/pages/send_view/sub_widgets/firo_balance_selection_sheet.dart';
|
import 'package:stackwallet/pages/send_view/sub_widgets/firo_balance_selection_sheet.dart';
|
||||||
import 'package:stackwallet/pages/send_view/sub_widgets/transaction_fee_selection_sheet.dart';
|
import 'package:stackwallet/pages/send_view/sub_widgets/transaction_fee_selection_sheet.dart';
|
||||||
|
import 'package:stackwallet/pages_desktop_specific/home/my_stack_view/wallet_view/sub_widgets/address_book_address_chooser/address_book_address_chooser.dart';
|
||||||
import 'package:stackwallet/providers/providers.dart';
|
import 'package:stackwallet/providers/providers.dart';
|
||||||
import 'package:stackwallet/providers/ui/fee_rate_type_state_provider.dart';
|
import 'package:stackwallet/providers/ui/fee_rate_type_state_provider.dart';
|
||||||
import 'package:stackwallet/providers/ui/preview_tx_button_state_provider.dart';
|
import 'package:stackwallet/providers/ui/preview_tx_button_state_provider.dart';
|
||||||
import 'package:stackwallet/providers/wallet/public_private_balance_state_provider.dart';
|
import 'package:stackwallet/providers/wallet/public_private_balance_state_provider.dart';
|
||||||
import 'package:stackwallet/route_generator.dart';
|
|
||||||
import 'package:stackwallet/services/coins/firo/firo_wallet.dart';
|
import 'package:stackwallet/services/coins/firo/firo_wallet.dart';
|
||||||
import 'package:stackwallet/services/coins/manager.dart';
|
import 'package:stackwallet/services/coins/manager.dart';
|
||||||
import 'package:stackwallet/utilities/address_utils.dart';
|
import 'package:stackwallet/utilities/address_utils.dart';
|
||||||
|
@ -42,7 +42,6 @@ import 'package:stackwallet/widgets/icon_widgets/qrcode_icon.dart';
|
||||||
import 'package:stackwallet/widgets/icon_widgets/x_icon.dart';
|
import 'package:stackwallet/widgets/icon_widgets/x_icon.dart';
|
||||||
import 'package:stackwallet/widgets/stack_text_field.dart';
|
import 'package:stackwallet/widgets/stack_text_field.dart';
|
||||||
import 'package:stackwallet/widgets/textfield_icon_button.dart';
|
import 'package:stackwallet/widgets/textfield_icon_button.dart';
|
||||||
import 'package:tuple/tuple.dart';
|
|
||||||
|
|
||||||
class DesktopSend extends ConsumerStatefulWidget {
|
class DesktopSend extends ConsumerStatefulWidget {
|
||||||
const DesktopSend({
|
const DesktopSend({
|
||||||
|
@ -1184,11 +1183,34 @@ class _DesktopSendState extends ConsumerState<DesktopSend> {
|
||||||
if (sendToController.text.isEmpty)
|
if (sendToController.text.isEmpty)
|
||||||
TextFieldIconButton(
|
TextFieldIconButton(
|
||||||
key: const Key("sendViewAddressBookButtonKey"),
|
key: const Key("sendViewAddressBookButtonKey"),
|
||||||
onTap: () {
|
onTap: () async {
|
||||||
Navigator.of(context).pushNamed(
|
final entry =
|
||||||
AddressBookView.routeName,
|
await showDialog<ContactAddressEntry?>(
|
||||||
arguments: coin,
|
context: context,
|
||||||
|
builder: (context) => DesktopDialog(
|
||||||
|
maxWidth: 696,
|
||||||
|
maxHeight: 600,
|
||||||
|
child: AddressBookAddressChooser(
|
||||||
|
coin: coin,
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (entry != null) {
|
||||||
|
sendToController.text =
|
||||||
|
entry.other ?? entry.label;
|
||||||
|
|
||||||
|
_address = entry.address;
|
||||||
|
|
||||||
|
_updatePreviewButtonState(
|
||||||
|
_address,
|
||||||
|
_amountToSend,
|
||||||
|
);
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_addressToggleFlag = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
},
|
},
|
||||||
child: const AddressBookIcon(),
|
child: const AddressBookIcon(),
|
||||||
),
|
),
|
||||||
|
|
|
@ -8,6 +8,8 @@ import 'package:stackwallet/utilities/constants.dart';
|
||||||
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
import 'package:stackwallet/utilities/enums/coin_enum.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/widgets/conditional_parent.dart';
|
||||||
import 'package:stackwallet/widgets/rounded_white_container.dart';
|
import 'package:stackwallet/widgets/rounded_white_container.dart';
|
||||||
|
|
||||||
class AddressBookCard extends ConsumerStatefulWidget {
|
class AddressBookCard extends ConsumerStatefulWidget {
|
||||||
|
@ -21,11 +23,12 @@ class AddressBookCard extends ConsumerStatefulWidget {
|
||||||
|
|
||||||
class _AddressBookCardState extends ConsumerState<AddressBookCard> {
|
class _AddressBookCardState extends ConsumerState<AddressBookCard> {
|
||||||
late final String contactId;
|
late final String contactId;
|
||||||
|
late final bool isDesktop;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
contactId = widget.contactId;
|
contactId = widget.contactId;
|
||||||
|
isDesktop = Util.isDesktop;
|
||||||
super.initState();
|
super.initState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -51,29 +54,8 @@ class _AddressBookCardState extends ConsumerState<AddressBookCard> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return RoundedWhiteContainer(
|
return ConditionalParent(
|
||||||
padding: const EdgeInsets.all(4),
|
condition: !isDesktop,
|
||||||
child: RawMaterialButton(
|
|
||||||
// splashColor: Theme.of(context).extension<StackColors>()!.highlight,
|
|
||||||
padding: const EdgeInsets.all(0),
|
|
||||||
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(
|
|
||||||
Constants.size.circularBorderRadius,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
onPressed: () {
|
|
||||||
showDialog<void>(
|
|
||||||
context: context,
|
|
||||||
useSafeArea: true,
|
|
||||||
barrierDismissible: true,
|
|
||||||
builder: (_) => ContactPopUp(
|
|
||||||
contactId: contact.id,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(8.0),
|
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
Container(
|
Container(
|
||||||
|
@ -110,6 +92,21 @@ class _AddressBookCardState extends ConsumerState<AddressBookCard> {
|
||||||
const SizedBox(
|
const SizedBox(
|
||||||
width: 12,
|
width: 12,
|
||||||
),
|
),
|
||||||
|
if (isDesktop)
|
||||||
|
Text(
|
||||||
|
contact.name,
|
||||||
|
style: STextStyles.itemSubtitle12(context),
|
||||||
|
),
|
||||||
|
if (isDesktop)
|
||||||
|
const SizedBox(
|
||||||
|
width: 16,
|
||||||
|
),
|
||||||
|
if (isDesktop)
|
||||||
|
Text(
|
||||||
|
coinsString,
|
||||||
|
style: STextStyles.label(context),
|
||||||
|
),
|
||||||
|
if (!isDesktop)
|
||||||
Column(
|
Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
|
@ -128,6 +125,31 @@ class _AddressBookCardState extends ConsumerState<AddressBookCard> {
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
builder: (child) => RoundedWhiteContainer(
|
||||||
|
padding: const EdgeInsets.all(4),
|
||||||
|
child: RawMaterialButton(
|
||||||
|
// splashColor: Theme.of(context).extension<StackColors>()!.highlight,
|
||||||
|
padding: const EdgeInsets.all(0),
|
||||||
|
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(
|
||||||
|
Constants.size.circularBorderRadius,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
showDialog<void>(
|
||||||
|
context: context,
|
||||||
|
useSafeArea: true,
|
||||||
|
barrierDismissible: true,
|
||||||
|
builder: (_) => ContactPopUp(
|
||||||
|
contactId: contact.id,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: child,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
Loading…
Reference in a new issue