make copy work, also some abstraction and new provider

This commit is contained in:
RafayAhmad7548 2025-08-16 10:12:51 +05:00
parent a66ed70532
commit 869f2c14c3
6 changed files with 145 additions and 107 deletions

View file

@ -0,0 +1,40 @@
import 'package:flutter/material.dart';
class SftpLoadingProvider extends ChangeNotifier {
double? _uploadProgress;
double? _downloadProgress;
double? _copyProgress;
List<String>? _toBeMovedOrCopied;
bool _isCopy = false;
double? get copyProgress => _copyProgress;
double? get uploadProgress => _uploadProgress;
double? get downloadProgress => _downloadProgress;
List<String>? get toBeMovedOrCopied => _toBeMovedOrCopied;
bool get isCopy => _isCopy;
void setUploadProgress(double? progress) {
_uploadProgress = progress;
notifyListeners();
}
void setDownloadProgress(double? progress) {
_downloadProgress = progress;
notifyListeners();
}
void setCopyProgress(double? progress) {
_copyProgress = progress;
notifyListeners();
}
void setCopyOrMoveFiles(List<String>? files, bool isCopy) {
_toBeMovedOrCopied = files;
_isCopy = isCopy;
notifyListeners();
}
}

View file

@ -9,12 +9,6 @@ class SftpProvider extends ChangeNotifier {
bool _isLoading = false; bool _isLoading = false;
late List<SftpName> _dirContents; late List<SftpName> _dirContents;
double? _uploadProgress;
double? _downloadProgress;
List<String>? _toBeMovedOrCopied;
bool _isCopy = false;
SftpProvider(this._sftpWorker) { SftpProvider(this._sftpWorker) {
listDir(); listDir();
} }
@ -25,12 +19,6 @@ class SftpProvider extends ChangeNotifier {
bool get isLoading => _isLoading; bool get isLoading => _isLoading;
List<SftpName> get dirContents => _dirContents; 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 { Future<void> listDir() async {
_isLoading = true; _isLoading = true;
notifyListeners(); notifyListeners();
@ -50,20 +38,4 @@ class SftpProvider extends ChangeNotifier {
listDir(); 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();
}
} }

View file

@ -3,12 +3,14 @@ import 'dart:convert';
import 'package:flutter/material.dart'; 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/providers/sftp_loading_provider.dart';
import 'package:fluxcloud/providers/sftp_provider.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'; import 'package:provider/provider.dart';
class SftpConnectionList extends StatefulWidget { class SftpConnectionList extends StatefulWidget {
const SftpConnectionList({ const SftpConnectionList({
super.key, super.key,
@ -82,10 +84,13 @@ 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) => ChangeNotifierProvider( Navigator.push(context, MaterialPageRoute(builder: (context) => MultiProvider(
create: (_) => SftpProvider(sftpWorker), providers: [
child: SftpExplorer() ChangeNotifierProvider<SftpProvider>(create: (_) => SftpProvider(sftpWorker)),
))); ChangeNotifierProvider<SftpLoadingProvider>(create: (_) => SftpLoadingProvider()),
],
child: SftpExplorer())
));
} }
}, },
borderRadius: BorderRadius.circular(10), borderRadius: BorderRadius.circular(10),

View file

@ -1,15 +1,15 @@
import 'dart:io'; import 'dart:io';
import 'package:dartssh2/dartssh2.dart';
import 'package:file_picker/file_picker.dart'; 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_provider.dart'; import 'package:fluxcloud/providers/sftp_loading_provider.dart';
import 'package:fluxcloud/providers/sftp_provider.dart';
import 'package:fluxcloud/widgets/operation_buttons.dart';
import 'package:path/path.dart'; import 'package:path/path.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'widgets/operation_buttons.dart';
class SftpExplorer extends StatelessWidget { class SftpExplorer extends StatelessWidget {
const SftpExplorer({super.key}); const SftpExplorer({super.key});
@ -34,46 +34,7 @@ class SftpExplorer extends StatelessWidget {
}, },
icon: Icon(Icons.arrow_back) icon: Icon(Icons.arrow_back)
), ),
actions: [ actions: _buildLoadingButtons,
Selector<SftpProvider, double?>(
selector: (_, sftpProvider) => sftpProvider.uploadProgress,
builder: (_, uploadProgress, __) => uploadProgress != null ? Stack(
alignment: Alignment.center,
children: [
TweenAnimationBuilder<double>(
tween: Tween(begin: 0, end: uploadProgress),
duration: Duration(milliseconds: 300),
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
),
IconButton(
onPressed: () {
// TODO: show upload details here
},
icon: Icon(Icons.upload)
),
]
) : const SizedBox.shrink(),
),
Selector<SftpProvider, double?>(
selector: (_, sftpProvider) => sftpProvider.downloadProgress,
builder: (_, downloadProgress, __) => downloadProgress != null ? Stack(
alignment: Alignment.center,
children: [
TweenAnimationBuilder<double>(
tween: Tween(begin: 0, end: downloadProgress),
duration: Duration(milliseconds: 300),
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
),
IconButton(
onPressed: () {
// TODO: show donwload details here
},
icon: Icon(Icons.download)
),
]
) : const SizedBox.shrink(),
),
],
), ),
floatingActionButton: _buildFABs(context), floatingActionButton: _buildFABs(context),
bottomNavigationBar: _buildCopyMoveButton(context), bottomNavigationBar: _buildCopyMoveButton(context),
@ -127,6 +88,8 @@ class SftpExplorer extends StatelessWidget {
} }
Widget _buildFABs(BuildContext context) { Widget _buildFABs(BuildContext context) {
final sftpProvider = context.read<SftpProvider>();
final sftpLoadingProvider = context.read<SftpLoadingProvider>();
return Row( return Row(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
spacing: 10, spacing: 10,
@ -149,7 +112,6 @@ class SftpExplorer extends StatelessWidget {
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 sftpProvider.sftpWorker.mkdir('${sftpProvider.path}${nameController.text}'); await sftpProvider.sftpWorker.mkdir('${sftpProvider.path}${nameController.text}');
sftpProvider.listDir(); sftpProvider.listDir();
@ -174,7 +136,6 @@ class SftpExplorer extends StatelessWidget {
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);
@ -187,7 +148,7 @@ class SftpExplorer extends StatelessWidget {
for (final filePath in filePaths) { for (final filePath in filePaths) {
try { try {
await for (final progress in sftpProvider.sftpWorker.uploadFile(sftpProvider.path, filePath)) { await for (final progress in sftpProvider.sftpWorker.uploadFile(sftpProvider.path, filePath)) {
sftpProvider.setUploadProgress(progress); sftpLoadingProvider.setUploadProgress(progress);
} }
await sftpProvider.listDir(); await sftpProvider.listDir();
} }
@ -197,7 +158,7 @@ class SftpExplorer extends StatelessWidget {
} }
} }
} }
sftpProvider.setUploadProgress(null); sftpLoadingProvider.setUploadProgress(null);
sftpProvider.listDir(); sftpProvider.listDir();
}, },
child: Icon(Icons.upload), child: Icon(Icons.upload),
@ -207,8 +168,8 @@ class SftpExplorer extends StatelessWidget {
} }
Widget _buildCopyMoveButton(BuildContext context) { Widget _buildCopyMoveButton(BuildContext context) {
return Selector<SftpProvider, (List<String>?, bool)>( return Selector<SftpLoadingProvider, (List<String>?, bool)>(
selector: (_, sftpProvider) => (sftpProvider.toBeMovedOrCopied, sftpProvider.isCopy), selector: (_, sftpLoadingProvider) => (sftpLoadingProvider.toBeMovedOrCopied, sftpLoadingProvider.isCopy),
builder: (_, data, __) { builder: (_, data, __) {
final (toBeMovedOrCopied, isCopy) = data; final (toBeMovedOrCopied, isCopy) = data;
if (toBeMovedOrCopied == null) { if (toBeMovedOrCopied == null) {
@ -222,13 +183,16 @@ class SftpExplorer extends StatelessWidget {
Expanded(child: ElevatedButton( Expanded(child: ElevatedButton(
onPressed: () async { onPressed: () async {
final sftpProvider = context.read<SftpProvider>(); final sftpProvider = context.read<SftpProvider>();
final sftpLoadingProvider = context.read<SftpLoadingProvider>();
for (final filePath in toBeMovedOrCopied) { for (final filePath in toBeMovedOrCopied) {
try { try {
final fileName = basename(filePath);
if (isCopy) { if (isCopy) {
await for (final progress in sftpProvider.sftpWorker.copy(filePath, '${sftpProvider.path}$fileName')) {
sftpLoadingProvider.setCopyProgress(progress);
}
} }
else { else {
final fileName = basename(filePath);
await sftpProvider.sftpWorker.rename(filePath, '${sftpProvider.path}$fileName'); await sftpProvider.sftpWorker.rename(filePath, '${sftpProvider.path}$fileName');
} }
} }
@ -238,7 +202,9 @@ class SftpExplorer extends StatelessWidget {
} }
} }
} }
sftpProvider.setCopyOrMoveFiles(null, isCopy); // TODO: figure out where to put this line
sftpLoadingProvider.setCopyOrMoveFiles(null, isCopy);
sftpLoadingProvider.setCopyProgress(null);
sftpProvider.listDir(); sftpProvider.listDir();
}, },
style: ElevatedButton.styleFrom( style: ElevatedButton.styleFrom(
@ -252,7 +218,7 @@ class SftpExplorer extends StatelessWidget {
)), )),
IconButton( IconButton(
onPressed: () { onPressed: () {
context.read<SftpProvider>().setCopyOrMoveFiles(null, isCopy); context.read<SftpLoadingProvider>().setCopyOrMoveFiles(null, isCopy);
}, },
style: ElevatedButton.styleFrom( style: ElevatedButton.styleFrom(
backgroundColor: Theme.of(context).colorScheme.secondaryContainer, backgroundColor: Theme.of(context).colorScheme.secondaryContainer,
@ -270,4 +236,38 @@ class SftpExplorer extends StatelessWidget {
); );
} }
List<Widget> get _buildLoadingButtons => [
Selector<SftpLoadingProvider, double?>(
selector: (_, sftpLoadingProvider) => sftpLoadingProvider.uploadProgress,
builder: (_, uploadProgress, __) => _buildLoader(uploadProgress, Icons.upload)
),
Selector<SftpLoadingProvider, double?>(
selector: (_, sftpLoadingProvider) => sftpLoadingProvider.downloadProgress,
builder: (_, downloadProgress, __) => _buildLoader(downloadProgress, Icons.download)
),
Selector<SftpLoadingProvider, double?>(
selector: (_, sftpLoadingProvider) => sftpLoadingProvider.copyProgress,
builder: (_, copyProgress, __) => _buildLoader(copyProgress, Icons.copy)
),
];
Widget _buildLoader(double? progress, IconData icon) {
return progress != null ? Stack(
alignment: Alignment.center,
children: [
TweenAnimationBuilder<double>(
tween: Tween(begin: 0, end: progress),
duration: Duration(milliseconds: 300),
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
),
IconButton(
onPressed: () {
// TODO: show details here
},
icon: Icon(icon)
),
]
) : const SizedBox.shrink();
}
} }

View file

@ -176,6 +176,32 @@ class SftpWorker {
sendPort.send((id, 1.0)); sendPort.send((id, 1.0));
}).listen((_) {}); }).listen((_) {});
final StreamController<(int, Copy)> copyController = StreamController();
copyController.stream.asyncMap((cmd) async {
final (id, copyCmd) = cmd;
try {
final srcFile = await sftpClient.open(copyCmd.filePath);
final dstFile = await sftpClient.open(copyCmd.copyToPath, mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive);
final fileSize = (await srcFile.stat()).size!;
bool timeout = true;
await dstFile.write(
srcFile.read(),
onProgress: (progress) {
if (timeout) {
timeout = false;
sendPort.send((id, progress/fileSize));
Future.delayed(Duration(seconds: 2), () => timeout = true);
}
}
);
}
on SftpStatusError catch (e) {
sendPort.send((id, RemoteError(e.message, '')));
}
sendPort.send((id, 1.0));
}).listen((_) {});
receivePort.listen((message) async { receivePort.listen((message) async {
final (int id, SftpCommand command) = message; final (int id, SftpCommand command) = message;
switch (command) { switch (command) {
@ -234,15 +260,8 @@ class SftpWorker {
} }
case DownloadFile(): case DownloadFile():
downloadController.add((id, command)); downloadController.add((id, command));
case Copy(:final filePath, :final copyToPath): case Copy():
try { copyController.add((id, command));
// TODO: complete this
sendPort.send((id, 0));
}
on SftpStatusError catch (e) {
sendPort.send((id, RemoteError(e.message, '')));
}
} }
}); });
} }
@ -325,12 +344,12 @@ class SftpWorker {
return controller.stream; return controller.stream;
} }
Future<void> copy(String filePath, String copyToPath) async { Stream<double> copy(String filePath, String copyToPath) {
final completer = Completer.sync(); final controller = StreamController<double>();
final id = _idCounter++; final id = _idCounter++;
_activeRequests[id] = completer; _activeRequests[id] = controller;
_commands.send((id, Copy(filePath, copyToPath))); _commands.send((id, Copy(filePath, copyToPath)));
await completer.future; return controller.stream;
} }
} }

