mirror of
https://github.com/lollipopkit/flutter_server_box.git
synced 2025-12-17 07:14:28 +01:00
使用单独的页面编辑服务器信息
This commit is contained in:
@@ -104,9 +104,9 @@ class _ConvertPageState extends State<ConvertPage>
|
|||||||
child: ExpansionTile(
|
child: ExpansionTile(
|
||||||
leading: TextButton(
|
leading: TextButton(
|
||||||
child: SizedBox(
|
child: SizedBox(
|
||||||
width: _media.size.width * 0.2,
|
width: _media.size.width * 0.3,
|
||||||
child: Row(
|
child: Row(
|
||||||
children: const [Icon(Icons.change_circle), Text('上下交换')],
|
children: const [Icon(Icons.change_circle), Text('Upside down')],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import 'package:toolbox/data/res/build_data.dart';
|
|||||||
import 'package:toolbox/locator.dart';
|
import 'package:toolbox/locator.dart';
|
||||||
import 'package:toolbox/view/page/convert.dart';
|
import 'package:toolbox/view/page/convert.dart';
|
||||||
import 'package:toolbox/view/page/debug.dart';
|
import 'package:toolbox/view/page/debug.dart';
|
||||||
import 'package:toolbox/view/page/server.dart';
|
import 'package:toolbox/view/page/server/server_tab.dart';
|
||||||
|
|
||||||
class MyHomePage extends StatefulWidget {
|
class MyHomePage extends StatefulWidget {
|
||||||
const MyHomePage({Key? key, required this.title}) : super(key: key);
|
const MyHomePage({Key? key, required this.title}) : super(key: key);
|
||||||
|
|||||||
149
lib/view/page/server/server_edit.dart
Normal file
149
lib/view/page/server/server_edit.dart
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
import 'package:after_layout/after_layout.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:toolbox/data/model/server_private_info.dart';
|
||||||
|
import 'package:toolbox/data/provider/server.dart';
|
||||||
|
import 'package:toolbox/locator.dart';
|
||||||
|
|
||||||
|
class ServerEditPage extends StatefulWidget {
|
||||||
|
const ServerEditPage({Key? key, this.spi}) : super(key: key);
|
||||||
|
|
||||||
|
final ServerPrivateInfo? spi;
|
||||||
|
|
||||||
|
@override
|
||||||
|
_ServerEditPageState createState() => _ServerEditPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ServerEditPageState extends State<ServerEditPage> with AfterLayoutMixin {
|
||||||
|
final nameController = TextEditingController();
|
||||||
|
final ipController = TextEditingController();
|
||||||
|
final portController = TextEditingController();
|
||||||
|
final usernameController = TextEditingController();
|
||||||
|
final passwordController = TextEditingController();
|
||||||
|
final keyController = TextEditingController();
|
||||||
|
|
||||||
|
late ServerProvider _serverProvider;
|
||||||
|
|
||||||
|
bool usePublicKey = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_serverProvider = locator<ServerProvider>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(title: const Text('Edit')),
|
||||||
|
body: SingleChildScrollView(
|
||||||
|
padding: const EdgeInsets.all(17),
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
TextField(
|
||||||
|
controller: nameController,
|
||||||
|
keyboardType: TextInputType.text,
|
||||||
|
decoration: _buildDecoration('Name', icon: Icons.info),
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: ipController,
|
||||||
|
keyboardType: TextInputType.text,
|
||||||
|
decoration: _buildDecoration('Host', icon: Icons.storage),
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: portController,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
decoration:
|
||||||
|
_buildDecoration('Port', icon: Icons.format_list_numbered),
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: usernameController,
|
||||||
|
keyboardType: TextInputType.text,
|
||||||
|
decoration: _buildDecoration('User', icon: Icons.account_box),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 7),
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
const Text('Public Key Auth'),
|
||||||
|
Switch(
|
||||||
|
value: usePublicKey,
|
||||||
|
onChanged: (val) => setState(() => usePublicKey = val)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
usePublicKey
|
||||||
|
? TextField(
|
||||||
|
controller: keyController,
|
||||||
|
keyboardType: TextInputType.text,
|
||||||
|
autocorrect: false,
|
||||||
|
maxLines: 10,
|
||||||
|
minLines: 5,
|
||||||
|
decoration:
|
||||||
|
_buildDecoration('Private Key', icon: Icons.vpn_key),
|
||||||
|
onSubmitted: (_) => {},
|
||||||
|
)
|
||||||
|
: const SizedBox(),
|
||||||
|
TextField(
|
||||||
|
controller: passwordController,
|
||||||
|
obscureText: true,
|
||||||
|
keyboardType: TextInputType.text,
|
||||||
|
decoration: _buildDecoration('Pwd', icon: Icons.password),
|
||||||
|
onSubmitted: (_) => {},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
floatingActionButton: FloatingActionButton(
|
||||||
|
child: const Icon(Icons.send),
|
||||||
|
onPressed: () {
|
||||||
|
final authorization = keyController.text.isEmpty
|
||||||
|
? passwordController.text
|
||||||
|
: {
|
||||||
|
"privateKey": keyController.text,
|
||||||
|
"passphrase": passwordController.text
|
||||||
|
};
|
||||||
|
final spi = ServerPrivateInfo(
|
||||||
|
name: nameController.text,
|
||||||
|
ip: ipController.text,
|
||||||
|
port: int.parse(portController.text),
|
||||||
|
user: usernameController.text,
|
||||||
|
authorization: authorization);
|
||||||
|
|
||||||
|
if (widget.spi == null) {
|
||||||
|
_serverProvider.addServer(spi);
|
||||||
|
} else {
|
||||||
|
_serverProvider.updateServer(widget.spi!, spi);
|
||||||
|
}
|
||||||
|
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
InputDecoration _buildDecoration(String label,
|
||||||
|
{TextStyle? textStyle, IconData? icon}) {
|
||||||
|
return InputDecoration(
|
||||||
|
labelText: label, labelStyle: textStyle, icon: Icon(icon));
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void afterFirstLayout(BuildContext context) {
|
||||||
|
if (widget.spi != null) {
|
||||||
|
nameController.text = widget.spi?.name ?? '';
|
||||||
|
ipController.text = widget.spi?.ip ?? '';
|
||||||
|
portController.text = (widget.spi?.port ?? 22).toString();
|
||||||
|
usernameController.text = widget.spi?.user ?? '';
|
||||||
|
if (widget.spi?.authorization is String) {
|
||||||
|
passwordController.text = widget.spi?.authorization as String? ?? '';
|
||||||
|
} else {
|
||||||
|
final auth = widget.spi?.authorization as Map;
|
||||||
|
passwordController.text = auth['passphrase'];
|
||||||
|
keyController.text = auth['privateKey'];
|
||||||
|
setState(() {
|
||||||
|
usePublicKey = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,11 +5,12 @@ import 'package:flutter/widgets.dart';
|
|||||||
import 'package:flutter_staggered_animations/flutter_staggered_animations.dart';
|
import 'package:flutter_staggered_animations/flutter_staggered_animations.dart';
|
||||||
import 'package:get_it/get_it.dart';
|
import 'package:get_it/get_it.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
import 'package:toolbox/core/utils.dart';
|
import 'package:toolbox/core/route.dart';
|
||||||
import 'package:toolbox/data/model/server_private_info.dart';
|
import 'package:toolbox/data/model/server_private_info.dart';
|
||||||
import 'package:toolbox/data/model/server_status.dart';
|
import 'package:toolbox/data/model/server_status.dart';
|
||||||
import 'package:toolbox/data/provider/server.dart';
|
import 'package:toolbox/data/provider/server.dart';
|
||||||
import 'package:toolbox/locator.dart';
|
import 'package:toolbox/locator.dart';
|
||||||
|
import 'package:toolbox/view/page/server/server_edit.dart';
|
||||||
import 'package:toolbox/view/widget/circle_pie.dart';
|
import 'package:toolbox/view/widget/circle_pie.dart';
|
||||||
|
|
||||||
class ServerPage extends StatefulWidget {
|
class ServerPage extends StatefulWidget {
|
||||||
@@ -23,22 +24,13 @@ class _ServerPageState extends State<ServerPage>
|
|||||||
with AutomaticKeepAliveClientMixin, AfterLayoutMixin {
|
with AutomaticKeepAliveClientMixin, AfterLayoutMixin {
|
||||||
late MediaQueryData _media;
|
late MediaQueryData _media;
|
||||||
late ThemeData _theme;
|
late ThemeData _theme;
|
||||||
bool useKey = false;
|
|
||||||
|
|
||||||
final nameController = TextEditingController();
|
late ServerProvider _serverProvider;
|
||||||
final ipController = TextEditingController();
|
|
||||||
final portController = TextEditingController();
|
|
||||||
final usernameController = TextEditingController();
|
|
||||||
final passwordController = TextEditingController();
|
|
||||||
final keyController = TextEditingController();
|
|
||||||
|
|
||||||
late ServerProvider serverProvider;
|
|
||||||
final cachedServerStatus = <ServerStatus?>[];
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
serverProvider = locator<ServerProvider>();
|
_serverProvider = locator<ServerProvider>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -74,7 +66,9 @@ class _ServerPageState extends State<ServerPage>
|
|||||||
})),
|
})),
|
||||||
),
|
),
|
||||||
floatingActionButton: FloatingActionButton(
|
floatingActionButton: FloatingActionButton(
|
||||||
onPressed: () => showAddServerDialog(),
|
onPressed: () =>
|
||||||
|
AppRoute(const ServerEditPage(), 'Add server info page')
|
||||||
|
.go(context),
|
||||||
tooltip: 'add a server',
|
tooltip: 'add a server',
|
||||||
heroTag: 'server page fab',
|
heroTag: 'server page fab',
|
||||||
child: const Icon(Icons.add),
|
child: const Icon(Icons.add),
|
||||||
@@ -82,151 +76,16 @@ class _ServerPageState extends State<ServerPage>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void showAddServerDialog() {
|
|
||||||
showRoundDialog(context, 'New', _buildTextInputField(context), [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
clearTextField();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: const Text('Close')),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
final authorization = keyController.text.isEmpty
|
|
||||||
? passwordController.text
|
|
||||||
: {
|
|
||||||
"privateKey": keyController.text,
|
|
||||||
"passphrase": passwordController.text
|
|
||||||
};
|
|
||||||
serverProvider.addServer(ServerPrivateInfo(
|
|
||||||
name: nameController.text,
|
|
||||||
ip: ipController.text,
|
|
||||||
port: int.parse(portController.text),
|
|
||||||
user: usernameController.text,
|
|
||||||
authorization: authorization));
|
|
||||||
clearTextField();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: const Text('Connect'))
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
InputDecoration _buildDecoration(String label, {TextStyle? textStyle}) {
|
|
||||||
return InputDecoration(labelText: label, labelStyle: textStyle);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildTextInputField(BuildContext ctx) {
|
|
||||||
return SingleChildScrollView(
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
TextField(
|
|
||||||
controller: nameController,
|
|
||||||
keyboardType: TextInputType.text,
|
|
||||||
decoration: _buildDecoration('Name'),
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: ipController,
|
|
||||||
keyboardType: TextInputType.text,
|
|
||||||
decoration: _buildDecoration('Host'),
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: portController,
|
|
||||||
keyboardType: TextInputType.number,
|
|
||||||
decoration: _buildDecoration('Port'),
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: usernameController,
|
|
||||||
keyboardType: TextInputType.text,
|
|
||||||
decoration: _buildDecoration('User'),
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: keyController,
|
|
||||||
keyboardType: TextInputType.text,
|
|
||||||
decoration: _buildDecoration('Key(Optional)'),
|
|
||||||
onSubmitted: (_) => {},
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: passwordController,
|
|
||||||
obscureText: true,
|
|
||||||
keyboardType: TextInputType.text,
|
|
||||||
decoration: _buildDecoration('Pwd'),
|
|
||||||
onSubmitted: (_) => {},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void clearTextField() {
|
|
||||||
nameController.clear();
|
|
||||||
ipController.clear();
|
|
||||||
portController.clear();
|
|
||||||
usernameController.clear();
|
|
||||||
passwordController.clear();
|
|
||||||
keyController.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildEachServerCard(ServerStatus ss, ServerPrivateInfo spi) {
|
Widget _buildEachServerCard(ServerStatus ss, ServerPrivateInfo spi) {
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
child: _buildEachCardContent(ss, spi),
|
child: _buildEachCardContent(ss, spi),
|
||||||
onLongPress: () {
|
onLongPress: () {
|
||||||
nameController.text = spi.name ?? '';
|
AppRoute(
|
||||||
ipController.text = spi.ip ?? '';
|
ServerEditPage(
|
||||||
portController.text = (spi.port ?? 22).toString();
|
spi: spi,
|
||||||
usernameController.text = spi.user ?? '';
|
),
|
||||||
if (spi.authorization is String) {
|
'Edit server info page')
|
||||||
passwordController.text = spi.authorization as String? ?? '';
|
.go(context);
|
||||||
} else {
|
|
||||||
final auth = spi.authorization as Map;
|
|
||||||
passwordController.text = auth['passphrase'];
|
|
||||||
keyController.text = auth['privateKey'];
|
|
||||||
}
|
|
||||||
|
|
||||||
showRoundDialog(
|
|
||||||
context,
|
|
||||||
'Edit',
|
|
||||||
_buildTextInputField(context),
|
|
||||||
[
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
clearTextField();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: const Text('Close')),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
final authorization = keyController.text.isEmpty
|
|
||||||
? passwordController.text
|
|
||||||
: {
|
|
||||||
"privateKey": keyController.text,
|
|
||||||
"passphrase": passwordController.text
|
|
||||||
};
|
|
||||||
serverProvider.updateServer(
|
|
||||||
spi,
|
|
||||||
ServerPrivateInfo(
|
|
||||||
name: nameController.text,
|
|
||||||
ip: ipController.text,
|
|
||||||
port: int.parse(portController.text),
|
|
||||||
user: usernameController.text,
|
|
||||||
authorization: authorization));
|
|
||||||
clearTextField();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: const Text('Save')),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
serverProvider.delServer(spi);
|
|
||||||
clearTextField();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
child: const Text(
|
|
||||||
'Delete',
|
|
||||||
style: TextStyle(color: Colors.red),
|
|
||||||
))
|
|
||||||
],
|
|
||||||
barrierDismiss: false);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -377,8 +236,8 @@ class _ServerPageState extends State<ServerPage>
|
|||||||
@override
|
@override
|
||||||
Future<void> afterFirstLayout(BuildContext context) async {
|
Future<void> afterFirstLayout(BuildContext context) async {
|
||||||
await GetIt.I.allReady();
|
await GetIt.I.allReady();
|
||||||
await serverProvider.loadLocalData();
|
await _serverProvider.loadLocalData();
|
||||||
await serverProvider.refreshData();
|
await _serverProvider.refreshData();
|
||||||
await serverProvider.startAutoRefresh();
|
await _serverProvider.startAutoRefresh();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user