// Haveno App extends the features of Haveno, supporting mobile devices and more.
// Copyright (C) 2024 Kewbit (https://kewbit.org)
// Source Code: https://git.haveno.com/haveno/haveno-app.git
//
// Author: Kewbit
// Website: https://kewbit.org
// Contact Email: me@kewbit.org
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
import 'dart:async';
import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:haveno/grpc_models.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_deposits_confirmed_buyer.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_deposits_published_buyer.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_deposits_unlocked_buyer.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_init_buyer.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_payment_received_buyer.dart';
import 'package:haveno_app/views/screens/trade_timeline/buyer_phases/phase_payment_sent_buyer.dart';
import 'package:haveno_app/utils/payment_utils.dart';
import 'package:provider/provider.dart';
import 'package:haveno_app/providers/haveno_client_providers/trades_provider.dart';
class ActiveBuyerTradeTimelineScreen extends StatefulWidget {
final TradeInfo trade;
const ActiveBuyerTradeTimelineScreen({super.key, required this.trade});
@override
_ActiveBuyerTradeTimelineScreenState createState() =>
_ActiveBuyerTradeTimelineScreenState();
}
class _ActiveBuyerTradeTimelineScreenState
extends State {
PageController _pageController = PageController();
int _currentPage = 0;
Timer? _countdownTimer;
late TradesProvider tradesProvider;
late final tradesProviderListener;
@override
void initState() {
super.initState();
tradesProvider = Provider.of(context, listen: false);
_initializePage();
_listenToPhaseUpdates();
}
void _initializePage() {
if (mounted) {
setState(() {
_currentPage = _getPhaseIndex(widget.trade.phase);
_pageController = PageController(initialPage: _currentPage);
});
}
}
void _listenToPhaseUpdates() {
tradesProviderListener = () {
final updatedTrade = tradesProvider.trades
.firstWhere((trade) => trade.tradeId == widget.trade.tradeId);
if (updatedTrade.phase != widget.trade.phase) {
if (mounted) {
setState(() {
widget.trade.phase = updatedTrade.phase;
_currentPage = _getPhaseIndex(updatedTrade.phase);
_pageController.animateToPage(
_currentPage,
duration: const Duration(milliseconds: 300),
curve: Curves.easeInOut,
);
});
}
}
};
tradesProvider.addListener(tradesProviderListener);
}
void _listenToDisputeStateUpdates() {
tradesProviderListener = () {
final updatedTrade = tradesProvider.trades
.firstWhere((trade) => trade.tradeId == widget.trade.tradeId);
if (updatedTrade.phase != widget.trade.phase) {
if (mounted) {
setState(() {
widget.trade.phase = updatedTrade.phase;
_currentPage = _getPhaseIndex(updatedTrade.phase);
_pageController.animateToPage(
_currentPage,
duration: const Duration(milliseconds: 300),
curve: Curves.easeInOut,
);
});
}
}
};
tradesProvider.addListener(tradesProviderListener);
}
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 _getPhaseWidget(int index) {
switch (index) {
case 0:
return const PhaseInitBuyer();
case 1:
return const PhaseDepositsPublishedBuyer();
case 2:
return const PhaseDepositsConfirmedBuyer();
case 3:
return PhaseDepositsUnlockedBuyer(
takerPaymentAccountPayload: _extractAccountPayload(
jsonDecode(jsonEncode(widget.trade.contract.takerPaymentAccountPayload.toProto3Json()))),
makerPaymentAccountPayload: _extractAccountPayload(
jsonDecode(jsonEncode(widget.trade.contract.makerPaymentAccountPayload.toProto3Json()))),
trade: widget.trade,
onPaidInFull: _onPaidInFull,
);
case 4:
return PhasePaymentSentBuyer(trade: widget.trade);
case 5:
return const PhasePaymentReceivedBuyer();
default:
return const PhaseInitBuyer(); // Fallback to an initial phase
}
}
Map _extractAccountPayload(Map json) {
return json.entries
.firstWhere((entry) => entry.key.contains('AccountPayload'))
.value as Map;
}
String formatFiat(double amount, String currencyCode) {
return isFiatCurrency(currencyCode)
? '${amount.toStringAsFixed(2)} $currencyCode'
: amount.toString();
}
Future _onPaidInFull() async {
try {
// Confirm the payment
await tradesProvider.confirmPaymentSent(widget.trade.tradeId);
// Move to the next phase screen, which is index 4 (PAYMENT_SENT phase)
setState(() {
_currentPage = 4;
});
_pageController.animateToPage(
_currentPage,
duration: const Duration(milliseconds: 300),
curve: Curves.easeInOut,
);
} catch (error) {
// Handle error and notify the user
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('Failed to confirm payment, please try again in a moment.')),
);
}
}
@override
Widget build(BuildContext context) {
const int totalPages = 6; // Number of phases
return Scaffold(
appBar: AppBar(
title: const Text('Buying XMR'),
),
body: Column(
children: [
Expanded(
child: PageView.builder(
controller: _pageController,
physics: const NeverScrollableScrollPhysics(),
itemCount: totalPages,
itemBuilder: (context, index) {
return _getPhaseWidget(index);
},
),
),
Padding(
padding: const EdgeInsets.only(bottom: 16.0, top: 4.0),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: List.generate(totalPages, (index) {
return Container(
margin: const EdgeInsets.symmetric(horizontal: 4.0),
width: 12.0,
height: 12.0,
decoration: BoxDecoration(
shape: BoxShape.circle,
color: index <= _currentPage ? Colors.green : Colors.grey,
),
);
}),
),
),
],
),
);
}
@override
void dispose() {
_countdownTimer?.cancel();
_pageController.dispose();
tradesProvider.removeListener(tradesProviderListener);
super.dispose();
}
}