Files
sendico/frontend/pshared/lib/provider/payment/quotation/intent_builder.dart

191 lines
6.5 KiB
Dart

import 'package:money2/money2.dart';
import 'package:pshared/controllers/payment/source.dart';
import 'package:pshared/models/payment/asset.dart';
import 'package:pshared/models/payment/chain_network.dart';
import 'package:pshared/models/payment/customer.dart';
import 'package:pshared/models/payment/currency_pair.dart';
import 'package:pshared/models/payment/fees/treatment.dart';
import 'package:pshared/models/payment/fx/intent.dart';
import 'package:pshared/models/payment/fx/side.dart';
import 'package:pshared/models/payment/kind.dart';
import 'package:pshared/models/payment/methods/card.dart';
import 'package:pshared/models/payment/methods/crypto_address.dart';
import 'package:pshared/models/payment/methods/data.dart';
import 'package:pshared/models/payment/methods/ledger.dart';
import 'package:pshared/models/payment/methods/managed_wallet.dart';
import 'package:pshared/models/payment/methods/type.dart';
import 'package:pshared/models/payment/settlement_mode.dart';
import 'package:pshared/models/payment/intent.dart';
import 'package:pshared/models/recipient/recipient.dart';
import 'package:pshared/provider/payment/amount.dart';
import 'package:pshared/provider/payment/flow.dart';
import 'package:pshared/provider/recipient/provider.dart';
import 'package:pshared/utils/currency.dart';
import 'package:pshared/utils/payment/fx_helpers.dart';
class QuotationIntentBuilder {
static const String _settlementCurrency = 'RUB';
PaymentIntent? build({
required PaymentAmountProvider payment,
required PaymentSourceController source,
required PaymentFlowProvider flow,
required RecipientsProvider recipients,
}) {
final sourceMethod = _resolveSourceMethod(source);
final sourceCurrency = source.selectedCurrencyCode;
final paymentData = flow.selectedPaymentData;
final selectedMethod = flow.selectedMethod;
final amountValue = payment.amount;
if (sourceCurrency == null) {
return null;
}
final customer = _buildCustomer(
recipient: recipients.currentObject,
method: selectedMethod,
data: paymentData,
);
final amountCurrency = payment.settlementMode == SettlementMode.fixReceived
? _settlementCurrency
: sourceCurrency;
final currency = money2CurrencyFromCode(amountCurrency);
if (currency == null) return null;
final amount = amountValue == null
? null
: Money.fromNumWithCurrency(amountValue, currency);
final isLedgerSource = source.selectedLedgerAccount != null;
final isCryptoToCrypto =
paymentData is CryptoAddressPaymentMethod &&
(paymentData.asset?.tokenSymbol ?? '').trim().toUpperCase() ==
sourceCurrency.trim().toUpperCase();
final fxIntent = _buildFxIntent(
sourceCurrency: sourceCurrency,
settlementMode: payment.settlementMode,
isLedgerSource: isLedgerSource,
enabled: !isCryptoToCrypto,
);
return PaymentIntent(
kind: PaymentKind.payout,
amount: amount,
destination: paymentData,
source: sourceMethod,
fx: fxIntent,
feeTreatment: payment.payerCoversFee
? FeeTreatment.addToSource
: FeeTreatment.deductFromDestination,
settlementMode: payment.settlementMode,
comment: payment.comment,
customer: customer,
);
}
FxIntent? _buildFxIntent({
required String sourceCurrency,
required SettlementMode settlementMode,
required bool isLedgerSource,
required bool enabled,
}) {
if (!enabled) return null;
// Ledger sources in fix_received mode need explicit reverse side.
// BFF maps only settlement currency + fx side, then quotation derives pair.
// For ledger this preserves source debit in ledger currency (e.g. USDT).
if (isLedgerSource && settlementMode == SettlementMode.fixReceived) {
return FxIntent(
pair: CurrencyPair(base: sourceCurrency, quote: _settlementCurrency),
side: FxSide.buyBaseSellQuote,
);
}
return FxIntentHelper.buildSellBaseBuyQuote(
baseCurrency: sourceCurrency,
quoteCurrency: _settlementCurrency, // TODO: exentd target currencies
enabled: true,
);
}
PaymentMethodData? _resolveSourceMethod(PaymentSourceController source) {
final wallet = source.selectedWallet;
if (wallet != null) {
return ManagedWalletPaymentMethod(
managedWalletRef: wallet.id,
asset: PaymentAsset(
tokenSymbol: wallet.tokenSymbol ?? '',
chain: wallet.network ?? ChainNetwork.unspecified,
),
);
}
final ledger = source.selectedLedgerAccount;
if (ledger != null) {
return LedgerPaymentMethod(ledgerAccountRef: ledger.ledgerAccountRef);
}
return null;
}
Customer? _buildCustomer({
required Recipient? recipient,
required PaymentMethod? method,
required PaymentMethodData? data,
}) {
final customerId = recipient?.id.trim() ?? '';
final card = _resolveCard(method: method, data: data);
final fromRecipient = _buildCustomerFromName(
customerId: customerId,
fullName: recipient?.name,
country: card?.country,
);
if (fromRecipient != null) return fromRecipient;
if (card != null) {
final firstName = _normalizedOrNull(card.firstName);
final lastName = _normalizedOrNull(card.lastName);
if (firstName == null && lastName == null) return null;
return Customer(
id: customerId,
firstName: firstName,
lastName: lastName,
country: card.country,
);
}
return null;
}
CardPaymentMethod? _resolveCard({
required PaymentMethod? method,
required PaymentMethodData? data,
}) => method?.cardData ?? (data is CardPaymentMethod ? data : null);
Customer? _buildCustomerFromName({
required String customerId,
required String? fullName,
String? country,
}) {
final normalizedName = _normalizedOrNull(fullName);
if (normalizedName == null) return null;
final parts = normalizedName.split(RegExp(r'\s+'));
final firstName = parts.isNotEmpty ? parts.first : null;
final lastName = parts.length >= 2 ? parts.last : null;
final middleName = parts.length > 2
? parts.sublist(1, parts.length - 1).join(' ')
: null;
return Customer(
id: customerId,
firstName: firstName,
middleName: middleName,
lastName: lastName,
country: country,
);
}
String? _normalizedOrNull(String? value) {
if (value == null) return null;
final normalized = value.trim();
return normalized.isEmpty ? null : normalized;
}
}