Merge pull request #1687 from Heap-Hop/master

fix flutter desktop Address book
This commit is contained in:
RustDesk 2022-10-09 00:29:41 +08:00 committed by GitHub
commit 672d5f31d0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 356 additions and 405 deletions

View File

@ -1,9 +1,10 @@
import 'package:contextmenu/contextmenu.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hbb/common/widgets/peer_card.dart';
import 'package:flutter_hbb/common/widgets/peers_view.dart'; import 'package:flutter_hbb/common/widgets/peers_view.dart';
import 'package:flutter_hbb/models/ab_model.dart'; import 'package:flutter_hbb/desktop/widgets/popup_menu.dart';
import '../../consts.dart';
import '../../desktop/widgets/material_mod_popup_menu.dart' as mod_menu;
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:provider/provider.dart';
import '../../common.dart'; import '../../common.dart';
import '../../desktop/pages/desktop_home_page.dart'; import '../../desktop/pages/desktop_home_page.dart';
@ -24,7 +25,7 @@ class _AddressBookState extends State<AddressBook> {
@override @override
void initState() { void initState() {
super.initState(); super.initState();
WidgetsBinding.instance.addPostFrameCallback((_) => gFFI.abModel.getAb()); WidgetsBinding.instance.addPostFrameCallback((_) => gFFI.abModel.pullAb());
} }
@override @override
@ -66,88 +67,72 @@ class _AddressBookState extends State<AddressBook> {
} }
final model = gFFI.abModel; final model = gFFI.abModel;
return FutureBuilder( return FutureBuilder(
future: model.getAb(), future: model.pullAb(),
builder: (context, snapshot) { builder: (context, snapshot) {
if (snapshot.hasData) { if (snapshot.hasData) {
return _buildAddressBook(context); return _buildAddressBook(context);
} else if (snapshot.hasError) { } else if (snapshot.hasError) {
return Column( return _buildShowError(snapshot.error.toString());
mainAxisAlignment: MainAxisAlignment.center,
children: [
Text(translate("${snapshot.error}")),
TextButton(
onPressed: () {
setState(() {});
},
child: Text(translate("Retry")))
],
);
} else { } else {
if (model.abLoading) { return Obx(() {
if (model.abLoading.value) {
return const Center( return const Center(
child: CircularProgressIndicator(), child: CircularProgressIndicator(),
); );
} else if (model.abError.isNotEmpty) { } else if (model.abError.isNotEmpty) {
return _buildShowError(model.abError.value);
} else {
return const Offstage();
}
});
}
});
}
Widget _buildShowError(String error) {
return Center( return Center(
child: Column( child: Column(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
Text(translate(model.abError)), Text(translate(error)),
TextButton( TextButton(
onPressed: () { onPressed: () {
setState(() {}); setState(() {});
}, },
child: Text(translate("Retry"))) child: Text(translate("Retry")))
], ],
), ));
);
} else {
return const Offstage();
}
}
});
} }
Widget _buildAddressBook(BuildContext context) { Widget _buildAddressBook(BuildContext context) {
return Consumer<AbModel>( var pos = RelativeRect.fill;
builder: (context, model, child) => Row( return Row(
children: [ children: [
Card( Card(
margin: EdgeInsets.symmetric(horizontal: 4.0),
shape: RoundedRectangleBorder( shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(20), borderRadius: BorderRadius.circular(12),
side: BorderSide( side:
color: Theme.of(context).scaffoldBackgroundColor)), BorderSide(color: Theme.of(context).scaffoldBackgroundColor)),
child: Container( child: Container(
width: 200, width: 200,
height: double.infinity, height: double.infinity,
padding: const EdgeInsets.symmetric( padding:
horizontal: 12.0, vertical: 8.0), const EdgeInsets.symmetric(horizontal: 12.0, vertical: 8.0),
child: Column( child: Column(
children: [ children: [
Row( Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Text(translate('Tags')), Text(translate('Tags')),
InkWell( GestureDetector(
child: PopupMenuButton( onTapDown: (e) {
itemBuilder: (context) => [ final x = e.globalPosition.dx;
PopupMenuItem( final y = e.globalPosition.dy;
value: 'add-id', pos = RelativeRect.fromLTRB(x, y, x, y);
child: Text(translate("Add ID")), },
), onTap: () => _showMenu(pos),
PopupMenuItem( child: ActionMore()),
value: 'add-tag',
child: Text(translate("Add Tag")),
),
PopupMenuItem(
value: 'unset-all-tag',
child: Text(
translate("Unselect all tags")),
),
],
onSelected: handleAbOp,
child: const Icon(Icons.more_vert_outlined)),
)
], ],
), ),
Expanded( Expanded(
@ -155,16 +140,16 @@ class _AddressBookState extends State<AddressBook> {
width: double.infinity, width: double.infinity,
height: double.infinity, height: double.infinity,
decoration: BoxDecoration( decoration: BoxDecoration(
border: Border.all(color: MyTheme.darkGray)), border: Border.all(color: MyTheme.darkGray),
borderRadius: BorderRadius.circular(2)),
child: Obx( child: Obx(
() => Wrap( () => Wrap(
children: gFFI.abModel.tags children: gFFI.abModel.tags
.map((e) => .map((e) => AddressBookTag(
buildTag(e, gFFI.abModel.selectedTags, name: e,
tags: gFFI.abModel.selectedTags,
onTap: () { onTap: () {
// if (gFFI.abModel.selectedTags.contains(e)) {
if (gFFI.abModel.selectedTags
.contains(e)) {
gFFI.abModel.selectedTags.remove(e); gFFI.abModel.selectedTags.remove(e);
} else { } else {
gFFI.abModel.selectedTags.add(e); gFFI.abModel.selectedTags.add(e);
@ -182,58 +167,36 @@ class _AddressBookState extends State<AddressBook> {
Expanded( Expanded(
child: Align( child: Align(
alignment: Alignment.topLeft, alignment: Alignment.topLeft,
child: AddressBookPeersView( child: Obx(() => AddressBookPeersView(
menuPadding: widget.menuPadding, menuPadding: widget.menuPadding,
)), initPeers: gFFI.abModel.peers.value,
))),
) )
], ],
));
}
Widget buildTag(String tagName, RxList<dynamic> rxTags, {Function()? onTap}) {
return ContextMenuArea(
width: 100,
builder: (context) => [
ListTile(
title: Text(translate("Delete")),
onTap: () {
gFFI.abModel.deleteTag(tagName);
gFFI.abModel.updateAb();
Future.delayed(Duration.zero, () => Get.back());
},
)
],
child: GestureDetector(
onTap: onTap,
child: Obx(
() => Container(
decoration: BoxDecoration(
color: rxTags.contains(tagName) ? Colors.blue : null,
border: Border.all(color: MyTheme.darkGray),
borderRadius: BorderRadius.circular(10)),
margin: const EdgeInsets.symmetric(horizontal: 4.0, vertical: 8.0),
padding: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 8.0),
child: Text(
tagName,
style: TextStyle(
color:
rxTags.contains(tagName) ? Colors.white : null), // TODO
),
),
),
),
); );
} }
/// tag operation void _showMenu(RelativeRect pos) {
void handleAbOp(String value) { final items = [
if (value == 'add-id') { getEntry(translate("Add ID"), abAddId),
abAddId(); getEntry(translate("Add Tag"), abAddTag),
} else if (value == 'add-tag') { getEntry(translate("Unselect all tags"), gFFI.abModel.unsetSelectedTags),
abAddTag(); ];
} else if (value == 'unset-all-tag') {
gFFI.abModel.unsetSelectedTags(); mod_menu.showMenu(
} context: context,
position: pos,
items: items
.map((e) => e.build(
context,
MenuConfig(
commonColor: CustomPopupMenuTheme.commonColor,
height: CustomPopupMenuTheme.height,
dividerHeight: CustomPopupMenuTheme.dividerHeight)))
.expand((i) => i)
.toList(),
elevation: 8,
);
} }
void abAddId() async { void abAddId() async {
@ -260,7 +223,7 @@ class _AddressBookState extends State<AddressBook> {
} }
gFFI.abModel.addId(newId); gFFI.abModel.addId(newId);
} }
await gFFI.abModel.updateAb(); await gFFI.abModel.pushAb();
this.setState(() {}); this.setState(() {});
// final currentPeers // final currentPeers
} }
@ -327,7 +290,7 @@ class _AddressBookState extends State<AddressBook> {
for (final tag in tags) { for (final tag in tags) {
gFFI.abModel.addTag(tag); gFFI.abModel.addTag(tag);
} }
await gFFI.abModel.updateAb(); await gFFI.abModel.pushAb();
// final currentPeers // final currentPeers
} }
close(); close();
@ -373,54 +336,88 @@ class _AddressBookState extends State<AddressBook> {
); );
}); });
} }
void abEditTag(String id) {
var isInProgress = false;
final tags = List.of(gFFI.abModel.tags);
var selectedTag = gFFI.abModel.getPeerTags(id).obs;
gFFI.dialogManager.show((setState, close) {
submit() async {
setState(() {
isInProgress = true;
});
gFFI.abModel.changeTagForPeer(id, selectedTag);
await gFFI.abModel.updateAb();
close();
} }
return CustomAlertDialog( class AddressBookTag extends StatelessWidget {
title: Text(translate("Edit Tag")), final String name;
content: Column( final RxList<dynamic> tags;
crossAxisAlignment: CrossAxisAlignment.start, final Function()? onTap;
children: [ final bool showActionMenu;
Container(
padding: const AddressBookTag(
const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), {Key? key,
child: Wrap( required this.name,
children: tags required this.tags,
.map((e) => buildTag(e, selectedTag, onTap: () { this.onTap,
if (selectedTag.contains(e)) { this.showActionMenu = true})
selectedTag.remove(e); : super(key: key);
} else {
selectedTag.add(e); @override
Widget build(BuildContext context) {
var pos = RelativeRect.fill;
void setPosition(TapDownDetails e) {
final x = e.globalPosition.dx;
final y = e.globalPosition.dy;
pos = RelativeRect.fromLTRB(x, y, x, y);
} }
}))
.toList(growable: false), return GestureDetector(
onTap: onTap,
onTapDown: showActionMenu ? setPosition : null,
onSecondaryTapDown: showActionMenu ? setPosition : null,
onSecondaryTap: showActionMenu ? () => _showMenu(context, pos) : null,
onLongPress: showActionMenu ? () => _showMenu(context, pos) : null,
child: Obx(
() => Container(
decoration: BoxDecoration(
color: tags.contains(name) ? Colors.blue : null,
border: Border.all(color: MyTheme.darkGray),
borderRadius: BorderRadius.circular(6)),
margin: const EdgeInsets.symmetric(horizontal: 4.0, vertical: 8.0),
padding: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 8.0),
child: Text(name,
style:
TextStyle(color: tags.contains(name) ? Colors.white : null)),
), ),
), ),
Offstage(
offstage: !isInProgress, child: const LinearProgressIndicator())
],
),
actions: [
TextButton(onPressed: close, child: Text(translate("Cancel"))),
TextButton(onPressed: submit, child: Text(translate("OK"))),
],
onSubmit: submit,
onCancel: close,
); );
}); }
void _showMenu(BuildContext context, RelativeRect pos) {
final items = [
getEntry(translate("Delete"), () {
gFFI.abModel.deleteTag(name);
gFFI.abModel.pushAb();
Future.delayed(Duration.zero, () => Get.back());
}),
];
mod_menu.showMenu(
context: context,
position: pos,
items: items
.map((e) => e.build(
context,
MenuConfig(
commonColor: CustomPopupMenuTheme.commonColor,
height: CustomPopupMenuTheme.height,
dividerHeight: CustomPopupMenuTheme.dividerHeight)))
.expand((i) => i)
.toList(),
elevation: 8,
);
} }
} }
MenuEntryButton<String> getEntry(String title, VoidCallback proc) {
return MenuEntryButton<String>(
childBuilder: (TextStyle? style) => Text(
title,
style: style,
),
proc: proc,
padding: kDesktopMenuPadding,
dismissOnClicked: true,
);
}

View File

@ -1,6 +1,6 @@
import 'package:contextmenu/contextmenu.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:flutter_hbb/common/widgets/address_book.dart';
import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/consts.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
@ -12,7 +12,7 @@ import '../../models/platform_model.dart';
import '../../desktop/widgets/material_mod_popup_menu.dart' as mod_menu; import '../../desktop/widgets/material_mod_popup_menu.dart' as mod_menu;
import '../../desktop/widgets/popup_menu.dart'; import '../../desktop/widgets/popup_menu.dart';
class _PopupMenuTheme { class CustomPopupMenuTheme {
static const Color commonColor = MyTheme.accent; static const Color commonColor = MyTheme.accent;
// kMinInteractiveDimension // kMinInteractiveDimension
static const double height = 20.0; static const double height = 20.0;
@ -46,9 +46,8 @@ class _PeerCard extends StatefulWidget {
class _PeerCardState extends State<_PeerCard> class _PeerCardState extends State<_PeerCard>
with AutomaticKeepAliveClientMixin { with AutomaticKeepAliveClientMixin {
var _menuPos = RelativeRect.fill; var _menuPos = RelativeRect.fill;
final double _cardRadis = 16; final double _cardRadius = 16;
final double _borderWidth = 2; final double _borderWidth = 2;
final RxBool _iconMoreHover = false.obs;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -122,23 +121,23 @@ class _PeerCardState extends State<_PeerCard>
var deco = Rx<BoxDecoration?>(BoxDecoration( var deco = Rx<BoxDecoration?>(BoxDecoration(
border: Border.all(color: Colors.transparent, width: _borderWidth), border: Border.all(color: Colors.transparent, width: _borderWidth),
borderRadius: peerCardUiType.value == PeerUiType.grid borderRadius: peerCardUiType.value == PeerUiType.grid
? BorderRadius.circular(_cardRadis) ? BorderRadius.circular(_cardRadius)
: null)); : null));
return MouseRegion( return MouseRegion(
onEnter: (evt) { onEnter: (evt) {
deco.value = BoxDecoration( deco.value = BoxDecoration(
border: Border.all( border: Border.all(
color: Theme.of(context).colorScheme.secondary, color: Theme.of(context).colorScheme.primary,
width: _borderWidth), width: _borderWidth),
borderRadius: peerCardUiType.value == PeerUiType.grid borderRadius: peerCardUiType.value == PeerUiType.grid
? BorderRadius.circular(_cardRadis) ? BorderRadius.circular(_cardRadius)
: null); : null);
}, },
onExit: (evt) { onExit: (evt) {
deco.value = BoxDecoration( deco.value = BoxDecoration(
border: Border.all(color: Colors.transparent, width: _borderWidth), border: Border.all(color: Colors.transparent, width: _borderWidth),
borderRadius: peerCardUiType.value == PeerUiType.grid borderRadius: peerCardUiType.value == PeerUiType.grid
? BorderRadius.circular(_cardRadis) ? BorderRadius.circular(_cardRadius)
: null); : null);
}, },
child: GestureDetector( child: GestureDetector(
@ -221,7 +220,7 @@ class _PeerCardState extends State<_PeerCard>
() => Container( () => Container(
foregroundDecoration: deco.value, foregroundDecoration: deco.value,
child: ClipRRect( child: ClipRRect(
borderRadius: BorderRadius.circular(_cardRadis - _borderWidth), borderRadius: BorderRadius.circular(_cardRadius - _borderWidth),
child: Column( child: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
@ -299,27 +298,7 @@ class _PeerCardState extends State<_PeerCard>
_menuPos = RelativeRect.fromLTRB(x, y, x, y); _menuPos = RelativeRect.fromLTRB(x, y, x, y);
}, },
onPointerUp: (_) => _showPeerMenu(peer.id), onPointerUp: (_) => _showPeerMenu(peer.id),
child: MouseRegion( child: ActionMore());
onEnter: (_) => _iconMoreHover.value = true,
onExit: (_) => _iconMoreHover.value = false,
child: CircleAvatar(
radius: 14,
backgroundColor: _iconMoreHover.value
? Theme.of(context).scaffoldBackgroundColor
: Theme.of(context).backgroundColor,
// ? Theme.of(context).scaffoldBackgroundColor!
// : Theme.of(context).backgroundColor!,
child: Icon(Icons.more_vert,
size: 18,
color: _iconMoreHover.value
? Theme.of(context).textTheme.titleLarge?.color
: Theme.of(context)
.textTheme
.titleLarge
?.color
?.withOpacity(0.5)))));
// ? MyTheme.color(context).text
// : MyTheme.color(context).lightText))));
/// Show the peer menu and handle user's choice. /// Show the peer menu and handle user's choice.
/// User might remove the peer or send a file to the peer. /// User might remove the peer or send a file to the peer.
@ -358,9 +337,9 @@ abstract class BasePeerCard extends StatelessWidget {
.map((e) => e.build( .map((e) => e.build(
context, context,
const MenuConfig( const MenuConfig(
commonColor: _PopupMenuTheme.commonColor, commonColor: CustomPopupMenuTheme.commonColor,
height: _PopupMenuTheme.height, height: CustomPopupMenuTheme.height,
dividerHeight: _PopupMenuTheme.dividerHeight))) dividerHeight: CustomPopupMenuTheme.dividerHeight)))
.expand((i) => i) .expand((i) => i)
.toList(); .toList();
@ -426,7 +405,7 @@ abstract class BasePeerCard extends StatelessWidget {
return MenuEntryButton<String>( return MenuEntryButton<String>(
childBuilder: (TextStyle? style) => Container( childBuilder: (TextStyle? style) => Container(
alignment: AlignmentDirectional.center, alignment: AlignmentDirectional.center,
height: _PopupMenuTheme.height, height: CustomPopupMenuTheme.height,
child: Row( child: Row(
children: [ children: [
Text( Text(
@ -601,11 +580,11 @@ abstract class BasePeerCard extends StatelessWidget {
var name = peer.alias; var name = peer.alias;
var controller = TextEditingController(text: name); var controller = TextEditingController(text: name);
if (isAddressBook) { if (isAddressBook) {
final peer = gFFI.abModel.peers.firstWhere((p) => id == p['id']); final peer = gFFI.abModel.peers.firstWhereOrNull((p) => id == p.id);
if (peer == null) { if (peer == null) {
// this should not happen // this should not happen
} else { } else {
name = peer['alias'] ?? ''; name = peer.alias;
} }
} }
gFFI.dialogManager.show((setState, close) { gFFI.dialogManager.show((setState, close) {
@ -614,11 +593,11 @@ abstract class BasePeerCard extends StatelessWidget {
name = controller.text; name = controller.text;
await bind.mainSetPeerOption(id: id, key: 'alias', value: name); await bind.mainSetPeerOption(id: id, key: 'alias', value: name);
if (isAddressBook) { if (isAddressBook) {
gFFI.abModel.setPeerOption(id, 'alias', name); gFFI.abModel.setPeerAlias(id, name);
await gFFI.abModel.updateAb(); await gFFI.abModel.pushAb();
} }
if (isAddressBook) { if (isAddressBook) {
gFFI.abModel.getAb(); gFFI.abModel.pullAb();
} else { } else {
bind.mainLoadRecentPeers(); bind.mainLoadRecentPeers();
bind.mainLoadFavPeers(); bind.mainLoadFavPeers();
@ -774,7 +753,9 @@ class AddressBookPeerCard extends BasePeerCard {
if (await bind.mainPeerHasPassword(id: peer.id)) { if (await bind.mainPeerHasPassword(id: peer.id)) {
menuItems.add(_unrememberPasswordAction(peer.id)); menuItems.add(_unrememberPasswordAction(peer.id));
} }
if (gFFI.abModel.tags.isNotEmpty) {
menuItems.add(_editTagAction(peer.id)); menuItems.add(_editTagAction(peer.id));
}
return menuItems; return menuItems;
} }
@ -791,7 +772,7 @@ class AddressBookPeerCard extends BasePeerCard {
proc: () { proc: () {
() async { () async {
gFFI.abModel.deletePeer(id); gFFI.abModel.deletePeer(id);
await gFFI.abModel.updateAb(); await gFFI.abModel.pushAb();
}(); }();
}, },
padding: super.menuPadding, padding: super.menuPadding,
@ -826,7 +807,7 @@ class AddressBookPeerCard extends BasePeerCard {
isInProgress = true; isInProgress = true;
}); });
gFFI.abModel.changeTagForPeer(id, selectedTag); gFFI.abModel.changeTagForPeer(id, selectedTag);
await gFFI.abModel.updateAb(); await gFFI.abModel.pushAb();
close(); close();
} }
@ -836,17 +817,20 @@ class AddressBookPeerCard extends BasePeerCard {
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Container( Container(
padding: padding: const EdgeInsets.symmetric(vertical: 8.0),
const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
child: Wrap( child: Wrap(
children: tags children: tags
.map((e) => _buildTag(e, selectedTag, onTap: () { .map((e) => AddressBookTag(
name: e,
tags: selectedTag,
onTap: () {
if (selectedTag.contains(e)) { if (selectedTag.contains(e)) {
selectedTag.remove(e); selectedTag.remove(e);
} else { } else {
selectedTag.add(e); selectedTag.add(e);
} }
})) },
showActionMenu: false))
.toList(growable: false), .toList(growable: false),
), ),
), ),
@ -863,41 +847,6 @@ class AddressBookPeerCard extends BasePeerCard {
); );
}); });
} }
Widget _buildTag(String tagName, RxList<dynamic> rxTags,
{Function()? onTap}) {
return ContextMenuArea(
width: 100,
builder: (context) => [
ListTile(
title: Text(translate("Delete")),
onTap: () {
gFFI.abModel.deleteTag(tagName);
gFFI.abModel.updateAb();
Future.delayed(Duration.zero, () => Get.back());
},
)
],
child: GestureDetector(
onTap: onTap,
child: Obx(
() => Container(
decoration: BoxDecoration(
color: rxTags.contains(tagName) ? Colors.blue : null,
border: Border.all(color: MyTheme.darkGray),
borderRadius: BorderRadius.circular(10)),
margin: const EdgeInsets.symmetric(horizontal: 4.0, vertical: 8.0),
padding: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 8.0),
child: Text(
tagName,
style: TextStyle(
color: rxTags.contains(tagName) ? Colors.white : null),
),
),
),
),
);
}
} }
void _rdpDialog(String id) async { void _rdpDialog(String id) async {
@ -905,7 +854,7 @@ void _rdpDialog(String id) async {
text: await bind.mainGetPeerOption(id: id, key: 'rdp_port')); text: await bind.mainGetPeerOption(id: id, key: 'rdp_port'));
final userController = TextEditingController( final userController = TextEditingController(
text: await bind.mainGetPeerOption(id: id, key: 'rdp_username')); text: await bind.mainGetPeerOption(id: id, key: 'rdp_username'));
final passwordContorller = TextEditingController( final passwordController = TextEditingController(
text: await bind.mainGetPeerOption(id: id, key: 'rdp_password')); text: await bind.mainGetPeerOption(id: id, key: 'rdp_password'));
RxBool secure = true.obs; RxBool secure = true.obs;
@ -916,7 +865,7 @@ void _rdpDialog(String id) async {
await bind.mainSetPeerOption( await bind.mainSetPeerOption(
id: id, key: 'rdp_username', value: userController.text); id: id, key: 'rdp_username', value: userController.text);
await bind.mainSetPeerOption( await bind.mainSetPeerOption(
id: id, key: 'rdp_password', value: passwordContorller.text); id: id, key: 'rdp_password', value: passwordController.text);
close(); close();
} }
@ -1000,7 +949,7 @@ void _rdpDialog(String id) async {
icon: Icon(secure.value icon: Icon(secure.value
? Icons.visibility_off ? Icons.visibility_off
: Icons.visibility))), : Icons.visibility))),
controller: passwordContorller, controller: passwordController,
)), )),
), ),
], ],
@ -1027,3 +976,28 @@ Widget getOnline(double rightPadding, bool online) {
child: CircleAvatar( child: CircleAvatar(
radius: 3, backgroundColor: online ? Colors.green : kColorWarn))); radius: 3, backgroundColor: online ? Colors.green : kColorWarn)));
} }
class ActionMore extends StatelessWidget {
final RxBool _iconMoreHover = false.obs;
@override
Widget build(BuildContext context) {
return MouseRegion(
onEnter: (_) => _iconMoreHover.value = true,
onExit: (_) => _iconMoreHover.value = false,
child: Obx(() => CircleAvatar(
radius: 14,
backgroundColor: _iconMoreHover.value
? Theme.of(context).scaffoldBackgroundColor
: Theme.of(context).backgroundColor,
child: Icon(Icons.more_vert,
size: 18,
color: _iconMoreHover.value
? Theme.of(context).textTheme.titleLarge?.color
: Theme.of(context)
.textTheme
.titleLarge
?.color
?.withOpacity(0.5)))));
}
}

View File

@ -54,7 +54,7 @@ class _PeerTabPageState extends State<PeerTabPage>
bind.mainDiscover(); bind.mainDiscover();
break; break;
case 3: case 3:
gFFI.abModel.getAb(); gFFI.abModel.pullAb();
break; break;
} }
} }

