Merge pull request #1687 from Heap-Hop/master
fix flutter desktop Address book
This commit is contained in:
commit
672d5f31d0
@ -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,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
@ -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)))));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -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";
|
||||||
|
|
||||||
|
@ -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()
|
],
|
||||||
|
))
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -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(
|
||||||
|
@ -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();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user