Skip to content
Snippets Groups Projects
Commit 970d5f97 authored by vjrj's avatar vjrj
Browse files

Improve contacts cache

parent 5a57e102
No related branches found
No related tags found
No related merge requests found
import 'dart:convert';
import 'dart:typed_data'; import 'dart:typed_data';
Uint8List? uIntFromList(dynamic value) { Uint8List? uIntFromList(dynamic value) {
if (value is List<int> && value.isNotEmpty) { if (value is List<int> && value.isNotEmpty) {
return Uint8List.fromList(value); return Uint8List.fromList(value);
} else if (value is String && value.isNotEmpty) { } else if (value is List<dynamic> && value.isNotEmpty) {
return base64Decode(value); return Uint8List.fromList(value.map((dynamic e) => e as int).toList());
} else { } else {
return null; return null;
} }
......
...@@ -17,11 +17,13 @@ class ContactsCache { ...@@ -17,11 +17,13 @@ class ContactsCache {
ContactsCache._internal(); ContactsCache._internal();
static ContactsCache? _instance; static ContactsCache? _instance;
final Map<String, List<Completer<Contact>>> _pendingRequests = {}; final Map<String, List<Completer<Contact>>> _pendingRequests =
<String, List<Completer<Contact>>>{};
Future<Contact> getContact(String pubKey) async { Future<Contact> getContact(String pubKey) async {
final String cacheKey = 'avatar-$pubKey'; final String cacheKey = 'contact-$pubKey';
const Duration duration = Duration(days: 3); const Duration duration =
kReleaseMode ? Duration(days: 3) : Duration(minutes: 5);
try { try {
final String? cachedValue = window.localStorage[cacheKey]; final String? cachedValue = window.localStorage[cacheKey];
...@@ -35,7 +37,7 @@ class ContactsCache { ...@@ -35,7 +37,7 @@ class ContactsCache {
final Contact contact = final Contact contact =
Contact.fromJson(decodedValue['data'] as Map<String, dynamic>); Contact.fromJson(decodedValue['data'] as Map<String, dynamic>);
if (!kReleaseMode) { if (!kReleaseMode) {
logger('Returning cached contact $contact'); // logger('Returning cached contact $contact');
} }
return contact; return contact;
} }
...@@ -61,7 +63,7 @@ class ContactsCache { ...@@ -61,7 +63,7 @@ class ContactsCache {
}); });
window.localStorage[cacheKey] = encodedValue; window.localStorage[cacheKey] = encodedValue;
if (!kReleaseMode) { if (!kReleaseMode) {
logger('Returning non cached contact $contact'); // logger('Returning non cached contact $contact');
} }
// Send to listeners // Send to listeners
for (final Completer<Contact> completer in _pendingRequests[pubKey]!) { for (final Completer<Contact> completer in _pendingRequests[pubKey]!) {
......
...@@ -25,176 +25,186 @@ class TransactionListItem extends StatelessWidget { ...@@ -25,176 +25,186 @@ class TransactionListItem extends StatelessWidget {
final int index; final int index;
@override @override
Widget build(BuildContext context) => BlocBuilder<TransactionsCubit, Widget build(BuildContext context) =>
BlocBuilder<TransactionsCubit,
TransactionsAndBalanceState>( TransactionsAndBalanceState>(
builder: (BuildContext context, builder: (BuildContext context,
TransactionsAndBalanceState transBalanceState) => TransactionsAndBalanceState transBalanceState) =>
FutureBuilder<Contact>( FutureBuilder<Contact>(
future: _fetchContact(pubKey, transaction), future: _fetchContact(pubKey, transaction),
builder: (BuildContext context, AsyncSnapshot<Contact> snapshot) { builder: (BuildContext context,
if (snapshot.hasData) { AsyncSnapshot<Contact> snapshot) {
IconData? icon; if (snapshot.hasData) {
Color? iconColor; return _buildTransactionItem(context, snapshot.data!);
String statusText; } else if (snapshot.hasError) {
final String amountS = return Text('Error ${snapshot.error}');
'${transaction.amount < 0 ? "" : "+"}${formatKAmount(context, transaction.amount)}'; } else {
statusText = tr('transaction_${transaction.type.name}'); return _buildTransactionItem(context, Contact(
pubKey: isIncoming(transaction.type) ? transaction
.from : transaction.to));
}
}));
switch (transaction.type) { Slidable _buildTransactionItem(BuildContext context,
case TransactionType.pending: Contact contact) {
icon = Icons.timelapse; IconData? icon;
iconColor = Colors.grey; Color? iconColor;
break; String statusText;
case TransactionType.sending: final String amountS =
icon = Icons.flight_takeoff; '${transaction.amount < 0 ? "" : "+"}${formatKAmount(
iconColor = Colors.grey; context, transaction.amount)}';
break; statusText = tr('transaction_${transaction.type.name}');
case TransactionType.receiving: switch (transaction.type) {
icon = Icons.flight_land; case TransactionType.pending:
iconColor = Colors.grey; icon = Icons.timelapse;
break; iconColor = Colors.grey;
case TransactionType.sent: break;
break; case TransactionType.sending:
case TransactionType.received: icon = Icons.flight_takeoff;
break; iconColor = Colors.grey;
} break;
final String myPubKey = SharedPreferencesHelper().getPubKey(); case TransactionType.receiving:
final ContactsCubit contactsCubit = icon = Icons.flight_land;
context.read<ContactsCubit>(); iconColor = Colors.grey;
return Slidable( break;
// Specify a key if the Slidable is dismissible. case TransactionType.sent:
key: const ValueKey<int>(0), break;
// The end action pane is the one at the right or the bottom side. case TransactionType.received:
endActionPane: ActionPane( break;
motion: const ScrollMotion(), }
children: <SlidableAction>[ final String myPubKey = SharedPreferencesHelper().getPubKey();
SlidableAction( final ContactsCubit contactsCubit =
onPressed: (BuildContext c) { context.read<ContactsCubit>();
if (snapshot.hasData) return Slidable(
contactsCubit.addContact(snapshot.data!); // Specify a key if the Slidable is dismissible.
ScaffoldMessenger.of(context).showSnackBar( key: const ValueKey<int>(0),
SnackBar( // The end action pane is the one at the right or the bottom side.
content: Text(tr('contact_added')), endActionPane: ActionPane(
), motion: const ScrollMotion(),
); children: <SlidableAction>[
}, SlidableAction(
backgroundColor: Theme.of(context).primaryColor, onPressed: (BuildContext c) {
foregroundColor: Colors.white, contactsCubit.addContact(contact);
icon: Icons.contacts, ScaffoldMessenger.of(context).showSnackBar(
label: tr('add_contact'), SnackBar(
), content: Text(tr('contact_added')),
], ),
);
},
backgroundColor: Theme
.of(context)
.primaryColor,
foregroundColor: Colors.white,
icon: Icons.contacts,
label: tr('add_contact'),
),
],
),
child: ListTile(
leading: (icon != null)
? Icon(
icon,
color: iconColor,
)
: null,
tileColor: tileColor(index, context),
title: Row(
children: <Widget>[
// if (avatar != null) avatar,
const SizedBox(width: 8.0),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Text(
statusText,
style: const TextStyle(
fontSize: 12.0,
color: Colors.grey,
), ),
child: ListTile( ),
leading: (icon != null) const SizedBox(height: 4.0),
? Icon( Text.rich(
icon, TextSpan(
color: iconColor, children: <InlineSpan>[
) /* TextSpan(
: null, text: isIncoming(transaction.type)
tileColor: tileColor(index, context), ? 'Recibido de '
title: Row( : 'Pago a ',
children: <Widget>[ style: const TextStyle(
// if (avatar != null) avatar, fontSize: 14.0,
const SizedBox(width: 8.0), // fontWeight: FontWeight.bold,
Expanded( ),
child: Column( ), */
crossAxisAlignment: CrossAxisAlignment.start, /* WidgetSpan(
children: <Widget>[ child: avatar != null
Text( ? const SizedBox(width: 8.0)
statusText, : const SizedBox.shrink(),
style: const TextStyle( ), */
fontSize: 12.0, WidgetSpan(
color: Colors.grey, child: Text(
), tr('transaction_from_to',
), namedArgs: <String, String>{
const SizedBox(height: 4.0), 'from': humanizeFromToPubKey(
Text.rich( myPubKey,
TextSpan( transaction.from),
children: <InlineSpan>[ 'to': humanizeFromToPubKey(
/* TextSpan( myPubKey, transaction.to)
text: isIncoming(transaction.type) }),
? 'Recibido de '
: 'Pago a ',
style: const TextStyle(
fontSize: 14.0,
// fontWeight: FontWeight.bold,
),
), */
/* WidgetSpan(
child: avatar != null
? const SizedBox(width: 8.0)
: const SizedBox.shrink(),
), */
WidgetSpan(
child: Text(
tr('transaction_from_to',
namedArgs: <String, String>{
'from': humanizeFromToPubKey(
myPubKey,
transaction.from),
'to': humanizeFromToPubKey(
myPubKey, transaction.to)
}),
style: const TextStyle(
fontSize: 14.0,
// fontWeight: FontWeight.bold,
),
),
),
],
),
),
],
),
),
],
),
subtitle: Padding(
padding: const EdgeInsets.fromLTRB(8, 0, 0, 10),
child: Text(transaction.comment,
style: const TextStyle( style: const TextStyle(
fontStyle: FontStyle.italic, fontSize: 14.0,
color: Colors.grey,
)),
),
trailing: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.end,
children: <Widget>[
Text(
amountS,
style: TextStyle(
// fontWeight: FontWeight.bold, // fontWeight: FontWeight.bold,
color: transaction.type ==
TransactionType.received ||
transaction.type ==
TransactionType.receiving
? Colors.blue
: Colors.red,
),
),
const SizedBox(height: 4.0),
Text(
humanizeTime(
transaction.time, context.locale.toString())!,
style: const TextStyle(
fontSize: 12.0,
color: Colors.grey,
), ),
), ),
], ),
), ],
)); ),
} else if (snapshot.hasError) { ),
return Text('Error ${snapshot.error}'); ],
} else { ),
return const Text('Loading'); ),
} ],
})); ),
subtitle: Padding(
padding: const EdgeInsets.fromLTRB(8, 0, 0, 10),
child: Text(transaction.comment,
style: const TextStyle(
fontStyle: FontStyle.italic,
color: Colors.grey,
)),
),
trailing: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.end,
children: <Widget>[
Text(
amountS,
style: TextStyle(
// fontWeight: FontWeight.bold,
color: transaction.type ==
TransactionType.received ||
transaction.type ==
TransactionType.receiving
? Colors.blue
: Colors.red,
),
),
const SizedBox(height: 4.0),
Text(
humanizeTime(
transaction.time, context.locale.toString())!,
style: const TextStyle(
fontSize: 12.0,
color: Colors.grey,
),
),
],
),
));
}
Future<Contact> _fetchContact(String pubKey, Transaction transaction) async { Future<Contact> _fetchContact(String pubKey, Transaction transaction) async {
return Contact(pubKey: pubKey); // return Contact(pubKey: pubKey);
if (pubKey == transaction.from) { if (pubKey == transaction.from) {
return ContactsCache().getContact(transaction.to); return ContactsCache().getContact(transaction.to);
} else { } else {
......
...@@ -4,13 +4,10 @@ import 'package:flutter/foundation.dart'; ...@@ -4,13 +4,10 @@ import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import '../../../data/models/contact.dart';
import '../../../data/models/node_list_cubit.dart'; import '../../../data/models/node_list_cubit.dart';
import '../../../data/models/transaction.dart'; import '../../../data/models/transaction.dart';
import '../../../data/models/transaction_cubit.dart'; import '../../../data/models/transaction_cubit.dart';
import '../../../shared_prefs.dart'; import '../../../shared_prefs.dart';
import '../../contacts_cache.dart';
import '../../logger.dart';
import '../../ui_helpers.dart'; import '../../ui_helpers.dart';
import 'transaction_chart.dart'; import 'transaction_chart.dart';
import 'transaction_item.dart'; import 'transaction_item.dart';
...@@ -64,15 +61,6 @@ class _TransactionsAndBalanceWidgetState ...@@ -64,15 +61,6 @@ class _TransactionsAndBalanceWidgetState
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final String myPubKey = SharedPreferencesHelper().getPubKey(); final String myPubKey = SharedPreferencesHelper().getPubKey();
if (!kReleaseMode) {
ContactsCache()
.getContact('6DrGg8cftpkgffv4Y4Lse9HSjgc8coEQor3yvMPHAnVH');
ContactsCache()
.getContact('A1Fc1VoCLKHyPYmXimYECSmjmsceqwRSZcTBXfgG9JaB');
ContactsCache()
.getContact('6DrGg8cftpkgffv4Y4Lse9HSjgc8coEQor3yvMPHAnVH');
ContactsCache().getContact(myPubKey).then((Contact c) => logger(c));
}
return BlocBuilder<TransactionsCubit, TransactionsAndBalanceState>(builder: return BlocBuilder<TransactionsCubit, TransactionsAndBalanceState>(builder:
(BuildContext context, TransactionsAndBalanceState transBalanceState) { (BuildContext context, TransactionsAndBalanceState transBalanceState) {
final List<Transaction> transactions = transBalanceState.transactions; final List<Transaction> transactions = transBalanceState.transactions;
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment