Merge pull request #5348 from 21pages/ab
ab: use cache for display and display peers while loading
This commit is contained in:
commit
ef022c91bb
@ -10,6 +10,7 @@ import 'package:get/get.dart';
|
|||||||
|
|
||||||
import '../../common.dart';
|
import '../../common.dart';
|
||||||
import 'dialog.dart';
|
import 'dialog.dart';
|
||||||
|
import 'loading_dot_widget.dart';
|
||||||
import 'login.dart';
|
import 'login.dart';
|
||||||
|
|
||||||
final hideAbTagsPanel = false.obs;
|
final hideAbTagsPanel = false.obs;
|
||||||
@ -39,7 +40,7 @@ class _AddressBookState extends State<AddressBook> {
|
|||||||
child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
onPressed: loginDialog, child: Text(translate("Login"))));
|
onPressed: loginDialog, child: Text(translate("Login"))));
|
||||||
} else {
|
} else {
|
||||||
if (gFFI.abModel.abLoading.value) {
|
if (gFFI.abModel.abLoading.value && gFFI.abModel.emtpy) {
|
||||||
return const Center(
|
return const Center(
|
||||||
child: CircularProgressIndicator(),
|
child: CircularProgressIndicator(),
|
||||||
);
|
);
|
||||||
@ -47,9 +48,15 @@ class _AddressBookState extends State<AddressBook> {
|
|||||||
if (gFFI.abModel.abError.isNotEmpty) {
|
if (gFFI.abModel.abError.isNotEmpty) {
|
||||||
return _buildShowError(gFFI.abModel.abError.value);
|
return _buildShowError(gFFI.abModel.abError.value);
|
||||||
}
|
}
|
||||||
return isDesktop
|
return Column(
|
||||||
|
children: [
|
||||||
|
_buildLoadingHavingPeers(),
|
||||||
|
Expanded(
|
||||||
|
child: isDesktop
|
||||||
? _buildAddressBookDesktop()
|
? _buildAddressBookDesktop()
|
||||||
: _buildAddressBookMobile();
|
: _buildAddressBookMobile())
|
||||||
|
],
|
||||||
|
);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -68,6 +75,22 @@ class _AddressBookState extends State<AddressBook> {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildLoadingHavingPeers() {
|
||||||
|
double size = 15;
|
||||||
|
return Obx(() => Offstage(
|
||||||
|
offstage: !(gFFI.abModel.abLoading.value && !gFFI.abModel.emtpy),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
SizedBox(
|
||||||
|
height: size,
|
||||||
|
child: Center(child: LoadingDotWidget(size: size)))
|
||||||
|
.marginSymmetric(vertical: 10)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
Widget _buildAddressBookDesktop() {
|
Widget _buildAddressBookDesktop() {
|
||||||
return Row(
|
return Row(
|
||||||
children: [
|
children: [
|
||||||
|
65
flutter/lib/common/widgets/loading_dot_widget.dart
Normal file
65
flutter/lib/common/widgets/loading_dot_widget.dart
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
|
||||||
|
class LoadingDotWidget extends StatefulWidget {
|
||||||
|
final int count;
|
||||||
|
final double size;
|
||||||
|
final int duration;
|
||||||
|
LoadingDotWidget(
|
||||||
|
{Key? key, required this.size, this.count = 3, this.duration = 200})
|
||||||
|
: super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<LoadingDotWidget> createState() => _LoadingDotWidgetState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _LoadingDotWidgetState extends State<LoadingDotWidget> {
|
||||||
|
int counter = 0;
|
||||||
|
Timer? timer;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
startAnimation();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
timer?.cancel();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
void startAnimation() {
|
||||||
|
timer = Timer.periodic(Duration(milliseconds: widget.duration), (timer) {
|
||||||
|
if (mounted) {
|
||||||
|
setState(() {
|
||||||
|
counter = (counter + 1) % widget.count;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
getChild(int index) {
|
||||||
|
return AnimatedContainer(
|
||||||
|
duration: Duration(milliseconds: widget.duration),
|
||||||
|
width: counter == index ? widget.size : widget.size / 2,
|
||||||
|
height: counter == index ? widget.size : widget.size / 2,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
color: Colors.grey,
|
||||||
|
),
|
||||||
|
).marginSymmetric(horizontal: widget.size);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Center(
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: List.generate(widget.count, (e) => e)
|
||||||
|
.map((e) => getChild(e))
|
||||||
|
.toList()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -987,7 +987,7 @@ class AddressBookPeerCard extends BasePeerCard {
|
|||||||
|
|
||||||
@protected
|
@protected
|
||||||
@override
|
@override
|
||||||
void _update() => gFFI.abModel.pullAb();
|
void _update() => gFFI.abModel.pullAb(quiet: true);
|
||||||
|
|
||||||
@protected
|
@protected
|
||||||
MenuEntryBase<String> _editTagAction(String id) {
|
MenuEntryBase<String> _editTagAction(String id) {
|
||||||
|
@ -28,6 +28,7 @@ class AbModel {
|
|||||||
final tags = [].obs;
|
final tags = [].obs;
|
||||||
final peers = List<Peer>.empty(growable: true).obs;
|
final peers = List<Peer>.empty(growable: true).obs;
|
||||||
final sortTags = shouldSortTags().obs;
|
final sortTags = shouldSortTags().obs;
|
||||||
|
bool get emtpy => peers.isEmpty && tags.isEmpty;
|
||||||
|
|
||||||
final selectedTags = List<String>.empty(growable: true).obs;
|
final selectedTags = List<String>.empty(growable: true).obs;
|
||||||
var initialized = false;
|
var initialized = false;
|
||||||
@ -46,10 +47,13 @@ class AbModel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<void> pullAb({force = true, quiet = false}) async {
|
Future<void> pullAb({force = true, quiet = false}) async {
|
||||||
debugPrint("pullAb, force:$force, quite:$quiet");
|
debugPrint("pullAb, force:$force, quiet:$quiet");
|
||||||
if (gFFI.userModel.userName.isEmpty) return;
|
if (gFFI.userModel.userName.isEmpty) return;
|
||||||
if (abLoading.value) return;
|
if (abLoading.value) return;
|
||||||
if (!force && initialized) return;
|
if (!force && initialized) return;
|
||||||
|
if (!initialized) {
|
||||||
|
await _loadCache();
|
||||||
|
}
|
||||||
if (!quiet) {
|
if (!quiet) {
|
||||||
abLoading.value = true;
|
abLoading.value = true;
|
||||||
abError.value = "";
|
abError.value = "";
|
||||||
@ -92,14 +96,21 @@ class AbModel {
|
|||||||
peers.add(Peer.fromJson(peer));
|
peers.add(Peer.fromJson(peer));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
save(); // save on success
|
_saveCache(); // save on success
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
abError.value = err.toString();
|
abError.value = err.toString();
|
||||||
} finally {
|
} finally {
|
||||||
|
if (initialized) {
|
||||||
|
// make loading effect obvious
|
||||||
|
Future.delayed(Duration(milliseconds: 300), () {
|
||||||
abLoading.value = false;
|
abLoading.value = false;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
abLoading.value = false;
|
||||||
|
}
|
||||||
initialized = true;
|
initialized = true;
|
||||||
sync_all_from_recent = true;
|
sync_all_from_recent = true;
|
||||||
_timerCounter = 0;
|
_timerCounter = 0;
|
||||||
@ -197,7 +208,7 @@ class AbModel {
|
|||||||
try {
|
try {
|
||||||
await http.Client().send(request);
|
await http.Client().send(request);
|
||||||
// await pullAb(quiet: true);
|
// await pullAb(quiet: true);
|
||||||
save(); // save on success
|
_saveCache(); // save on success
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
BotToast.showText(contentColor: Colors.red, text: e.toString());
|
BotToast.showText(contentColor: Colors.red, text: e.toString());
|
||||||
} finally {
|
} finally {
|
||||||
@ -369,21 +380,39 @@ class AbModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
save() {
|
_saveCache() {
|
||||||
try {
|
try {
|
||||||
final infos = peers
|
final peersJsonData = peers.map((e) => e.toAbUploadJson()).toList();
|
||||||
.map((e) => (<String, dynamic>{
|
|
||||||
"id": e.id,
|
|
||||||
"hash": e.hash,
|
|
||||||
}))
|
|
||||||
.toList();
|
|
||||||
final m = <String, dynamic>{
|
final m = <String, dynamic>{
|
||||||
"access_token": bind.mainGetLocalOption(key: 'access_token'),
|
"access_token": bind.mainGetLocalOption(key: 'access_token'),
|
||||||
"peers": infos,
|
"peers": peersJsonData,
|
||||||
|
"tags": tags.map((e) => e.toString()).toList(),
|
||||||
};
|
};
|
||||||
bind.mainSaveAb(json: jsonEncode(m));
|
bind.mainSaveAb(json: jsonEncode(m));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint('ab save:$e');
|
debugPrint('ab save:$e');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_loadCache() async {
|
||||||
|
try {
|
||||||
|
final access_token = bind.mainGetLocalOption(key: 'access_token');
|
||||||
|
if (access_token.isEmpty) return;
|
||||||
|
final cache = await bind.mainLoadAb();
|
||||||
|
final data = jsonDecode(cache);
|
||||||
|
if (data == null || data['access_token'] != access_token) return;
|
||||||
|
tags.clear();
|
||||||
|
peers.clear();
|
||||||
|
if (data['tags'] is List) {
|
||||||
|
tags.value = data['tags'];
|
||||||
|
}
|
||||||
|
if (data['peers'] is List) {
|
||||||
|
for (final peer in data['peers']) {
|
||||||
|
peers.add(Peer.fromJson(peer));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
debugPrint("load ab cache: $e");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1481,6 +1481,26 @@ pub struct AbPeer {
|
|||||||
skip_serializing_if = "String::is_empty"
|
skip_serializing_if = "String::is_empty"
|
||||||
)]
|
)]
|
||||||
pub hash: String,
|
pub hash: String,
|
||||||
|
#[serde(
|
||||||
|
default,
|
||||||
|
deserialize_with = "deserialize_string",
|
||||||
|
skip_serializing_if = "String::is_empty"
|
||||||
|
)]
|
||||||
|
pub username: String,
|
||||||
|
#[serde(
|
||||||
|
default,
|
||||||
|
deserialize_with = "deserialize_string",
|
||||||
|
skip_serializing_if = "String::is_empty"
|
||||||
|
)]
|
||||||
|
pub hostname: String,
|
||||||
|
#[serde(
|
||||||
|
default,
|
||||||
|
deserialize_with = "deserialize_string",
|
||||||
|
skip_serializing_if = "String::is_empty"
|
||||||
|
)]
|
||||||
|
pub platform: String,
|
||||||
|
#[serde(default, deserialize_with = "deserialize_vec_string")]
|
||||||
|
pub tags: Vec<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default, Serialize, Deserialize, Clone)]
|
#[derive(Debug, Default, Serialize, Deserialize, Clone)]
|
||||||
@ -1493,6 +1513,8 @@ pub struct Ab {
|
|||||||
pub access_token: String,
|
pub access_token: String,
|
||||||
#[serde(default, deserialize_with = "deserialize_vec_abpeer")]
|
#[serde(default, deserialize_with = "deserialize_vec_abpeer")]
|
||||||
pub peers: Vec<AbPeer>,
|
pub peers: Vec<AbPeer>,
|
||||||
|
#[serde(default, deserialize_with = "deserialize_vec_string")]
|
||||||
|
pub tags: Vec<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Ab {
|
impl Ab {
|
||||||
|
@ -1165,6 +1165,10 @@ pub fn main_clear_ab() {
|
|||||||
config::Ab::remove();
|
config::Ab::remove();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn main_load_ab() -> String {
|
||||||
|
serde_json::to_string(&config::Ab::load()).unwrap_or_default()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn session_send_pointer(session_id: SessionID, msg: String) {
|
pub fn session_send_pointer(session_id: SessionID, msg: String) {
|
||||||
if let Ok(m) = serde_json::from_str::<HashMap<String, serde_json::Value>>(&msg) {
|
if let Ok(m) = serde_json::from_str::<HashMap<String, serde_json::Value>>(&msg) {
|
||||||
let alt = m.get("alt").is_some();
|
let alt = m.get("alt").is_some();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user