A framework for building convergent cross-platform Nextcloud clients using Flutter.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

151 lines
5.4 KiB

part of '../neon_files.dart';
2 years ago
/// Mode to operate the [FilesBrowserView] in.
enum FilesBrowserMode {
/// Default file browser mode.
///
/// When a file is selected it will be opened or downloaded.
browser,
/// Select directory.
selectDirectory,
/// Don't show file actions.
noActions,
}
2 years ago
class FilesBrowserView extends StatefulWidget {
const FilesBrowserView({
required this.bloc,
required this.filesBloc,
this.mode = FilesBrowserMode.browser,
2 years ago
super.key,
});
2 years ago
final FilesBrowserBloc bloc;
final FilesBloc filesBloc;
final FilesBrowserMode mode;
2 years ago
@override
State<FilesBrowserView> createState() => _FilesBrowserViewState();
}
class _FilesBrowserViewState extends State<FilesBrowserView> {
@override
void initState() {
widget.bloc.errors.listen((final error) {
NeonError.showSnackbar(context, error);
2 years ago
});
super.initState();
2 years ago
}
@override
Widget build(final BuildContext context) => ResultBuilder<List<WebDavFile>>.behaviorSubject(
subject: widget.bloc.files,
builder: (final context, final filesSnapshot) => StreamBuilder<PathUri>(
stream: widget.bloc.uri,
builder: (final context, final uriSnapshot) => StreamBuilder<List<FilesTask>>(
stream: widget.filesBloc.tasks,
builder: (final context, final tasksSnapshot) {
if (!uriSnapshot.hasData || !tasksSnapshot.hasData) {
return const SizedBox();
}
return ValueListenableBuilder(
valueListenable: widget.bloc.options.showHiddenFilesOption,
builder: (final context, final showHiddenFiles, final _) {
final files = filesSnapshot.data?.where((final file) {
var hideFile = false;
if (widget.mode == FilesBrowserMode.selectDirectory && !file.isDirectory) {
hideFile = true;
}
if (!showHiddenFiles && file.isHidden) {
hideFile = true;
}
return !hideFile;
}).toList();
return BackButtonListener(
onBackButtonPressed: () async {
final parent = uriSnapshot.requireData.parent;
if (parent != null) {
widget.bloc.setPath(parent);
return true;
}
return false;
},
child: SortBoxBuilder<FilesSortProperty, WebDavFile>(
sortBox: filesSortBox,
sortProperty: widget.bloc.options.filesSortPropertyOption,
sortBoxOrder: widget.bloc.options.filesSortBoxOrderOption,
presort: const {
(property: FilesSortProperty.isFolder, order: SortBoxOrder.ascending),
},
input: files,
builder: (final context, final sorted) {
final uploadingTaskTiles = buildUploadTasks(tasksSnapshot.requireData, sorted);
return NeonListView(
scrollKey: 'files-${uriSnapshot.requireData.path}',
itemCount: sorted.length,
itemBuilder: (final context, final index) {
final file = sorted[index];
final matchingTask = tasksSnapshot.requireData.firstWhereOrNull(
(final task) => file.name == task.uri.name && widget.bloc.uri.value == task.uri.parent,
);
final details = matchingTask != null
? FileDetails.fromTask(
task: matchingTask,
file: file,
)
: FileDetails.fromWebDav(
file: file,
);
return FileListTile(
bloc: widget.filesBloc,
browserBloc: widget.bloc,
details: details,
mode: widget.mode,
);
},
isLoading: filesSnapshot.isLoading,
error: filesSnapshot.error,
onRefresh: widget.bloc.refresh,
topScrollingChildren: [
FilesBrowserNavigator(
uri: uriSnapshot.requireData,
bloc: widget.bloc,
),
...uploadingTaskTiles,
],
);
},
),
);
},
);
},
2 years ago
),
),
);
Iterable<Widget> buildUploadTasks(final List<FilesTask> tasks, final List<WebDavFile> files) sync* {
for (final task in tasks) {
if (task is! FilesUploadTask) {
continue;
}
yield FileListTile(
bloc: widget.filesBloc,
browserBloc: widget.bloc,
details: FileDetails.fromUploadTask(
task: task,
),
);
}
}
2 years ago
}