2024-07-14 15:34:24 +00:00
|
|
|
import 'dart:io';
|
|
|
|
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/offers_provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/payment_accounts_provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/prices_provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/trades_provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/wallets_provider.dart';
|
2024-07-15 19:14:30 +00:00
|
|
|
import 'package:haveno_flutter_app/screens/drawer/payment_accounts_screen.dart';
|
|
|
|
import 'package:haveno_flutter_app/screens/drawer/settings_screen.dart';
|
2024-07-14 15:34:24 +00:00
|
|
|
import 'package:haveno_flutter_app/tabs/trades_tab.dart';
|
|
|
|
import 'package:haveno_flutter_app/tabs/buy_tab.dart';
|
|
|
|
import 'package:haveno_flutter_app/tabs/sell_tab.dart';
|
2024-07-15 19:14:30 +00:00
|
|
|
import 'package:haveno_flutter_app/screens/drawer/wallet_screen.dart';
|
2024-07-14 15:34:24 +00:00
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:path_provider/path_provider.dart';
|
|
|
|
import 'package:provider/provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/services/haveno_service.dart';
|
|
|
|
import 'package:haveno_flutter_app/services/http_service.dart';
|
|
|
|
import 'package:haveno_flutter_app/services/monero_service.dart';
|
2024-07-16 23:21:28 +00:00
|
|
|
import 'package:haveno_flutter_app/services/tor_service.dart';
|
2024-07-14 15:34:24 +00:00
|
|
|
import 'package:haveno_flutter_app/providers/get_version_provider.dart';
|
|
|
|
import 'package:haveno_flutter_app/providers/account_provider.dart';
|
|
|
|
import 'dart:async';
|
2024-07-16 23:21:28 +00:00
|
|
|
import 'package:badges/badges.dart' as badges;
|
|
|
|
import 'package:tor/tor.dart'; // Import the badges package
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
void main() async {
|
|
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
|
|
|
|
|
|
// Setup logging
|
|
|
|
_setupLogging();
|
|
|
|
|
|
|
|
// Initialize services
|
2024-07-16 23:21:28 +00:00
|
|
|
Tor tor = Tor(); // Constructing the Tor instance
|
|
|
|
TorService torService = TorService(tor);
|
|
|
|
|
|
|
|
// Strart service and listen
|
|
|
|
torService.startService();
|
|
|
|
//torService.listenToTorServiceEvents();
|
|
|
|
torService.checkControlPort();
|
|
|
|
torService.checkSocks5Port();
|
|
|
|
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
final httpService = HttpService();
|
2024-07-16 23:21:28 +00:00
|
|
|
final myIP = await httpService.request('GET', 'https://api.ipify.org?format=json');
|
|
|
|
debugPrint(myIP.toString());
|
2024-07-14 15:34:24 +00:00
|
|
|
final moneroService = MoneroService();
|
|
|
|
|
2024-07-15 19:14:30 +00:00
|
|
|
final havenoService = HavenoService('127.0.0.1', 3201, 'apitest');
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
runApp(
|
|
|
|
MultiProvider(
|
|
|
|
providers: [
|
2024-07-15 19:14:30 +00:00
|
|
|
//Provider(create: (_) => torService),
|
2024-07-14 15:34:24 +00:00
|
|
|
Provider(create: (_) => httpService),
|
|
|
|
Provider(create: (_) => moneroService),
|
|
|
|
Provider(create: (_) => havenoService),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => GetVersionProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => AccountProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => WalletsProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => OffersProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => TradesProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => PaymentAccountsProvider(havenoService),
|
|
|
|
),
|
|
|
|
ChangeNotifierProvider(
|
|
|
|
create: (context) => PricesProvider(havenoService),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
child: MyApp(),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _setupLogging() async {
|
|
|
|
Logger.root.level = Level.ALL; // Capture all log levels
|
|
|
|
|
|
|
|
// Get the application documents directory
|
|
|
|
final directory = await getApplicationDocumentsDirectory();
|
|
|
|
final logFile = File('${directory.path}/app.log');
|
|
|
|
|
|
|
|
// Setup the logger to write to the file and print to console
|
|
|
|
Logger.root.onRecord.listen((record) {
|
|
|
|
final logMessage =
|
|
|
|
'${record.level.name}: ${record.time}: ${record.loggerName}: ${record.message}\n';
|
|
|
|
logFile.writeAsStringSync(logMessage,
|
|
|
|
mode: FileMode.append, flush: true); // Ensure it's flushed to file
|
|
|
|
print(logMessage); // Also print to console
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
final Logger _logger = Logger('Haveno');
|
|
|
|
|
|
|
|
class MyApp extends StatelessWidget {
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return MaterialApp(
|
2024-07-15 19:14:30 +00:00
|
|
|
debugShowCheckedModeBanner: false,
|
2024-07-14 15:34:24 +00:00
|
|
|
theme: ThemeData(
|
|
|
|
useMaterial3: true,
|
|
|
|
colorScheme: ColorScheme.fromSeed(
|
|
|
|
primary: Color(0xFFF4511E),
|
|
|
|
seedColor: Color(0xFFF4511E), // Reddish-orange primary color
|
|
|
|
brightness: Brightness.dark,
|
|
|
|
// Ensures dark mode with light text
|
|
|
|
),
|
|
|
|
scaffoldBackgroundColor: Color(0xFF303030),
|
2024-07-15 19:14:30 +00:00
|
|
|
appBarTheme: const AppBarTheme(
|
2024-07-14 15:34:24 +00:00
|
|
|
backgroundColor: Color(0xFF303030),
|
|
|
|
),
|
2024-07-15 19:14:30 +00:00
|
|
|
drawerTheme: const DrawerThemeData(
|
2024-07-14 15:34:24 +00:00
|
|
|
backgroundColor: Color(0xFF303030),
|
|
|
|
),
|
2024-07-15 19:14:30 +00:00
|
|
|
bottomNavigationBarTheme: const BottomNavigationBarThemeData(
|
2024-07-14 15:34:24 +00:00
|
|
|
backgroundColor: Color(0xFF303030),
|
|
|
|
selectedItemColor: Color(0xFFF4511E),
|
|
|
|
unselectedItemColor: Colors.white,
|
|
|
|
),
|
|
|
|
elevatedButtonTheme: ElevatedButtonThemeData(
|
|
|
|
style: ElevatedButton.styleFrom(
|
|
|
|
backgroundColor: Color(0xFFF4511E), // Button background color
|
|
|
|
foregroundColor: Colors.white, // Button text color
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
borderRadius: BorderRadius.circular(5), // Button border radius
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
2024-07-15 19:14:30 +00:00
|
|
|
cardTheme: const CardTheme(
|
2024-07-14 15:34:24 +00:00
|
|
|
color: Color(0xFF424242), // Card background color
|
|
|
|
),
|
|
|
|
),
|
|
|
|
home: HomeScreen(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class HomeScreen extends StatefulWidget {
|
|
|
|
@override
|
|
|
|
_HomeScreenState createState() => _HomeScreenState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _HomeScreenState extends State<HomeScreen> {
|
|
|
|
String _statusMessage = "Connecting to Tor...";
|
|
|
|
Timer? _timer;
|
|
|
|
int _selectedIndex = 0;
|
2024-07-15 19:14:30 +00:00
|
|
|
int _notificationCount = 5; // Mock notification count
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
static final List<Widget> _widgetOptions = <Widget>[
|
|
|
|
BuyTab(),
|
|
|
|
SellTab(),
|
|
|
|
TradesTab(),
|
|
|
|
];
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
2024-07-15 19:14:30 +00:00
|
|
|
// final torService = context.read<TorService>();
|
|
|
|
// torService.statusStream.listen((String status) {
|
|
|
|
// print(status);
|
|
|
|
// setState(() {
|
|
|
|
// if (status.contains("started")) {
|
|
|
|
// _statusMessage = "Connecting to the Monero network...";
|
|
|
|
// } else {
|
|
|
|
// _statusMessage = status;
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
// });
|
2024-07-14 15:34:24 +00:00
|
|
|
|
|
|
|
_initializeServices();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _initializeServices() async {
|
|
|
|
try {
|
|
|
|
await context.read<GetVersionProvider>().fetchVersion();
|
|
|
|
await context.read<AccountProvider>().checkAccountExists();
|
|
|
|
} catch (e) {
|
|
|
|
setState(() {
|
|
|
|
_statusMessage = "Initialization failed: ${e.toString()}";
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onItemTapped(int index) {
|
|
|
|
setState(() {
|
|
|
|
_selectedIndex = index;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> checkMoneroConnection() async {
|
|
|
|
try {
|
|
|
|
final info = await context.read<MoneroService>().getInfo();
|
|
|
|
setState(() {
|
|
|
|
_statusMessage =
|
|
|
|
"Connected to Monero Mainnet (via Tor) at block ${info['height']}";
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
setState(() {
|
|
|
|
_statusMessage =
|
|
|
|
"Failed to connect to the Monero network: ${e.toString()}";
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
2024-07-15 19:14:30 +00:00
|
|
|
//context.read<TorService>().dispose();
|
2024-07-14 15:34:24 +00:00
|
|
|
context.read<HttpService>().close();
|
|
|
|
context.read<MoneroService>().close();
|
|
|
|
_timer?.cancel();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Scaffold(
|
|
|
|
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
|
|
|
appBar: AppBar(
|
2024-07-15 19:14:30 +00:00
|
|
|
title: const Text('Haveno'),
|
|
|
|
actions: [
|
|
|
|
Padding(
|
|
|
|
padding: const EdgeInsets.only(right: 15.0), // Adjust padding to move the bell and badge
|
|
|
|
child: badges.Badge(
|
|
|
|
position: badges.BadgePosition.topEnd(top: 5, end: 5),
|
|
|
|
badgeContent: Text(
|
|
|
|
'$_notificationCount',
|
|
|
|
style: TextStyle(color: Colors.white, fontSize: 10),
|
|
|
|
),
|
|
|
|
child: IconButton(
|
|
|
|
icon: Icon(Icons.notifications),
|
|
|
|
onPressed: () {
|
|
|
|
// Handle notification bell tap
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
2024-07-14 15:34:24 +00:00
|
|
|
),
|
|
|
|
drawer: _buildDrawer(context),
|
|
|
|
body: Center(
|
|
|
|
child: _widgetOptions.elementAt(_selectedIndex),
|
|
|
|
),
|
|
|
|
bottomNavigationBar: NavigationBar(
|
|
|
|
selectedIndex: _selectedIndex,
|
|
|
|
onDestinationSelected: _onItemTapped,
|
|
|
|
destinations: const <NavigationDestination>[
|
|
|
|
NavigationDestination(
|
|
|
|
icon: Icon(Icons.shopping_cart),
|
|
|
|
label: 'Buy',
|
|
|
|
),
|
|
|
|
NavigationDestination(
|
|
|
|
icon: Icon(Icons.sell),
|
|
|
|
label: 'Sell',
|
|
|
|
),
|
|
|
|
NavigationDestination(
|
|
|
|
icon: Icon(Icons.swap_vert),
|
|
|
|
label: 'Trades',
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildDrawer(BuildContext context) {
|
|
|
|
return Drawer(
|
|
|
|
child: ListView(
|
|
|
|
padding: EdgeInsets.zero,
|
|
|
|
children: <Widget>[
|
|
|
|
DrawerHeader(
|
|
|
|
child: Center(
|
|
|
|
child: Image.asset(
|
|
|
|
'assets/haveno-logo.png',
|
|
|
|
height: 60,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
color: Color(0xFF303030), // Header background color
|
|
|
|
),
|
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
leading: Icon(Icons.account_balance_wallet, color: Colors.white),
|
|
|
|
title: Text('Wallet', style: TextStyle(color: Colors.white)),
|
|
|
|
onTap: () {
|
|
|
|
Navigator.pop(context);
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(builder: (context) => WalletScreen()),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
leading: Icon(Icons.account_circle, color: Colors.white),
|
|
|
|
title:
|
|
|
|
Text('Payment Accounts', style: TextStyle(color: Colors.white)),
|
|
|
|
onTap: () {
|
|
|
|
Navigator.pop(context);
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
2024-07-15 19:14:30 +00:00
|
|
|
MaterialPageRoute(builder: (context) => PaymentAccountsScreen()),
|
2024-07-14 15:34:24 +00:00
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
leading: Icon(Icons.settings, color: Colors.white),
|
|
|
|
title: Text('Settings', style: TextStyle(color: Colors.white)),
|
|
|
|
onTap: () {
|
|
|
|
Navigator.pop(context);
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(builder: (context) => SettingsScreen()),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
leading: Icon(Icons.sync, color: Colors.white),
|
|
|
|
title: Text('Daemons', style: TextStyle(color: Colors.white)),
|
|
|
|
onTap: () {
|
|
|
|
// Handle logout
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|