/* * 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:tuple/tuple.dart'; import '../models/isar/models/isar_models.dart'; import '../notifications/show_flush_bar.dart'; import '../pages/wallet_view/sub_widgets/tx_icon.dart'; import '../pages/wallet_view/transaction_views/transaction_details_view.dart'; import '../providers/db/main_db_provider.dart'; import '../providers/providers.dart'; import '../themes/stack_colors.dart'; import '../utilities/amount/amount.dart'; import '../utilities/amount/amount_formatter.dart'; import '../utilities/constants.dart'; import '../utilities/format.dart'; import '../utilities/text_styles.dart'; import '../utilities/util.dart'; import '../wallets/crypto_currency/crypto_currency.dart'; import 'desktop/desktop_dialog.dart'; class TransactionCard extends ConsumerStatefulWidget { const TransactionCard({ super.key, required this.transaction, required this.walletId, }); final Transaction transaction; final String walletId; @override ConsumerState createState() => _TransactionCardState(); } class _TransactionCardState extends ConsumerState { late final Transaction _transaction; late final String walletId; late final bool isTokenTx; late final String prefix; late final String unit; late final CryptoCurrency coin; late final EthContract? tokenContract; late final int minConfirms; String whatIsIt( TransactionType type, CryptoCurrency coin, int currentHeight, ) { if (coin is Epiccash && _transaction.slateId == null) { return "Restored Funds"; } final confirmedStatus = _transaction.isConfirmed( currentHeight, minConfirms, ); if (type != TransactionType.incoming && _transaction.subType == TransactionSubType.mint) { // if (type == "Received") { if (confirmedStatus) { return "Anonymized"; } else { return "Anonymizing"; } // } else if (type == "Sent") { // if (_transaction.confirmedStatus) { // return "Sent MINT"; // } else { // return "Sending MINT"; // } // } else { // return type; // } } if (type == TransactionType.incoming) { // if (_transaction.isMinting) { // return "Minting"; // } else if (confirmedStatus) { return "Received"; } else { return "Receiving"; } } else if (type == TransactionType.outgoing) { if (confirmedStatus) { return "Sent"; } else { return "Sending"; } } else if (type == TransactionType.sentToSelf) { return "Sent to self"; } else { return type.name; } } @override void initState() { walletId = widget.walletId; minConfirms = ref.read(pWallets).getWallet(walletId).cryptoCurrency.minConfirms; _transaction = widget.transaction; isTokenTx = _transaction.subType == TransactionSubType.ethToken; if (Util.isDesktop) { if (_transaction.type == TransactionType.outgoing) { prefix = "-"; } else if (_transaction.type == TransactionType.incoming) { prefix = "+"; } else { prefix = ""; } } else { prefix = ""; } coin = ref.read(pWallets).getWallet(widget.walletId).info.coin; tokenContract = ref .read(mainDBProvider) .getEthContractSync(_transaction.otherData ?? ""); unit = isTokenTx ? tokenContract!.symbol : coin.ticker; super.initState(); } @override Widget build(BuildContext context) { final locale = ref.watch( localeServiceChangeNotifierProvider.select((value) => value.locale), ); final baseCurrency = ref .watch(prefsChangeNotifierProvider.select((value) => value.currency)); final price = ref .watch( priceAnd24hChangeNotifierProvider.select( (value) => isTokenTx ? value.getTokenPrice(_transaction.otherData!) : value.getPrice(coin), ), ) .item1; final currentHeight = ref.watch( pWallets .select((value) => value.getWallet(walletId).info.cachedChainHeight), ); return Material( color: Theme.of(context).extension()!.popupBG, elevation: 0, shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(Constants.size.circularBorderRadius), ), child: Padding( padding: const EdgeInsets.all(6), child: RawMaterialButton( shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular( Constants.size.circularBorderRadius, ), ), onPressed: () async { if (coin is Epiccash && _transaction.slateId == null) { unawaited( showFloatingFlushBar( context: context, message: "Restored Epic funds from your Seed have no Data.\nUse Stack Backup to keep your transaction history.", type: FlushBarType.warning, duration: const Duration(seconds: 5), ), ); return; } if (Util.isDesktop) { await showDialog( context: context, builder: (context) => DesktopDialog( maxHeight: MediaQuery.of(context).size.height - 64, maxWidth: 580, child: TransactionDetailsView( transaction: _transaction, coin: coin, walletId: walletId, ), ), ); } else { unawaited( Navigator.of(context).pushNamed( TransactionDetailsView.routeName, arguments: Tuple3( _transaction, coin, walletId, ), ), ); } }, child: Padding( padding: const EdgeInsets.all(8), child: Row( children: [ TxIcon( transaction: _transaction, coin: coin, currentHeight: currentHeight, ), const SizedBox( width: 14, ), Expanded( child: Column( mainAxisAlignment: MainAxisAlignment.spaceBetween, // crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Flexible( child: FittedBox( fit: BoxFit.scaleDown, child: Text( _transaction.isCancelled ? coin is Ethereum ? "Failed" : "Cancelled" : whatIsIt( _transaction.type, coin, currentHeight, ), style: STextStyles.itemSubtitle12(context), ), ), ), const SizedBox( width: 10, ), Flexible( child: FittedBox( fit: BoxFit.scaleDown, child: Builder( builder: (_) { final amount = _transaction.realAmount; return Text( "$prefix${ref.watch(pAmountFormatter(coin)).format(amount, ethContract: tokenContract)}", style: STextStyles.itemSubtitle12(context), ); }, ), ), ), ], ), const SizedBox( height: 4, ), Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, // crossAxisAlignment: CrossAxisAlignment.end, children: [ Flexible( child: FittedBox( fit: BoxFit.scaleDown, child: Text( Format.extractDateFrom(_transaction.timestamp), style: STextStyles.label(context), ), ), ), if (ref.watch( prefsChangeNotifierProvider .select((value) => value.externalCalls), )) const SizedBox( width: 10, ), if (ref.watch( prefsChangeNotifierProvider .select((value) => value.externalCalls), )) Flexible( child: FittedBox( fit: BoxFit.scaleDown, child: Builder( builder: (_) { final amount = _transaction.realAmount; return Text( "$prefix${Amount.fromDecimal( amount.decimal * price, fractionDigits: 2, ).fiatString( locale: locale, )} $baseCurrency", style: STextStyles.label(context), ); }, ), ), ), ], ), ], ), ), ], ), ), ), ), ); } }