delete operation, new folder, upload files
This commit is contained in:
parent
0740c4b82d
commit
eb723b74fd
4 changed files with 192 additions and 7 deletions
|
@ -1,4 +1,5 @@
|
|||
import 'package:dartssh2/dartssh2.dart';
|
||||
import 'package:file_selector/file_selector.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class SftpExplorer extends StatefulWidget {
|
||||
|
@ -15,7 +16,10 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
|||
|
||||
bool _isLoading = true;
|
||||
late List<SftpName> _dirContents;
|
||||
|
||||
|
||||
SftpFileWriter? _loader;
|
||||
String _loadingFileName = '';
|
||||
double _progress = 0;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
|
@ -23,11 +27,10 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
|||
_listDir();
|
||||
}
|
||||
|
||||
void _listDir() async {
|
||||
Future<void> _listDir() async {
|
||||
setState(() => _isLoading = true);
|
||||
_dirContents = await widget.sftpClient.listdir(widget.path);
|
||||
setState(() {
|
||||
_isLoading = false;
|
||||
});
|
||||
setState(() => _isLoading = false);
|
||||
}
|
||||
|
||||
|
||||
|
@ -37,6 +40,8 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
|||
appBar: AppBar(
|
||||
title: Text('Explorer'),
|
||||
),
|
||||
bottomNavigationBar: _buildLoadingWidget(context),
|
||||
floatingActionButton: _buildFABs(context),
|
||||
body: _isLoading ? Center(child: CircularProgressIndicator()) : ListView.builder(
|
||||
itemCount: _dirContents.length,
|
||||
itemBuilder: (context, index) {
|
||||
|
@ -44,6 +49,71 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
|||
return ListTile(
|
||||
leading: Icon(dirEntry.attr.isDirectory ? Icons.folder : Icons.description),
|
||||
title: Text(dirEntry.filename),
|
||||
trailing: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
|
||||
},
|
||||
icon: Icon(Icons.drive_file_move)
|
||||
),
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
|
||||
},
|
||||
icon: Icon(Icons.copy)
|
||||
),
|
||||
IconButton(
|
||||
onPressed: () {},
|
||||
icon: Icon(Icons.drive_file_rename_outline)
|
||||
),
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (context) => AlertDialog(
|
||||
title: Text('Delete Permanently?'),
|
||||
content: Text(dirEntry.attr.isDirectory ? 'The contents of this folder will be deleted as well\nThis action cannot be undone' : 'This action cannot be undone'),
|
||||
actions: [
|
||||
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancel')),
|
||||
TextButton(
|
||||
onPressed: () async {
|
||||
if (dirEntry.attr.isDirectory) {
|
||||
Future<void> removeRecursively (String path) async {
|
||||
final dirContents = await widget.sftpClient.listdir(path);
|
||||
for (SftpName entry in dirContents) {
|
||||
final fullPath = '$path${entry.filename}';
|
||||
if (entry.attr.isDirectory) {
|
||||
await removeRecursively('$fullPath/');
|
||||
await widget.sftpClient.rmdir('$fullPath/');
|
||||
}
|
||||
else {
|
||||
await widget.sftpClient.remove(fullPath);
|
||||
}
|
||||
}
|
||||
await widget.sftpClient.rmdir(path);
|
||||
}
|
||||
await removeRecursively('${widget.path}${dirEntry.filename}/');
|
||||
}
|
||||
else {
|
||||
await widget.sftpClient.remove('${widget.path}${dirEntry.filename}');
|
||||
}
|
||||
_listDir();
|
||||
if (context.mounted) {
|
||||
Navigator.pop(context);
|
||||
}
|
||||
},
|
||||
child: Text('Yes')
|
||||
),
|
||||
],
|
||||
)
|
||||
);
|
||||
},
|
||||
icon: Icon(Icons.delete)
|
||||
),
|
||||
],
|
||||
),
|
||||
onTap: () {
|
||||
if (dirEntry.attr.isDirectory) {
|
||||
Navigator.push(context, MaterialPageRoute(
|
||||
|
@ -59,4 +129,119 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
|||
)
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildFABs(BuildContext context) {
|
||||
return Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
spacing: 10,
|
||||
children: [
|
||||
FloatingActionButton(
|
||||
heroTag: 'create-new-folder',
|
||||
onPressed: () {
|
||||
final nameController = TextEditingController();
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (context) => AlertDialog(
|
||||
title: Text('Create new folder'),
|
||||
content: TextField(
|
||||
controller: nameController,
|
||||
decoration: InputDecoration(
|
||||
labelText: 'Enter folder name'
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancel')),
|
||||
TextButton(
|
||||
onPressed: () async {
|
||||
try {
|
||||
await widget.sftpClient.mkdir('${widget.path}${nameController.text}');
|
||||
_listDir();
|
||||
}
|
||||
on SftpStatusError catch (e) {
|
||||
if (context.mounted) {
|
||||
if (e.code == 4) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
behavior: SnackBarBehavior.floating,
|
||||
content: Text('Folder Already Exists')
|
||||
)
|
||||
);
|
||||
}
|
||||
else {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
behavior: SnackBarBehavior.floating,
|
||||
content: Text('Error: ${e.message}')
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context.mounted) {
|
||||
Navigator.pop(context);
|
||||
}
|
||||
},
|
||||
child: Text('Ok')
|
||||
),
|
||||
],
|
||||
)
|
||||
);
|
||||
},
|
||||
child: Icon(Icons.create_new_folder),
|
||||
),
|
||||
FloatingActionButton(
|
||||
heroTag: 'upload-file',
|
||||
onPressed: () async {
|
||||
final List<XFile> files = await openFiles();
|
||||
for (XFile file in files) {
|
||||
final remoteFile = await widget.sftpClient.open('${widget.path}${file.name}', mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive);
|
||||
final fileSize = await file.length();
|
||||
final uploader = remoteFile.write(
|
||||
file.openRead().cast(),
|
||||
onProgress: (progress) => setState(() => _progress = progress/fileSize)
|
||||
);
|
||||
setState(() {
|
||||
_loader = uploader;
|
||||
_loadingFileName = file.name;
|
||||
});
|
||||
await uploader.done;
|
||||
}
|
||||
setState(() => _loader = null);
|
||||
_listDir();
|
||||
},
|
||||
child: Icon(Icons.upload),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildLoadingWidget(BuildContext context) {
|
||||
return _loader != null ? Container(
|
||||
color: Theme.of(context).colorScheme.secondaryContainer,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(10),
|
||||
child: Column(
|
||||
spacing: 10,
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Row(
|
||||
spacing: 10,
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Text('Uploading file: $_loadingFileName', style: TextStyle(fontSize: 16),),
|
||||
TextButton(
|
||||
onPressed: () {
|
||||
_loader!.abort();
|
||||
widget.sftpClient.remove('${widget.path}$_loadingFileName');
|
||||
},
|
||||
child: Text('Cancel')
|
||||
),
|
||||
],
|
||||
),
|
||||
LinearProgressIndicator(value: _progress,)
|
||||
],
|
||||
),
|
||||
),
|
||||
) : SizedBox();
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue