import 'dart:async'; import 'package:cake_wallet/buy/buy_provider.dart'; import 'package:cake_wallet/buy/buy_quote.dart'; import 'package:cake_wallet/buy/onramper/onramper_buy_provider.dart'; import 'package:cake_wallet/buy/payment_method.dart'; import 'package:cake_wallet/buy/sell_buy_states.dart'; import 'package:cake_wallet/core/selectable_option.dart'; import 'package:cake_wallet/core/wallet_change_listener_view_model.dart'; import 'package:cake_wallet/entities/fiat_currency.dart'; import 'package:cake_wallet/entities/provider_types.dart'; import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/routes.dart'; import 'package:cake_wallet/store/app_store.dart'; import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/themes/theme_base.dart'; import 'package:cw_core/crypto_currency.dart'; import 'package:cw_core/currency_for_wallet_type.dart'; import 'package:flutter/cupertino.dart'; import 'package:intl/intl.dart'; import 'package:mobx/mobx.dart'; part 'buy_sell_view_model.g.dart'; class BuySellViewModel = BuySellViewModelBase with _$BuySellViewModel; abstract class BuySellViewModelBase extends WalletChangeListenerViewModel with Store { BuySellViewModelBase( AppStore appStore, ) : _cryptoNumberFormat = NumberFormat(), cryptoAmount = '', fiatAmount = '', cryptoCurrencyAddress = '', isCryptoCurrencyAddressEnabled = false, cryptoCurrencies = [], fiatCurrencies = [], paymentMethodState = InitialPaymentMethod(), buySellQuotState = InitialBuySellQuotState(), cryptoCurrency = appStore.wallet!.currency, fiatCurrency = appStore.settingsStore.fiatCurrency, providerList = [], sortedRecommendedQuotes = ObservableList(), sortedQuotes = ObservableList(), paymentMethods = ObservableList(), settingsStore = appStore.settingsStore, super(appStore: appStore) { const excludeFiatCurrencies = []; const excludeCryptoCurrencies = []; fiatCurrencies = FiatCurrency.all.where((currency) => !excludeFiatCurrencies.contains(currency)).toList(); cryptoCurrencies = CryptoCurrency.all .where((currency) => !excludeCryptoCurrencies.contains(currency)) .toList(); _initialize(); isCryptoCurrencyAddressEnabled = !(cryptoCurrency == wallet.currency); } final NumberFormat _cryptoNumberFormat; late Timer bestRateSync; List get availableBuyProviders { final providerTypes = ProvidersHelper.getAvailableBuyProviderTypes( walletTypeForCurrency(cryptoCurrency) ?? wallet.type); return providerTypes .map((type) => ProvidersHelper.getProviderByType(type)) .where((provider) => provider != null) .cast() .toList(); } List get availableSellProviders { final providerTypes = ProvidersHelper.getAvailableSellProviderTypes( walletTypeForCurrency(cryptoCurrency) ?? wallet.type); return providerTypes .map((type) => ProvidersHelper.getProviderByType(type)) .where((provider) => provider != null) .cast() .toList(); } @override void onWalletChange(wallet) { cryptoCurrency = wallet.currency; } bool get isDarkTheme => settingsStore.currentTheme.type == ThemeType.dark; double get amount { final formattedFiatAmount = double.tryParse(fiatAmount) ?? 200.0; final formattedCryptoAmount = double.tryParse(cryptoAmount) ?? (cryptoCurrency == CryptoCurrency.btc ? 0.001 : 1); return isBuyAction ? formattedFiatAmount : formattedCryptoAmount; } SettingsStore settingsStore; Quote? bestRateQuote; Quote? selectedQuote; @observable List cryptoCurrencies; @observable List fiatCurrencies; @observable bool isBuyAction = true; @observable List providerList; @observable ObservableList sortedRecommendedQuotes; @observable ObservableList sortedQuotes; @observable ObservableList paymentMethods; @observable FiatCurrency fiatCurrency; @observable CryptoCurrency cryptoCurrency; @observable String cryptoAmount; @observable String fiatAmount; @observable String cryptoCurrencyAddress; @observable bool isCryptoCurrencyAddressEnabled; @observable PaymentMethod? selectedPaymentMethod; @observable PaymentMethodLoadingState paymentMethodState; @observable BuySellQuotLoadingState buySellQuotState; @computed bool get isReadyToTrade { final hasSelectedQuote = selectedQuote != null; final hasSelectedPaymentMethod = selectedPaymentMethod != null; final isPaymentMethodLoaded = paymentMethodState is PaymentMethodLoaded; final isBuySellQuotLoaded = buySellQuotState is BuySellQuotLoaded; final isBuySellQuotFailed = buySellQuotState is BuySellQuotFailed; return hasSelectedQuote && hasSelectedPaymentMethod && isPaymentMethodLoaded && isBuySellQuotLoaded && !isBuySellQuotFailed; } @computed bool get isBuySellQuotFailed => buySellQuotState is BuySellQuotFailed; @action void reset() { cryptoCurrency = wallet.currency; fiatCurrency = settingsStore.fiatCurrency; isCryptoCurrencyAddressEnabled = !(cryptoCurrency == wallet.currency); _initialize(); } @action void changeBuySellAction() { isBuyAction = !isBuyAction; _initialize(); } @action void changeFiatCurrency({required FiatCurrency currency}) { fiatCurrency = currency; _onPairChange(); } @action void changeCryptoCurrency({required CryptoCurrency currency}) { cryptoCurrency = currency; _onPairChange(); isCryptoCurrencyAddressEnabled = !(cryptoCurrency == wallet.currency); } @action void changeCryptoCurrencyAddress(String address) => cryptoCurrencyAddress = address; @action Future changeFiatAmount({required String amount}) async { fiatAmount = amount; if (amount.isEmpty) { fiatAmount = ''; cryptoAmount = ''; return; } final enteredAmount = double.tryParse(amount.replaceAll(',', '.')) ?? 0; if (!isReadyToTrade) { cryptoAmount = S.current.fetching; return; } if (bestRateQuote != null) { _cryptoNumberFormat.maximumFractionDigits = cryptoCurrency.decimals; cryptoAmount = _cryptoNumberFormat .format(enteredAmount / bestRateQuote!.rate) .toString() .replaceAll(RegExp('\\,'), ''); } else { await calculateBestRate(); } } @action Future changeCryptoAmount({required String amount}) async { cryptoAmount = amount; if (amount.isEmpty) { fiatAmount = ''; cryptoAmount = ''; return; } final enteredAmount = double.tryParse(amount.replaceAll(',', '.')) ?? 0; if (!isReadyToTrade) { fiatAmount = S.current.fetching; } if (bestRateQuote != null) { fiatAmount = _cryptoNumberFormat .format(enteredAmount * bestRateQuote!.rate) .toString() .replaceAll(RegExp('\\,'), ''); } else { await calculateBestRate(); } } @action void changeOption(SelectableOption option) { if (option is Quote) { sortedRecommendedQuotes.forEach((element) => element.setIsSelected = false); sortedQuotes.forEach((element) => element.setIsSelected = false); option.setIsSelected = true; selectedQuote = option; } else if (option is PaymentMethod) { paymentMethods.forEach((element) => element.isSelected = false); option.isSelected = true; selectedPaymentMethod = option; } else { throw ArgumentError('Unknown option type'); } } void onTapChoseProvider(BuildContext context) async { final initialQuotes = List.from(sortedRecommendedQuotes + sortedQuotes); await calculateBestRate(); final newQuotes = (sortedRecommendedQuotes + sortedQuotes); for (var quote in newQuotes) quote.limits = null; final newQuoteProviders = newQuotes .map((quote) => quote.provider.isAggregator ? quote.rampName : quote.provider.title) .toSet(); final outOfLimitQuotes = initialQuotes.where((initialQuote) { return !newQuoteProviders.contains( initialQuote.provider.isAggregator ? initialQuote.rampName : initialQuote.provider.title); }).map((missingQuote) { final quote = Quote( rate: missingQuote.rate, feeAmount: missingQuote.feeAmount, networkFee: missingQuote.networkFee, transactionFee: missingQuote.transactionFee, payout: missingQuote.payout, rampId: missingQuote.rampId, rampName: missingQuote.rampName, rampIconPath: missingQuote.rampIconPath, paymentType: missingQuote.paymentType, quoteId: missingQuote.quoteId, recommendations: missingQuote.recommendations, provider: missingQuote.provider, isBuyAction: missingQuote.isBuyAction, limits: missingQuote.limits, ); quote.setFiatCurrency = missingQuote.fiatCurrency; quote.setCryptoCurrency = missingQuote.cryptoCurrency; return quote; }).toList(); final updatedQuoteOptions = List.from([ OptionTitle(title: 'Recommended'), ...sortedRecommendedQuotes, if (sortedQuotes.isNotEmpty) OptionTitle(title: 'All Providers'), ...sortedQuotes, if (outOfLimitQuotes.isNotEmpty) OptionTitle(title: 'Out of Limits'), ...outOfLimitQuotes, ]); await Navigator.of(context).pushNamed( Routes.buyOptionsPage, arguments: [ updatedQuoteOptions, changeOption, launchTrade, ], ).then((value) => calculateBestRate()); } void _onPairChange() { _initialize(); } void _setProviders() => providerList = isBuyAction ? availableBuyProviders : availableSellProviders; Future _initialize() async { _setProviders(); cryptoAmount = ''; fiatAmount = ''; cryptoCurrencyAddress = _getInitialCryptoCurrencyAddress(); paymentMethodState = InitialPaymentMethod(); buySellQuotState = InitialBuySellQuotState(); await _getAvailablePaymentTypes(); await calculateBestRate(); } String _getInitialCryptoCurrencyAddress() { return cryptoCurrency == wallet.currency ? wallet.walletAddresses.address : ''; } @action Future _getAvailablePaymentTypes() async { paymentMethodState = PaymentMethodLoading(); selectedPaymentMethod = null; final result = await Future.wait(providerList.map((element) => element .getAvailablePaymentTypes(fiatCurrency.title, cryptoCurrency.title, isBuyAction) .timeout( Duration(seconds: 10), onTimeout: () => [], ))); final Map uniquePaymentMethods = {}; for (var methods in result) { for (var method in methods) { uniquePaymentMethods[method.paymentMethodType] = method; } } paymentMethods = ObservableList.of(uniquePaymentMethods.values); if (paymentMethods.isNotEmpty) { paymentMethods.insert(0, PaymentMethod.all()); selectedPaymentMethod = paymentMethods.first; selectedPaymentMethod!.isSelected = true; paymentMethodState = PaymentMethodLoaded(); } else { paymentMethodState = PaymentMethodFailed(); } } @action Future calculateBestRate() async { buySellQuotState = BuySellQuotLoading(); final result = await Future.wait?>(providerList.map((element) => element .fetchQuote( cryptoCurrency: cryptoCurrency, fiatCurrency: fiatCurrency, amount: amount, paymentType: selectedPaymentMethod?.paymentMethodType, isBuyAction: isBuyAction, walletAddress: wallet.walletAddresses.address, ) .timeout( Duration(seconds: 10), onTimeout: () => null, ))); sortedRecommendedQuotes.clear(); sortedQuotes.clear(); final validQuotes = result .where((element) => element != null && element.isNotEmpty) .expand((element) => element!) .toList(); if (validQuotes.isEmpty) { buySellQuotState = BuySellQuotFailed(); return; } validQuotes.sort((a, b) => a.rate.compareTo(b.rate)); final Set addedProviders = {}; final List uniqueProviderQuotes = validQuotes.where((element) { if (addedProviders.contains(element.provider.title)) return false; addedProviders.add(element.provider.title); return true; }).toList(); sortedRecommendedQuotes.addAll(uniqueProviderQuotes); sortedQuotes = ObservableList.of( validQuotes.where((element) => !uniqueProviderQuotes.contains(element)).toList()); if (sortedRecommendedQuotes.isNotEmpty) { sortedRecommendedQuotes.first ..setIsBestRate = true ..recommendations.insert(0, ProviderRecommendation.bestRate); bestRateQuote = sortedRecommendedQuotes.first; sortedRecommendedQuotes.sort((a, b) { if (a.provider is OnRamperBuyProvider) return -1; if (b.provider is OnRamperBuyProvider) return 1; return 0; }); selectedQuote = sortedRecommendedQuotes.first; sortedRecommendedQuotes.first.setIsSelected = true; } buySellQuotState = BuySellQuotLoaded(); } @action Future launchTrade(BuildContext context) async { final provider = selectedQuote!.provider; await provider.launchProvider( context: context, quote: selectedQuote!, amount: amount, isBuyAction: isBuyAction, cryptoCurrencyAddress: cryptoCurrencyAddress, ); } }