// 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 'package:flutter/material.dart';
import 'package:haveno/grpc_models.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_deposits_confirmed_seller.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_deposits_published_seller.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_deposits_unlocked_seller.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_init_seller.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_payment_received_seller.dart';
import 'package:haveno_app/views/screens/trade_timeline/seller_phases/phase_payment_sent_seller.dart';
import 'package:provider/provider.dart';
import 'package:haveno_app/providers/haveno_client_providers/trades_provider.dart';
class ActiveSellerTradeTimelineScreen extends StatefulWidget {
final TradeInfo trade;
const ActiveSellerTradeTimelineScreen({super.key, required this.trade});
@override
_ActiveSellerTradeTimelineScreenState createState() =>
_ActiveSellerTradeTimelineScreenState();
}
class _ActiveSellerTradeTimelineScreenState
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);
}
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 PhaseInitSeller();
case 1:
return const PhaseDepositsPublishedSeller();
case 2:
return const PhaseDepositsConfirmedSeller();
case 3:
return PhaseDepositsUnlockedSeller(trade: widget.trade);
case 4:
return PhasePaymentSentSeller(trade: widget.trade);
case 5:
return const PhasePaymentReceivedSeller();
default:
return const PhaseInitSeller(); // Fallback to an initial phase
}
}
@override
Widget build(BuildContext context) {
const int totalPages = 6; // Number of phases
return Scaffold(
appBar: AppBar(
title: const Text('Selling 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();
}
}