token transaction display fixes

This commit is contained in:
julian 2024-01-11 12:12:31 -06:00
parent 90cc8ced9c
commit 46d454fad1
6 changed files with 115 additions and 55 deletions

View file

@ -6,7 +6,6 @@ import 'package:stackwallet/models/isar/models/blockchain_data/transaction.dart'
import 'package:stackwallet/models/isar/models/blockchain_data/v2/input_v2.dart'; import 'package:stackwallet/models/isar/models/blockchain_data/v2/input_v2.dart';
import 'package:stackwallet/models/isar/models/blockchain_data/v2/output_v2.dart'; import 'package:stackwallet/models/isar/models/blockchain_data/v2/output_v2.dart';
import 'package:stackwallet/utilities/amount/amount.dart'; import 'package:stackwallet/utilities/amount/amount.dart';
import 'package:stackwallet/utilities/enums/coin_enum.dart';
import 'package:stackwallet/utilities/extensions/extensions.dart'; import 'package:stackwallet/utilities/extensions/extensions.dart';
import 'package:stackwallet/wallets/wallet/wallet_mixin_interfaces/spark_interface.dart'; import 'package:stackwallet/wallets/wallet/wallet_mixin_interfaces/spark_interface.dart';
@ -79,7 +78,7 @@ class TransactionV2 {
return confirmations >= minimumConfirms; return confirmations >= minimumConfirms;
} }
Amount getFee({required Coin coin}) { Amount getFee({required int fractionDigits}) {
// check for override fee // check for override fee
final fee = _getOverrideFee(); final fee = _getOverrideFee();
if (fee != null) { if (fee != null) {
@ -92,44 +91,47 @@ class TransactionV2 {
.map((e) => e.value) .map((e) => e.value)
.reduce((value, element) => value += element); .reduce((value, element) => value += element);
return Amount(rawValue: inSum - outSum, fractionDigits: coin.decimals); return Amount(rawValue: inSum - outSum, fractionDigits: fractionDigits);
} }
Amount getAmountReceivedInThisWallet({required Coin coin}) { Amount getAmountReceivedInThisWallet({required int fractionDigits}) {
final outSum = outputs final outSum = outputs
.where((e) => e.walletOwns) .where((e) => e.walletOwns)
.fold(BigInt.zero, (p, e) => p + e.value); .fold(BigInt.zero, (p, e) => p + e.value);
return Amount(rawValue: outSum, fractionDigits: coin.decimals); return Amount(rawValue: outSum, fractionDigits: fractionDigits);
} }
Amount getAmountSparkSelfMinted({required Coin coin}) { Amount getAmountSparkSelfMinted({required int fractionDigits}) {
final outSum = outputs.where((e) { final outSum = outputs.where((e) {
final op = e.scriptPubKeyHex.substring(0, 2).toUint8ListFromHex.first; final op = e.scriptPubKeyHex.substring(0, 2).toUint8ListFromHex.first;
return e.walletOwns && (op == OP_SPARKMINT); return e.walletOwns && (op == OP_SPARKMINT);
}).fold(BigInt.zero, (p, e) => p + e.value); }).fold(BigInt.zero, (p, e) => p + e.value);
return Amount(rawValue: outSum, fractionDigits: coin.decimals); return Amount(rawValue: outSum, fractionDigits: fractionDigits);
} }
Amount getAmountSentFromThisWallet({required Coin coin}) { Amount getAmountSentFromThisWallet({required int fractionDigits}) {
final inSum = inputs final inSum = inputs
.where((e) => e.walletOwns) .where((e) => e.walletOwns)
.fold(BigInt.zero, (p, e) => p + e.value); .fold(BigInt.zero, (p, e) => p + e.value);
final amount = Amount( Amount amount = Amount(
rawValue: inSum, rawValue: inSum,
fractionDigits: coin.decimals, fractionDigits: fractionDigits,
) - ) -
getAmountReceivedInThisWallet( getAmountReceivedInThisWallet(
coin: coin, fractionDigits: fractionDigits,
) - );
getFee(coin: coin);
if (subType != TransactionSubType.ethToken) {
amount = amount - getFee(fractionDigits: fractionDigits);
}
// negative amounts are likely an error or can happen with coins such as eth // negative amounts are likely an error or can happen with coins such as eth
// that don't use the btc style inputs/outputs // that don't use the btc style inputs/outputs
if (amount.raw < BigInt.zero) { if (amount.raw < BigInt.zero) {
return Amount.zeroWith(fractionDigits: coin.decimals); return Amount.zeroWith(fractionDigits: fractionDigits);
} }
return amount; return amount;

View file

@ -843,6 +843,9 @@ class _DesktopTransactionCardRowState
late final TransactionV2 _transaction; late final TransactionV2 _transaction;
late final String walletId; late final String walletId;
late final int minConfirms; late final int minConfirms;
late final EthContract? ethContract;
bool get isTokenTx => ethContract != null;
String whatIsIt(TransactionV2 tx, int height) => tx.statusLabel( String whatIsIt(TransactionV2 tx, int height) => tx.statusLabel(
currentChainHeight: height, currentChainHeight: height,
@ -858,6 +861,15 @@ class _DesktopTransactionCardRowState
.cryptoCurrency .cryptoCurrency
.minConfirms; .minConfirms;
_transaction = widget.transaction; _transaction = widget.transaction;
if (_transaction.subType == TransactionSubType.ethToken) {
ethContract = ref
.read(mainDBProvider)
.getEthContractSync(_transaction.contractAddress!);
} else {
ethContract == null;
}
super.initState(); super.initState();
} }
@ -892,19 +904,22 @@ class _DesktopTransactionCardRowState
final currentHeight = ref.watch(pWalletChainHeight(walletId)); final currentHeight = ref.watch(pWalletChainHeight(walletId));
final Amount amount; final Amount amount;
final fractionDigits = ethContract?.decimals ?? coin.decimals;
if (_transaction.subType == TransactionSubType.cashFusion) { if (_transaction.subType == TransactionSubType.cashFusion) {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
} else { } else {
switch (_transaction.type) { switch (_transaction.type) {
case TransactionType.outgoing: case TransactionType.outgoing:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
break; break;
case TransactionType.incoming: case TransactionType.incoming:
case TransactionType.sentToSelf: case TransactionType.sentToSelf:
if (_transaction.subType == TransactionSubType.sparkMint) { if (_transaction.subType == TransactionSubType.sparkMint) {
amount = _transaction.getAmountSparkSelfMinted(coin: coin); amount = _transaction.getAmountSparkSelfMinted(
fractionDigits: fractionDigits);
} else if (_transaction.subType == TransactionSubType.sparkSpend) { } else if (_transaction.subType == TransactionSubType.sparkSpend) {
final changeAddress = final changeAddress =
(ref.watch(pWallets).getWallet(walletId) as SparkInterface) (ref.watch(pWallets).getWallet(walletId) as SparkInterface)
@ -917,12 +932,14 @@ class _DesktopTransactionCardRowState
fractionDigits: coin.decimals, fractionDigits: coin.decimals,
); );
} else { } else {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
} }
break; break;
case TransactionType.unknown: case TransactionType.unknown:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
break; break;
} }
} }
@ -1005,7 +1022,7 @@ class _DesktopTransactionCardRowState
Expanded( Expanded(
flex: 6, flex: 6,
child: Text( child: Text(
"$prefix${ref.watch(pAmountFormatter(coin)).format(amount)}", "$prefix${ref.watch(pAmountFormatter(coin)).format(amount, ethContract: ethContract)}",
style: style:
STextStyles.desktopTextExtraExtraSmall(context).copyWith( STextStyles.desktopTextExtraExtraSmall(context).copyWith(
color: Theme.of(context).extension<StackColors>()!.textDark, color: Theme.of(context).extension<StackColors>()!.textDark,

View file

@ -112,18 +112,23 @@ class _TransactionCardStateV2 extends ConsumerState<TransactionCardV2> {
final Amount amount; final Amount amount;
final fractionDigits = tokenContract?.decimals ?? coin.decimals;
if (_transaction.subType == TransactionSubType.cashFusion) { if (_transaction.subType == TransactionSubType.cashFusion) {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
} else { } else {
switch (_transaction.type) { switch (_transaction.type) {
case TransactionType.outgoing: case TransactionType.outgoing:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
break; break;
case TransactionType.incoming: case TransactionType.incoming:
case TransactionType.sentToSelf: case TransactionType.sentToSelf:
if (_transaction.subType == TransactionSubType.sparkMint) { if (_transaction.subType == TransactionSubType.sparkMint) {
amount = _transaction.getAmountSparkSelfMinted(coin: coin); amount = _transaction.getAmountSparkSelfMinted(
fractionDigits: fractionDigits);
} else if (_transaction.subType == TransactionSubType.sparkSpend) { } else if (_transaction.subType == TransactionSubType.sparkSpend) {
final changeAddress = final changeAddress =
(ref.watch(pWallets).getWallet(walletId) as SparkInterface) (ref.watch(pWallets).getWallet(walletId) as SparkInterface)
@ -136,12 +141,14 @@ class _TransactionCardStateV2 extends ConsumerState<TransactionCardV2> {
fractionDigits: coin.decimals, fractionDigits: coin.decimals,
); );
} else { } else {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
} }
break; break;
case TransactionType.unknown: case TransactionType.unknown:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
break; break;
} }
} }

View file

@ -17,11 +17,13 @@ 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/isar/models/blockchain_data/transaction.dart'; import 'package:stackwallet/models/isar/models/blockchain_data/transaction.dart';
import 'package:stackwallet/models/isar/models/blockchain_data/v2/transaction_v2.dart'; import 'package:stackwallet/models/isar/models/blockchain_data/v2/transaction_v2.dart';
import 'package:stackwallet/models/isar/models/ethereum/eth_contract.dart';
import 'package:stackwallet/notifications/show_flush_bar.dart'; import 'package:stackwallet/notifications/show_flush_bar.dart';
import 'package:stackwallet/pages/wallet_view/sub_widgets/tx_icon.dart'; import 'package:stackwallet/pages/wallet_view/sub_widgets/tx_icon.dart';
import 'package:stackwallet/pages/wallet_view/transaction_views/dialogs/cancelling_transaction_progress_dialog.dart'; import 'package:stackwallet/pages/wallet_view/transaction_views/dialogs/cancelling_transaction_progress_dialog.dart';
import 'package:stackwallet/pages/wallet_view/transaction_views/edit_note_view.dart'; import 'package:stackwallet/pages/wallet_view/transaction_views/edit_note_view.dart';
import 'package:stackwallet/pages/wallet_view/wallet_view.dart'; import 'package:stackwallet/pages/wallet_view/wallet_view.dart';
import 'package:stackwallet/providers/db/main_db_provider.dart';
import 'package:stackwallet/providers/global/address_book_service_provider.dart'; import 'package:stackwallet/providers/global/address_book_service_provider.dart';
import 'package:stackwallet/providers/providers.dart'; import 'package:stackwallet/providers/providers.dart';
import 'package:stackwallet/themes/stack_colors.dart'; import 'package:stackwallet/themes/stack_colors.dart';
@ -84,6 +86,9 @@ class _TransactionV2DetailsViewState
late final String amountPrefix; late final String amountPrefix;
late final String unit; late final String unit;
late final int minConfirms; late final int minConfirms;
late final EthContract? ethContract;
bool get isTokenTx => ethContract != null;
late final List<({List<String> addresses, Amount amount})> data; late final List<({List<String> addresses, Amount amount})> data;
@ -96,10 +101,24 @@ class _TransactionV2DetailsViewState
walletId = widget.walletId; walletId = widget.walletId;
coin = widget.coin; coin = widget.coin;
if (_transaction.subType == TransactionSubType.ethToken) {
ethContract = ref
.read(mainDBProvider)
.getEthContractSync(_transaction.contractAddress!);
unit = ethContract!.symbol;
} else {
ethContract == null;
unit = coin.ticker;
}
minConfirms = minConfirms =
ref.read(pWallets).getWallet(walletId).cryptoCurrency.minConfirms; ref.read(pWallets).getWallet(walletId).cryptoCurrency.minConfirms;
fee = _transaction.getFee(coin: coin); final fractionDigits = ethContract?.decimals ?? coin.decimals;
fee = _transaction.getFee(fractionDigits: fractionDigits);
if (_transaction.subType == TransactionSubType.cashFusion || if (_transaction.subType == TransactionSubType.cashFusion ||
_transaction.type == TransactionType.sentToSelf) { _transaction.type == TransactionType.sentToSelf) {
@ -108,18 +127,18 @@ class _TransactionV2DetailsViewState
amountPrefix = _transaction.type == TransactionType.outgoing ? "-" : "+"; amountPrefix = _transaction.type == TransactionType.outgoing ? "-" : "+";
} }
unit = coin.ticker;
if (_transaction.isEpiccashTransaction) { if (_transaction.isEpiccashTransaction) {
switch (_transaction.type) { switch (_transaction.type) {
case TransactionType.outgoing: case TransactionType.outgoing:
case TransactionType.unknown: case TransactionType.unknown:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
break; break;
case TransactionType.incoming: case TransactionType.incoming:
case TransactionType.sentToSelf: case TransactionType.sentToSelf:
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
break; break;
} }
data = _transaction.outputs data = _transaction.outputs
@ -129,7 +148,8 @@ class _TransactionV2DetailsViewState
)) ))
.toList(); .toList();
} else if (_transaction.subType == TransactionSubType.cashFusion) { } else if (_transaction.subType == TransactionSubType.cashFusion) {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
data = _transaction.outputs data = _transaction.outputs
.where((e) => e.walletOwns) .where((e) => e.walletOwns)
.map((e) => ( .map((e) => (
@ -140,7 +160,8 @@ class _TransactionV2DetailsViewState
} else { } else {
switch (_transaction.type) { switch (_transaction.type) {
case TransactionType.outgoing: case TransactionType.outgoing:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
data = _transaction.outputs data = _transaction.outputs
.where((e) => !e.walletOwns) .where((e) => !e.walletOwns)
.map((e) => ( .map((e) => (
@ -154,7 +175,8 @@ class _TransactionV2DetailsViewState
case TransactionType.incoming: case TransactionType.incoming:
case TransactionType.sentToSelf: case TransactionType.sentToSelf:
if (_transaction.subType == TransactionSubType.sparkMint) { if (_transaction.subType == TransactionSubType.sparkMint) {
amount = _transaction.getAmountSparkSelfMinted(coin: coin); amount = _transaction.getAmountSparkSelfMinted(
fractionDigits: fractionDigits);
} else if (_transaction.subType == TransactionSubType.sparkSpend) { } else if (_transaction.subType == TransactionSubType.sparkSpend) {
final changeAddress = final changeAddress =
(ref.read(pWallets).getWallet(walletId) as SparkInterface) (ref.read(pWallets).getWallet(walletId) as SparkInterface)
@ -167,7 +189,8 @@ class _TransactionV2DetailsViewState
fractionDigits: coin.decimals, fractionDigits: coin.decimals,
); );
} else { } else {
amount = _transaction.getAmountReceivedInThisWallet(coin: coin); amount = _transaction.getAmountReceivedInThisWallet(
fractionDigits: fractionDigits);
} }
data = _transaction.outputs data = _transaction.outputs
.where((e) => e.walletOwns) .where((e) => e.walletOwns)
@ -180,7 +203,8 @@ class _TransactionV2DetailsViewState
break; break;
case TransactionType.unknown: case TransactionType.unknown:
amount = _transaction.getAmountSentFromThisWallet(coin: coin); amount = _transaction.getAmountSentFromThisWallet(
fractionDigits: fractionDigits);
data = _transaction.inputs data = _transaction.inputs
.where((e) => e.walletOwns) .where((e) => e.walletOwns)
.map((e) => ( .map((e) => (
@ -515,7 +539,7 @@ class _TransactionV2DetailsViewState
: CrossAxisAlignment.start, : CrossAxisAlignment.start,
children: [ children: [
SelectableText( SelectableText(
"$amountPrefix${ref.watch(pAmountFormatter(coin)).format(amount)}", "$amountPrefix${ref.watch(pAmountFormatter(coin)).format(amount, ethContract: ethContract)}",
style: isDesktop style: isDesktop
? STextStyles ? STextStyles
.desktopTextExtraExtraSmall( .desktopTextExtraExtraSmall(

View file

@ -105,21 +105,6 @@ class EthereumWallet extends Bip39Wallet with PrivateKeyInterface {
await mainDB.updateOrPutAddresses([address]); await mainDB.updateOrPutAddresses([address]);
} }
// delete
@override
Future<Address> getCurrentReceivingAddress() async {
return Address(
walletId: walletId,
value:
checksumEthereumAddress("0x6Cc3006944070B32D80107D51d843a66EaC00686"),
publicKey: [], // maybe store address bytes here? seems a waste of space though
derivationIndex: 0,
derivationPath: DerivationPath()..value = "$hdPathEthereum/0",
type: AddressType.ethereum,
subType: AddressSubType.receiving,
);
}
// ==================== Overrides ============================================ // ==================== Overrides ============================================
@override @override

View file

@ -407,13 +407,17 @@ class EthTokenWallet extends Wallet {
for (final tuple in data) { for (final tuple in data) {
// ignore all non Transfer events (for now) // ignore all non Transfer events (for now)
if (tuple.tx.topics[0] == kTransferEventSignature) { if (tuple.tx.topics[0] == kTransferEventSignature) {
final Amount txFee = tuple.extra.gasUsed * tuple.extra.gasPrice;
final amount = Amount( final amount = Amount(
rawValue: tuple.tx.data.toBigIntFromHex, rawValue: tuple.tx.data.toBigIntFromHex,
fractionDigits: tokenContract.decimals, fractionDigits: tokenContract.decimals,
); );
if (amount.raw == BigInt.zero) {
// probably don't need to show this
continue;
}
final Amount txFee = tuple.extra.gasUsed * tuple.extra.gasPrice;
final addressFrom = _addressFromTopic( final addressFrom = _addressFromTopic(
tuple.tx.topics[1], tuple.tx.topics[1],
); );
@ -451,7 +455,28 @@ class EthTokenWallet extends Wallet {
final List<OutputV2> outputs = []; final List<OutputV2> outputs = [];
final List<InputV2> inputs = []; final List<InputV2> inputs = [];
// TODO: ins outs OutputV2 output = OutputV2.isarCantDoRequiredInDefaultConstructor(
scriptPubKeyHex: "00",
valueStringSats: amount.raw.toString(),
addresses: [
addressTo,
],
walletOwns: addressTo == addressString,
);
InputV2 input = InputV2.isarCantDoRequiredInDefaultConstructor(
scriptSigHex: null,
sequence: null,
outpoint: null,
addresses: [addressFrom],
valueStringSats: amount.raw.toString(),
witness: null,
innerRedeemScriptAsm: null,
coinbase: null,
walletOwns: addressFrom == addressString,
);
outputs.add(output);
inputs.add(input);
final txn = TransactionV2( final txn = TransactionV2(
walletId: walletId, walletId: walletId,