stack_wallet/lib/widgets/node_options_sheet.dart

434 lines
15 KiB
Dart
Raw Normal View History

/*
2023-05-26 21:21:16 +00:00
* This file is part of Stack Wallet.
*
2023-05-26 21:21:16 +00:00
* Copyright (c) 2023 Cypher Stack
* All Rights Reserved.
* The code is distributed under GPLv3 license, see LICENSE file for details.
* Generated by Cypher Stack on 2023-05-26
*
*/
import 'dart:async';
2022-08-26 08:11:35 +00:00
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_svg/svg.dart';
2024-03-20 00:50:42 +00:00
import 'package:solana/solana.dart';
2024-05-27 23:56:22 +00:00
import 'package:tuple/tuple.dart';
import '../models/node_model.dart';
import '../notifications/show_flush_bar.dart';
import '../pages/settings_views/global_settings_view/manage_nodes_views/add_edit_node_view.dart';
import '../pages/settings_views/global_settings_view/manage_nodes_views/node_details_view.dart';
import '../providers/global/active_wallet_provider.dart';
import '../providers/providers.dart';
import '../services/tor_service.dart';
import '../themes/stack_colors.dart';
import '../utilities/assets.dart';
import '../utilities/connection_check/electrum_connection_check.dart';
import '../utilities/constants.dart';
import '../utilities/default_nodes.dart';
import '../utilities/enums/sync_type_enum.dart';
import '../utilities/logger.dart';
import '../utilities/test_epic_box_connection.dart';
import '../utilities/test_eth_node_connection.dart';
import '../utilities/test_monero_node_connection.dart';
import '../utilities/text_styles.dart';
import '../wallets/crypto_currency/crypto_currency.dart';
import 'rounded_white_container.dart';
2022-08-26 08:11:35 +00:00
class NodeOptionsSheet extends ConsumerWidget {
const NodeOptionsSheet({
super.key,
2022-08-26 08:11:35 +00:00
required this.nodeId,
required this.coin,
required this.popBackToRoute,
});
2022-08-26 08:11:35 +00:00
final String nodeId;
2024-05-15 21:20:45 +00:00
final CryptoCurrency coin;
2022-08-26 08:11:35 +00:00
final String popBackToRoute;
Future<void> _notifyWalletsOfUpdatedNode(WidgetRef ref) async {
final wallets =
ref.read(pWallets).wallets.where((e) => e.info.coin == coin);
2022-08-26 08:11:35 +00:00
final prefs = ref.read(prefsChangeNotifierProvider);
switch (prefs.syncType) {
case SyncingType.currentWalletOnly:
for (final wallet in wallets) {
if (ref.read(currentWalletIdProvider) == wallet.walletId) {
unawaited(wallet.updateNode().then((value) => wallet.refresh()));
2022-08-26 08:11:35 +00:00
} else {
unawaited(wallet.updateNode());
2022-08-26 08:11:35 +00:00
}
}
break;
case SyncingType.selectedWalletsAtStartup:
final List<String> walletIdsToSync = prefs.walletIdsSyncOnStartup;
for (final wallet in wallets) {
if (walletIdsToSync.contains(wallet.walletId)) {
unawaited(wallet.updateNode().then((value) => wallet.refresh()));
2022-08-26 08:11:35 +00:00
} else {
unawaited(wallet.updateNode());
2022-08-26 08:11:35 +00:00
}
}
break;
case SyncingType.allWalletsOnStartup:
for (final wallet in wallets) {
unawaited(wallet.updateNode().then((value) => wallet.refresh()));
2022-08-26 08:11:35 +00:00
}
break;
}
}
Future<bool> _testConnection(
NodeModel node,
BuildContext context,
WidgetRef ref,
) async {
2022-08-26 08:11:35 +00:00
bool testPassed = false;
2024-05-15 21:20:45 +00:00
switch (coin.runtimeType) {
case const (Epiccash):
2022-08-26 08:11:35 +00:00
try {
2022-12-12 21:59:06 +00:00
testPassed = await testEpicNodeConnection(
NodeFormData()
..host = node.host
..useSSL = node.useSSL
..port = node.port,
) !=
null;
2022-08-26 08:11:35 +00:00
} catch (e, s) {
Logging.instance.log("$e\n$s", level: LogLevel.Warning);
}
break;
2024-05-15 21:20:45 +00:00
case const (Monero):
case const (Wownero):
2022-08-26 08:11:35 +00:00
try {
final uri = Uri.parse(node.host);
if (uri.scheme.startsWith("http")) {
final String path = uri.path.isEmpty ? "/json_rpc" : uri.path;
2024-05-15 21:20:45 +00:00
final String uriString =
"${uri.scheme}://${uri.host}:${node.port}$path";
2022-08-26 08:11:35 +00:00
2022-11-08 16:18:48 +00:00
final response = await testMoneroNodeConnection(
Uri.parse(uriString),
false,
);
if (response.cert != null) {
// if (mounted) {
final shouldAllowBadCert = await showBadX509CertificateDialog(
response.cert!,
response.url!,
response.port!,
context,
);
if (shouldAllowBadCert) {
final response =
await testMoneroNodeConnection(Uri.parse(uriString), true);
testPassed = response.success;
}
// }
} else {
testPassed = response.success;
}
2022-08-26 08:11:35 +00:00
}
} catch (e, s) {
Logging.instance.log("$e\n$s", level: LogLevel.Warning);
}
break;
2024-05-15 21:20:45 +00:00
case const (Bitcoin):
case const (Litecoin):
case const (Dogecoin):
case const (Firo):
case const (Particl):
case const (Bitcoincash):
case const (Namecoin):
case const (Ecash):
case const (BitcoinFrost):
case const (Peercoin):
2022-08-26 08:11:35 +00:00
try {
2024-04-18 23:17:45 +00:00
testPassed = await checkElectrumServer(
host: node.host,
port: node.port,
useSSL: node.useSSL,
overridePrefs: ref.read(prefsChangeNotifierProvider),
overrideTorService: ref.read(pTorService),
);
2022-08-26 08:11:35 +00:00
} catch (_) {
testPassed = false;
}
break;
2023-05-30 16:38:47 +00:00
case const (Ethereum):
2023-05-30 16:38:47 +00:00
try {
testPassed = await testEthNodeConnection(node.host);
} catch (_) {
testPassed = false;
}
break;
2023-06-05 20:55:41 +00:00
2024-05-15 21:20:45 +00:00
case const (Nano):
case const (Banano):
case const (Tezos):
case const (Stellar):
2023-07-19 17:08:46 +00:00
throw UnimplementedError();
2023-07-28 16:50:05 +00:00
//TODO: check network/node
2024-03-20 00:50:42 +00:00
2024-05-15 21:20:45 +00:00
case const (Solana):
2024-03-20 00:50:42 +00:00
try {
RpcClient rpcClient;
if (node.host.startsWith("http") || node.host.startsWith("https")) {
rpcClient = RpcClient("${node.host}:${node.port}");
} else {
rpcClient = RpcClient("http://${node.host}:${node.port}");
}
await rpcClient.getEpochInfo().then((value) => testPassed = true);
} catch (_) {
testPassed = false;
}
break;
2022-08-26 08:11:35 +00:00
}
if (testPassed) {
// showFloatingFlushBar(
// type: FlushBarType.success,
// message: "Server ping success",
// context: context,
// );
} else {
unawaited(
showFloatingFlushBar(
type: FlushBarType.warning,
iconAsset: Assets.svg.circleAlert,
message: "Could not connect to node",
context: context,
),
);
2022-08-26 08:11:35 +00:00
}
return testPassed;
}
@override
Widget build(BuildContext context, WidgetRef ref) {
final maxHeight = MediaQuery.of(context).size.height * 0.60;
final node = ref.watch(
nodeServiceChangeNotifierProvider
.select((value) => value.getNodeById(id: nodeId)),
)!;
2022-08-26 08:11:35 +00:00
final status = ref
.watch(
nodeServiceChangeNotifierProvider.select(
2024-05-27 23:56:22 +00:00
(value) => value.getPrimaryNodeFor(currency: coin),
),
)
2022-08-26 08:11:35 +00:00
?.id !=
nodeId
? "Disconnected"
: "Connected";
return Container(
decoration: BoxDecoration(
color: Theme.of(context).extension<StackColors>()!.popupBG,
borderRadius: const BorderRadius.vertical(
2022-08-26 08:11:35 +00:00
top: Radius.circular(20),
),
),
child: LimitedBox(
maxHeight: maxHeight,
child: Padding(
padding: const EdgeInsets.only(
left: 24,
right: 24,
top: 10,
bottom: 0,
),
child: SingleChildScrollView(
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Center(
child: Container(
decoration: BoxDecoration(
color: Theme.of(context)
.extension<StackColors>()!
.textFieldDefaultBG,
2022-08-26 08:11:35 +00:00
borderRadius: BorderRadius.circular(
Constants.size.circularBorderRadius,
),
),
width: 60,
height: 4,
),
),
const SizedBox(
height: 36,
),
Text(
"Node options",
2022-09-22 22:17:21 +00:00
style: STextStyles.pageTitleH2(context),
2022-08-26 08:11:35 +00:00
textAlign: TextAlign.left,
),
RoundedWhiteContainer(
padding: const EdgeInsets.symmetric(vertical: 38),
child: Row(
children: [
Container(
width: 32,
height: 32,
decoration: BoxDecoration(
color: node.id
.startsWith(DefaultNodes.defaultNodeIdPrefix)
? Theme.of(context)
.extension<StackColors>()!
.textSubtitle4
: Theme.of(context)
.extension<StackColors>()!
.infoItemIcons
2022-09-21 00:46:07 +00:00
.withOpacity(0.2),
2022-08-26 08:11:35 +00:00
borderRadius: BorderRadius.circular(100),
),
child: Center(
child: SvgPicture.asset(
Assets.svg.node,
height: 15,
width: 19,
color: node.id.startsWith(
DefaultNodes.defaultNodeIdPrefix,
)
? Theme.of(context)
.extension<StackColors>()!
.accentColorDark
: Theme.of(context)
.extension<StackColors>()!
.infoItemIcons,
2022-08-26 08:11:35 +00:00
),
),
),
const SizedBox(
width: 12,
),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
node.name,
2022-09-22 22:17:21 +00:00
style: STextStyles.titleBold12(context),
2022-08-26 08:11:35 +00:00
),
const SizedBox(
height: 2,
),
Text(
status,
2022-09-22 22:17:21 +00:00
style: STextStyles.label(context),
2022-08-26 08:11:35 +00:00
),
],
),
const Spacer(),
SvgPicture.asset(
Assets.svg.network,
color: status == "Connected"
? Theme.of(context)
.extension<StackColors>()!
.accentColorGreen
: Theme.of(context)
.extension<StackColors>()!
.buttonBackSecondary,
2022-08-26 08:11:35 +00:00
width: 18,
),
],
),
),
Row(
children: [
// if (!node.id.startsWith("default"))
Expanded(
child: TextButton(
style: Theme.of(context)
.extension<StackColors>()!
2023-01-24 19:29:12 +00:00
.getSecondaryEnabledButtonStyle(context),
2022-08-26 08:11:35 +00:00
onPressed: () {
Navigator.pop(context);
Navigator.of(context).pushNamed(
NodeDetailsView.routeName,
arguments: Tuple3(
coin,
node.id,
popBackToRoute,
),
);
},
child: Text(
"Details",
2022-09-22 22:17:21 +00:00
style: STextStyles.button(context).copyWith(
color: Theme.of(context)
.extension<StackColors>()!
.accentColorDark,
),
2022-08-26 08:11:35 +00:00
),
),
),
// if (!node.id.startsWith("default"))
const SizedBox(
width: 12,
),
Expanded(
child: TextButton(
2022-09-21 00:46:07 +00:00
style: status == "Connected"
? Theme.of(context)
.extension<StackColors>()!
2023-01-24 19:29:12 +00:00
.getPrimaryDisabledButtonStyle(context)
: Theme.of(context)
.extension<StackColors>()!
2023-01-24 19:29:12 +00:00
.getPrimaryEnabledButtonStyle(context),
2022-08-26 08:11:35 +00:00
onPressed: status == "Connected"
? null
: () async {
final canConnect =
await _testConnection(node, context, ref);
if (!canConnect) {
return;
}
await ref
2022-08-26 08:11:35 +00:00
.read(nodeServiceChangeNotifierProvider)
.setPrimaryNodeFor(
coin: coin,
node: node,
shouldNotifyListeners: true,
);
await _notifyWalletsOfUpdatedNode(ref);
2022-08-26 08:11:35 +00:00
},
child: Text(
// status == "Connected" ? "Disconnect" : "Connect",
"Connect",
2022-09-22 22:17:21 +00:00
style: STextStyles.button(context),
2022-08-26 08:11:35 +00:00
),
),
),
],
),
const SizedBox(
height: 24,
),
],
),
),
),
),
);
}
}