mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2024-10-31 17:37:41 +00:00
4c60b178be
* Fix web3dart versioning issue * Add primary receive address extracted from private key * Implement open wallet functionality * Implement restore wallet from seed functionality * Fixate web3dart version as higher versions cause some issues * Add Initial Transaction priorities for eth Add estimated gas price * Rename priority value to tip * Re-order wallet types * Change ethereum node Fix connection issues * Fix estimating gas for priority * Add case for ethereum to fetch it's seeds * Add case for ethereum to request node * Fix Exchange screen initial pairs * Add initial send transaction flow * Add missing configure for ethereum class * Add Eth address initial setup * Fix Private key for Ethereum wallets * Change sign/send transaction flow * - Fix Conflicts with main - Remove unused function from Haven configure.dart * Add build command for ethereum package * Add missing Node list file to pubspec * - Fix balance display - Fix parsing of Ethereum amount - Add more Ethereum Nodes [skip ci] * - Fix extracting Ethereum Private key from seeds - Integrate signing/sending transaction with the send view model * - Update and Fix Conflicts with main * Add Balances for ERC20 tokens * Fix conflicts with main * Add erc20 abi json * Add send erc20 tokens initial function * add missing getHeightByDate in Haven [skip ci] * Allow contacts and wallets from the same tag * Add Shiba Inu icon * Add send ERC-20 tokens initial flow * Add missing import in generated file * Add initial approach for transaction sending for ERC-20 tokens * Refactor signing/sending transactions * Add initial flow for transactions subscription * Refactor signing/sending transactions * Add home settings icon * Fix conflicts with main * Initial flow for home settings * Add logic flow for adding erc20 tokens * Fix initial UI * Finalize UI for Tokens * Integrate UI with Ethereum flow * Add "Enable/Disable" feature for ERC20 tokens * Add initial Erc20 tokens * Add Sorting and Pin Native Token features * Fix price sorting * Sort tokens list as well when Sort criteria changes * - Improve sorting balances flow - Add initial add token from search bar flow * Fix Accounts Popup UI * Fix Pin native token * Fix Enabling/Disabling tokens Fix sorting by fiat once app is opened Improve token availability mechanism * Fix deleting token Fix renaming tokens * Fix issue with search * Add more tokens * - Fix scroll issue - Add ERC20 tokens placeholder image in picker * - Separate and organize default erc20 tokens - Fix scrolling - Add token placeholder images in picker - Sort disabled tokens alphabetically * Change BNB token initial availability [skip ci] * Fix Conflicts with main * Fix Conflicts with main * Add Verse ERC20 token to the initial tokens list * Add rename wallet to Ethereum * Integrate EtherScan API for fetching address transactions Generate Ethereum specific secrets in Ethereum package * Adjust transactions fiat price for ERC20 tokens * Free Up GitHub Actions Ubuntu Runner Disk Space * Free Up GitHub Actions Ubuntu Runner Disk space (trial 2) * Fix Transaction Fee display * Save transaction history * Enhance loading time for erc20 tokens transactions * Minor Fixes and Enhancements * Fix sending erc20 fix block explorer issue * Fix int overflow * Fix transaction amount conversions * Minor: `slow` -> `Slow` [skip-ci] * initial changes * more base config stuff * config changes * successfully builds! * save * successfully add nano wallet * save * seed generation * receive screen + node screen working * tx history working and fiat fixes * balance working * derivation updates * nano-unfinished * sends working * remove fees from send screen, send and receive transactions working * fixes + auto receive incoming txs * fix for scanning QR codes * save * update translations * fixes * more fixes * more strings * small fix * fix github actions workflow * potential fix * potential fix * ci/cd fix * change rep working * seed generation fixes * fixes * save * change rep screen functional * save * banano changes * fixes, start adding ui for PoW * pow node changes * update translations * fix * account changing barely working * save * disable account generation * small fix * save * UI work * save * fixes after merge main * fixes * remove monero stuff, work on derivation ui * lots of fixes + finish up seed derivation * last minute fixes * node related fixes * more fixes * small fix * more fixes * fixes * pretty big refactor for pow, still some bugs * finally works! * get transactions after send * fix * merge conflict fixes * save * fix pow node showing up twice * done * initial changes * small fix * more merge fixes * fixes * more fixes * fix * save * fix manage pow nodes setting appearing on other wallets * fix contact bug * fixes * fiat fixes * save * save * save * save * updates * cleanup * restore fix * fixes * remove deprecated alert * fix * small fix * remove outdated warning * electrum restore fixes * fixes * fixes * fix * derivation fixes * nano fixes pt.1 * nano fixes pt.2 * bip39 fixes * pownode refactor * nodes pages fixes * observer fix * ssl fix * remove old references * remove unused imports * code cleanup * small fix * small potential fix * save * undo all bitcoin related changes * remove dead code * review fixes * more fixes * fix * fix * review fix * small fix * nano derivation and nanoutil fixes * exchange nano fix * nano review fixes pt.1 * nano fixes pt.2 * nano fixes pt.3 * remove old imports + stop using dynamic in di * nanoutil fixes * add nano.dart to gitignore, configure fixes * review fixes, getnanowalletservice removed * fix settings screen, add changeRep to configure.dart, other minor fixes * remove manage_pow_nodes_page, key derivation edge case handled * remove old refs * more small fixes * Generic Enhancements/Minor fixes * review fixes * hopefully final fixes * review fixes * node connection fixes --------- Co-authored-by: OmarHatem <omarh.ismail1@gmail.com> Co-authored-by: Justin Ehrenhofer <justin.ehrenhofer@gmail.com> Co-authored-by: fossephate <fosse@book.local>
433 lines
14 KiB
Dart
433 lines
14 KiB
Dart
import 'package:cake_wallet/entities/auto_generate_subaddress_status.dart';
|
|
import 'package:cake_wallet/entities/buy_provider_types.dart';
|
|
import 'package:cake_wallet/entities/exchange_api_mode.dart';
|
|
import 'package:cake_wallet/store/anonpay/anonpay_transactions_store.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/anonpay_transaction_list_item.dart';
|
|
import 'package:cake_wallet/view_model/settings/sync_mode.dart';
|
|
import 'package:cake_wallet/wallet_type_utils.dart';
|
|
import 'package:cw_core/transaction_history.dart';
|
|
import 'package:cw_core/balance.dart';
|
|
import 'package:cake_wallet/entities/balance_display_mode.dart';
|
|
import 'package:cw_core/transaction_info.dart';
|
|
import 'package:cake_wallet/exchange/exchange_provider_description.dart';
|
|
import 'package:cake_wallet/store/settings_store.dart';
|
|
import 'package:cake_wallet/store/dashboard/orders_store.dart';
|
|
import 'package:cake_wallet/store/yat/yat_store.dart';
|
|
import 'package:cake_wallet/utils/mobx.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/balance_view_model.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/filter_item.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/order_list_item.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/trade_list_item.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/transaction_list_item.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/action_list_item.dart';
|
|
import 'package:mobx/mobx.dart';
|
|
import 'package:cw_core/wallet_base.dart';
|
|
import 'package:cw_core/sync_status.dart';
|
|
import 'package:cw_core/wallet_type.dart';
|
|
import 'package:cake_wallet/store/app_store.dart';
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
import 'package:cake_wallet/store/dashboard/trades_store.dart';
|
|
import 'package:cake_wallet/store/dashboard/trade_filter_store.dart';
|
|
import 'package:cake_wallet/store/dashboard/transaction_filter_store.dart';
|
|
import 'package:cake_wallet/view_model/dashboard/formatted_item_list.dart';
|
|
import 'package:cake_wallet/monero/monero.dart';
|
|
|
|
part 'dashboard_view_model.g.dart';
|
|
|
|
class DashboardViewModel = DashboardViewModelBase with _$DashboardViewModel;
|
|
|
|
abstract class DashboardViewModelBase with Store {
|
|
DashboardViewModelBase(
|
|
{required this.balanceViewModel,
|
|
required this.appStore,
|
|
required this.tradesStore,
|
|
required this.tradeFilterStore,
|
|
required this.transactionFilterStore,
|
|
required this.settingsStore,
|
|
required this.yatStore,
|
|
required this.ordersStore,
|
|
required this.anonpayTransactionsStore})
|
|
: hasSellAction = false,
|
|
hasBuyAction = false,
|
|
hasExchangeAction = false,
|
|
isShowFirstYatIntroduction = false,
|
|
isShowSecondYatIntroduction = false,
|
|
isShowThirdYatIntroduction = false,
|
|
filterItems = {
|
|
S.current.transactions: [
|
|
FilterItem(
|
|
value: () => transactionFilterStore.displayAll,
|
|
caption: S.current.all_transactions,
|
|
onChanged: transactionFilterStore.toggleAll),
|
|
FilterItem(
|
|
value: () => transactionFilterStore.displayIncoming,
|
|
caption: S.current.incoming,
|
|
onChanged:transactionFilterStore.toggleIncoming),
|
|
FilterItem(
|
|
value: () => transactionFilterStore.displayOutgoing,
|
|
caption: S.current.outgoing,
|
|
onChanged: transactionFilterStore.toggleOutgoing),
|
|
// FilterItem(
|
|
// value: () => false,
|
|
// caption: S.current.transactions_by_date,
|
|
// onChanged: null),
|
|
],
|
|
S.current.trades: [
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displayAllTrades,
|
|
caption: S.current.all_trades,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.all)),
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displayChangeNow,
|
|
caption: ExchangeProviderDescription.changeNow.title,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.changeNow)),
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displaySideShift,
|
|
caption: ExchangeProviderDescription.sideShift.title,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.sideShift)),
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displaySimpleSwap,
|
|
caption: ExchangeProviderDescription.simpleSwap.title,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.simpleSwap)),
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displayTrocador,
|
|
caption: ExchangeProviderDescription.trocador.title,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.trocador)),
|
|
FilterItem(
|
|
value: () => tradeFilterStore.displayExolix,
|
|
caption: ExchangeProviderDescription.exolix.title,
|
|
onChanged: () => tradeFilterStore
|
|
.toggleDisplayExchange(ExchangeProviderDescription.exolix)),
|
|
]
|
|
},
|
|
subname = '',
|
|
name = appStore.wallet!.name,
|
|
type = appStore.wallet!.type,
|
|
transactions = ObservableList<TransactionListItem>(),
|
|
wallet = appStore.wallet! {
|
|
name = wallet.name;
|
|
type = wallet.type;
|
|
isShowFirstYatIntroduction = false;
|
|
isShowSecondYatIntroduction = false;
|
|
isShowThirdYatIntroduction = false;
|
|
updateActions();
|
|
|
|
final _wallet = wallet;
|
|
|
|
if (_wallet.type == WalletType.monero) {
|
|
subname = monero!.getCurrentAccount(_wallet).label;
|
|
|
|
_onMoneroAccountChangeReaction = reaction(
|
|
(_) => monero!.getMoneroWalletDetails(wallet).account,
|
|
(Account account) => _onMoneroAccountChange(_wallet));
|
|
|
|
_onMoneroBalanceChangeReaction = reaction(
|
|
(_) => monero!.getMoneroWalletDetails(wallet).balance,
|
|
(MoneroBalance balance) => _onMoneroTransactionsUpdate(_wallet));
|
|
|
|
final _accountTransactions = _wallet.transactionHistory.transactions.values
|
|
.where((tx) =>
|
|
monero!.getTransactionInfoAccountId(tx) == monero!.getCurrentAccount(wallet).id)
|
|
.toList();
|
|
|
|
transactions = ObservableList.of(_accountTransactions.map((transaction) =>
|
|
TransactionListItem(
|
|
transaction: transaction,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore)));
|
|
} else {
|
|
transactions = ObservableList.of(wallet.transactionHistory.transactions.values.map(
|
|
(transaction) => TransactionListItem(
|
|
transaction: transaction,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore)));
|
|
}
|
|
|
|
// TODO: nano sub-account generation is disabled:
|
|
// if (_wallet.type == WalletType.nano || _wallet.type == WalletType.banano) {
|
|
// subname = nano!.getCurrentAccount(_wallet).label;
|
|
// }
|
|
|
|
reaction((_) => appStore.wallet, _onWalletChange);
|
|
|
|
connectMapToListWithTransform(
|
|
appStore.wallet!.transactionHistory.transactions,
|
|
transactions,
|
|
(TransactionInfo? transaction) => TransactionListItem(
|
|
transaction: transaction!,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore), filter: (TransactionInfo? transaction) {
|
|
if (transaction == null) {
|
|
return false;
|
|
}
|
|
|
|
final wallet = _wallet;
|
|
if (wallet.type == WalletType.monero) {
|
|
return monero!.getTransactionInfoAccountId(transaction) ==
|
|
monero!.getCurrentAccount(wallet).id;
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
|
|
@observable
|
|
WalletType type;
|
|
|
|
@observable
|
|
String name;
|
|
|
|
@observable
|
|
ObservableList<TransactionListItem> transactions;
|
|
|
|
@observable
|
|
String subname;
|
|
|
|
@observable
|
|
bool isShowFirstYatIntroduction;
|
|
|
|
@observable
|
|
bool isShowSecondYatIntroduction;
|
|
|
|
@observable
|
|
bool isShowThirdYatIntroduction;
|
|
|
|
@computed
|
|
String get address => wallet.walletAddresses.address;
|
|
|
|
@computed
|
|
SyncStatus get status => wallet.syncStatus;
|
|
|
|
@computed
|
|
String get syncStatusText {
|
|
var statusText = '';
|
|
|
|
if (status is SyncingSyncStatus) {
|
|
statusText = S.current.Blocks_remaining(status.toString());
|
|
}
|
|
|
|
if (status is FailedSyncStatus || status is LostConnectionSyncStatus) {
|
|
statusText = S.current.please_try_to_connect_to_another_node;
|
|
}
|
|
|
|
return statusText;
|
|
}
|
|
|
|
@computed
|
|
BalanceDisplayMode get balanceDisplayMode => appStore.settingsStore.balanceDisplayMode;
|
|
|
|
@computed
|
|
bool get shouldShowMarketPlaceInDashboard {
|
|
return appStore.settingsStore.shouldShowMarketPlaceInDashboard;
|
|
}
|
|
|
|
@computed
|
|
List<TradeListItem> get trades =>
|
|
tradesStore.trades.where((trade) => trade.trade.walletId == wallet.id).toList();
|
|
|
|
@computed
|
|
List<OrderListItem> get orders =>
|
|
ordersStore.orders.where((item) => item.order.walletId == wallet.id).toList();
|
|
|
|
@computed
|
|
List<AnonpayTransactionListItem> get anonpayTransactons => anonpayTransactionsStore.transactions
|
|
.where((item) => item.transaction.walletId == wallet.id)
|
|
.toList();
|
|
|
|
@computed
|
|
double get price => balanceViewModel.price;
|
|
|
|
@computed
|
|
bool get isAutoGenerateSubaddressesEnabled =>
|
|
settingsStore.autoGenerateSubaddressStatus != AutoGenerateSubaddressStatus.disabled;
|
|
|
|
@computed
|
|
List<ActionListItem> get items {
|
|
final _items = <ActionListItem>[];
|
|
|
|
_items.addAll(
|
|
transactionFilterStore.filtered(transactions: [...transactions, ...anonpayTransactons]));
|
|
_items.addAll(tradeFilterStore.filtered(trades: trades, wallet: wallet));
|
|
_items.addAll(orders);
|
|
|
|
return formattedItemsList(_items);
|
|
}
|
|
|
|
@observable
|
|
WalletBase<Balance, TransactionHistoryBase<TransactionInfo>, TransactionInfo> wallet;
|
|
|
|
bool get hasRescan => wallet.type == WalletType.monero || wallet.type == WalletType.haven;
|
|
|
|
BalanceViewModel balanceViewModel;
|
|
|
|
AppStore appStore;
|
|
|
|
SettingsStore settingsStore;
|
|
|
|
YatStore yatStore;
|
|
|
|
TradesStore tradesStore;
|
|
|
|
OrdersStore ordersStore;
|
|
|
|
TradeFilterStore tradeFilterStore;
|
|
|
|
AnonpayTransactionsStore anonpayTransactionsStore;
|
|
|
|
TransactionFilterStore transactionFilterStore;
|
|
|
|
Map<String, List<FilterItem>> filterItems;
|
|
|
|
BuyProviderType get defaultBuyProvider => settingsStore.defaultBuyProvider;
|
|
|
|
bool get isBuyEnabled => settingsStore.isBitcoinBuyEnabled;
|
|
|
|
bool get shouldShowYatPopup => settingsStore.shouldShowYatPopup;
|
|
|
|
@action
|
|
void furtherShowYatPopup(bool shouldShow) => settingsStore.shouldShowYatPopup = shouldShow;
|
|
|
|
@computed
|
|
bool get isEnabledExchangeAction => settingsStore.exchangeStatus != ExchangeApiMode.disabled;
|
|
|
|
@observable
|
|
bool hasExchangeAction;
|
|
|
|
@computed
|
|
bool get isEnabledBuyAction => !settingsStore.disableBuy && wallet.type != WalletType.haven;
|
|
|
|
@observable
|
|
bool hasBuyAction;
|
|
|
|
@computed
|
|
bool get isEnabledSellAction =>
|
|
!settingsStore.disableSell &&
|
|
wallet.type != WalletType.haven &&
|
|
wallet.type != WalletType.monero;
|
|
|
|
@observable
|
|
bool hasSellAction;
|
|
|
|
ReactionDisposer? _onMoneroAccountChangeReaction;
|
|
|
|
ReactionDisposer? _onMoneroBalanceChangeReaction;
|
|
|
|
@computed
|
|
bool get hasPowNodes => wallet.type == WalletType.nano || wallet.type == WalletType.banano;
|
|
|
|
Future<void> reconnect() async {
|
|
final node = appStore.settingsStore.getCurrentNode(wallet.type);
|
|
await wallet.connectToNode(node: node);
|
|
if (hasPowNodes) {
|
|
final powNode = settingsStore.getCurrentPowNode(wallet.type);
|
|
await wallet.connectToPowNode(node: powNode);
|
|
}
|
|
}
|
|
|
|
@action
|
|
void _onWalletChange(
|
|
WalletBase<Balance, TransactionHistoryBase<TransactionInfo>, TransactionInfo>? wallet) {
|
|
if (wallet == null) {
|
|
return;
|
|
}
|
|
|
|
this.wallet = wallet;
|
|
type = wallet.type;
|
|
name = wallet.name;
|
|
updateActions();
|
|
|
|
if (wallet.type == WalletType.monero) {
|
|
subname = monero!.getCurrentAccount(wallet).label;
|
|
|
|
_onMoneroAccountChangeReaction?.reaction.dispose();
|
|
_onMoneroBalanceChangeReaction?.reaction.dispose();
|
|
|
|
_onMoneroAccountChangeReaction = reaction(
|
|
(_) => monero!.getMoneroWalletDetails(wallet).account,
|
|
(Account account) => _onMoneroAccountChange(wallet));
|
|
|
|
_onMoneroBalanceChangeReaction = reaction(
|
|
(_) => monero!.getMoneroWalletDetails(wallet).balance,
|
|
(MoneroBalance balance) => _onMoneroTransactionsUpdate(wallet));
|
|
|
|
_onMoneroTransactionsUpdate(wallet);
|
|
} else {
|
|
// FIX-ME: Check for side effects
|
|
// subname = null;
|
|
subname = '';
|
|
|
|
transactions.clear();
|
|
|
|
transactions.addAll(wallet.transactionHistory.transactions.values.map((transaction) =>
|
|
TransactionListItem(
|
|
transaction: transaction,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore)));
|
|
}
|
|
|
|
connectMapToListWithTransform(
|
|
appStore.wallet!.transactionHistory.transactions,
|
|
transactions,
|
|
(TransactionInfo? transaction) => TransactionListItem(
|
|
transaction: transaction!,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore), filter: (TransactionInfo? tx) {
|
|
if (tx == null) {
|
|
return false;
|
|
}
|
|
|
|
if (wallet.type == WalletType.monero) {
|
|
return monero!.getTransactionInfoAccountId(tx) == monero!.getCurrentAccount(wallet).id;
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
|
|
@action
|
|
void _onMoneroAccountChange(WalletBase wallet) {
|
|
subname = monero!.getCurrentAccount(wallet).label;
|
|
_onMoneroTransactionsUpdate(wallet);
|
|
}
|
|
|
|
@action
|
|
void _onMoneroTransactionsUpdate(WalletBase wallet) {
|
|
transactions.clear();
|
|
|
|
final _accountTransactions = monero!
|
|
.getTransactionHistory(wallet)
|
|
.transactions
|
|
.values
|
|
.where(
|
|
(tx) => monero!.getTransactionInfoAccountId(tx) == monero!.getCurrentAccount(wallet).id)
|
|
.toList();
|
|
|
|
transactions.addAll(_accountTransactions.map((transaction) => TransactionListItem(
|
|
transaction: transaction,
|
|
balanceViewModel: balanceViewModel,
|
|
settingsStore: appStore.settingsStore)));
|
|
}
|
|
|
|
void updateActions() {
|
|
hasExchangeAction = !isHaven;
|
|
hasBuyAction = !isHaven;
|
|
hasSellAction = !isHaven;
|
|
}
|
|
|
|
@computed
|
|
SyncMode get syncMode => settingsStore.currentSyncMode;
|
|
|
|
@action
|
|
void setSyncMode(SyncMode syncMode) => settingsStore.currentSyncMode = syncMode;
|
|
|
|
@computed
|
|
bool get syncAll => settingsStore.currentSyncAll;
|
|
|
|
@action
|
|
void setSyncAll(bool value) => settingsStore.currentSyncAll = value;
|
|
}
|