mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-23 17:26:18 +01:00
383 lines
12 KiB
Dart
383 lines
12 KiB
Dart
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<T extends StatefulWidget> on State<T> {
|
|
@override
|
|
initState() {
|
|
super.initState();
|
|
_root = LsDirBlocItem(File(path: getPickerRoot()), []);
|
|
_initBloc();
|
|
}
|
|
|
|
@protected
|
|
Widget buildDirPicker(BuildContext context) {
|
|
return BlocListener<LsDirBloc, LsDirBlocState>(
|
|
bloc: _bloc,
|
|
listener: (context, state) => _onStateChange(context, state),
|
|
child: BlocBuilder<LsDirBloc, LsDirBlocState>(
|
|
bloc: _bloc,
|
|
builder: (context, state) => _buildContent(context, state),
|
|
),
|
|
);
|
|
}
|
|
|
|
@protected
|
|
String getPickerRoot();
|
|
|
|
@protected
|
|
Account getAccount();
|
|
|
|
@protected
|
|
List<File> getPickedDirs() => UnmodifiableListView(_picks);
|
|
|
|
@protected
|
|
bool canPickDir(File file) => true;
|
|
|
|
@protected
|
|
void pickAll(List<File> 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: <Widget>[
|
|
...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.global().rootPickerNavigateUpItemText),
|
|
onTap: () {
|
|
try {
|
|
_navigateInto(File(path: path.dirname(_currentPath)));
|
|
} catch (e) {
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
content: Text(exception_util.toUserString(e)),
|
|
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)),
|
|
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)),
|
|
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 ?? <LsDirBlocItem>[]) {
|
|
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<File> _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 = <File>[];
|
|
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.global().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<LsDirBlocItem> _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 = <LsDirBlocItem>[];
|
|
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<File> 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 = <File>[];
|
|
|
|
static final _log = Logger("widget.dir_picker_mixin.DirPickerMixin");
|
|
}
|
|
|
|
enum _PickState {
|
|
notPicked,
|
|
picked,
|
|
childPicked,
|
|
}
|