Compare commits
4 commits
075003f1ad
...
a66ed70532
Author | SHA1 | Date | |
---|---|---|---|
![]() |
a66ed70532 | ||
![]() |
7ea9e98789 | ||
![]() |
1ccadcf200 | ||
![]() |
08bbeb2619 |
9 changed files with 348 additions and 216 deletions
|
@ -1 +1,4 @@
|
||||||
include: package:flutter_lints/flutter.yaml
|
include: package:flutter_lints/flutter.yaml
|
||||||
|
analyzer:
|
||||||
|
errors:
|
||||||
|
todo: ignore
|
||||||
|
|
|
@ -4,8 +4,10 @@ import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
||||||
import 'package:fluxcloud/connection.dart';
|
import 'package:fluxcloud/connection.dart';
|
||||||
import 'package:fluxcloud/sftp_explorer.dart';
|
import 'package:fluxcloud/sftp_explorer.dart';
|
||||||
|
import 'package:fluxcloud/sftp_provider.dart';
|
||||||
import 'package:fluxcloud/sftp_worker.dart';
|
import 'package:fluxcloud/sftp_worker.dart';
|
||||||
import 'package:fluxcloud/widgets/add_server_modal.dart';
|
import 'package:fluxcloud/widgets/add_server_modal.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
|
||||||
class SftpConnectionList extends StatefulWidget {
|
class SftpConnectionList extends StatefulWidget {
|
||||||
const SftpConnectionList({
|
const SftpConnectionList({
|
||||||
|
@ -80,7 +82,10 @@ class _SftpConnectionListState extends State<SftpConnectionList> {
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
final sftpWorker = await SftpWorker.spawn(_connections[index]);
|
final sftpWorker = await SftpWorker.spawn(_connections[index]);
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
Navigator.push(context, MaterialPageRoute(builder: (context) => SftpExplorer(sftpWorker: sftpWorker,)));
|
Navigator.push(context, MaterialPageRoute(builder: (context) => ChangeNotifierProvider(
|
||||||
|
create: (_) => SftpProvider(sftpWorker),
|
||||||
|
child: SftpExplorer()
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
borderRadius: BorderRadius.circular(10),
|
borderRadius: BorderRadius.circular(10),
|
||||||
|
|
|
@ -5,50 +5,14 @@ import 'package:file_picker/file_picker.dart';
|
||||||
import 'package:file_selector/file_selector.dart';
|
import 'package:file_selector/file_selector.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:fluxcloud/main.dart';
|
import 'package:fluxcloud/main.dart';
|
||||||
import 'package:fluxcloud/sftp_worker.dart';
|
import 'package:fluxcloud/sftp_provider.dart';
|
||||||
|
import 'package:path/path.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
|
||||||
import 'widgets/operation_buttons.dart';
|
import 'widgets/operation_buttons.dart';
|
||||||
|
|
||||||
class SftpExplorer extends StatefulWidget {
|
class SftpExplorer extends StatelessWidget {
|
||||||
const SftpExplorer({super.key, required this.sftpWorker});
|
const SftpExplorer({super.key});
|
||||||
|
|
||||||
final SftpWorker sftpWorker;
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<SftpExplorer> createState() => _SftpExplorerState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _SftpExplorerState extends State<SftpExplorer> {
|
|
||||||
|
|
||||||
String path = '/';
|
|
||||||
|
|
||||||
bool _isLoading = true;
|
|
||||||
late List<SftpName> _dirContents;
|
|
||||||
|
|
||||||
double? _uploadProgress;
|
|
||||||
double? _downloadProgress;
|
|
||||||
|
|
||||||
void _setDownloadProgress(double? progress) => setState(() => _downloadProgress = progress);
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_listDir();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _listDir() async {
|
|
||||||
setState(() => _isLoading = true);
|
|
||||||
try {
|
|
||||||
_dirContents = await widget.sftpWorker.listdir(path);
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
if (mounted) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(buildErrorSnackBar(context, e.toString()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setState(() => _isLoading = false);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
@ -60,25 +24,24 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
actionsPadding: EdgeInsets.only(right: 20),
|
actionsPadding: EdgeInsets.only(right: 20),
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
if (path == '/') {
|
if (context.read<SftpProvider>().path == '/') {
|
||||||
// TODO: figure this out
|
// TODO: figure this out
|
||||||
// Navigator.pop(context);
|
// Navigator.pop(context);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
path = path.substring(0, path.length - 1);
|
context.read<SftpProvider>().goToPrevDir();
|
||||||
path = path.substring(0, path.lastIndexOf('/')+1);
|
|
||||||
_listDir();
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
icon: Icon(Icons.arrow_back)
|
icon: Icon(Icons.arrow_back)
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
if (_uploadProgress != null)
|
Selector<SftpProvider, double?>(
|
||||||
Stack(
|
selector: (_, sftpProvider) => sftpProvider.uploadProgress,
|
||||||
|
builder: (_, uploadProgress, __) => uploadProgress != null ? Stack(
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
children: [
|
children: [
|
||||||
TweenAnimationBuilder<double>(
|
TweenAnimationBuilder<double>(
|
||||||
tween: Tween(begin: 0, end: _uploadProgress),
|
tween: Tween(begin: 0, end: uploadProgress),
|
||||||
duration: Duration(milliseconds: 300),
|
duration: Duration(milliseconds: 300),
|
||||||
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
|
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
|
||||||
),
|
),
|
||||||
|
@ -89,13 +52,15 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
icon: Icon(Icons.upload)
|
icon: Icon(Icons.upload)
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
) : const SizedBox.shrink(),
|
||||||
),
|
),
|
||||||
if (_downloadProgress != null)
|
Selector<SftpProvider, double?>(
|
||||||
Stack(
|
selector: (_, sftpProvider) => sftpProvider.downloadProgress,
|
||||||
|
builder: (_, downloadProgress, __) => downloadProgress != null ? Stack(
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
children: [
|
children: [
|
||||||
TweenAnimationBuilder<double>(
|
TweenAnimationBuilder<double>(
|
||||||
tween: Tween(begin: 0, end: _downloadProgress),
|
tween: Tween(begin: 0, end: downloadProgress),
|
||||||
duration: Duration(milliseconds: 300),
|
duration: Duration(milliseconds: 300),
|
||||||
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
|
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
|
||||||
),
|
),
|
||||||
|
@ -106,20 +71,21 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
icon: Icon(Icons.download)
|
icon: Icon(Icons.download)
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
) : const SizedBox.shrink(),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
floatingActionButton: _buildFABs(context),
|
floatingActionButton: _buildFABs(context),
|
||||||
|
bottomNavigationBar: _buildCopyMoveButton(context),
|
||||||
body: PopScope(
|
body: PopScope(
|
||||||
canPop: false,
|
canPop: false,
|
||||||
onPopInvokedWithResult: (_, _) {
|
onPopInvokedWithResult: (_, _) {
|
||||||
if (path != '/') {
|
if (context.read<SftpProvider>().path != '/') {
|
||||||
path = path.substring(0, path.length - 1);
|
context.read<SftpProvider>().goToPrevDir();
|
||||||
path = path.substring(0, path.lastIndexOf('/')+1);
|
|
||||||
_listDir();
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
child: AnimatedSwitcher(
|
child: Consumer<SftpProvider>(
|
||||||
|
builder: (_, sftpProvider, __) => AnimatedSwitcher(
|
||||||
duration: Duration(milliseconds: 300),
|
duration: Duration(milliseconds: 300),
|
||||||
transitionBuilder: (child, animation) {
|
transitionBuilder: (child, animation) {
|
||||||
final curved = CurvedAnimation(
|
final curved = CurvedAnimation(
|
||||||
|
@ -137,25 +103,25 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
child: _isLoading ? Center(child: CircularProgressIndicator()) : ListView.builder(
|
child: sftpProvider.isLoading ? Center(child: CircularProgressIndicator()) : ListView.builder(
|
||||||
key: ValueKey(path),
|
key: ValueKey(sftpProvider.path),
|
||||||
itemCount: _dirContents.length,
|
itemCount: sftpProvider.dirContents.length,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final dirEntry = _dirContents[index];
|
final dirEntry = sftpProvider.dirContents[index];
|
||||||
return ListTile(
|
return ListTile(
|
||||||
leading: Icon(dirEntry.attr.isDirectory ? Icons.folder : Icons.description),
|
leading: Icon(dirEntry.attr.isDirectory ? Icons.folder : Icons.description),
|
||||||
title: Text(dirEntry.filename),
|
title: Text(dirEntry.filename),
|
||||||
trailing: OperationButtons(sftpWorker: widget.sftpWorker, path: path, dirEntries: [dirEntry], listDir: _listDir, setDownloadProgress: _setDownloadProgress,),
|
trailing: OperationButtons(dirEntries: [dirEntry],),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
if (dirEntry.attr.isDirectory) {
|
if (dirEntry.attr.isDirectory) {
|
||||||
path = '$path${dirEntry.filename}/';
|
sftpProvider.goToDir('${sftpProvider.path}${dirEntry.filename}/');
|
||||||
_listDir();
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -183,9 +149,10 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancel')),
|
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancel')),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
|
final sftpProvider = context.read<SftpProvider>();
|
||||||
try {
|
try {
|
||||||
await widget.sftpWorker.mkdir('$path${nameController.text}');
|
await sftpProvider.sftpWorker.mkdir('${sftpProvider.path}${nameController.text}');
|
||||||
_listDir();
|
sftpProvider.listDir();
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
|
@ -207,6 +174,7 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
FloatingActionButton(
|
FloatingActionButton(
|
||||||
heroTag: 'upload-file',
|
heroTag: 'upload-file',
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
|
final sftpProvider = context.read<SftpProvider>();
|
||||||
final List<String> filePaths;
|
final List<String> filePaths;
|
||||||
if (Platform.isAndroid | Platform.isIOS) {
|
if (Platform.isAndroid | Platform.isIOS) {
|
||||||
final res = await FilePicker.platform.pickFiles(allowMultiple: true);
|
final res = await FilePicker.platform.pickFiles(allowMultiple: true);
|
||||||
|
@ -218,10 +186,10 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
}
|
}
|
||||||
for (final filePath in filePaths) {
|
for (final filePath in filePaths) {
|
||||||
try {
|
try {
|
||||||
await for (final progress in widget.sftpWorker.uploadFile(path, filePath)) {
|
await for (final progress in sftpProvider.sftpWorker.uploadFile(sftpProvider.path, filePath)) {
|
||||||
setState(() => _uploadProgress = progress);
|
sftpProvider.setUploadProgress(progress);
|
||||||
}
|
}
|
||||||
await _listDir();
|
await sftpProvider.listDir();
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
|
@ -229,8 +197,8 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
setState(() => _uploadProgress = null);
|
sftpProvider.setUploadProgress(null);
|
||||||
_listDir();
|
sftpProvider.listDir();
|
||||||
},
|
},
|
||||||
child: Icon(Icons.upload),
|
child: Icon(Icons.upload),
|
||||||
),
|
),
|
||||||
|
@ -238,4 +206,68 @@ class _SftpExplorerState extends State<SftpExplorer> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildCopyMoveButton(BuildContext context) {
|
||||||
|
return Selector<SftpProvider, (List<String>?, bool)>(
|
||||||
|
selector: (_, sftpProvider) => (sftpProvider.toBeMovedOrCopied, sftpProvider.isCopy),
|
||||||
|
builder: (_, data, __) {
|
||||||
|
final (toBeMovedOrCopied, isCopy) = data;
|
||||||
|
if (toBeMovedOrCopied == null) {
|
||||||
|
return const SizedBox.shrink();
|
||||||
|
}
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.all(12),
|
||||||
|
child: Row(
|
||||||
|
spacing: 10,
|
||||||
|
children: [
|
||||||
|
Expanded(child: ElevatedButton(
|
||||||
|
onPressed: () async {
|
||||||
|
final sftpProvider = context.read<SftpProvider>();
|
||||||
|
for (final filePath in toBeMovedOrCopied) {
|
||||||
|
try {
|
||||||
|
if (isCopy) {
|
||||||
|
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
final fileName = basename(filePath);
|
||||||
|
await sftpProvider.sftpWorker.rename(filePath, '${sftpProvider.path}$fileName');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
if (context.mounted) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(buildErrorSnackBar(context, e.toString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sftpProvider.setCopyOrMoveFiles(null, isCopy);
|
||||||
|
sftpProvider.listDir();
|
||||||
|
},
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Theme.of(context).colorScheme.secondaryContainer,
|
||||||
|
foregroundColor: Theme.of(context).colorScheme.onSecondaryContainer
|
||||||
|
),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Text(isCopy ? 'Copy Here' : 'Move Here'),
|
||||||
|
),
|
||||||
|
)),
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
context.read<SftpProvider>().setCopyOrMoveFiles(null, isCopy);
|
||||||
|
},
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: Theme.of(context).colorScheme.secondaryContainer,
|
||||||
|
foregroundColor: Theme.of(context).colorScheme.onSecondaryContainer
|
||||||
|
),
|
||||||
|
icon: Padding(
|
||||||
|
padding: const EdgeInsets.all(4),
|
||||||
|
child: Icon(Icons.close),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
69
lib/sftp_provider.dart
Normal file
69
lib/sftp_provider.dart
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
import 'package:dartssh2/dartssh2.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:fluxcloud/sftp_worker.dart';
|
||||||
|
|
||||||
|
class SftpProvider extends ChangeNotifier {
|
||||||
|
final SftpWorker _sftpWorker;
|
||||||
|
|
||||||
|
String _path = '/';
|
||||||
|
bool _isLoading = false;
|
||||||
|
late List<SftpName> _dirContents;
|
||||||
|
|
||||||
|
double? _uploadProgress;
|
||||||
|
double? _downloadProgress;
|
||||||
|
|
||||||
|
List<String>? _toBeMovedOrCopied;
|
||||||
|
bool _isCopy = false;
|
||||||
|
|
||||||
|
SftpProvider(this._sftpWorker) {
|
||||||
|
listDir();
|
||||||
|
}
|
||||||
|
|
||||||
|
SftpWorker get sftpWorker => _sftpWorker;
|
||||||
|
|
||||||
|
String get path => _path;
|
||||||
|
bool get isLoading => _isLoading;
|
||||||
|
List<SftpName> get dirContents => _dirContents;
|
||||||
|
|
||||||
|
double? get uploadProgress => _uploadProgress;
|
||||||
|
double? get downloadProgress => _downloadProgress;
|
||||||
|
|
||||||
|
List<String>? get toBeMovedOrCopied => _toBeMovedOrCopied;
|
||||||
|
bool get isCopy => _isCopy;
|
||||||
|
|
||||||
|
Future<void> listDir() async {
|
||||||
|
_isLoading = true;
|
||||||
|
notifyListeners();
|
||||||
|
_dirContents = await _sftpWorker.listdir(_path);
|
||||||
|
_isLoading = false;
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
void goToPrevDir() {
|
||||||
|
_path = _path.substring(0, _path.length - 1);
|
||||||
|
_path = _path.substring(0, _path.lastIndexOf('/')+1);
|
||||||
|
listDir();
|
||||||
|
}
|
||||||
|
|
||||||
|
void goToDir(String path) {
|
||||||
|
_path = path;
|
||||||
|
listDir();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setUploadProgress(double? progress) {
|
||||||
|
_uploadProgress = progress;
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setDownloadProgress(double? progress) {
|
||||||
|
_downloadProgress = progress;
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setCopyOrMoveFiles(List<String>? files, bool isCopy) {
|
||||||
|
_toBeMovedOrCopied = files;
|
||||||
|
_isCopy = isCopy;
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -50,6 +50,13 @@ class DownloadFile extends SftpCommand {
|
||||||
DownloadFile(this.file, this.path, this.downloadPath);
|
DownloadFile(this.file, this.path, this.downloadPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class Copy extends SftpCommand {
|
||||||
|
final String filePath;
|
||||||
|
final String copyToPath;
|
||||||
|
|
||||||
|
Copy(this.filePath, this.copyToPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class SftpWorker {
|
class SftpWorker {
|
||||||
|
|
||||||
|
@ -109,23 +116,46 @@ class SftpWorker {
|
||||||
|
|
||||||
|
|
||||||
static void _sftpCmdHandler(SendPort sendPort, ReceivePort receivePort, SftpClient sftpClient) {
|
static void _sftpCmdHandler(SendPort sendPort, ReceivePort receivePort, SftpClient sftpClient) {
|
||||||
receivePort.listen((message) async {
|
|
||||||
final (int id, SftpCommand command) = message;
|
final StreamController<(int, DownloadFile)> downloadController = StreamController();
|
||||||
switch (command) {
|
downloadController.stream.asyncMap((cmd) async {
|
||||||
case ListDir(:final path):
|
final (int id, DownloadFile downloadCmd) = cmd;
|
||||||
try {
|
try {
|
||||||
final files = await sftpClient.listdir(path);
|
final localFile = File('${downloadCmd.downloadPath}/${downloadCmd.file.filename}');
|
||||||
sendPort.send((id, files));
|
if (await localFile.exists()) {
|
||||||
|
sendPort.send((id, RemoteError('File Already Exists', '')));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final localFileWriter = await localFile.open(mode: FileMode.write);
|
||||||
|
final remoteFile = await sftpClient.open('${downloadCmd.path}${downloadCmd.file.filename}');
|
||||||
|
final fileSize = downloadCmd.file.attr.size!;
|
||||||
|
bool timeout = true;
|
||||||
|
await for (final bytes in remoteFile.read(
|
||||||
|
onProgress: (progress) {
|
||||||
|
if (timeout) {
|
||||||
|
timeout = false;
|
||||||
|
sendPort.send((id, progress/fileSize));
|
||||||
|
Future.delayed(Duration(seconds: 2), () => timeout = true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)) {
|
||||||
|
await localFileWriter.writeFrom(bytes);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
on SftpStatusError catch (e) {
|
on SftpStatusError catch (e) {
|
||||||
sendPort.send((id, RemoteError(e.message, '')));
|
sendPort.send((id, RemoteError(e.message, '')));
|
||||||
}
|
}
|
||||||
case UploadFile(:final path, :final filePath):
|
sendPort.send((id, 1.0));
|
||||||
|
}).listen((_) {});
|
||||||
|
|
||||||
|
final StreamController<(int, UploadFile)> uploadController = StreamController();
|
||||||
|
uploadController.stream.asyncMap((cmd) async {
|
||||||
|
final (int id, UploadFile uploadCmd) = cmd;
|
||||||
try {
|
try {
|
||||||
final file = File(filePath);
|
final file = File(uploadCmd.filePath);
|
||||||
final fileSize = await file.length();
|
final fileSize = await file.length();
|
||||||
final remoteFile = await sftpClient.open(
|
final remoteFile = await sftpClient.open(
|
||||||
'$path${basename(filePath)}',
|
'${uploadCmd.path}${basename(uploadCmd.filePath)}',
|
||||||
mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive
|
mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive
|
||||||
);
|
);
|
||||||
bool timeout = true;
|
bool timeout = true;
|
||||||
|
@ -144,6 +174,21 @@ class SftpWorker {
|
||||||
sendPort.send((id, RemoteError(e.message, '')));
|
sendPort.send((id, RemoteError(e.message, '')));
|
||||||
}
|
}
|
||||||
sendPort.send((id, 1.0));
|
sendPort.send((id, 1.0));
|
||||||
|
}).listen((_) {});
|
||||||
|
|
||||||
|
receivePort.listen((message) async {
|
||||||
|
final (int id, SftpCommand command) = message;
|
||||||
|
switch (command) {
|
||||||
|
case ListDir(:final path):
|
||||||
|
try {
|
||||||
|
final files = await sftpClient.listdir(path);
|
||||||
|
sendPort.send((id, files));
|
||||||
|
}
|
||||||
|
on SftpStatusError catch (e) {
|
||||||
|
sendPort.send((id, RemoteError(e.message, '')));
|
||||||
|
}
|
||||||
|
case UploadFile():
|
||||||
|
uploadController.add((id, command));
|
||||||
case MkDir(:final path):
|
case MkDir(:final path):
|
||||||
try {
|
try {
|
||||||
await sftpClient.mkdir(path);
|
await sftpClient.mkdir(path);
|
||||||
|
@ -187,33 +232,17 @@ class SftpWorker {
|
||||||
on SftpStatusError catch (e) {
|
on SftpStatusError catch (e) {
|
||||||
sendPort.send((id, RemoteError(e.message, '')));
|
sendPort.send((id, RemoteError(e.message, '')));
|
||||||
}
|
}
|
||||||
case DownloadFile(:final file, :final path, :final downloadPath):
|
case DownloadFile():
|
||||||
|
downloadController.add((id, command));
|
||||||
|
case Copy(:final filePath, :final copyToPath):
|
||||||
try {
|
try {
|
||||||
final localFile = File('$downloadPath/${file.filename}');
|
// TODO: complete this
|
||||||
if (await localFile.exists()) {
|
sendPort.send((id, 0));
|
||||||
sendPort.send((id, RemoteError('File Already Exists', '')));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
final localFileWriter = await localFile.open(mode: FileMode.write);
|
|
||||||
final remoteFile = await sftpClient.open('$path${file.filename}');
|
|
||||||
final fileSize = file.attr.size!;
|
|
||||||
bool timeout = true;
|
|
||||||
await for (final bytes in remoteFile.read(
|
|
||||||
onProgress: (progress) {
|
|
||||||
if (timeout) {
|
|
||||||
timeout = false;
|
|
||||||
sendPort.send((id, progress/fileSize));
|
|
||||||
Future.delayed(Duration(seconds: 2), () => timeout = true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)) {
|
|
||||||
await localFileWriter.writeFrom(bytes);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
on SftpStatusError catch (e) {
|
on SftpStatusError catch (e) {
|
||||||
sendPort.send((id, RemoteError(e.message, '')));
|
sendPort.send((id, RemoteError(e.message, '')));
|
||||||
}
|
}
|
||||||
sendPort.send((id, 1.0));
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -296,4 +325,12 @@ class SftpWorker {
|
||||||
return controller.stream;
|
return controller.stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> copy(String filePath, String copyToPath) async {
|
||||||
|
final completer = Completer.sync();
|
||||||
|
final id = _idCounter++;
|
||||||
|
_activeRequests[id] = completer;
|
||||||
|
_commands.send((id, Copy(filePath, copyToPath)));
|
||||||
|
await completer.future;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,35 +1,34 @@
|
||||||
import 'package:dartssh2/dartssh2.dart';
|
import 'package:dartssh2/dartssh2.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:fluxcloud/main.dart';
|
import 'package:fluxcloud/main.dart';
|
||||||
import 'package:fluxcloud/sftp_worker.dart';
|
import 'package:fluxcloud/sftp_provider.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
import 'package:provider/provider.dart';
|
||||||
|
|
||||||
class OperationButtons extends StatelessWidget {
|
class OperationButtons extends StatelessWidget {
|
||||||
const OperationButtons({
|
const OperationButtons({
|
||||||
super.key,
|
super.key, required this.dirEntries,
|
||||||
required this.sftpWorker, required this.path, required this.dirEntries, required this.listDir, required this.setDownloadProgress,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
final SftpWorker sftpWorker;
|
|
||||||
final String path;
|
|
||||||
final List<SftpName> dirEntries;
|
final List<SftpName> dirEntries;
|
||||||
final Function listDir;
|
|
||||||
final Function(double? progress) setDownloadProgress;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
final sftpProvider = context.read<SftpProvider>();
|
||||||
return Row(
|
return Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
|
final filePaths = dirEntries.map((dirEntry) => '${sftpProvider.path}${dirEntry.filename}').toList();
|
||||||
|
sftpProvider.setCopyOrMoveFiles(filePaths, false);
|
||||||
},
|
},
|
||||||
icon: Icon(Icons.drive_file_move)
|
icon: Icon(Icons.drive_file_move)
|
||||||
),
|
),
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
|
final filePaths = dirEntries.map((dirEntry) => '${sftpProvider.path}${dirEntry.filename}').toList();
|
||||||
|
sftpProvider.setCopyOrMoveFiles(filePaths, true);
|
||||||
},
|
},
|
||||||
icon: Icon(Icons.copy)
|
icon: Icon(Icons.copy)
|
||||||
),
|
),
|
||||||
|
@ -54,8 +53,8 @@ class OperationButtons extends StatelessWidget {
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
try {
|
try {
|
||||||
await sftpWorker.rename('$path${dirEntry.filename}', '$path${newNameController.text}');
|
await sftpProvider.sftpWorker.rename('${sftpProvider.path}${dirEntry.filename}', '${sftpProvider.path}${newNameController.text}');
|
||||||
listDir();
|
sftpProvider.listDir();
|
||||||
}
|
}
|
||||||
on SftpStatusError catch (e) {
|
on SftpStatusError catch (e) {
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
|
@ -97,14 +96,14 @@ class OperationButtons extends StatelessWidget {
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
for (final dirEntry in dirEntries) {
|
for (final dirEntry in dirEntries) {
|
||||||
try {
|
try {
|
||||||
await sftpWorker.remove(dirEntry, path);
|
await sftpProvider.sftpWorker.remove(dirEntry, sftpProvider.path);
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(buildErrorSnackBar(context, e.toString()));
|
ScaffoldMessenger.of(context).showSnackBar(buildErrorSnackBar(context, e.toString()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
listDir();
|
sftpProvider.listDir();
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
Navigator.pop(context);
|
Navigator.pop(context);
|
||||||
}
|
}
|
||||||
|
@ -125,8 +124,8 @@ class OperationButtons extends StatelessWidget {
|
||||||
if (downloadsDir == null) return;
|
if (downloadsDir == null) return;
|
||||||
for (final dirEntry in dirEntries) {
|
for (final dirEntry in dirEntries) {
|
||||||
try {
|
try {
|
||||||
await for (final progress in sftpWorker.downloadFile(dirEntry, path, downloadsDir.path)) {
|
await for (final progress in sftpProvider.sftpWorker.downloadFile(dirEntry, sftpProvider.path, downloadsDir.path)) {
|
||||||
setDownloadProgress(progress);
|
sftpProvider.setDownloadProgress(progress);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
|
@ -135,7 +134,7 @@ class OperationButtons extends StatelessWidget {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
setDownloadProgress(null);
|
sftpProvider.setDownloadProgress(null);
|
||||||
},
|
},
|
||||||
icon: Icon(Icons.download)
|
icon: Icon(Icons.download)
|
||||||
)
|
)
|
||||||
|
|
16
pubspec.lock
16
pubspec.lock
|
@ -320,6 +320,14 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.16.0"
|
version: "1.16.0"
|
||||||
|
nested:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: nested
|
||||||
|
sha256: "03bac4c528c64c95c722ec99280375a6f2fc708eec17c7b3f07253b626cd2a20"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "1.0.0"
|
||||||
path:
|
path:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
|
@ -408,6 +416,14 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.9.1"
|
version: "3.9.1"
|
||||||
|
provider:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: provider
|
||||||
|
sha256: "4abbd070a04e9ddc287673bf5a030c7ca8b685ff70218720abab8b092f53dd84"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "6.1.5"
|
||||||
sky_engine:
|
sky_engine:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description: flutter
|
description: flutter
|
||||||
|
|
|
@ -15,6 +15,7 @@ dependencies:
|
||||||
path: ^1.9.1
|
path: ^1.9.1
|
||||||
file_picker: ^10.2.0
|
file_picker: ^10.2.0
|
||||||
path_provider: ^2.1.5
|
path_provider: ^2.1.5
|
||||||
|
provider: ^6.1.5
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
|
|
@ -1,30 +0,0 @@
|
||||||
// This is a basic Flutter widget test.
|
|
||||||
//
|
|
||||||
// To perform an interaction with a widget in your test, use the WidgetTester
|
|
||||||
// utility in the flutter_test package. For example, you can send tap and scroll
|
|
||||||
// gestures. You can also use WidgetTester to find child widgets in the widget
|
|
||||||
// tree, read text, and verify that the values of widget properties are correct.
|
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
|
|
||||||
import 'package:fluxcloud/main.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
testWidgets('Counter increments smoke test', (WidgetTester tester) async {
|
|
||||||
// Build our app and trigger a frame.
|
|
||||||
await tester.pumpWidget(const MyApp());
|
|
||||||
|
|
||||||
// Verify that our counter starts at 0.
|
|
||||||
expect(find.text('0'), findsOneWidget);
|
|
||||||
expect(find.text('1'), findsNothing);
|
|
||||||
|
|
||||||
// Tap the '+' icon and trigger a frame.
|
|
||||||
await tester.tap(find.byIcon(Icons.add));
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
// Verify that our counter has incremented.
|
|
||||||
expect(find.text('0'), findsNothing);
|
|
||||||
expect(find.text('1'), findsOneWidget);
|
|
||||||
});
|
|
||||||
}
|
|
Loading…
Add table
Add a link
Reference in a new issue