#1216 settings: hidden path filters are merged with others and can be toggled

This commit is contained in:
Thibault Deckers 2024-10-03 23:00:19 +02:00
parent d859887319
commit c07dc36d26
4 changed files with 83 additions and 177 deletions

View file

@ -946,14 +946,9 @@
"settingsHiddenItemsTile": "Hidden items",
"settingsHiddenItemsPageTitle": "Hidden Items",
"settingsHiddenItemsTabFilters": "Hidden Filters",
"settingsHiddenFiltersBanner": "Photos and videos matching hidden filters will not appear in your collection.",
"settingsHiddenFiltersEmpty": "No hidden filters",
"settingsHiddenItemsTabPaths": "Hidden Paths",
"settingsHiddenPathsBanner": "Photos and videos in these folders, or any of their subfolders, will not appear in your collection.",
"addPathTooltip": "Add path",
"settingsStorageAccessTile": "Storage access",
"settingsStorageAccessPageTitle": "Storage Access",
"settingsStorageAccessBanner": "Some directories require an explicit access grant to modify files in them. You can review here directories to which you previously gave access.",

View file

@ -55,7 +55,6 @@ class _SearchPageState extends State<SearchPage> {
_unregisterWidget(widget);
widget.animation.removeStatusListener(_onAnimationStatusChanged);
_searchFieldFocusNode.dispose();
widget.delegate.dispose();
super.dispose();
}

View file

@ -21,6 +21,14 @@ class SearchPageRoute<T> extends PageRoute<T> {
delegate.route = this;
}
@override
void dispose() {
// `delegate` is always created by the caller at route creation time,
// so it should always be disposed when the route is disposed
delegate.dispose();
super.dispose();
}
final AvesSearchDelegate delegate;
@override

View file

@ -1,17 +1,12 @@
import 'package:aves/model/filters/filters.dart';
import 'package:aves/model/filters/path.dart';
import 'package:aves/model/settings/settings.dart';
import 'package:aves/theme/durations.dart';
import 'package:aves/theme/icons.dart';
import 'package:aves/widgets/common/basic/font_size_icon_theme.dart';
import 'package:aves/widgets/common/basic/scaffold.dart';
import 'package:aves/widgets/common/extensions/build_context.dart';
import 'package:aves/widgets/common/identity/aves_filter_chip.dart';
import 'package:aves/widgets/common/identity/buttons/outlined_button.dart';
import 'package:aves/widgets/common/identity/empty.dart';
import 'package:aves/widgets/settings/privacy/file_picker/file_picker_page.dart';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';
import 'package:provider/provider.dart';
class HiddenItemsPage extends StatelessWidget {
@ -22,48 +17,17 @@ class HiddenItemsPage extends StatelessWidget {
@override
Widget build(BuildContext context) {
final l10n = context.l10n;
final tabs = <(Tab, Widget)>[
(
Tab(text: l10n.settingsHiddenItemsTabFilters),
const _HiddenFilters(),
),
(
Tab(text: l10n.settingsHiddenItemsTabPaths),
const _HiddenPaths(),
),
];
return DefaultTabController(
length: tabs.length,
child: AvesScaffold(
return AvesScaffold(
appBar: AppBar(
automaticallyImplyLeading: !settings.useTvLayout,
title: Text(l10n.settingsHiddenItemsPageTitle),
bottom: TabBar(
tabs: tabs.map((t) => t.$1).toList(),
),
),
body: SafeArea(
child: TabBarView(
children: tabs.map((t) => t.$2).toList(),
),
),
),
);
}
}
class _HiddenFilters extends StatelessWidget {
const _HiddenFilters();
@override
Widget build(BuildContext context) {
bool filterPredicate(CollectionFilter v) => v is! PathFilter;
return Selector<Settings, Set<CollectionFilter>>(
selector: (context, s) => settings.hiddenFilters.where(filterPredicate).toSet(),
child: Selector<Settings, Set<CollectionFilter>>(
selector: (context, s) => settings.hiddenFilters.toSet(),
builder: (context, activatedHiddenFilters, child) {
return Selector<Settings, Set<CollectionFilter>>(
selector: (context, s) => settings.deactivatedHiddenFilters.where(filterPredicate).toSet(),
selector: (context, s) => settings.deactivatedHiddenFilters.toSet(),
builder: (context, deactivatedHiddenFilters, child) {
final allHiddenFilters = {
...activatedHiddenFilters,
@ -132,68 +96,8 @@ class _HiddenFilters extends StatelessWidget {
},
);
},
);
}
}
class _HiddenPaths extends StatelessWidget {
const _HiddenPaths();
@override
Widget build(BuildContext context) {
return Selector<Settings, Set<PathFilter>>(
selector: (context, s) => {
...settings.hiddenFilters,
...settings.deactivatedHiddenFilters,
}.whereType<PathFilter>().toSet(),
builder: (context, hiddenPaths, child) {
final pathList = hiddenPaths.toList()..sort();
return Column(
children: [
_Banner(bannerText: context.l10n.settingsHiddenPathsBanner),
const Divider(height: 0),
Flexible(
child: ListView(
shrinkWrap: true,
children: [
...pathList.map((pathFilter) {
void onPressed() => settings.changeFilterVisibility({pathFilter}, true);
return ListTile(
title: Text(pathFilter.path),
dense: true,
trailing: IconButton(
icon: const Icon(AIcons.clear),
onPressed: onPressed,
tooltip: context.l10n.actionRemove,
),
onTap: settings.useTvLayout ? onPressed : null,
);
}),
],
),
),
const Divider(height: 0),
const SizedBox(height: 8),
AvesOutlinedButton(
icon: const Icon(AIcons.add),
label: context.l10n.addPathTooltip,
onPressed: () async {
final path = await Navigator.maybeOf(context)?.push(
MaterialPageRoute<String>(
settings: const RouteSettings(name: FilePickerPage.routeName),
builder: (context) => const FilePickerPage(),
),
);
// wait for the dialog to hide as applying the change may block the UI
await Future.delayed(ADurations.pageTransitionLoose * timeDilation);
if (path != null && path.isNotEmpty) {
settings.changeFilterVisibility({PathFilter(path)}, false);
}
},
),
],
);
},
);
}
}