part of '../neon_files.dart'; abstract interface class FilesBlocEvents { void uploadFile(final PathUri uri, final String localPath); void syncFile(final PathUri uri); void openFile(final PathUri uri, final String etag, final String? mimeType); void shareFileNative(final PathUri uri, final String etag); void delete(final PathUri uri); void rename(final PathUri uri, final String name); void move(final PathUri uri, final PathUri destination); void copy(final PathUri uri, final PathUri destination); void addFavorite(final PathUri uri); void removeFavorite(final PathUri uri); } abstract interface class FilesBlocStates { BehaviorSubject> get tasks; } class FilesBloc extends InteractiveBloc implements FilesBlocEvents, FilesBlocStates { FilesBloc( this.options, this.account, ) { options.uploadQueueParallelism.addListener(_uploadParallelismListener); options.downloadQueueParallelism.addListener(_downloadParallelismListener); } final FilesAppSpecificOptions options; final Account account; late final browser = getNewFilesBrowserBloc(); final _uploadQueue = Queue(); final _downloadQueue = Queue(); @override void dispose() { _uploadQueue.dispose(); _downloadQueue.dispose(); unawaited(tasks.close()); options.uploadQueueParallelism.removeListener(_uploadParallelismListener); options.downloadQueueParallelism.removeListener(_downloadParallelismListener); super.dispose(); } @override BehaviorSubject> tasks = BehaviorSubject>.seeded([]); @override void addFavorite(final PathUri uri) { wrapAction( () async => account.client.webdav.proppatch( uri, set: WebDavProp(ocfavorite: 1), ), ); } @override void copy(final PathUri uri, final PathUri destination) { wrapAction(() async => account.client.webdav.copy(uri, destination)); } @override void delete(final PathUri uri) { wrapAction(() async => account.client.webdav.delete(uri)); } @override void move(final PathUri uri, final PathUri destination) { wrapAction(() async => account.client.webdav.move(uri, destination)); } @override void openFile(final PathUri uri, final String etag, final String? mimeType) { wrapAction( () async { final file = await _cacheFile(uri, etag); final result = await OpenFile.open(file.path, type: mimeType); if (result.type != ResultType.done) { throw const UnableToOpenFileException(); } }, disableTimeout: true, ); } @override void shareFileNative(final PathUri uri, final String etag) { wrapAction( () async { final file = await _cacheFile(uri, etag); await Share.shareXFiles([XFile(file.path)]); }, disableTimeout: true, ); } @override Future refresh() async { await browser.refresh(); } @override void removeFavorite(final PathUri uri) { wrapAction( () async => account.client.webdav.proppatch( uri, set: WebDavProp(ocfavorite: 0), ), ); } @override void rename(final PathUri uri, final String name) { wrapAction( () async => account.client.webdav.move( uri, uri.rename(name), ), ); } @override void syncFile(final PathUri uri) { wrapAction( () async { final file = File( p.joinAll([ await NeonPlatform.instance.userAccessibleAppDataPath, account.humanReadableID, 'files', ...uri.pathSegments, ]), ); if (!file.parent.existsSync()) { file.parent.createSync(recursive: true); } await _downloadFile(uri, file); }, disableTimeout: true, ); } @override void uploadFile(final PathUri uri, final String localPath) { wrapAction( () async { final task = FilesUploadTask( uri: uri, file: File(localPath), ); tasks.add(tasks.value..add(task)); await _uploadQueue.add(() => task.execute(account.client)); tasks.add(tasks.value..remove(task)); }, disableTimeout: true, ); } Future _cacheFile(final PathUri uri, final String etag) async { final cacheDir = TargetPlatform.aurora == defaultTargetPlatform ? await getTemporaryDirectory() : await getApplicationCacheDirectory(); final file = File(p.join(cacheDir.path, 'files', etag.replaceAll('"', ''), uri.name)); if (!file.existsSync()) { debugPrint('Downloading $uri since it does not exist'); if (!file.parent.existsSync()) { await file.parent.create(recursive: true); } await _downloadFile(uri, file); } return file; } Future _downloadFile( final PathUri uri, final File file, ) async { final task = FilesDownloadTask( uri: uri, file: file, ); tasks.add(tasks.value..add(task)); await _downloadQueue.add(() => task.execute(account.client)); tasks.add(tasks.value..remove(task)); } FilesBrowserBloc getNewFilesBrowserBloc({ final PathUri? initialUri, }) => FilesBrowserBloc( options, account, initialPath: initialUri, ); void _downloadParallelismListener() { _downloadQueue.parallel = options.downloadQueueParallelism.value; } void _uploadParallelismListener() { _uploadQueue.parallel = options.uploadQueueParallelism.value; } } @immutable class UnableToOpenFileException extends NeonException { const UnableToOpenFileException(); @override NeonExceptionDetails get details => NeonExceptionDetails( getText: (final context) => FilesLocalizations.of(context).errorUnableToOpenFile, ); }