Compare commits

...

4 commits

Author SHA1 Message Date
RafayAhmad7548
a66ed70532 move working 2025-08-14 15:45:41 +05:00
RafayAhmad7548
7ea9e98789 remove test and remove todo from my diagnostics 2025-08-14 07:45:59 +05:00
RafayAhmad7548
1ccadcf200 make upload/download work sequentialy so animation stays cool 2025-08-14 07:34:45 +05:00
RafayAhmad7548
08bbeb2619 provider yayy 2025-08-14 07:07:50 +05:00
9 changed files with 348 additions and 216 deletions

View file

@ -1 +1,4 @@
include: package:flutter_lints/flutter.yaml include: package:flutter_lints/flutter.yaml
analyzer:
errors:
todo: ignore

View file

@ -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),

View file

@ -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,101 +24,103 @@ 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,
alignment: Alignment.center, builder: (_, uploadProgress, __) => uploadProgress != null ? Stack(
children: [ alignment: Alignment.center,
TweenAnimationBuilder<double>( children: [
tween: Tween(begin: 0, end: _uploadProgress), TweenAnimationBuilder<double>(
duration: Duration(milliseconds: 300), tween: Tween(begin: 0, end: uploadProgress),
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,) duration: Duration(milliseconds: 300),
), builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
IconButton( ),
onPressed: () { IconButton(
// TODO: show upload details here onPressed: () {
}, // TODO: show upload details here
icon: Icon(Icons.upload) },
), icon: Icon(Icons.upload)
] ),
]
) : const SizedBox.shrink(),
), ),
if (_downloadProgress != null) Selector<SftpProvider, double?>(
Stack( selector: (_, sftpProvider) => sftpProvider.downloadProgress,
alignment: Alignment.center, builder: (_, downloadProgress, __) => downloadProgress != null ? Stack(
children: [ alignment: Alignment.center,
TweenAnimationBuilder<double>( children: [
tween: Tween(begin: 0, end: _downloadProgress), TweenAnimationBuilder<double>(
duration: Duration(milliseconds: 300), tween: Tween(begin: 0, end: downloadProgress),
builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,) duration: Duration(milliseconds: 300),
), builder: (context, value, _) => CircularProgressIndicator(strokeWidth: 3, value: value,)
IconButton( ),
onPressed: () { IconButton(
// TODO: show donwload details here onPressed: () {
}, // TODO: show donwload details here
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>(
duration: Duration(milliseconds: 300), builder: (_, sftpProvider, __) => AnimatedSwitcher(
transitionBuilder: (child, animation) { duration: Duration(milliseconds: 300),
final curved = CurvedAnimation( transitionBuilder: (child, animation) {
parent: animation, final curved = CurvedAnimation(
curve: Curves.fastOutSlowIn parent: animation,
); curve: Curves.fastOutSlowIn
return FadeTransition(
opacity: curved,
child: ScaleTransition(
scale: Tween<double>(
begin: 0.92,
end: 1
).animate(curved),
child: child,
),
);
},
child: _isLoading ? Center(child: CircularProgressIndicator()) : ListView.builder(
key: ValueKey(path),
itemCount: _dirContents.length,
itemBuilder: (context, index) {
final dirEntry = _dirContents[index];
return ListTile(
leading: Icon(dirEntry.attr.isDirectory ? Icons.folder : Icons.description),
title: Text(dirEntry.filename),
trailing: OperationButtons(sftpWorker: widget.sftpWorker, path: path, dirEntries: [dirEntry], listDir: _listDir, setDownloadProgress: _setDownloadProgress,),
onTap: () {
if (dirEntry.attr.isDirectory) {
path = '$path${dirEntry.filename}/';
_listDir();
}
},
); );
}, return FadeTransition(
) opacity: curved,
child: ScaleTransition(
scale: Tween<double>(
begin: 0.92,
end: 1
).animate(curved),
child: child,
),
);
},
child: sftpProvider.isLoading ? Center(child: CircularProgressIndicator()) : ListView.builder(
key: ValueKey(sftpProvider.path),
itemCount: sftpProvider.dirContents.length,
itemBuilder: (context, index) {
final dirEntry = sftpProvider.dirContents[index];
return ListTile(
leading: Icon(dirEntry.attr.isDirectory ? Icons.folder : Icons.description),
title: Text(dirEntry.filename),
trailing: OperationButtons(dirEntries: [dirEntry],),
onTap: () {
if (dirEntry.attr.isDirectory) {
sftpProvider.goToDir('${sftpProvider.path}${dirEntry.filename}/');
}
},
);
},
)
),
), ),
) )
); );
@ -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
View 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();
}
}

View file

@ -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,6 +116,66 @@ class SftpWorker {
static void _sftpCmdHandler(SendPort sendPort, ReceivePort receivePort, SftpClient sftpClient) { static void _sftpCmdHandler(SendPort sendPort, ReceivePort receivePort, SftpClient sftpClient) {
final StreamController<(int, DownloadFile)> downloadController = StreamController();
downloadController.stream.asyncMap((cmd) async {
final (int id, DownloadFile downloadCmd) = cmd;
try {
final localFile = File('${downloadCmd.downloadPath}/${downloadCmd.file.filename}');
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) {
sendPort.send((id, RemoteError(e.message, '')));
}
sendPort.send((id, 1.0));
}).listen((_) {});
final StreamController<(int, UploadFile)> uploadController = StreamController();
uploadController.stream.asyncMap((cmd) async {
final (int id, UploadFile uploadCmd) = cmd;
try {
final file = File(uploadCmd.filePath);
final fileSize = await file.length();
final remoteFile = await sftpClient.open(
'${uploadCmd.path}${basename(uploadCmd.filePath)}',
mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive
);
bool timeout = true;
await remoteFile.write(
file.openRead().cast(),
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) {
@ -120,30 +187,8 @@ class SftpWorker {
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): case UploadFile():
try { uploadController.add((id, command));
final file = File(filePath);
final fileSize = await file.length();
final remoteFile = await sftpClient.open(
'$path${basename(filePath)}',
mode: SftpFileOpenMode.create | SftpFileOpenMode.write | SftpFileOpenMode.exclusive
);
bool timeout = true;
await remoteFile.write(
file.openRead().cast(),
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));
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;
}
} }

View file

@ -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)
) )

View file

@ -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

View file

@ -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:

View file

@ -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);
});
}