mobile_nebula/lib/screens/SiteTunnelsScreen.dart

145 lines
4.1 KiB
Dart
Raw Normal View History

2020-07-27 20:43:58 +00:00
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:mobile_nebula/components/SimplePage.dart';
import 'package:mobile_nebula/components/config/ConfigPageItem.dart';
import 'package:mobile_nebula/components/config/ConfigSection.dart';
import 'package:mobile_nebula/models/HostInfo.dart';
import 'package:mobile_nebula/models/Site.dart';
import 'package:mobile_nebula/screens/HostInfoScreen.dart';
import 'package:mobile_nebula/services/utils.dart';
import 'package:pull_to_refresh/pull_to_refresh.dart';
class SiteTunnelsScreen extends StatefulWidget {
const SiteTunnelsScreen({
Key? key,
required this.site,
required this.tunnels,
required this.pending,
required this.onChanged,
required this.supportsQRScanning,
}) : super(key: key);
2020-07-27 20:43:58 +00:00
final Site site;
final List<HostInfo> tunnels;
final bool pending;
final Function(List<HostInfo>)? onChanged;
2020-07-27 20:43:58 +00:00
final bool supportsQRScanning;
2020-07-27 20:43:58 +00:00
@override
_SiteTunnelsScreenState createState() => _SiteTunnelsScreenState();
}
class _SiteTunnelsScreenState extends State<SiteTunnelsScreen> {
late Site site;
late List<HostInfo> tunnels;
2020-07-27 20:43:58 +00:00
RefreshController refreshController = RefreshController(initialRefresh: false);
@override
void initState() {
site = widget.site;
tunnels = widget.tunnels;
2020-07-27 20:43:58 +00:00
_sortTunnels();
super.initState();
}
@override
void dispose() {
refreshController.dispose();
2020-07-27 20:43:58 +00:00
super.dispose();
}
@override
Widget build(BuildContext context) {
final double ipWidth = Utils.textSize("000.000.000.000", CupertinoTheme.of(context).textTheme.textStyle).width + 32;
List<Widget> children = [];
tunnels.forEach((hostInfo) {
Widget icon;
final isLh = site.staticHostmap[hostInfo.vpnIp]?.lighthouse ?? false;
if (isLh) {
icon = Icon(Icons.lightbulb_outline, color: CupertinoColors.placeholderText.resolveFrom(context));
} else {
icon = Icon(Icons.computer, color: CupertinoColors.placeholderText.resolveFrom(context));
}
children.add(ConfigPageItem(
onPressed: () => Utils.openPage(
context,
(context) => HostInfoScreen(
isLighthouse: isLh,
hostInfo: hostInfo,
pending: widget.pending,
site: widget.site,
onChanged: () {
_listHostmap();
},
supportsQRScanning: widget.supportsQRScanning,
),
),
2020-07-27 20:43:58 +00:00
label: Row(children: <Widget>[Padding(child: icon, padding: EdgeInsets.only(right: 10)), Text(hostInfo.vpnIp)]),
labelWidth: ipWidth,
content: Container(alignment: Alignment.centerRight, child: Text(hostInfo.cert?.details.name ?? "")),
2020-07-27 20:43:58 +00:00
));
});
Widget child;
if (children.length == 0) {
child = Center(child: Padding(child: Text('No tunnels to show'), padding: EdgeInsets.only(top: 30)));
} else {
child = ConfigSection(children: children);
}
final title = widget.pending ? 'Pending' : 'Active';
return SimplePage(
title: Text('$title Tunnels'),
2020-07-27 20:43:58 +00:00
leadingAction: Utils.leadingBackWidget(context, onPressed: () {
Navigator.pop(context);
}),
refreshController: refreshController,
onRefresh: () async {
await _listHostmap();
refreshController.refreshCompleted();
},
child: child);
}
_sortTunnels() {
tunnels.sort((a, b) {
final aLh = _isLighthouse(a.vpnIp), bLh = _isLighthouse(b.vpnIp);
if (aLh && !bLh) {
return -1;
} else if (!aLh && bLh) {
return 1;
}
return Utils.ip2int(a.vpnIp) - Utils.ip2int(b.vpnIp);
});
}
bool _isLighthouse(String vpnIp) {
return site.staticHostmap[vpnIp]?.lighthouse ?? false;
}
_listHostmap() async {
try {
if (widget.pending) {
tunnels = await site.listPendingHostmap();
} else {
tunnels = await site.listHostmap();
}
_sortTunnels();
if (widget.onChanged != null) {
widget.onChanged!(tunnels);
2020-07-27 20:43:58 +00:00
}
setState(() {});
} catch (err) {
Utils.popError(context, 'Error while fetching hostmap', err.toString());
2020-07-27 20:43:58 +00:00
}
}
}