allow clearing of spark cache on desktop

This commit is contained in:
julian 2024-06-07 10:30:35 -06:00
parent 6e3d96cdfa
commit 82fb542740

View file

@ -8,12 +8,11 @@
* *
*/ */
import 'dart:convert';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_svg/flutter_svg.dart'; import 'package:flutter_svg/flutter_svg.dart';
import '../../../../../db/sqlite/firo_cache.dart';
import '../../../../../providers/db/main_db_provider.dart'; import '../../../../../providers/db/main_db_provider.dart';
import '../../../../../providers/global/prefs_provider.dart'; import '../../../../../providers/global/prefs_provider.dart';
import '../../../../../providers/global/wallets_provider.dart'; import '../../../../../providers/global/wallets_provider.dart';
@ -23,10 +22,13 @@ import '../../../../../utilities/text_styles.dart';
import '../../../../../wallets/crypto_currency/coins/banano.dart'; import '../../../../../wallets/crypto_currency/coins/banano.dart';
import '../../../../../wallets/crypto_currency/coins/firo.dart'; import '../../../../../wallets/crypto_currency/coins/firo.dart';
import '../../../../../wallets/isar/models/wallet_info.dart'; import '../../../../../wallets/isar/models/wallet_info.dart';
import '../../../../../wallets/isar/providers/wallet_info_provider.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/cash_fusion_interface.dart'; import '../../../../../wallets/wallet/wallet_mixin_interfaces/cash_fusion_interface.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/coin_control_interface.dart'; import '../../../../../wallets/wallet/wallet_mixin_interfaces/coin_control_interface.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/lelantus_interface.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/ordinals_interface.dart'; import '../../../../../wallets/wallet/wallet_mixin_interfaces/ordinals_interface.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/paynym_interface.dart'; import '../../../../../wallets/wallet/wallet_mixin_interfaces/paynym_interface.dart';
import '../../../../../wallets/wallet/wallet_mixin_interfaces/spark_interface.dart';
import '../../../../../widgets/custom_buttons/draggable_switch_button.dart'; import '../../../../../widgets/custom_buttons/draggable_switch_button.dart';
import '../../../../../widgets/desktop/desktop_dialog.dart'; import '../../../../../widgets/desktop/desktop_dialog.dart';
import '../../../../../widgets/desktop/desktop_dialog_close_button.dart'; import '../../../../../widgets/desktop/desktop_dialog_close_button.dart';
@ -59,9 +61,26 @@ class MoreFeaturesDialog extends ConsumerStatefulWidget {
} }
class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> { class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
bool _enableLelantusScanning = false;
bool _isUpdatingLelantusScanning = false; // Mutex. bool _isUpdatingLelantusScanning = false; // Mutex.
Future<void> _switchToggled(bool newValue) async {
if (_isUpdatingLelantusScanning) return;
_isUpdatingLelantusScanning = true; // Lock mutex.
try {
// Toggle enableLelantusScanning in wallet info.
await ref.read(pWalletInfo(widget.walletId)).updateOtherData(
newEntries: {
WalletInfoKeys.enableLelantusScanning: newValue,
},
isar: ref.read(mainDBProvider).isar,
);
} finally {
// ensure _isUpdatingLelantusScanning is set to false no matter what
_isUpdatingLelantusScanning = false;
}
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final wallet = ref.watch( final wallet = ref.watch(
@ -70,14 +89,6 @@ class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
), ),
); );
// Parse otherDataJsonString to get the enableLelantusScanning value.
if (wallet.info.otherDataJsonString != null) {
final otherDataJson = json.decode(wallet.info.otherDataJsonString!);
_enableLelantusScanning =
otherDataJson[WalletInfoKeys.enableLelantusScanning] as bool? ??
false;
}
final coinControlPrefEnabled = ref.watch( final coinControlPrefEnabled = ref.watch(
prefsChangeNotifierProvider.select( prefsChangeNotifierProvider.select(
(value) => value.enableCoinControl, (value) => value.enableCoinControl,
@ -108,7 +119,7 @@ class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
label: "Anonymize funds", label: "Anonymize funds",
detail: "Anonymize funds", detail: "Anonymize funds",
iconAsset: Assets.svg.recycle, iconAsset: Assets.svg.recycle,
onPressed: () => widget.onAnonymizeAllPressed?.call(), onPressed: () async => widget.onAnonymizeAllPressed?.call(),
), ),
// TODO: [prio=med] // TODO: [prio=med]
// if (manager.hasWhirlpoolSupport) // if (manager.hasWhirlpoolSupport)
@ -123,73 +134,53 @@ class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
label: "Coin control", label: "Coin control",
detail: "Control, freeze, and utilize outputs at your discretion", detail: "Control, freeze, and utilize outputs at your discretion",
iconAsset: Assets.svg.coinControl.gamePad, iconAsset: Assets.svg.coinControl.gamePad,
onPressed: () => widget.onCoinControlPressed?.call(), onPressed: () async => widget.onCoinControlPressed?.call(),
), ),
if (wallet is PaynymInterface) if (wallet is PaynymInterface)
_MoreFeaturesItem( _MoreFeaturesItem(
label: "PayNym", label: "PayNym",
detail: "Increased address privacy using BIP47", detail: "Increased address privacy using BIP47",
iconAsset: Assets.svg.robotHead, iconAsset: Assets.svg.robotHead,
onPressed: () => widget.onPaynymPressed?.call(), onPressed: () async => widget.onPaynymPressed?.call(),
), ),
if (wallet is OrdinalsInterface) if (wallet is OrdinalsInterface)
_MoreFeaturesItem( _MoreFeaturesItem(
label: "Ordinals", label: "Ordinals",
detail: "View and control your ordinals in Stack", detail: "View and control your ordinals in Stack",
iconAsset: Assets.svg.ordinal, iconAsset: Assets.svg.ordinal,
onPressed: () => widget.onOrdinalsPressed?.call(), onPressed: () async => widget.onOrdinalsPressed?.call(),
), ),
if (wallet.info.coin is Banano) if (wallet.info.coin is Banano)
_MoreFeaturesItem( _MoreFeaturesItem(
label: "MonKey", label: "MonKey",
detail: "Generate Banano MonKey", detail: "Generate Banano MonKey",
iconAsset: Assets.svg.monkey, iconAsset: Assets.svg.monkey,
onPressed: () => widget.onMonkeyPressed?.call(), onPressed: () async => widget.onMonkeyPressed?.call(),
), ),
if (wallet is CashFusionInterface) if (wallet is CashFusionInterface)
_MoreFeaturesItem( _MoreFeaturesItem(
label: "Fusion", label: "Fusion",
detail: "Decentralized mixing protocol", detail: "Decentralized mixing protocol",
iconAsset: Assets.svg.cashFusion, iconAsset: Assets.svg.cashFusion,
onPressed: () => widget.onFusionPressed?.call(), onPressed: () async => widget.onFusionPressed?.call(),
), ),
if (wallet.info.coin is Firo) if (wallet is SparkInterface)
Padding( const _MoreFeaturesClearSparkCacheItem(),
padding: const EdgeInsets.symmetric( if (wallet is LelantusInterface)
vertical: 6, _MoreFeaturesItemBase(
horizontal: 32,
),
child: RoundedContainer(
color: Colors.transparent,
borderColor: Theme.of(context)
.extension<StackColors>()!
.textFieldDefaultBG,
child: Row( child: Row(
children: [ children: [
SizedBox(width: 3), const SizedBox(width: 3),
SizedBox( SizedBox(
height: 20, height: 20,
width: 40, width: 40,
child: DraggableSwitchButton( child: DraggableSwitchButton(
isOn: _enableLelantusScanning, isOn: ref.watch(
onValueChanged: (newValue) async { pWalletInfo(widget.walletId)
if (_isUpdatingLelantusScanning) return; .select((value) => value.otherData),
_isUpdatingLelantusScanning = true; // Lock mutex. )[WalletInfoKeys.enableLelantusScanning] as bool? ??
false,
// Toggle enableLelantusScanning in wallet info. onValueChanged: _switchToggled,
await wallet.info.updateOtherData(
newEntries: {
WalletInfoKeys.enableLelantusScanning:
!_enableLelantusScanning,
},
isar: ref.read(mainDBProvider).isar,
);
setState(() {
_enableLelantusScanning = !_enableLelantusScanning;
_isUpdatingLelantusScanning = false; // Free mutex.
});
},
), ),
), ),
const SizedBox( const SizedBox(
@ -202,16 +193,11 @@ class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
"Scan for Lelantus transactions", "Scan for Lelantus transactions",
style: STextStyles.w600_20(context), style: STextStyles.w600_20(context),
), ),
// Text(
// detail,
// style: STextStyles.desktopTextExtraExtraSmall(context),
// ),
], ],
), ),
], ],
), ),
), ),
),
const SizedBox( const SizedBox(
height: 28, height: 28,
), ),
@ -221,7 +207,7 @@ class _MoreFeaturesDialogState extends ConsumerState<MoreFeaturesDialog> {
} }
} }
class _MoreFeaturesItem extends StatelessWidget { class _MoreFeaturesItem extends StatefulWidget {
const _MoreFeaturesItem({ const _MoreFeaturesItem({
super.key, super.key,
required this.label, required this.label,
@ -236,6 +222,78 @@ class _MoreFeaturesItem extends StatelessWidget {
final String label; final String label;
final String detail; final String detail;
final String iconAsset; final String iconAsset;
final Future<void> Function()? onPressed;
@override
State<_MoreFeaturesItem> createState() => _MoreFeaturesItemState();
}
class _MoreFeaturesItemState extends State<_MoreFeaturesItem> {
bool _onPressedLock = false;
@override
Widget build(BuildContext context) {
return _MoreFeaturesItemBase(
onPressed: () async {
if (_onPressedLock) {
return;
}
_onPressedLock = true;
try {
await widget.onPressed?.call();
} finally {
_onPressedLock = false;
}
},
child: Row(
children: [
RoundedContainer(
padding: const EdgeInsets.all(0),
color: Theme.of(context).extension<StackColors>()!.settingsIconBack,
width: _MoreFeaturesItem.iconSizeBG,
height: _MoreFeaturesItem.iconSizeBG,
radiusMultiplier: _MoreFeaturesItem.iconSizeBG,
child: Center(
child: SvgPicture.asset(
widget.iconAsset,
width: _MoreFeaturesItem.iconSize,
height: _MoreFeaturesItem.iconSize,
color: Theme.of(context)
.extension<StackColors>()!
.settingsIconIcon,
),
),
),
const SizedBox(
width: 16,
),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
widget.label,
style: STextStyles.w600_20(context),
),
Text(
widget.detail,
style: STextStyles.desktopTextExtraExtraSmall(context),
),
],
),
],
),
);
}
}
class _MoreFeaturesItemBase extends StatelessWidget {
const _MoreFeaturesItemBase({
super.key,
required this.child,
this.onPressed,
});
final Widget child;
final VoidCallback? onPressed; final VoidCallback? onPressed;
@override @override
@ -250,20 +308,61 @@ class _MoreFeaturesItem extends StatelessWidget {
borderColor: borderColor:
Theme.of(context).extension<StackColors>()!.textFieldDefaultBG, Theme.of(context).extension<StackColors>()!.textFieldDefaultBG,
onPressed: onPressed, onPressed: onPressed,
child: child,
),
);
}
}
class _MoreFeaturesClearSparkCacheItem extends StatefulWidget {
const _MoreFeaturesClearSparkCacheItem({
super.key,
});
static const double iconSizeBG = 46;
static const double iconSize = 24;
@override
State<_MoreFeaturesClearSparkCacheItem> createState() =>
_MoreFeaturesClearSparkCacheItemState();
}
class _MoreFeaturesClearSparkCacheItemState
extends State<_MoreFeaturesClearSparkCacheItem> {
bool _onPressedLock = false;
static const label = "Reset Spark electrumx cache";
@override
Widget build(BuildContext context) {
return _MoreFeaturesItemBase(
onPressed: () async {
if (_onPressedLock) {
return;
}
_onPressedLock = true;
try {
await FiroCacheCoordinator.clearSharedCache();
setState(() {
// trigger rebuild for cache size display
});
} finally {
_onPressedLock = false;
}
},
child: Row( child: Row(
children: [ children: [
RoundedContainer( RoundedContainer(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
color: color: Theme.of(context).extension<StackColors>()!.settingsIconBack,
Theme.of(context).extension<StackColors>()!.settingsIconBack, width: _MoreFeaturesItem.iconSizeBG,
width: iconSizeBG, height: _MoreFeaturesItem.iconSizeBG,
height: iconSizeBG, radiusMultiplier: _MoreFeaturesItem.iconSizeBG,
radiusMultiplier: iconSizeBG,
child: Center( child: Center(
child: SvgPicture.asset( child: SvgPicture.asset(
iconAsset, Assets.svg.x,
width: iconSize, width: _MoreFeaturesItem.iconSize,
height: iconSize, height: _MoreFeaturesItem.iconSize,
color: Theme.of(context) color: Theme.of(context)
.extension<StackColors>()! .extension<StackColors>()!
.settingsIconIcon, .settingsIconIcon,
@ -280,15 +379,19 @@ class _MoreFeaturesItem extends StatelessWidget {
label, label,
style: STextStyles.w600_20(context), style: STextStyles.w600_20(context),
), ),
Text( FutureBuilder(
detail, future: FiroCacheCoordinator.getSparkCacheSize(),
builder: (_, snapshot) {
return Text(
snapshot.data ?? "",
style: STextStyles.desktopTextExtraExtraSmall(context), style: STextStyles.desktopTextExtraExtraSmall(context),
);
},
), ),
], ],
), ),
], ],
), ),
),
); );
} }
} }