mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-09 12:29:31 +00:00
c0cd68a823
Some checks are pending
Cache Dependencies / test (push) Waiting to run
* update monero_c commit * fix: no element in getAllUnusedSubAddresses * fix: Wallet created with empty seed and 0 as private key The error that was there is caused when wallet is being created, but it errors out, so better handling of errors should be all that's needed, as it is not an error on it's own, but rather lack of handling. * fix: create transaction multi dest function is missing * update monero_c hash * fix: receiving on 2 different addresses shows as 1
162 lines
4.7 KiB
Dart
162 lines
4.7 KiB
Dart
import 'package:cw_core/subaddress.dart';
|
|
import 'package:cw_monero/api/coins_info.dart';
|
|
import 'package:cw_monero/api/subaddress_list.dart' as subaddress_list;
|
|
import 'package:flutter/services.dart';
|
|
import 'package:mobx/mobx.dart';
|
|
|
|
part 'monero_subaddress_list.g.dart';
|
|
|
|
class MoneroSubaddressList = MoneroSubaddressListBase with _$MoneroSubaddressList;
|
|
|
|
abstract class MoneroSubaddressListBase with Store {
|
|
MoneroSubaddressListBase()
|
|
: _isRefreshing = false,
|
|
_isUpdating = false,
|
|
subaddresses = ObservableList<Subaddress>();
|
|
|
|
final List<String> _usedAddresses = [];
|
|
|
|
@observable
|
|
ObservableList<Subaddress> subaddresses;
|
|
|
|
bool _isRefreshing;
|
|
bool _isUpdating;
|
|
|
|
void update({required int accountIndex}) {
|
|
refreshCoins(accountIndex);
|
|
|
|
if (_isUpdating) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
_isUpdating = true;
|
|
refresh(accountIndex: accountIndex);
|
|
subaddresses.clear();
|
|
subaddresses.addAll(getAll());
|
|
_isUpdating = false;
|
|
} catch (e) {
|
|
_isUpdating = false;
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
List<Subaddress> getAll() {
|
|
var subaddresses = subaddress_list.getAllSubaddresses();
|
|
|
|
if (subaddresses.length > 2) {
|
|
final primary = subaddresses.first;
|
|
final rest = subaddresses.sublist(1).reversed;
|
|
subaddresses = [primary] + rest.toList();
|
|
}
|
|
|
|
return subaddresses.map((s) {
|
|
final address = s.address;
|
|
final label = s.label;
|
|
final id = s.addressIndex;
|
|
final hasDefaultAddressName =
|
|
label.toLowerCase() == 'Primary account'.toLowerCase() ||
|
|
label.toLowerCase() == 'Untitled account'.toLowerCase();
|
|
final isPrimaryAddress = id == 0 && hasDefaultAddressName;
|
|
return Subaddress(
|
|
id: id,
|
|
address: address,
|
|
label: isPrimaryAddress
|
|
? 'Primary address'
|
|
: hasDefaultAddressName
|
|
? ''
|
|
: label);
|
|
}).toList();
|
|
}
|
|
|
|
Future<void> addSubaddress({required int accountIndex, required String label}) async {
|
|
await subaddress_list.addSubaddress(accountIndex: accountIndex, label: label);
|
|
update(accountIndex: accountIndex);
|
|
}
|
|
|
|
Future<void> setLabelSubaddress(
|
|
{required int accountIndex, required int addressIndex, required String label}) async {
|
|
await subaddress_list.setLabelForSubaddress(
|
|
accountIndex: accountIndex, addressIndex: addressIndex, label: label);
|
|
update(accountIndex: accountIndex);
|
|
}
|
|
|
|
void refresh({required int accountIndex}) {
|
|
if (_isRefreshing) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
_isRefreshing = true;
|
|
subaddress_list.refreshSubaddresses(accountIndex: accountIndex);
|
|
_isRefreshing = false;
|
|
} on PlatformException catch (e) {
|
|
_isRefreshing = false;
|
|
print(e);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> updateWithAutoGenerate({
|
|
required int accountIndex,
|
|
required String defaultLabel,
|
|
required List<String> usedAddresses,
|
|
}) async {
|
|
_usedAddresses.addAll(usedAddresses);
|
|
if (_isUpdating) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
_isUpdating = true;
|
|
refresh(accountIndex: accountIndex);
|
|
subaddresses.clear();
|
|
final newSubAddresses =
|
|
await _getAllUnusedAddresses(accountIndex: accountIndex, label: defaultLabel);
|
|
subaddresses.addAll(newSubAddresses);
|
|
} catch (e) {
|
|
rethrow;
|
|
} finally {
|
|
_isUpdating = false;
|
|
}
|
|
}
|
|
|
|
Future<List<Subaddress>> _getAllUnusedAddresses(
|
|
{required int accountIndex, required String label}) async {
|
|
final allAddresses = subaddress_list.getAllSubaddresses();
|
|
final lastAddress = allAddresses.length == 0 ? allAddresses.last.address : Subaddress(id: -1, address: "", label: "");
|
|
if (allAddresses.isEmpty || _usedAddresses.contains(lastAddress)) {
|
|
final isAddressUnused = await _newSubaddress(accountIndex: accountIndex, label: label);
|
|
if (!isAddressUnused) {
|
|
return await _getAllUnusedAddresses(accountIndex: accountIndex, label: label);
|
|
}
|
|
}
|
|
|
|
return allAddresses
|
|
.map((s) {
|
|
final id = s.addressIndex;
|
|
final address = s.address;
|
|
final label = s.label;
|
|
return Subaddress(
|
|
id: id,
|
|
address: address,
|
|
label: id == 0 &&
|
|
label.toLowerCase() == 'Primary account'.toLowerCase()
|
|
? 'Primary address'
|
|
: label);
|
|
})
|
|
.toList();
|
|
}
|
|
|
|
Future<bool> _newSubaddress({required int accountIndex, required String label}) async {
|
|
await subaddress_list.addSubaddress(accountIndex: accountIndex, label: label);
|
|
|
|
return subaddress_list
|
|
.getAllSubaddresses()
|
|
.where((s) {
|
|
final address = s.address;
|
|
return !_usedAddresses.contains(address);
|
|
})
|
|
.isNotEmpty;
|
|
}
|
|
}
|