View file

@ -1,7 +1,8 @@
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_provider.dart'; import 'package:fluxcloud/providers/sftp_loading_provider.dart';
import 'package:fluxcloud/providers/sftp_provider.dart';
import 'package:path_provider/path_provider.dart'; import 'package:path_provider/path_provider.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@ -15,20 +16,21 @@ class OperationButtons extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final sftpProvider = context.read<SftpProvider>(); final sftpProvider = context.read<SftpProvider>();
final sftpLoadingProvider = context.read<SftpLoadingProvider>();
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(); final filePaths = dirEntries.map((dirEntry) => '${sftpProvider.path}${dirEntry.filename}').toList();
sftpProvider.setCopyOrMoveFiles(filePaths, false); sftpLoadingProvider.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(); final filePaths = dirEntries.map((dirEntry) => '${sftpProvider.path}${dirEntry.filename}').toList();
sftpProvider.setCopyOrMoveFiles(filePaths, true); sftpLoadingProvider.setCopyOrMoveFiles(filePaths, true);
}, },
icon: Icon(Icons.copy) icon: Icon(Icons.copy)
), ),
@ -125,7 +127,7 @@ class OperationButtons extends StatelessWidget {
for (final dirEntry in dirEntries) { for (final dirEntry in dirEntries) {
try { try {
await for (final progress in sftpProvider.sftpWorker.downloadFile(dirEntry, sftpProvider.path, downloadsDir.path)) { await for (final progress in sftpProvider.sftpWorker.downloadFile(dirEntry, sftpProvider.path, downloadsDir.path)) {
sftpProvider.setDownloadProgress(progress); sftpLoadingProvider.setDownloadProgress(progress);
} }
} }
catch (e) { catch (e) {
@ -134,7 +136,7 @@ class OperationButtons extends StatelessWidget {
} }
} }
} }
sftpProvider.setDownloadProgress(null); sftpLoadingProvider.setDownloadProgress(null);
}, },
icon: Icon(Icons.download) icon: Icon(Icons.download)
) )