/* * This file is part of Stack Wallet. * * Copyright (c) 2023 Cypher Stack * All Rights Reserved. * The code is distributed under GPLv3 license, see LICENSE file for details. * Generated by Cypher Stack on 2023-05-26 * */ import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; import '../../models/paynym/paynym_account.dart'; import '../../providers/global/paynym_api_provider.dart'; import '../../providers/global/wallets_provider.dart'; import '../../providers/wallet/my_paynym_account_state_provider.dart'; import '../../themes/stack_colors.dart'; import '../../utilities/assets.dart'; import '../../utilities/text_styles.dart'; import '../../utilities/util.dart'; import '../../wallets/wallet/wallet_mixin_interfaces/paynym_interface.dart'; import '../../widgets/conditional_parent.dart'; import '../../widgets/custom_buttons/app_bar_icon_button.dart'; import '../../widgets/desktop/desktop_app_bar.dart'; import '../../widgets/desktop/desktop_scaffold.dart'; import '../../widgets/desktop/primary_button.dart'; import '../wallet_view/wallet_view.dart'; import 'dialogs/claiming_paynym_dialog.dart'; import 'paynym_home_view.dart'; class PaynymClaimView extends ConsumerStatefulWidget { const PaynymClaimView({ super.key, required this.walletId, }); final String walletId; static const String routeName = "/claimPaynym"; @override ConsumerState createState() => _PaynymClaimViewState(); } class _PaynymClaimViewState extends ConsumerState { Future _addSegwitCode(PaynymAccount myAccount) async { final wallet = ref.read(pWallets).getWallet(widget.walletId) as PaynymInterface; final token = await ref .read(paynymAPIProvider) .token(myAccount.nonSegwitPaymentCode.code); final signature = await wallet.signStringWithNotificationKey(token.value!); final pCodeSegwit = await wallet.getPaymentCode(isSegwit: true); final addResult = await ref.read(paynymAPIProvider).add( token.value!, signature, myAccount.nymID, pCodeSegwit.toString(), ); return addResult.value ?? false; } @override Widget build(BuildContext context) { debugPrint("BUILD: $runtimeType"); final isDesktop = Util.isDesktop; return MasterScaffold( isDesktop: isDesktop, appBar: isDesktop ? DesktopAppBar( isCompactHeight: true, background: Theme.of(context).extension()!.popupBG, leading: Row( children: [ Padding( padding: const EdgeInsets.only( left: 24, right: 20, ), child: AppBarIconButton( size: 32, color: Theme.of(context) .extension()! .textFieldDefaultBG, shadows: const [], icon: SvgPicture.asset( Assets.svg.arrowLeft, width: 18, height: 18, color: Theme.of(context) .extension()! .topNavIconPrimary, ), onPressed: Navigator.of(context).pop, ), ), SvgPicture.asset( Assets.svg.user, width: 42, height: 42, color: Theme.of(context).extension()!.textDark, ), const SizedBox( width: 10, ), Text( "PayNym", style: STextStyles.desktopH3(context), ), ], ), ) : AppBar( leading: const AppBarBackButton(), titleSpacing: 0, title: Text( "PayNym", style: STextStyles.navBarTitle(context), overflow: TextOverflow.ellipsis, ), ), body: ConditionalParent( condition: !isDesktop, builder: (child) => SafeArea( child: Padding( padding: const EdgeInsets.all(16), child: child, ), ), child: ConditionalParent( condition: isDesktop, builder: (child) => SizedBox( width: 328, child: child, ), child: Column( children: [ const Spacer( flex: 1, ), SvgPicture.asset( Assets.svg.unclaimedPaynym, width: MediaQuery.of(context).size.width / 2, ), const SizedBox( height: 20, ), Text( "You do not have a PayNym yet.\nClaim yours now!", style: isDesktop ? STextStyles.desktopSubtitleH2(context).copyWith( color: Theme.of(context) .extension()! .textSubtitle1, ) : STextStyles.baseXS(context).copyWith( color: Theme.of(context) .extension()! .textSubtitle1, ), textAlign: TextAlign.center, ), if (isDesktop) const SizedBox( height: 30, ), if (!isDesktop) const Spacer( flex: 2, ), PrimaryButton( label: "Claim", onPressed: () async { bool shouldCancel = false; unawaited( showDialog( context: context, barrierDismissible: false, builder: (context) => const ClaimingPaynymDialog(), ).then((value) => shouldCancel = value == true), ); final wallet = ref.read(pWallets).getWallet(widget.walletId) as PaynymInterface; if (shouldCancel) return; // get payment code final pCode = await wallet.getPaymentCode(isSegwit: false); if (shouldCancel) return; // attempt to create new entry in [PaynymIsApi.baseURL] db final created = await ref .read(paynymAPIProvider) .create(pCode.toString()); debugPrint("created:$created"); if (shouldCancel) return; if (created.value!.claimed) { // payment code already claimed debugPrint("pcode already claimed!!"); final account = await ref.read(paynymAPIProvider).nym(pCode.toString()); if (!account.value!.segwit) { for (int i = 0; i < 100; i++) { final result = await _addSegwitCode(account.value!); if (result == true) { break; } } } if (mounted) { if (isDesktop) { Navigator.of(context, rootNavigator: true).pop(); Navigator.of(context).pop(); } else { Navigator.of(context).popUntil( ModalRoute.withName( WalletView.routeName, ), ); } } return; } if (shouldCancel) return; final token = await ref.read(paynymAPIProvider).token(pCode.toString()); if (shouldCancel) return; // sign token with notification private key final signature = await wallet.signStringWithNotificationKey(token.value!); if (shouldCancel) return; // claim paynym account final claim = await ref .read(paynymAPIProvider) .claim(token.value!, signature); if (shouldCancel) return; if (claim.value?.claimed == pCode.toString()) { final account = await ref.read(paynymAPIProvider).nym(pCode.toString()); if (!account.value!.segwit) { for (int i = 0; i < 100; i++) { final result = await _addSegwitCode(account.value!); if (result == true) { break; } } } ref.read(myPaynymAccountStateProvider.state).state = account.value!; if (mounted) { if (isDesktop) { Navigator.of(context, rootNavigator: true).pop(); Navigator.of(context).pop(); } else { Navigator.of(context).popUntil( ModalRoute.withName( WalletView.routeName, ), ); } await Navigator.of(context).pushNamed( PaynymHomeView.routeName, arguments: widget.walletId, ); } } else if (mounted && !shouldCancel) { Navigator.of(context, rootNavigator: isDesktop).pop(); } }, ), if (isDesktop) const Spacer( flex: 2, ), ], ), ), ), ); } }