import 'dart:convert'; import 'package:fixnum/fixnum.dart'; import 'package:flutter/material.dart'; import 'package:haveno_flutter_app/utils/payment_utils.dart'; import 'package:provider/provider.dart'; import 'package:haveno_flutter_app/providers/trades_provider.dart'; import 'package:haveno_flutter_app/proto/compiled/grpc.pbgrpc.dart'; import 'package:flutter/services.dart'; class ActiveBuyerTradeTimelineScreen extends StatefulWidget { final TradeInfo trade; const ActiveBuyerTradeTimelineScreen({required this.trade}); @override _ActiveBuyerTradeTimelineScreenState createState() => _ActiveBuyerTradeTimelineScreenState(); } class _ActiveBuyerTradeTimelineScreenState extends State { PageController _pageController = PageController(); int _currentPage = 0; @override void initState() { super.initState(); _initializePage(); _listenToPhaseUpdates(); } void _initializePage() { setState(() { _currentPage = _getPhaseIndex(widget.trade.phase); _pageController = PageController(initialPage: _currentPage); }); } Future _completeTrade() async { final tradesProvider = Provider.of(context, listen: false); await tradesProvider.completeTrade(widget.trade.tradeId); //final fetchedTrade = await tradesProvider.getTrade(widget.trade.tradeId); //debugPrint("Fetched Trade ${fetchedTrade?.role}"); } void _listenToPhaseUpdates() { final tradesProvider = Provider.of(context, listen: false); debugPrint( "From getTrades() function: ${tradesProvider.trades?.last.amount}"); tradesProvider.addListener(() { final updatedTrade = tradesProvider.trades ?.firstWhere((trade) => trade.tradeId == widget.trade.tradeId); if (updatedTrade != null && updatedTrade.phase != widget.trade.phase) { setState(() { widget.trade.phase = updatedTrade.phase; _currentPage = _getPhaseIndex(updatedTrade.phase); _pageController.animateToPage( _currentPage, duration: Duration(milliseconds: 300), curve: Curves.easeInOut, ); }); } }); } String _getPhaseText(String phase) { switch (phase) { case 'INIT': return "Initializing Trade"; case 'DEPOSITS_PUBLISHED': return "Transferring to Escrow"; case 'DEPOSITS_CONFIRMED': return "Deposit Received"; case 'DEPOSITS_UNLOCKED': return "Awaiting Your Payment"; case 'PAYMENT_SENT': return "Seller Confirming Payment"; case 'PAYMENT_RECEIVED': return "Payment Received"; default: return phase; } } int _getPhaseIndex(String phase) { switch (phase) { case 'INIT': return 0; case 'DEPOSITS_PUBLISHED': return 1; case 'DEPOSITS_CONFIRMED': return 2; case 'DEPOSITS_UNLOCKED': return 3; case 'PAYMENT_SENT': return 4; case 'PAYMENT_RECEIVED': return 5; default: return 0; } } Widget _buildPaymentDetails(String title, Map payload) { return Card( margin: const EdgeInsets.all(8.0), child: Padding( padding: const EdgeInsets.all(16.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( title, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16), ), SizedBox(height: 10), ...payload.entries.map((entry) { return Padding( padding: const EdgeInsets.symmetric(vertical: 8.0), child: TextFormField( initialValue: entry.value, readOnly: true, decoration: InputDecoration( labelText: entry.key, border: OutlineInputBorder(), ), ), ); }).toList(), SizedBox(height: 20), ], ), ), ); } Widget _buildCopyablePaymentDetails( String title, Map payload) { return Card( margin: const EdgeInsets.all(8.0), child: Padding( padding: const EdgeInsets.all(16.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( title, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16), ), SizedBox(height: 10), ...payload.entries.map((entry) { return Padding( padding: const EdgeInsets.symmetric(vertical: 8.0), child: TextFormField( initialValue: entry.value, readOnly: true, decoration: InputDecoration( labelText: entry.key, border: OutlineInputBorder(), suffixIcon: IconButton( icon: Icon(Icons.copy), onPressed: () { Clipboard.setData(ClipboardData(text: entry.value)); ScaffoldMessenger.of(context).showSnackBar( SnackBar(content: Text('Copied to clipboard')), ); }, ), ), ), ); }).toList(), SizedBox(height: 20), ], ), ), ); } Map _extractAccountPayload(Map json) { return json.entries .firstWhere((entry) => entry.key.contains('AccountPayload')) .value as Map; } void _onPaidInFull() { final tradesProvider = Provider.of(context, listen: false); tradesProvider.confirmPaymentSent(widget.trade.tradeId).catchError((error) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text( 'Failed to confirm payment, please try again in a moment.')), ); }); } @override Widget build(BuildContext context) { final phases = [ 'INIT', 'DEPOSITS_PUBLISHED', 'DEPOSITS_CONFIRMED', 'DEPOSITS_UNLOCKED', 'PAYMENT_SENT', 'PAYMENT_RECEIVED', ]; final int totalPages = phases.length; final price = double.parse(widget.trade.price); final amount = formatXmr(widget.trade.amount, returnString: false) as double; final totalAmount = amount * price; final totalAmountFormatted = formatFiat(totalAmount, widget.trade.offer.counterCurrencyCode); final String currencyCode = widget.trade.offer.counterCurrencyCode; return Scaffold( appBar: AppBar( title: Text('Active Trade Details'), ), body: Column( children: [ Expanded( child: PageView.builder( controller: _pageController, physics: NeverScrollableScrollPhysics(), itemCount: totalPages, itemBuilder: (context, index) { final phase = phases[index]; if (phase == 'DEPOSITS_UNLOCKED') { final takerPaymentAccountJson = widget .trade.contract.takerPaymentAccountPayload .toProto3Json(); final makerPaymentAccountJson = widget .trade.contract.makerPaymentAccountPayload .toProto3Json(); debugPrint(jsonEncode(takerPaymentAccountJson)); debugPrint(jsonEncode(makerPaymentAccountJson)); final takerPaymentAccountPayload = _extractAccountPayload( jsonDecode(jsonEncode(takerPaymentAccountJson))); final makerPaymentAccountPayload = _extractAccountPayload( jsonDecode(jsonEncode(makerPaymentAccountJson))); return Center( child: SingleChildScrollView( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ CircularProgressIndicator(), SizedBox(height: 20), Text( _getPhaseText(phase), style: TextStyle(fontSize: 24), ), SizedBox(height: 20), Card( margin: const EdgeInsets.all(8.0), child: Padding( padding: const EdgeInsets.all(16.0), child: Text( 'You must pay a total of $totalAmountFormatted. Please be sure the amount is exact.', style: TextStyle( fontWeight: FontWeight.bold, fontSize: 16, ), ), ), ), _buildPaymentDetails('You\'ll send from...', takerPaymentAccountPayload), _buildCopyablePaymentDetails( 'To the sellers account...', makerPaymentAccountPayload), SizedBox(height: 20), ElevatedButton( onPressed: _onPaidInFull, child: Text('I have paid in full'), ), ], ), ), ); } return Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ CircularProgressIndicator(), SizedBox(height: 20), Text( _getPhaseText(phase), style: TextStyle(fontSize: 24), ), ], ), ); }, ), ), Padding( padding: const EdgeInsets.only(bottom: 18.0, top: 8.0), child: Row( mainAxisAlignment: MainAxisAlignment.center, children: List.generate(totalPages, (index) { return Container( margin: EdgeInsets.symmetric(horizontal: 4.0), width: 12.0, height: 12.0, decoration: BoxDecoration( shape: BoxShape.circle, color: index <= _currentPage ? Colors.green : Colors.grey, ), ); }), ), ), ], ), ); } dynamic formatXmr(Int64? atomicUnits, {bool returnString = true}) { if (atomicUnits == null) { return returnString ? 'N/A' : null; } double value = atomicUnits.toInt() / 1e12; return returnString ? value.toStringAsFixed(5) : value; } String formatFiat(double amount, String currencyCode) { return isFiatCurrency(currencyCode) ? '${amount.toStringAsFixed(2)} $currencyCode' : amount.toString(); } @override void dispose() { _pageController.dispose(); super.dispose(); } }