View File

@ -3,7 +3,6 @@ import 'dart:async';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/consts.dart';
import 'package:flutter_hbb/desktop/widgets/scroll_wrapper.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'package:visibility_detector/visibility_detector.dart'; import 'package:visibility_detector/visibility_detector.dart';
@ -14,8 +13,7 @@ import '../../models/peer_model.dart';
import '../../models/platform_model.dart'; import '../../models/platform_model.dart';
import 'peer_card.dart'; import 'peer_card.dart';
typedef OffstageFunc = bool Function(Peer peer); typedef PeerCardBuilder = Widget Function(Peer peer);
typedef PeerCardBuilder = BasePeerCard Function(Peer peer);
/// for peer search text, global obs value /// for peer search text, global obs value
final peerSearchText = "".obs; final peerSearchText = "".obs;
@ -24,16 +22,10 @@ final peerSearchTextController =
class _PeersView extends StatefulWidget { class _PeersView extends StatefulWidget {
final Peers peers; final Peers peers;
final OffstageFunc offstageFunc;
final PeerCardBuilder peerCardBuilder; final PeerCardBuilder peerCardBuilder;
final ScrollController? scrollController;
const _PeersView( const _PeersView(
{required this.peers, {required this.peers, required this.peerCardBuilder, Key? key})
required this.offstageFunc,
required this.peerCardBuilder,
Key? key,
this.scrollController})
: super(key: key); : super(key: key);
@override @override
@ -124,20 +116,16 @@ class _PeersViewState extends State<_PeersView> with WindowListener {
}, },
child: widget.peerCardBuilder(peer), child: widget.peerCardBuilder(peer),
); );
cards.add(Offstage( cards.add(isDesktop
key: ValueKey("off${peer.id}"),
offstage: widget.offstageFunc(peer),
child: isDesktop
? Obx( ? Obx(
() => SizedBox( () => SizedBox(
width: 220, width: 220,
height: peerCardUiType.value == PeerUiType.grid height:
? 140 peerCardUiType.value == PeerUiType.grid ? 140 : 42,
: 42,
child: visibilityChild, child: visibilityChild,
), ),
) )
: SizedBox(width: mobileWidth, child: visibilityChild))); : SizedBox(width: mobileWidth, child: visibilityChild));
} }
return Wrap(spacing: space, runSpacing: space, children: cards); return Wrap(spacing: space, runSpacing: space, children: cards);
} else { } else {
@ -190,7 +178,6 @@ class _PeersViewState extends State<_PeersView> with WindowListener {
abstract class BasePeersView extends StatelessWidget { abstract class BasePeersView extends StatelessWidget {
final String name; final String name;
final String loadEvent; final String loadEvent;
final OffstageFunc offstageFunc;
final PeerCardBuilder peerCardBuilder; final PeerCardBuilder peerCardBuilder;
final List<Peer> initPeers; final List<Peer> initPeers;
@ -198,7 +185,6 @@ abstract class BasePeersView extends StatelessWidget {
Key? key, Key? key,
required this.name, required this.name,
required this.loadEvent, required this.loadEvent,
required this.offstageFunc,
required this.peerCardBuilder, required this.peerCardBuilder,
required this.initPeers, required this.initPeers,
}) : super(key: key); }) : super(key: key);
@ -207,7 +193,6 @@ abstract class BasePeersView extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return _PeersView( return _PeersView(
peers: Peers(name: name, loadEvent: loadEvent, peers: initPeers), peers: Peers(name: name, loadEvent: loadEvent, peers: initPeers),
offstageFunc: offstageFunc,
peerCardBuilder: peerCardBuilder); peerCardBuilder: peerCardBuilder);
} }
} }
@ -219,7 +204,6 @@ class RecentPeersView extends BasePeersView {
key: key, key: key,
name: 'recent peer', name: 'recent peer',
loadEvent: 'load_recent_peers', loadEvent: 'load_recent_peers',
offstageFunc: (Peer peer) => false,
peerCardBuilder: (Peer peer) => RecentPeerCard( peerCardBuilder: (Peer peer) => RecentPeerCard(
peer: peer, peer: peer,
menuPadding: menuPadding, menuPadding: menuPadding,
@ -242,7 +226,6 @@ class FavoritePeersView extends BasePeersView {
key: key, key: key,
name: 'favorite peer', name: 'favorite peer',
loadEvent: 'load_fav_peers', loadEvent: 'load_fav_peers',
offstageFunc: (Peer peer) => false,
peerCardBuilder: (Peer peer) => FavoritePeerCard( peerCardBuilder: (Peer peer) => FavoritePeerCard(
peer: peer, peer: peer,
menuPadding: menuPadding, menuPadding: menuPadding,
@ -265,7 +248,6 @@ class DiscoveredPeersView extends BasePeersView {
key: key, key: key,
name: 'discovered peer', name: 'discovered peer',
loadEvent: 'load_lan_peers', loadEvent: 'load_lan_peers',
offstageFunc: (Peer peer) => false,
peerCardBuilder: (Peer peer) => DiscoveredPeerCard( peerCardBuilder: (Peer peer) => DiscoveredPeerCard(
peer: peer, peer: peer,
menuPadding: menuPadding, menuPadding: menuPadding,
@ -283,27 +265,24 @@ class DiscoveredPeersView extends BasePeersView {
class AddressBookPeersView extends BasePeersView { class AddressBookPeersView extends BasePeersView {
AddressBookPeersView( AddressBookPeersView(
{Key? key, EdgeInsets? menuPadding, ScrollController? scrollController}) {Key? key,
EdgeInsets? menuPadding,
ScrollController? scrollController,
required List<Peer> initPeers})
: super( : super(
key: key, key: key,
name: 'address book peer', name: 'address book peer',
loadEvent: 'load_address_book_peers', loadEvent: 'load_address_book_peers',
offstageFunc: (Peer peer) => peerCardBuilder: (Peer peer) => Obx(() => Offstage(
!_hitTag(gFFI.abModel.selectedTags, peer.tags), key: ValueKey("off${peer.id}"),
peerCardBuilder: (Peer peer) => AddressBookPeerCard( offstage: !_hitTag(gFFI.abModel.selectedTags, peer.tags),
child: AddressBookPeerCard(
peer: peer, peer: peer,
menuPadding: menuPadding, menuPadding: menuPadding,
), ))),
initPeers: _loadPeers(), initPeers: initPeers,
); );
static List<Peer> _loadPeers() {
debugPrint("_loadPeers : ${gFFI.abModel.peers.toString()}");
return gFFI.abModel.peers.map((e) {
return Peer.fromJson(e);
}).toList();
}
static bool _hitTag(List<dynamic> selectedTags, List<dynamic> idents) { static bool _hitTag(List<dynamic> selectedTags, List<dynamic> idents) {
if (selectedTags.isEmpty) { if (selectedTags.isEmpty) {
return true; return true;
@ -318,11 +297,4 @@ class AddressBookPeersView extends BasePeersView {
} }
return true; return true;
} }
@override
Widget build(BuildContext context) {
final widget = super.build(context);
// gFFI.abModel.updateAb();
return widget;
}
} }

View File

@ -32,6 +32,7 @@ const kDefaultMouseWheelThrottleDuration = Duration(milliseconds: 50);
const kFullScreenEdgeSize = 0.0; const kFullScreenEdgeSize = 0.0;
var kWindowEdgeSize = Platform.isWindows ? 1.0 : 5.0; var kWindowEdgeSize = Platform.isWindows ? 1.0 : 5.0;
const kWindowBorderWidth = 1.0; const kWindowBorderWidth = 1.0;
const kDesktopMenuPadding = EdgeInsets.only(left: 12.0, right: 3.0);
const kInvalidValueStr = "InvalidValueStr"; const kInvalidValueStr = "InvalidValueStr";

View File

@ -86,16 +86,16 @@ class _ConnectionPageState extends State<ConnectionPage>
], ],
children: [ children: [
RecentPeersView( RecentPeersView(
menuPadding: EdgeInsets.only(left: 12.0, right: 3.0), menuPadding: kDesktopMenuPadding,
), ),
FavoritePeersView( FavoritePeersView(
menuPadding: EdgeInsets.only(left: 12.0, right: 3.0), menuPadding: kDesktopMenuPadding,
), ),
DiscoveredPeersView( DiscoveredPeersView(
menuPadding: EdgeInsets.only(left: 12.0, right: 3.0), menuPadding: kDesktopMenuPadding,
), ),
const AddressBook( const AddressBook(
menuPadding: EdgeInsets.only(left: 12.0, right: 3.0), menuPadding: kDesktopMenuPadding,
), ),
], ],
).paddingOnly(right: 12.0), ).paddingOnly(right: 12.0),
@ -288,17 +288,23 @@ class _ConnectionPageState extends State<ConnectionPage>
children: [ children: [
light, light,
Text(translate('Ready'), style: textStyle), Text(translate('Ready'), style: textStyle),
Offstage(
offstage: !svcIsUsingPublicServer.value,
child: Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Text(', ', style: textStyle), Text(', ', style: textStyle),
svcIsUsingPublicServer.value InkWell(
? InkWell(
onTap: onUsePublicServerGuide, onTap: onUsePublicServerGuide,
child: Text( child: Text(
translate('setup_server_tip'), translate('setup_server_tip'),
style: TextStyle( style: TextStyle(
decoration: TextDecoration.underline, fontSize: fontSize), decoration: TextDecoration.underline,
fontSize: fontSize),
), ),
) )
: Offstage() ],
))
], ],
); );
} }

View File

@ -267,7 +267,6 @@ class _AppState extends State<App> {
ChangeNotifierProvider.value(value: gFFI.imageModel), ChangeNotifierProvider.value(value: gFFI.imageModel),
ChangeNotifierProvider.value(value: gFFI.cursorModel), ChangeNotifierProvider.value(value: gFFI.cursorModel),
ChangeNotifierProvider.value(value: gFFI.canvasModel), ChangeNotifierProvider.value(value: gFFI.canvasModel),
ChangeNotifierProvider.value(value: gFFI.abModel),
ChangeNotifierProvider.value(value: gFFI.userModel), ChangeNotifierProvider.value(value: gFFI.userModel),
], ],
child: GetMaterialApp( child: GetMaterialApp(

View File

@ -2,17 +2,18 @@ import 'dart:convert';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hbb/models/model.dart'; import 'package:flutter_hbb/models/model.dart';
import 'package:flutter_hbb/models/peer_model.dart';
import 'package:flutter_hbb/models/platform_model.dart'; import 'package:flutter_hbb/models/platform_model.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:http/http.dart' as http; import 'package:http/http.dart' as http;
import '../common.dart'; import '../common.dart';
class AbModel with ChangeNotifier { class AbModel {
var abLoading = false; var abLoading = false.obs;
var abError = ""; var abError = "".obs;
var tags = [].obs; var tags = [].obs;
var peers = [].obs; var peers = List<Peer>.empty(growable: true).obs;
var selectedTags = List<String>.empty(growable: true).obs; var selectedTags = List<String>.empty(growable: true).obs;
@ -22,11 +23,10 @@ class AbModel with ChangeNotifier {
FFI? get _ffi => parent.target; FFI? get _ffi => parent.target;
Future<dynamic> getAb() async { Future<dynamic> pullAb() async {
abLoading = true; abLoading.value = true;
notifyListeners();
// request // request
final api = "${await getApiServer()}/api/ab/get"; final api = "${await bind.mainGetApiServer()}/api/ab/get";
try { try {
final resp = final resp =
await http.post(Uri.parse(api), headers: await getHttpHeaders()); await http.post(Uri.parse(api), headers: await getHttpHeaders());
@ -37,38 +37,34 @@ class AbModel with ChangeNotifier {
} else if (json.containsKey('data')) { } else if (json.containsKey('data')) {
final data = jsonDecode(json['data']); final data = jsonDecode(json['data']);
tags.value = data['tags']; tags.value = data['tags'];
peers.value = data['peers']; peers.clear();
for (final peer in data['peers']) {
peers.add(Peer.fromJson(peer));
}
} }
notifyListeners();
return resp.body; return resp.body;
} else { } else {
return ""; return "";
} }
} catch (err) { } catch (err) {
abError = err.toString(); err.printError();
abError.value = err.toString();
} finally { } finally {
abLoading = false; abLoading.value = false;
notifyListeners();
} }
return null; return null;
} }
Future<String> getApiServer() async {
return await bind.mainGetApiServer();
}
void reset() { void reset() {
tags.clear(); tags.clear();
peers.clear(); peers.clear();
notifyListeners();
} }
void addId(String id) async { void addId(String id) async {
if (idContainBy(id)) { if (idContainBy(id)) {
return; return;
} }
peers.add({"id": id}); peers.add(Peer.fromJson({"id": id}));
notifyListeners();
} }
void addTag(String tag) async { void addTag(String tag) async {
@ -76,42 +72,40 @@ class AbModel with ChangeNotifier {
return; return;
} }
tags.add(tag); tags.add(tag);
notifyListeners();
} }
void changeTagForPeer(String id, List<dynamic> tags) { void changeTagForPeer(String id, List<dynamic> tags) {
final it = peers.where((element) => element['id'] == id); final it = peers.where((element) => element.id == id);
if (it.isEmpty) { if (it.isEmpty) {
return; return;
} }
it.first['tags'] = tags; it.first.tags = tags;
} }
Future<void> updateAb() async { Future<void> pushAb() async {
abLoading = true; abLoading.value = true;
notifyListeners(); final api = "${await bind.mainGetApiServer()}/api/ab";
final api = "${await getApiServer()}/api/ab";
var authHeaders = await getHttpHeaders(); var authHeaders = await getHttpHeaders();
authHeaders['Content-Type'] = "application/json"; authHeaders['Content-Type'] = "application/json";
final peersJsonData = peers.map((e) => e.toJson()).toList();
final body = jsonEncode({ final body = jsonEncode({
"data": jsonEncode({"tags": tags, "peers": peers}) "data": jsonEncode({"tags": tags, "peers": peersJsonData})
}); });
try { try {
final resp = final resp =
await http.post(Uri.parse(api), headers: authHeaders, body: body); await http.post(Uri.parse(api), headers: authHeaders, body: body);
abError = ""; abError.value = "";
await getAb(); await pullAb();
debugPrint("resp: ${resp.body}"); debugPrint("resp: ${resp.body}");
} catch (e) { } catch (e) {
abError = e.toString(); abError.value = e.toString();
} finally { } finally {
abLoading = false; abLoading.value = false;
} }
notifyListeners();
} }
bool idContainBy(String id) { bool idContainBy(String id) {
return peers.where((element) => element['id'] == id).isNotEmpty; return peers.where((element) => element.id == id).isNotEmpty;
} }
bool tagContainBy(String tag) { bool tagContainBy(String tag) {
@ -119,50 +113,47 @@ class AbModel with ChangeNotifier {
} }
void deletePeer(String id) { void deletePeer(String id) {
peers.removeWhere((element) => element['id'] == id); peers.removeWhere((element) => element.id == id);
notifyListeners();
} }
void deleteTag(String tag) { void deleteTag(String tag) {
gFFI.abModel.selectedTags.remove(tag);
tags.removeWhere((element) => element == tag); tags.removeWhere((element) => element == tag);
for (var peer in peers) { for (var peer in peers) {
if (peer['tags'] == null) { if (peer.tags.isEmpty) {
continue; continue;
} }
if (((peer['tags']) as List<dynamic>).contains(tag)) { if (peer.tags.contains(tag)) {
((peer['tags']) as List<dynamic>).remove(tag); ((peer.tags)).remove(tag);
} }
} }
notifyListeners();
} }
void unsetSelectedTags() { void unsetSelectedTags() {
selectedTags.clear(); selectedTags.clear();
notifyListeners();
} }
List<dynamic> getPeerTags(String id) { List<dynamic> getPeerTags(String id) {
final it = peers.where((p0) => p0['id'] == id); final it = peers.where((p0) => p0.id == id);
if (it.isEmpty) { if (it.isEmpty) {
return []; return [];
} else { } else {
return it.first['tags'] ?? []; return it.first.tags;
} }
} }
void setPeerOption(String id, String key, String value) { void setPeerAlias(String id, String value) {
final it = peers.where((p0) => p0['id'] == id); final it = peers.where((p0) => p0.id == id);
if (it.isEmpty) { if (it.isEmpty) {
debugPrint("${id} is not exists"); debugPrint("$id is not exists");
return; return;
} else { } else {
it.first[key] = value; it.first.alias = value;
} }
} }
void clear() { void clear() {
peers.clear(); peers.clear();
tags.clear(); tags.clear();
notifyListeners();
} }
} }

View File

@ -7,8 +7,8 @@ class Peer {
final String username; final String username;
final String hostname; final String hostname;
final String platform; final String platform;
final String alias; String alias;
final List<dynamic> tags; List<dynamic> tags;
bool online = false; bool online = false;
Peer.fromJson(Map<String, dynamic> json) Peer.fromJson(Map<String, dynamic> json)
@ -19,6 +19,17 @@ class Peer {
alias = json['alias'] ?? '', alias = json['alias'] ?? '',
tags = json['tags'] ?? []; tags = json['tags'] ?? [];
Map<String, dynamic> toJson() {
return <String, dynamic>{
"id": id,
"username": username,
"hostname": hostname,
"platform": platform,
"alias": alias,
"tags": tags,
};
}
Peer({ Peer({
required this.id, required this.id,
required this.username, required this.username,