mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-01-26 04:05:53 +00:00
334 lines
16 KiB
Dart
334 lines
16 KiB
Dart
|
import 'package:flutter/material.dart';
|
||
|
import 'package:flutter/services.dart';
|
||
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||
|
import 'package:flutter_svg/svg.dart';
|
||
|
import 'package:stackwallet/models/send_view_auto_fill_data.dart';
|
||
|
import 'package:stackwallet/notifications/show_flush_bar.dart';
|
||
|
import 'package:stackwallet/pages/address_book_views/subviews/contact_details_view.dart';
|
||
|
import 'package:stackwallet/pages/send_view/send_view.dart';
|
||
|
import 'package:stackwallet/providers/exchange/exchange_flow_is_active_state_provider.dart';
|
||
|
import 'package:stackwallet/providers/global/address_book_service_provider.dart';
|
||
|
import 'package:stackwallet/providers/providers.dart';
|
||
|
import 'package:stackwallet/utilities/assets.dart';
|
||
|
import 'package:stackwallet/utilities/cfcolors.dart';
|
||
|
import 'package:stackwallet/utilities/clipboard_interface.dart';
|
||
|
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
||
|
import 'package:stackwallet/utilities/enums/flush_bar_type.dart';
|
||
|
import 'package:stackwallet/utilities/text_styles.dart';
|
||
|
import 'package:stackwallet/widgets/rounded_container.dart';
|
||
|
import 'package:stackwallet/widgets/rounded_white_container.dart';
|
||
|
import 'package:tuple/tuple.dart';
|
||
|
|
||
|
class ContactPopUp extends ConsumerWidget {
|
||
|
const ContactPopUp({
|
||
|
Key? key,
|
||
|
required this.contactId,
|
||
|
this.clipboard = const ClipboardWrapper(),
|
||
|
}) : super(key: key);
|
||
|
|
||
|
final String contactId;
|
||
|
final ClipboardInterface clipboard;
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context, WidgetRef ref) {
|
||
|
final maxHeight = MediaQuery.of(context).size.height * 0.6;
|
||
|
final contact = ref.watch(addressBookServiceProvider
|
||
|
.select((value) => value.getContactById(contactId)));
|
||
|
|
||
|
final active = ref
|
||
|
.read(walletsChangeNotifierProvider)
|
||
|
.managers
|
||
|
.where((e) => e.isActiveWallet)
|
||
|
.toList(growable: false);
|
||
|
|
||
|
assert(active.isEmpty || active.length == 1);
|
||
|
|
||
|
bool hasActiveWallet = active.length == 1;
|
||
|
bool isExchangeFlow =
|
||
|
ref.watch(exchangeFlowIsActiveStateProvider.state).state;
|
||
|
|
||
|
final addresses = contact.addresses.where((e) {
|
||
|
if (hasActiveWallet && !isExchangeFlow) {
|
||
|
return e.coin == active[0].coin;
|
||
|
} else {
|
||
|
return true;
|
||
|
}
|
||
|
}).toList(growable: false);
|
||
|
|
||
|
return Column(
|
||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||
|
children: [
|
||
|
Padding(
|
||
|
padding: const EdgeInsets.symmetric(horizontal: 16),
|
||
|
child: LimitedBox(
|
||
|
maxHeight: maxHeight,
|
||
|
child: Column(
|
||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||
|
children: [
|
||
|
Material(
|
||
|
borderRadius: BorderRadius.circular(
|
||
|
20,
|
||
|
),
|
||
|
child: Container(
|
||
|
decoration: BoxDecoration(
|
||
|
color: CFColors.white,
|
||
|
borderRadius: BorderRadius.circular(
|
||
|
20,
|
||
|
),
|
||
|
),
|
||
|
child: LimitedBox(
|
||
|
maxHeight: maxHeight,
|
||
|
child: ClipRRect(
|
||
|
borderRadius: BorderRadius.circular(
|
||
|
20,
|
||
|
),
|
||
|
child: SingleChildScrollView(
|
||
|
child: Column(
|
||
|
// spacing: 10,
|
||
|
children: [
|
||
|
const SizedBox(
|
||
|
height: 24,
|
||
|
),
|
||
|
Padding(
|
||
|
padding:
|
||
|
const EdgeInsets.symmetric(horizontal: 24),
|
||
|
child: Row(
|
||
|
children: [
|
||
|
Container(
|
||
|
width: 32,
|
||
|
height: 32,
|
||
|
decoration: BoxDecoration(
|
||
|
color: CFColors.contactIconBackground,
|
||
|
borderRadius: BorderRadius.circular(32),
|
||
|
),
|
||
|
child: contact.id == "default"
|
||
|
? Center(
|
||
|
child: SvgPicture.asset(
|
||
|
Assets.svg.stackIcon,
|
||
|
width: 20,
|
||
|
),
|
||
|
)
|
||
|
: contact.emojiChar != null
|
||
|
? Center(
|
||
|
child:
|
||
|
Text(contact.emojiChar!),
|
||
|
)
|
||
|
: Center(
|
||
|
child: SvgPicture.asset(
|
||
|
Assets.svg.user,
|
||
|
width: 18,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
const SizedBox(
|
||
|
width: 12,
|
||
|
),
|
||
|
Expanded(
|
||
|
child: Text(
|
||
|
contact.name,
|
||
|
style: STextStyles.itemSubtitle12,
|
||
|
),
|
||
|
),
|
||
|
if (contact.id != "default")
|
||
|
TextButton(
|
||
|
onPressed: () {
|
||
|
Navigator.pop(context);
|
||
|
Navigator.of(context).pushNamed(
|
||
|
ContactDetailsView.routeName,
|
||
|
arguments: contact.id,
|
||
|
);
|
||
|
},
|
||
|
style: ButtonStyle(
|
||
|
minimumSize:
|
||
|
MaterialStateProperty.all<Size>(
|
||
|
const Size(46, 32)),
|
||
|
backgroundColor:
|
||
|
MaterialStateProperty.all(
|
||
|
CFColors.buttonGray,
|
||
|
),
|
||
|
),
|
||
|
child: Padding(
|
||
|
padding: const EdgeInsets.symmetric(
|
||
|
horizontal: 18),
|
||
|
child: Text("Details",
|
||
|
style: STextStyles.buttonSmall),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
SizedBox(
|
||
|
height: contact.id == "default" ? 16 : 8,
|
||
|
),
|
||
|
Container(
|
||
|
height: 1,
|
||
|
color: CFColors.almostWhite,
|
||
|
),
|
||
|
if (addresses.isEmpty)
|
||
|
Padding(
|
||
|
padding: const EdgeInsets.symmetric(
|
||
|
horizontal: 20),
|
||
|
child: RoundedWhiteContainer(
|
||
|
child: Center(
|
||
|
child: Text(
|
||
|
"No ${active[0].coin.prettyName} addresses found",
|
||
|
style: STextStyles.itemSubtitle,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
...addresses.map(
|
||
|
(e) => Padding(
|
||
|
padding: const EdgeInsets.only(
|
||
|
top: 12,
|
||
|
bottom: 12,
|
||
|
left: 28,
|
||
|
right: 24,
|
||
|
),
|
||
|
child: Row(
|
||
|
crossAxisAlignment:
|
||
|
CrossAxisAlignment.start,
|
||
|
children: [
|
||
|
Column(
|
||
|
children: [
|
||
|
const SizedBox(
|
||
|
height: 2,
|
||
|
),
|
||
|
SvgPicture.asset(
|
||
|
Assets.svg.iconFor(coin: e.coin),
|
||
|
height: 24,
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
const SizedBox(
|
||
|
width: 12,
|
||
|
),
|
||
|
Expanded(
|
||
|
child: Column(
|
||
|
crossAxisAlignment:
|
||
|
CrossAxisAlignment.start,
|
||
|
children: [
|
||
|
Text(
|
||
|
"${e.label} (${e.coin.ticker})",
|
||
|
style: STextStyles.itemSubtitle12,
|
||
|
),
|
||
|
const SizedBox(
|
||
|
height: 2,
|
||
|
),
|
||
|
Text(
|
||
|
e.address,
|
||
|
style: STextStyles.itemSubtitle
|
||
|
.copyWith(
|
||
|
fontSize: 8,
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
const SizedBox(
|
||
|
width: 10,
|
||
|
),
|
||
|
Column(
|
||
|
children: [
|
||
|
const SizedBox(
|
||
|
height: 2,
|
||
|
),
|
||
|
GestureDetector(
|
||
|
onTap: () {
|
||
|
clipboard.setData(
|
||
|
ClipboardData(text: e.address),
|
||
|
);
|
||
|
showFloatingFlushBar(
|
||
|
type: FlushBarType.info,
|
||
|
message: "Copied to clipboard",
|
||
|
iconAsset: Assets.svg.copy,
|
||
|
context: context,
|
||
|
);
|
||
|
},
|
||
|
child: RoundedContainer(
|
||
|
color: CFColors.fieldGray,
|
||
|
padding: const EdgeInsets.all(4),
|
||
|
child: SvgPicture.asset(
|
||
|
Assets.svg.copy,
|
||
|
width: 12,
|
||
|
height: 12,
|
||
|
color: CFColors.stackAccent,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
if (contact.id != "default" &&
|
||
|
hasActiveWallet &&
|
||
|
!isExchangeFlow)
|
||
|
const SizedBox(
|
||
|
width: 4,
|
||
|
),
|
||
|
if (contact.id != "default" &&
|
||
|
hasActiveWallet &&
|
||
|
!isExchangeFlow)
|
||
|
Column(
|
||
|
children: [
|
||
|
const SizedBox(
|
||
|
height: 2,
|
||
|
),
|
||
|
GestureDetector(
|
||
|
onTap: () {
|
||
|
final String contactLabel =
|
||
|
"${contact.name} (${e.label})";
|
||
|
final String address =
|
||
|
e.address;
|
||
|
|
||
|
if (hasActiveWallet) {
|
||
|
Navigator.of(context)
|
||
|
.pushNamed(
|
||
|
SendView.routeName,
|
||
|
arguments: Tuple3(
|
||
|
active[0].walletId,
|
||
|
active[0].coin,
|
||
|
SendViewAutoFillData(
|
||
|
address: address,
|
||
|
contactLabel:
|
||
|
contactLabel,
|
||
|
),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
},
|
||
|
child: RoundedContainer(
|
||
|
color: CFColors.fieldGray,
|
||
|
padding:
|
||
|
const EdgeInsets.all(4),
|
||
|
child: SvgPicture.asset(
|
||
|
Assets.svg.circleArrowUpRight,
|
||
|
width: 12,
|
||
|
height: 12,
|
||
|
color: CFColors.stackAccent,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
const SizedBox(
|
||
|
height: 12,
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
);
|
||
|
}
|
||
|
}
|