Merge pull request #5348 from 21pages/ab

ab: use cache for display and display peers while loading
This commit is contained in:
RustDesk 2023-08-11 15:59:12 +08:00 committed by GitHub
commit ef022c91bb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 160 additions and 17 deletions

View File

@ -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: [

View 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()),
);
}
}

View File

@ -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) {

View File

@ -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");
}
}
} }

View File

@ -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 {

View File

@ -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();