new: iperf

This commit is contained in:
lollipopkit
2024-01-21 18:53:17 +08:00
parent 50d6ed919b
commit 1434556e0b
34 changed files with 432 additions and 117 deletions

View File

@@ -33,7 +33,7 @@ class BackupPage extends StatelessWidget {
Widget build(BuildContext context) {
return Scaffold(
appBar: CustomAppBar(
title: Text(l10n.backupAndRestore, style: UIs.text18),
title: Text(l10n.backup, style: UIs.text18),
),
body: _buildBody(context),
);

View File

@@ -232,7 +232,7 @@ class _HomePageState extends State<HomePage>
),
ListTile(
leading: const Icon(Icons.import_export),
title: Text(l10n.backupAndRestore),
title: Text(l10n.backup),
onTap: () => AppRoute.backup().go(context),
),
ListTile(
@@ -286,7 +286,7 @@ ${GithubIds.contributors.map((e) => '[$e](${e.url})').join(' ')}
#### Participants
${GithubIds.participants.map((e) => '[$e](${e.url})').join(' ')}
#### My apps
#### My other apps
- [GPT Box](https://github.com/lollipopkit/flutter_gpt_box)
''',
),

67
lib/view/page/iperf.dart Normal file
View File

@@ -0,0 +1,67 @@
import 'package:flutter/material.dart';
import 'package:toolbox/core/extension/context/locale.dart';
import 'package:toolbox/core/extension/context/snackbar.dart';
import 'package:toolbox/core/route.dart';
import 'package:toolbox/data/model/server/server_private_info.dart';
import 'package:toolbox/view/widget/appbar.dart';
import 'package:toolbox/view/widget/input_field.dart';
class IPerfPage extends StatefulWidget {
final ServerPrivateInfo spi;
const IPerfPage({super.key, required this.spi});
@override
State<IPerfPage> createState() => _IPerfPageState();
}
class _IPerfPageState extends State<IPerfPage> {
final _hostCtrl = TextEditingController();
final _portCtrl = TextEditingController();
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: const CustomAppBar(
title: Text('iperf'),
),
body: _buildBody(),
floatingActionButton: _buildFAB(),
);
}
Widget _buildFAB() {
return FloatingActionButton(
heroTag: 'iperf',
child: const Icon(Icons.send),
onPressed: () {
if (_hostCtrl.text.isEmpty || _portCtrl.text.isEmpty) {
context.showSnackBar(l10n.fieldMustNotEmpty);
return;
}
AppRoute.ssh(
spi: widget.spi,
initCmd: 'iperf -c ${_hostCtrl.text} -p ${_portCtrl.text}',
).go(context);
},
);
}
Widget _buildBody() {
return ListView(
padding: const EdgeInsets.symmetric(horizontal: 17),
children: [
Input(
controller: _hostCtrl,
label: l10n.host,
icon: Icons.computer,
),
Input(
controller: _portCtrl,
label: l10n.port,
type: TextInputType.number,
icon: Icons.numbers,
),
],
);
}
}

View File

@@ -452,7 +452,7 @@ class _ServerDetailPageState extends State<ServerDetailPage>
Widget _buildDiskItem(Disk disk, ServerStatus ss) {
final (read, write) = ss.diskIO.getSpeed(disk.dev);
final text = () {
final use = '${disk.used.kb2Str} / ${disk.size.kb2Str}';
final use = '${l10n.used} ${disk.avail.kb2Str} / ${disk.size.kb2Str}';
if (read == null || write == null) return use;
return '$use\n${l10n.read} $read | ${l10n.write} $write';
}();

View File

@@ -206,7 +206,8 @@ class _SettingPageState extends State<SettingPage> {
children: [
_buildCollapseUI(),
_buildServerFuncBtns(),
_buildSequence(),
_buildServerSeq(),
_buildServerDetailCardSeq(),
_buildNetViewType(),
_buildUpdateInterval(),
_buildMaxRetry(),
@@ -985,6 +986,16 @@ class _SettingPageState extends State<SettingPage> {
}
Widget _buildServerFuncBtns() {
return ExpandTile(
title: Text(l10n.serverFuncBtns),
children: [
_buildServerFuncBtnsSwitch(),
_buildServerFuncBtnsOrder(),
],
);
}
Widget _buildServerFuncBtnsSwitch() {
return ListTile(
title: Text(l10n.location),
subtitle: Text(l10n.moveOutServerFuncBtnsHelp, style: UIs.text13Grey),
@@ -992,26 +1003,28 @@ class _SettingPageState extends State<SettingPage> {
);
}
Widget _buildSequence() {
return ExpandTile(
Widget _buildServerFuncBtnsOrder() {
return ListTile(
title: Text(l10n.sequence),
subtitle: Text(
'${l10n.serverOrder} / ${l10n.serverDetailOrder} ...',
style: UIs.textGrey,
),
children: [
ListTile(
title: Text(l10n.serverOrder),
trailing: const Icon(Icons.keyboard_arrow_right),
onTap: () => AppRoute.serverOrder().go(context),
),
ListTile(
trailing: const Icon(Icons.keyboard_arrow_right),
onTap: () => AppRoute.serverFuncBtnsOrder().go(context),
);
}
Widget _buildServerSeq() {
return ListTile(
title: Text(l10n.serverOrder),
trailing: const Icon(Icons.keyboard_arrow_right),
onTap: () => AppRoute.serverOrder().go(context),
);
}
Widget _buildServerDetailCardSeq() {
return ListTile(
title: Text(l10n.serverDetailOrder),
trailing: const Icon(Icons.keyboard_arrow_right),
onTap: () => AppRoute.serverDetailOrder().go(context),
),
],
);
);
}
Widget _buildEditorFontSize() {

View File

@@ -9,7 +9,7 @@ import 'package:toolbox/core/extension/context/snackbar.dart';
import 'package:toolbox/core/utils/platform/auth.dart';
import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/data/res/ui.dart';
import 'package:toolbox/view/page/setting/platform_pub.dart';
import 'package:toolbox/view/page/setting/platform/platform_pub.dart';
import 'package:toolbox/view/widget/appbar.dart';
import 'package:toolbox/view/widget/input_field.dart';
import 'package:toolbox/view/widget/cardx.dart';

View File

@@ -12,7 +12,7 @@ import 'package:toolbox/data/res/logger.dart';
import 'package:toolbox/data/res/misc.dart';
import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/data/res/ui.dart';
import 'package:toolbox/view/page/setting/platform_pub.dart';
import 'package:toolbox/view/page/setting/platform/platform_pub.dart';
import 'package:toolbox/view/widget/appbar.dart';
import 'package:toolbox/view/widget/future_widget.dart';
import 'package:toolbox/view/widget/cardx.dart';

View File

@@ -5,9 +5,9 @@ import 'package:toolbox/core/utils/platform/base.dart';
import 'package:toolbox/data/res/default.dart';
import 'package:toolbox/data/res/store.dart';
import '../../../core/extension/order.dart';
import '../../widget/appbar.dart';
import '../../widget/cardx.dart';
import '../../../../core/extension/order.dart';
import '../../../widget/appbar.dart';
import '../../../widget/cardx.dart';
class ServerDetailOrderPage extends StatefulWidget {
const ServerDetailOrderPage({super.key});

View File

@@ -0,0 +1,88 @@
import 'package:flutter/material.dart';
import 'package:toolbox/core/extension/context/locale.dart';
import 'package:toolbox/core/extension/context/snackbar.dart';
import 'package:toolbox/core/utils/platform/base.dart';
import 'package:toolbox/data/model/app/menu/server_func.dart';
import 'package:toolbox/data/res/store.dart';
import '../../../../core/extension/order.dart';
import '../../../widget/appbar.dart';
import '../../../widget/cardx.dart';
class ServerFuncBtnsOrderPage extends StatefulWidget {
const ServerFuncBtnsOrderPage({super.key});
@override
State<ServerFuncBtnsOrderPage> createState() => _ServerDetailOrderPageState();
}
class _ServerDetailOrderPageState extends State<ServerFuncBtnsOrderPage> {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: CustomAppBar(
title: Text(l10n.serverDetailOrder),
),
body: _buildBody(),
);
}
Widget _buildBody() {
final keys_ = Stores.setting.serverFuncBtns.fetch();
final keys = <ServerFuncBtn>[];
for (final key in keys_) {
keys.add(key);
}
final disabled =
ServerFuncBtn.values.where((e) => !keys.contains(e)).toList();
final allKeys = [...keys, ...disabled];
return ReorderableListView.builder(
padding: const EdgeInsets.all(7),
itemBuilder: (_, idx) {
final key = allKeys[idx];
return CardX(
key: ValueKey(idx),
child: ListTile(
title: Text(key.toStr),
leading: _buildCheckBox(keys, key, idx, idx < keys.length),
trailing: isDesktop ? null : const Icon(Icons.drag_handle),
),
);
},
itemCount: allKeys.length,
onReorder: (o, n) {
if (o >= keys.length || n >= keys.length) {
context.showSnackBar(l10n.disabled);
return;
}
keys.moveByItem(keys, o, n, property: Stores.setting.serverFuncBtns);
setState(() {});
},
);
}
Widget _buildCheckBox(
List<ServerFuncBtn> keys,
ServerFuncBtn key,
int idx,
bool value,
) {
return Checkbox(
value: value,
onChanged: (val) {
if (val == null) return;
if (val) {
if (idx >= keys.length) {
keys.add(key);
} else {
keys.insert(idx - 1, key);
}
} else {
keys.remove(key);
}
Stores.setting.serverFuncBtns.put(keys);
setState(() {});
},
);
}
}

View File

@@ -6,7 +6,7 @@ import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/data/res/ui.dart';
import 'package:toolbox/view/widget/cardx.dart';
import '../../widget/appbar.dart';
import '../../../widget/appbar.dart';
class ServerOrderPage extends StatefulWidget {
const ServerOrderPage({super.key});

View File

@@ -8,7 +8,7 @@ import 'package:toolbox/data/res/store.dart';
import 'package:toolbox/data/res/ui.dart';
import 'package:toolbox/view/widget/cardx.dart';
import '../../widget/appbar.dart';
import '../../../widget/appbar.dart';
class SSHVirtKeySettingPage extends StatefulWidget {
const SSHVirtKeySettingPage({super.key});

View File

@@ -180,9 +180,8 @@ class _LocalStoragePageState extends State<LocalStoragePage> {
? const Icon(Icons.folder_open)
: const Icon(Icons.insert_drive_file),
title: Text(fileName),
subtitle: isDir
? null
: Text(stat.size.bytes2Str, style: UIs.textGrey),
subtitle:
isDir ? null : Text(stat.size.bytes2Str, style: UIs.textGrey),
trailing: Text(
stat.modified
.toString()

View File

@@ -16,6 +16,7 @@ import 'package:toolbox/data/model/server/dist.dart';
import 'package:toolbox/data/model/server/snippet.dart';
import 'package:toolbox/data/res/path.dart';
import 'package:toolbox/data/res/provider.dart';
import 'package:toolbox/data/res/store.dart';
import '../../core/route.dart';
import '../../core/utils/server.dart';
@@ -33,9 +34,9 @@ class ServerFuncBtnsTopRight extends StatelessWidget {
@override
Widget build(BuildContext context) {
return PopupMenu<ServerTabMenu>(
items: ServerTabMenu.values
.map((e) => PopupMenuItem<ServerTabMenu>(
return PopupMenu<ServerFuncBtn>(
items: ServerFuncBtn.values
.map((e) => PopupMenuItem<ServerFuncBtn>(
value: e,
child: Row(
children: [
@@ -94,7 +95,7 @@ class ServerFuncBtns extends StatelessWidget {
// );
return Row(
mainAxisAlignment: MainAxisAlignment.spaceAround,
children: ServerTabMenu.values
children: Stores.setting.serverFuncBtns.fetch()
.map(
(e) => IconButton(
onPressed: () => _onTapMoreBtns(e, spi, context),
@@ -109,21 +110,21 @@ class ServerFuncBtns extends StatelessWidget {
}
void _onTapMoreBtns(
ServerTabMenu value,
ServerFuncBtn value,
ServerPrivateInfo spi,
BuildContext context,
) async {
switch (value) {
case ServerTabMenu.pkg:
case ServerFuncBtn.pkg:
_onPkg(context, spi);
break;
case ServerTabMenu.sftp:
case ServerFuncBtn.sftp:
AppRoute.sftp(spi: spi).checkGo(
context: context,
check: () => _checkClient(context, spi.id),
);
break;
case ServerTabMenu.snippet:
case ServerFuncBtn.snippet:
final snippet = await context.showPickSingleDialog<Snippet>(
items: Pros.snippet.snippets,
name: (e) => e.name,
@@ -135,21 +136,27 @@ void _onTapMoreBtns(
check: () => _checkClient(context, spi.id),
);
break;
case ServerTabMenu.container:
case ServerFuncBtn.container:
AppRoute.docker(spi: spi).checkGo(
context: context,
check: () => _checkClient(context, spi.id),
);
break;
case ServerTabMenu.process:
case ServerFuncBtn.process:
AppRoute.process(spi: spi).checkGo(
context: context,
check: () => _checkClient(context, spi.id),
);
break;
case ServerTabMenu.terminal:
case ServerFuncBtn.terminal:
_gotoSSH(spi, context);
break;
case ServerFuncBtn.iperf:
AppRoute.iperf(spi: spi).checkGo(
context: context,
check: () => _checkClient(context, spi.id),
);
break;
}
}