2024-07-14 15:34:24 +00:00
|
|
|
import 'package:fixnum/fixnum.dart' as fixnum;
|
|
|
|
import 'package:flutter/material.dart';
|
2024-09-20 17:16:54 +00:00
|
|
|
import 'package:haveno/proto/compiled/grpc.pbgrpc.dart';
|
|
|
|
import 'package:haveno/proto/compiled/pb.pb.dart';
|
|
|
|
import 'package:haveno/utils/xmr_math.dart';
|
2024-07-14 15:34:24 +00:00
|
|
|
import 'package:provider/provider.dart';
|
2024-09-20 17:16:54 +00:00
|
|
|
import 'package:haveno/providers/offers_provider.dart';
|
|
|
|
import 'package:haveno/providers/trades_provider.dart';
|
|
|
|
import 'package:haveno/providers/payment_accounts_provider.dart';
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
class OfferDetailScreen extends StatefulWidget {
|
|
|
|
final OfferInfo offer;
|
|
|
|
|
|
|
|
OfferDetailScreen({required this.offer});
|
|
|
|
|
|
|
|
@override
|
|
|
|
_OfferDetailScreenState createState() => _OfferDetailScreenState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _OfferDetailScreenState extends State<OfferDetailScreen> {
|
|
|
|
final _formKey = GlobalKey<FormState>();
|
|
|
|
final TextEditingController _payController = TextEditingController();
|
|
|
|
final TextEditingController _receiveController = TextEditingController();
|
|
|
|
bool _isLoading = true;
|
|
|
|
bool _isWithinLimits = true;
|
|
|
|
OfferInfo? _offer;
|
|
|
|
String? _selectedPaymentAccountId;
|
|
|
|
List<PaymentAccount> _paymentAccounts = [];
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
_offer = widget.offer;
|
|
|
|
_loadPaymentAccounts();
|
|
|
|
_initializeReceiveAmount();
|
|
|
|
_payController.addListener(_updateReceiveAmount);
|
|
|
|
_receiveController.addListener(_updatePayAmount);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _initializeReceiveAmount() {
|
|
|
|
if (_offer != null) {
|
|
|
|
final minAmount = _offer!.minAmount.toDouble() / 1e12;
|
|
|
|
_receiveController.text = minAmount.toStringAsFixed(12);
|
|
|
|
_updatePayAmount();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _loadPaymentAccounts() async {
|
|
|
|
final paymentAccountsProvider =
|
|
|
|
Provider.of<PaymentAccountsProvider>(context, listen: false);
|
|
|
|
await paymentAccountsProvider.getPaymentAccounts();
|
|
|
|
setState(() {
|
|
|
|
_paymentAccounts = paymentAccountsProvider.paymentAccounts
|
|
|
|
?.where((account) =>
|
|
|
|
account.paymentMethod.id == _offer?.paymentMethodId)
|
|
|
|
.toList() ??
|
|
|
|
[];
|
|
|
|
_isLoading = false; // Ensure the loading state is updated
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void _updateReceiveAmount() {
|
|
|
|
if (_offer != null && _payController.text.isNotEmpty) {
|
|
|
|
final payAmount = double.tryParse(_payController.text) ?? 0;
|
|
|
|
final receiveAmount = payAmount / double.parse(_offer!.price);
|
|
|
|
_receiveController.removeListener(_updatePayAmount);
|
|
|
|
final minAmount = _offer!.minAmount.toDouble() / 1e12;
|
|
|
|
_receiveController.text = receiveAmount < minAmount
|
|
|
|
? minAmount.toStringAsFixed(12)
|
|
|
|
: receiveAmount.toStringAsFixed(12);
|
|
|
|
_receiveController.addListener(_updatePayAmount);
|
|
|
|
_checkLimits(payAmount);
|
|
|
|
} else {
|
|
|
|
_receiveController.text = '';
|
|
|
|
_isWithinLimits = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _updatePayAmount() {
|
|
|
|
if (_offer != null && _receiveController.text.isNotEmpty) {
|
|
|
|
final receiveAmount = double.tryParse(_receiveController.text) ?? 0;
|
|
|
|
final payAmount = receiveAmount * double.parse(_offer!.price);
|
|
|
|
_payController.removeListener(_updateReceiveAmount);
|
|
|
|
_payController.text = payAmount.toStringAsFixed(2);
|
|
|
|
_payController.addListener(_updateReceiveAmount);
|
|
|
|
_checkLimits(payAmount);
|
|
|
|
} else {
|
|
|
|
_payController.text = '';
|
|
|
|
_isWithinLimits = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _checkLimits(double payAmount) {
|
|
|
|
final minVolume = double.parse(_offer!.minVolume);
|
|
|
|
final maxVolume = double.parse(_offer!.volume);
|
|
|
|
final minAmount =
|
|
|
|
_offer!.minAmount.toDouble() / 1e12; // Convert from atomic units to XMR
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isWithinLimits = true; //payAmount >= minVolume &&
|
|
|
|
//payAmount <= maxVolume &&
|
|
|
|
//(double.tryParse(_receiveController.text) ?? 0) >= minAmount;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void _confirmOrder() async {
|
|
|
|
if (_formKey.currentState?.validate() ?? false) {
|
|
|
|
final tradesProvider =
|
|
|
|
Provider.of<TradesProvider>(context, listen: false);
|
|
|
|
final receiveAmountDouble = double.parse(_receiveController.text);
|
|
|
|
final receiveAmountBigInt =
|
|
|
|
BigInt.from((receiveAmountDouble * 1e12).round());
|
|
|
|
|
|
|
|
await tradesProvider.takeOffer(
|
|
|
|
widget.offer.id,
|
|
|
|
_selectedPaymentAccountId, // Use selected payment account ID
|
|
|
|
fixnum.Int64(receiveAmountBigInt.toInt()),
|
|
|
|
);
|
|
|
|
final _sentAmount = fixnum.Int64(receiveAmountBigInt.toInt());
|
|
|
|
debugPrint(
|
|
|
|
"Sent amount: $_sentAmount and Minimum Amount: ${widget.offer.minAmount}");
|
|
|
|
|
|
|
|
// Handle success or navigate to another screen
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_payController.dispose();
|
|
|
|
_receiveController.dispose();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
|
|
|
title: Text('Offer View'),
|
|
|
|
),
|
|
|
|
body: _isLoading
|
|
|
|
? Center(child: CircularProgressIndicator())
|
|
|
|
: _offer == null
|
|
|
|
? Center(child: Text('Offer not found'))
|
|
|
|
: Padding(
|
|
|
|
padding: const EdgeInsets.all(16.0),
|
|
|
|
child: SingleChildScrollView(
|
|
|
|
child: Form(
|
|
|
|
key: _formKey,
|
|
|
|
child: Column(
|
|
|
|
children: <Widget>[
|
|
|
|
Card(
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(16.0),
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Text('Offer Details',
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 18,
|
|
|
|
fontWeight: FontWeight.bold)),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
TextFormField(
|
|
|
|
controller: _payController,
|
|
|
|
decoration: InputDecoration(
|
|
|
|
labelText: 'I will pay',
|
|
|
|
suffixText: _offer?.counterCurrencyCode,
|
|
|
|
border: OutlineInputBorder(),
|
|
|
|
),
|
|
|
|
keyboardType: TextInputType.number,
|
|
|
|
validator: (value) {
|
|
|
|
if (value == null || value.isEmpty) {
|
|
|
|
return 'Please enter the amount to pay';
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
TextFormField(
|
|
|
|
controller: _receiveController,
|
|
|
|
decoration: InputDecoration(
|
|
|
|
labelText: 'I will receive',
|
|
|
|
suffixText: 'XMR',
|
|
|
|
border: OutlineInputBorder(),
|
|
|
|
),
|
|
|
|
keyboardType: TextInputType.number,
|
|
|
|
validator: (value) {
|
|
|
|
if (value == null || value.isEmpty) {
|
|
|
|
return 'Please enter the amount to receive';
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
),
|
|
|
|
if (!_isWithinLimits)
|
|
|
|
Padding(
|
|
|
|
padding: const EdgeInsets.only(top: 8.0),
|
|
|
|
child: Text(
|
|
|
|
'Amount is out of the buy limits',
|
|
|
|
style: TextStyle(color: Colors.red),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
2024-07-15 19:14:30 +00:00
|
|
|
SizedBox(height: 10.0),
|
2024-07-14 15:34:24 +00:00
|
|
|
Card(
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(16.0),
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Text('Payment Account',
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 18,
|
|
|
|
fontWeight: FontWeight.bold)),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
DropdownButtonFormField<String>(
|
|
|
|
decoration: InputDecoration(
|
|
|
|
labelText: "You'll pay from",
|
|
|
|
border: OutlineInputBorder(),
|
|
|
|
),
|
|
|
|
items: _paymentAccounts.map((account) {
|
|
|
|
return DropdownMenuItem<String>(
|
|
|
|
value: account.id,
|
|
|
|
child: Text(account.accountName),
|
|
|
|
);
|
|
|
|
}).toList(),
|
|
|
|
onChanged: (value) {
|
|
|
|
setState(() {
|
|
|
|
_selectedPaymentAccountId = value;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
validator: (value) {
|
|
|
|
if (value == null) {
|
|
|
|
return 'Please select a payment account';
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
2024-07-15 19:14:30 +00:00
|
|
|
SizedBox(height: 10.0),
|
2024-07-14 15:34:24 +00:00
|
|
|
Card(
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(16.0),
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Text('About this Offer',
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 18,
|
|
|
|
fontWeight: FontWeight.bold)),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment:
|
|
|
|
MainAxisAlignment.spaceBetween,
|
|
|
|
children: [
|
|
|
|
Text('Seller\'s Rate'),
|
|
|
|
Text(_isFiatCurrency(
|
|
|
|
_offer!.counterCurrencyCode)
|
|
|
|
? '${double.parse(_offer!.price).toStringAsFixed(2)} ${_offer!.counterCurrencyCode}/${_offer!.baseCurrencyCode}'
|
|
|
|
: _offer!.price),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment:
|
|
|
|
MainAxisAlignment.spaceBetween,
|
|
|
|
children: [
|
|
|
|
Text('Minimum Amount'),
|
|
|
|
Text(
|
|
|
|
'${formatXmr(_offer!.minAmount)} XMR'),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
Text('Offer ID'),
|
|
|
|
Text(_offer!.id),
|
|
|
|
SizedBox(height: 16.0),
|
|
|
|
Text('Payment Method'),
|
|
|
|
Text(_offer!.paymentMethodShortName),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
bottomNavigationBar: Padding(
|
|
|
|
padding: const EdgeInsets.all(16.0),
|
|
|
|
child: ElevatedButton(
|
|
|
|
onPressed: _isWithinLimits ? _confirmOrder : null,
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
padding: EdgeInsets.symmetric(vertical: 16.0),
|
|
|
|
backgroundColor: Theme.of(context).colorScheme.primary,
|
|
|
|
),
|
|
|
|
child: Text('Confirm Trade'),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool _isFiatCurrency(String currencyCode) {
|
|
|
|
const fiatCurrencies = {
|
|
|
|
'GBP',
|
|
|
|
'USD',
|
|
|
|
'EUR',
|
|
|
|
'JPY',
|
|
|
|
'AUD',
|
|
|
|
'CAD',
|
|
|
|
'CHF',
|
|
|
|
'CNY',
|
|
|
|
'SEK',
|
|
|
|
'NZD'
|
|
|
|
};
|
|
|
|
return fiatCurrencies.contains(currencyCode);
|
|
|
|
}
|
|
|
|
}
|