mirror of
https://github.com/lollipopkit/flutter_server_box.git
synced 2025-12-17 07:14:28 +01:00
353 lines
9.7 KiB
Dart
353 lines
9.7 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:flutter/material.dart';
|
|
import 'package:toolbox/core/extension/context/common.dart';
|
|
import 'package:toolbox/core/extension/context/dialog.dart';
|
|
import 'package:toolbox/core/extension/context/locale.dart';
|
|
import 'package:toolbox/core/extension/context/snackbar.dart';
|
|
import 'package:toolbox/core/utils/share.dart';
|
|
import 'package:toolbox/data/model/server/server_private_info.dart';
|
|
import 'package:toolbox/data/model/sftp/req.dart';
|
|
import 'package:toolbox/data/res/misc.dart';
|
|
import 'package:toolbox/data/res/provider.dart';
|
|
import 'package:toolbox/view/widget/input_field.dart';
|
|
import 'package:toolbox/view/widget/omit_start_text.dart';
|
|
import 'package:toolbox/view/widget/cardx.dart';
|
|
|
|
import '../../../core/extension/numx.dart';
|
|
import '../../../core/route.dart';
|
|
import '../../../core/utils/misc.dart';
|
|
import '../../../data/model/app/path_with_prefix.dart';
|
|
import '../../../data/res/path.dart';
|
|
import '../../../data/res/ui.dart';
|
|
import '../../widget/appbar.dart';
|
|
import '../../widget/fade_in.dart';
|
|
|
|
class LocalStoragePage extends StatefulWidget {
|
|
final bool isPickFile;
|
|
final String? initDir;
|
|
const LocalStoragePage({
|
|
super.key,
|
|
required this.isPickFile,
|
|
this.initDir,
|
|
});
|
|
|
|
@override
|
|
State<LocalStoragePage> createState() => _LocalStoragePageState();
|
|
}
|
|
|
|
class _LocalStoragePageState extends State<LocalStoragePage> {
|
|
LocalPath? _path;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
if (widget.initDir != null) {
|
|
setState(() {
|
|
_path = LocalPath(widget.initDir!);
|
|
});
|
|
} else {
|
|
Paths.sftp.then((dir) {
|
|
setState(() {
|
|
_path = LocalPath(dir);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Scaffold(
|
|
appBar: CustomAppBar(
|
|
leading: IconButton(
|
|
icon: const BackButtonIcon(),
|
|
onPressed: () {
|
|
if (_path != null) {
|
|
_path!.update('/');
|
|
}
|
|
context.pop();
|
|
},
|
|
),
|
|
title: Text(l10n.files),
|
|
actions: [
|
|
IconButton(
|
|
icon: const Icon(Icons.downloading),
|
|
onPressed: () => AppRoute.sftpMission().go(context),
|
|
)
|
|
],
|
|
),
|
|
body: FadeIn(
|
|
key: UniqueKey(),
|
|
child: _buildBody(),
|
|
),
|
|
bottomNavigationBar: SafeArea(child: _buildPath()),
|
|
);
|
|
}
|
|
|
|
Widget _buildPath() {
|
|
return Container(
|
|
padding: const EdgeInsets.fromLTRB(11, 7, 11, 11),
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
OmitStartText(_path?.path ?? l10n.loadingFiles),
|
|
_buildBtns(),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildBtns() {
|
|
return Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
children: [
|
|
IconButton(
|
|
onPressed: () {
|
|
_path?.update('..');
|
|
setState(() {});
|
|
},
|
|
icon: const Icon(Icons.arrow_back),
|
|
),
|
|
IconButton(
|
|
onPressed: () async {
|
|
final path = await pickOneFile();
|
|
if (path == null) return;
|
|
final name = getFileName(path) ?? 'imported';
|
|
await File(path).copy(pathJoin(_path!.path, name));
|
|
setState(() {});
|
|
},
|
|
icon: const Icon(Icons.add),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
Widget _buildBody() {
|
|
if (_path == null) {
|
|
return const Center(
|
|
child: CircularProgressIndicator(),
|
|
);
|
|
}
|
|
final dir = Directory(_path!.path);
|
|
final files = dir.listSync();
|
|
return ListView.builder(
|
|
itemCount: files.length,
|
|
padding: const EdgeInsets.symmetric(vertical: 3, horizontal: 7),
|
|
itemBuilder: (context, index) {
|
|
var file = files[index];
|
|
var fileName = file.path.split('/').last;
|
|
var stat = file.statSync();
|
|
var isDir = stat.type == FileSystemEntityType.directory;
|
|
|
|
return CardX(
|
|
child: ListTile(
|
|
leading: isDir
|
|
? const Icon(Icons.folder)
|
|
: const Icon(Icons.insert_drive_file),
|
|
title: Text(fileName),
|
|
subtitle: isDir
|
|
? null
|
|
: Text(stat.size.convertBytes, style: UIs.textGrey),
|
|
trailing: Text(
|
|
stat.modified
|
|
.toString()
|
|
.substring(0, stat.modified.toString().length - 4),
|
|
style: UIs.textGrey,
|
|
),
|
|
onLongPress: () {
|
|
if (!isDir) return;
|
|
_showDirActionDialog(file);
|
|
},
|
|
onTap: () async {
|
|
if (!isDir) {
|
|
await _showFileActionDialog(file);
|
|
return;
|
|
}
|
|
_path!.update(fileName);
|
|
setState(() {});
|
|
},
|
|
),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
Future<void> _showDirActionDialog(FileSystemEntity file) async {
|
|
context.showRoundDialog(
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
ListTile(
|
|
onTap: () {
|
|
context.pop();
|
|
_showRenameDialog(file);
|
|
},
|
|
title: Text(l10n.rename),
|
|
leading: const Icon(Icons.abc),
|
|
),
|
|
ListTile(
|
|
onTap: () {
|
|
context.pop();
|
|
_showDeleteDialog(file);
|
|
},
|
|
title: Text(l10n.delete),
|
|
leading: const Icon(Icons.delete),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
Future<void> _showFileActionDialog(FileSystemEntity file) async {
|
|
final fileName = file.path.split('/').last;
|
|
if (widget.isPickFile) {
|
|
await context.showRoundDialog(
|
|
title: Text(l10n.pickFile),
|
|
child: Text(fileName),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
context.pop();
|
|
context.pop(file.path);
|
|
},
|
|
child: Text(l10n.ok),
|
|
),
|
|
]);
|
|
return;
|
|
}
|
|
context.showRoundDialog(
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
ListTile(
|
|
leading: const Icon(Icons.edit),
|
|
title: Text(l10n.edit),
|
|
onTap: () async {
|
|
context.pop();
|
|
final stat = await file.stat();
|
|
if (stat.size > Miscs.editorMaxSize) {
|
|
context.showRoundDialog(
|
|
title: Text(l10n.attention),
|
|
child: Text(l10n.fileTooLarge(fileName, stat.size, '1m')),
|
|
);
|
|
return;
|
|
}
|
|
final result = await AppRoute.editor(
|
|
path: file.absolute.path,
|
|
).go<bool>(context);
|
|
if (result == true) {
|
|
context.showSnackBar(l10n.saved);
|
|
setState(() {});
|
|
}
|
|
},
|
|
),
|
|
ListTile(
|
|
leading: const Icon(Icons.abc),
|
|
title: Text(l10n.rename),
|
|
onTap: () {
|
|
context.pop();
|
|
_showRenameDialog(file);
|
|
},
|
|
),
|
|
ListTile(
|
|
leading: const Icon(Icons.delete),
|
|
title: Text(l10n.delete),
|
|
onTap: () {
|
|
context.pop();
|
|
_showDeleteDialog(file);
|
|
},
|
|
),
|
|
ListTile(
|
|
leading: const Icon(Icons.upload),
|
|
title: Text(l10n.upload),
|
|
onTap: () async {
|
|
context.pop();
|
|
|
|
final spi = await context.showPickSingleDialog<ServerPrivateInfo>(
|
|
items: Pros.server.serverOrder
|
|
.map((e) => Pros.server.pick(id: e)?.spi)
|
|
.toList(),
|
|
name: (e) => e.name,
|
|
);
|
|
if (spi == null) return;
|
|
|
|
final remotePath = await AppRoute.sftp(
|
|
spi: spi,
|
|
isSelect: true,
|
|
).go<String>(context);
|
|
if (remotePath == null) {
|
|
return;
|
|
}
|
|
|
|
Pros.sftp.add(SftpReq(
|
|
spi,
|
|
'$remotePath/$fileName',
|
|
file.absolute.path,
|
|
SftpReqType.upload,
|
|
));
|
|
context.showSnackBar(l10n.added2List);
|
|
},
|
|
),
|
|
ListTile(
|
|
leading: const Icon(Icons.open_in_new),
|
|
title: Text(l10n.open),
|
|
onTap: () {
|
|
Shares.files([file.absolute.path]);
|
|
},
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
void _showRenameDialog(FileSystemEntity file) {
|
|
final fileName = file.path.split('/').last;
|
|
context.showRoundDialog(
|
|
title: Text(l10n.rename),
|
|
child: Input(
|
|
autoFocus: true,
|
|
controller: TextEditingController(text: fileName),
|
|
onSubmitted: (p0) {
|
|
context.pop();
|
|
final newPath = '${file.parent.path}/$p0';
|
|
try {
|
|
file.renameSync(newPath);
|
|
} catch (e) {
|
|
context.showSnackBar('${l10n.failed}:\n$e');
|
|
return;
|
|
}
|
|
|
|
setState(() {});
|
|
},
|
|
),
|
|
);
|
|
}
|
|
|
|
void _showDeleteDialog(FileSystemEntity file) {
|
|
final fileName = file.path.split('/').last;
|
|
context.showRoundDialog(
|
|
title: Text(l10n.delete),
|
|
child: Text(l10n.askContinue('${l10n.delete} $fileName')),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () => context.pop(),
|
|
child: Text(l10n.cancel),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
context.pop();
|
|
try {
|
|
file.deleteSync(recursive: true);
|
|
} catch (e) {
|
|
context.showSnackBar('${l10n.failed}:\n$e');
|
|
return;
|
|
}
|
|
setState(() {});
|
|
},
|
|
child: Text(l10n.ok),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
}
|