import 'dart:collection'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/app_localizations.dart'; import 'package:nc_photos/bloc/ls_dir.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/exception_util.dart' as exception_util; import 'package:nc_photos/k.dart' as k; import 'package:nc_photos/snack_bar_manager.dart'; import 'package:nc_photos/theme.dart'; import 'package:path/path.dart' as path; mixin DirPickerMixin on State { @override initState() { super.initState(); _root = LsDirBlocItem(File(path: getPickerRoot()), []); _initBloc(); } @protected Widget buildDirPicker(BuildContext context) { return BlocListener( bloc: _bloc, listener: (context, state) => _onStateChange(context, state), child: BlocBuilder( bloc: _bloc, builder: (context, state) => _buildContent(context, state), ), ); } @protected String getPickerRoot(); @protected Account getAccount(); @protected List getPickedDirs() => UnmodifiableListView(_picks); @protected bool canPickDir(File file) => true; @protected void pickAll(List dirs) { _picks.addAll(dirs); } void _initBloc() { _log.info("[_initBloc] Initialize bloc"); _bloc = LsDirBloc(); _navigateInto(File(path: getPickerRoot())); } Widget _buildContent(BuildContext context, LsDirBlocState state) { return Stack( children: [ Align( alignment: Alignment.center, child: state is LsDirBlocLoading ? Container() : _buildList(context, state), ), if (state is LsDirBlocLoading) Align( alignment: Alignment.topCenter, child: const LinearProgressIndicator(), ), ], ); } Widget _buildList(BuildContext context, LsDirBlocState state) { final isTopLevel = _currentPath == getPickerRoot(); return Theme( data: Theme.of(context).copyWith( accentColor: AppTheme.getOverscrollIndicatorColor(context), ), child: AnimatedSwitcher( duration: k.animationDurationNormal, // see AnimatedSwitcher.defaultLayoutBuilder layoutBuilder: (currentChild, previousChildren) => Stack( children: [ ...previousChildren, if (currentChild != null) currentChild, ], alignment: Alignment.topLeft, ), child: ListView.separated( key: Key(_currentPath), itemBuilder: (context, index) { if (!isTopLevel && index == 0) { return ListTile( dense: true, leading: const SizedBox(width: 24), title: Text(L10n.of(context).rootPickerNavigateUpItemText), onTap: () { try { _navigateInto(File(path: path.dirname(_currentPath))); } catch (e) { SnackBarManager().showSnackBar(SnackBar( content: Text(exception_util.toUserString(e, context)), duration: k.snackBarDurationNormal, )); } }, ); } else { return _buildItem( context, state.items[index - (isTopLevel ? 0 : 1)]); } }, separatorBuilder: (context, index) => const Divider(), itemCount: state.items.length + (isTopLevel ? 0 : 1), ), ), ); } Widget _buildItem(BuildContext context, LsDirBlocItem item) { final canPick = canPickDir(item.file); final pickState = _isItemPicked(item); IconData? iconData; if (canPick) { switch (pickState) { case _PickState.picked: iconData = Icons.check_box; break; case _PickState.childPicked: iconData = Icons.indeterminate_check_box; break; case _PickState.notPicked: default: iconData = Icons.check_box_outline_blank; break; } } return ListTile( dense: true, leading: canPick ? IconButton( icon: AnimatedSwitcher( duration: k.animationDurationShort, transitionBuilder: (child, animation) => ScaleTransition(child: child, scale: animation), child: Icon( iconData, key: ValueKey(pickState), ), ), onPressed: () { if (pickState == _PickState.picked) { _unpick(item); } else { _pick(item); } }, ) : IconButton( icon: Icon(null), onPressed: null, ), title: Text(path.basename(item.file.path)), trailing: item.children?.isNotEmpty == true ? const Icon(Icons.arrow_forward_ios) : null, onTap: item.children?.isNotEmpty == true ? () { try { _navigateInto(item.file); } catch (e) { SnackBarManager().showSnackBar(SnackBar( content: Text(exception_util.toUserString(e, context)), duration: k.snackBarDurationNormal, )); } } : null, ); } void _onStateChange(BuildContext context, LsDirBlocState state) { if (state is LsDirBlocSuccess) { if (!_fillResult(_root, state)) { _log.shout("[_onStateChange] Failed while _fillResult" + (kDebugMode ? ", root:\n${_root.toString(isDeep: true)}\nstate: ${state.root.path}" : "")); } } else if (state is LsDirBlocFailure) { SnackBarManager().showSnackBar(SnackBar( content: Text(exception_util.toUserString(state.exception, context)), duration: k.snackBarDurationNormal, )); } } /// Fill query results from bloc to our item tree bool _fillResult(LsDirBlocItem root, LsDirBlocSuccess state) { if (root.file.path == state.root.path) { root.children = state.items; return true; } else if (state.root.path.startsWith(root.file.path)) { for (final child in root.children ?? []) { if (_fillResult(child, state)) { return true; } } return false; } else { // not us, not child of us return false; } } /// Pick an item void _pick(LsDirBlocItem item) { setState(() { _picks.add(item.file); _picks = _optimizePicks(_root); }); _log.fine("[_pick] Picked: ${_pickListToString(_picks)}"); } /// Optimize the picked array /// /// 1) If a parent directory is picked, all children will be ignored List _optimizePicks(LsDirBlocItem item) { if (_picks.any((element) => element.path == item.file.path)) { // this dir is explicitly picked, nothing more to do return [item.file]; } if (item.children == null || item.children!.isEmpty) { return []; } final products = []; for (final i in item.children!) { products.addAll(_optimizePicks(i)); } // // see if all children are being picked // if (item != _root && // products.length >= item.children.length && // item.children.every((element) => products.contains(element))) { // // all children are being picked, add [item] to list and remove its // // children // _log.fine( // "[_optimizePicks] All children under '${item.file.path}' are being picked, optimized"); // return products // .where((element) => !item.children.contains(element)) // .toList() // ..add(item); // } return products; } /// Unpick an item void _unpick(LsDirBlocItem item) { setState(() { if (_picks.any((element) => element.path == item.file.path)) { // ourself is being picked, simple _picks = _picks.where((element) => element.path != item.file.path).toList(); } else { // Look for the closest picked dir final parents = _picks .where((element) => item.file.path.startsWith(element.path)) .toList() ..sort((a, b) => b.path.length.compareTo(a.path.length)); final parent = parents.first; try { _picks.addAll(_pickedAllExclude(path: parent.path, exclude: item) .map((e) => e.file)); _picks.removeWhere((element) => identical(element, parent)); } catch (_) { SnackBarManager().showSnackBar(SnackBar( content: Text(L10n.of(context).rootPickerUnpickFailureNotification))); } } }); _log.fine("[_unpick] Picked: ${_pickListToString(_picks)}"); } /// Return a list where all children of [path] or [item], except [exclude], /// are picked /// /// Either [path] or [item] must be set, If both are set, [item] takes /// priority List _pickedAllExclude({ String? path, LsDirBlocItem? item, required LsDirBlocItem exclude, }) { assert(path != null || item != null); if (item == null) { final item = _findChildItemByPath(_root, path!); return _pickedAllExclude(item: item, exclude: exclude); } if (item.file.path == exclude.file.path) { return []; } _log.fine( "[_pickedAllExclude] Unpicking '${item.file.path}' and picking children"); final products = []; for (final i in item.children ?? []) { if (exclude.file.path == i.file.path || exclude.file.path.startsWith("${i.file.path}/")) { // [i] is a parent of exclude products.addAll(_pickedAllExclude(item: i, exclude: exclude)); } else { products.add(i); } } return products; } /// Return the child/grandchild/... item of [parent] with [path] LsDirBlocItem _findChildItemByPath(LsDirBlocItem parent, String path) { if (path == parent.file.path) { return parent; } for (final c in parent.children ?? []) { if (path == c.file.path || path.startsWith("${c.file.path}/")) { return _findChildItemByPath(c, path); } } // ??? _log.shout( "[_findChildItemByPath] Failed finding child item for '$path' under '${parent.file.path}'"); throw ArgumentError("Path not found"); } _PickState _isItemPicked(LsDirBlocItem item) { var product = _PickState.notPicked; for (final p in _picks) { // exact match, or parent is picked if (p.path == item.file.path || item.file.path.startsWith("${p.path}/")) { product = _PickState.picked; // no need to check the remaining ones break; } if (p.path.startsWith("${item.file.path}/")) { product = _PickState.childPicked; } } if (product == _PickState.childPicked) {} return product; } /// Return the string representation of a list of LsDirBlocItem static String _pickListToString(List items) => "['${items.map((e) => e.path).join('\', \'')}']"; void _navigateInto(File file) { _currentPath = file.path; _bloc.add(LsDirBlocQuery(getAccount(), file, depth: 2)); } late LsDirBloc _bloc; late LsDirBlocItem _root; /// Track where the user is navigating in [_backingFiles] late String _currentPath; var _picks = []; static final _log = Logger("widget.dir_picker_mixin.DirPickerMixin"); } enum _PickState { notPicked, picked, childPicked, }