nc-photos/app/lib/widget/collection_browser/app_bar.dart
2023-08-15 00:03:34 +08:00

454 lines
15 KiB
Dart

part of '../collection_browser.dart';
class _AppBar extends StatelessWidget {
const _AppBar();
@override
Widget build(BuildContext context) {
final c = KiwiContainer().resolve<DiContainer>();
return _BlocBuilder(
buildWhen: (previous, current) =>
previous.items != current.items ||
previous.collection != current.collection,
builder: (context, state) {
final bloc = context.read<_Bloc>();
final adapter = CollectionAdapter.of(c, bloc.account, state.collection);
final canRename = adapter.isPermitted(CollectionCapability.rename);
final canManualCover =
adapter.isPermitted(CollectionCapability.manualCover);
final canShare = adapter.isPermitted(CollectionCapability.share);
return SliverAppBar(
floating: true,
expandedHeight: 160,
flexibleSpace: FlexibleSpaceBar(
background: const _AppBarCover(),
title: Text(
state.collection.name,
style: TextStyle(
color: Theme.of(context).appBarTheme.foregroundColor,
),
),
),
actions: [
if (canShare)
IconButton(
onPressed: () => _onSharePressed(context),
icon: const Icon(Icons.share),
tooltip: L10n.global().shareTooltip,
),
if (state.collection.isPendingSharedAlbum)
IconButton(
onPressed: () => _onAddToCollectionsViewPressed(context),
icon: const AssetIcon(asset.icAddCollectionsOutlined24),
tooltip: L10n.global().addToCollectionsViewTooltip,
),
if (state.items.isNotEmpty || canRename)
PopupMenuButton<_MenuOption>(
tooltip: MaterialLocalizations.of(context).moreButtonTooltip,
itemBuilder: (_) => [
if (canRename)
PopupMenuItem(
value: _MenuOption.edit,
child: Text(L10n.global().editTooltip),
),
if (canManualCover && adapter.isManualCover())
PopupMenuItem(
value: _MenuOption.unsetCover,
child: Text(L10n.global().unsetAlbumCoverTooltip),
),
if (state.items.isNotEmpty) ...[
PopupMenuItem(
value: _MenuOption.download,
child: Text(L10n.global().downloadTooltip),
),
PopupMenuItem(
value: _MenuOption.export,
child: Text(L10n.global().exportCollectionTooltip),
),
],
if (state.collection.contentProvider
is CollectionAlbumProvider &&
CollectionAdapter.of(c, bloc.account, state.collection)
.isPermitted(CollectionCapability.share))
PopupMenuItem(
value: _MenuOption.albumFixShare,
child: Text(L10n.global().fixSharesTooltip),
),
],
onSelected: (option) {
_onMenuSelected(context, option);
},
),
],
);
},
);
}
void _onMenuSelected(BuildContext context, _MenuOption option) {
switch (option) {
case _MenuOption.edit:
context.read<_Bloc>().add(const _BeginEdit());
break;
case _MenuOption.unsetCover:
context.read<_Bloc>().add(const _UnsetCover());
break;
case _MenuOption.download:
context.read<_Bloc>().add(const _Download());
break;
case _MenuOption.export:
_onExportSelected(context);
break;
case _MenuOption.albumFixShare:
_onAlbumFixShareSelected(context);
break;
}
}
Future<void> _onExportSelected(BuildContext context) async {
final bloc = context.read<_Bloc>();
final result = await showDialog<Collection>(
context: context,
builder: (_) => ExportCollectionDialog(
account: bloc.account,
collection: bloc.state.collection,
items: bloc.state.items,
),
);
if (result != null) {
Navigator.of(context).pop();
}
}
void _onAlbumFixShareSelected(BuildContext context) {
final bloc = context.read<_Bloc>();
final collection = bloc.state.collection;
final album = (collection.contentProvider as CollectionAlbumProvider).album;
Navigator.of(context).pushNamed(
AlbumShareOutlierBrowser.routeName,
arguments: AlbumShareOutlierBrowserArguments(bloc.account, album),
);
}
Future<void> _onSharePressed(BuildContext context) async {
final bloc = context.read<_Bloc>();
await showDialog(
context: context,
builder: (_) => ShareCollectionDialog(
account: bloc.account,
collection: bloc.state.collection,
),
);
}
Future<void> _onAddToCollectionsViewPressed(BuildContext context) async {
context.read<_Bloc>().add(const _ImportPendingSharedCollection());
}
}
class _AppBarCover extends StatelessWidget {
const _AppBarCover();
@override
Widget build(BuildContext context) {
return _BlocBuilder(
buildWhen: (previous, current) => previous.coverUrl != current.coverUrl,
builder: (context, state) {
if (state.coverUrl != null) {
return Opacity(
opacity:
Theme.of(context).brightness == Brightness.light ? 0.25 : 0.35,
child: FittedBox(
clipBehavior: Clip.hardEdge,
fit: BoxFit.cover,
child: CachedNetworkImage(
cacheManager: CoverCacheManager.inst,
imageUrl: state.coverUrl!,
httpHeaders: {
"Authorization":
AuthUtil.fromAccount(context.read<_Bloc>().account)
.toHeaderValue(),
},
fadeInDuration: const Duration(),
filterQuality: FilterQuality.high,
errorWidget: (context, url, error) {
// just leave it empty
return Container();
},
imageRenderMethodForWeb: ImageRenderMethodForWeb.HttpGet,
),
),
);
} else {
return const SizedBox.shrink();
}
},
);
}
}
@npLog
class _SelectionAppBar extends StatelessWidget {
const _SelectionAppBar();
@override
Widget build(BuildContext context) {
return _BlocBuilder(
buildWhen: (previous, current) =>
previous.selectedItems != current.selectedItems,
builder: (context, state) => SelectionAppBar(
count: state.selectedItems.length,
onClosePressed: () {
context.read<_Bloc>().add(const _SetSelectedItems(items: {}));
},
actions: [
IconButton(
icon: const Icon(Icons.share_outlined),
tooltip: L10n.global().shareTooltip,
onPressed: () {
_onSelectionSharePressed(context);
},
),
IconButton(
icon: const Icon(Icons.add_outlined),
tooltip: L10n.global().addItemToCollectionTooltip,
onPressed: () {
_onSelectionAddPressed(context);
},
),
PopupMenuButton<_SelectionMenuOption>(
tooltip: MaterialLocalizations.of(context).moreButtonTooltip,
itemBuilder: (context) => [
if (context.read<_Bloc>().isCollectionCapabilityPermitted(
CollectionCapability.manualItem) &&
state.isSelectionRemovable)
PopupMenuItem(
value: _SelectionMenuOption.removeFromAlbum,
child: Text(L10n.global().removeFromAlbumTooltip),
),
PopupMenuItem(
value: _SelectionMenuOption.download,
child: Text(L10n.global().downloadTooltip),
),
if (state.isSelectionManageableFile) ...[
PopupMenuItem(
value: _SelectionMenuOption.archive,
child: Text(L10n.global().archiveTooltip),
),
PopupMenuItem(
value: _SelectionMenuOption.delete,
child: Text(L10n.global().deleteTooltip),
),
],
],
onSelected: (option) {
_onSelectionMenuSelected(context, option);
},
),
],
),
);
}
void _onSelectionMenuSelected(
BuildContext context, _SelectionMenuOption option) {
switch (option) {
case _SelectionMenuOption.download:
context.read<_Bloc>().add(const _DownloadSelectedItems());
break;
case _SelectionMenuOption.removeFromAlbum:
context.read<_Bloc>().add(const _RemoveSelectedItemsFromCollection());
break;
case _SelectionMenuOption.archive:
context.read<_Bloc>().add(const _ArchiveSelectedItems());
break;
case _SelectionMenuOption.delete:
context.read<_Bloc>().add(const _DeleteSelectedItems());
break;
default:
_log.shout("[_onSelectionMenuSelected] Unknown option: $option");
break;
}
}
Future<void> _onSelectionSharePressed(BuildContext context) async {
final bloc = context.read<_Bloc>();
final selected = bloc.state.selectedItems
.whereType<_FileItem>()
.map((e) => e.file)
.toList();
if (selected.isEmpty) {
SnackBarManager().showSnackBar(SnackBar(
content: Text(L10n.global().shareSelectedEmptyNotification),
duration: k.snackBarDurationNormal,
));
return;
}
final result = await showDialog(
context: context,
builder: (context) => FileSharerDialog(
account: bloc.account,
files: selected,
),
);
if (result ?? false) {
bloc.add(const _SetSelectedItems(items: {}));
}
}
Future<void> _onSelectionAddPressed(BuildContext context) async {
final collection = await Navigator.of(context)
.pushNamed<Collection>(CollectionPicker.routeName);
if (collection == null) {
return;
}
context.read<_Bloc>().add(_AddSelectedItemsToCollection(collection));
}
}
class _EditAppBar extends StatelessWidget {
const _EditAppBar();
@override
Widget build(BuildContext context) {
final capabilitiesAdapter = CollectionAdapter.of(
KiwiContainer().resolve<DiContainer>(),
context.read<_Bloc>().account,
context.read<_Bloc>().state.collection,
);
return SliverAppBar(
floating: true,
expandedHeight: 160,
flexibleSpace: FlexibleSpaceBar(
background: const _AppBarCover(),
title: TextFormField(
initialValue: context.read<_Bloc>().state.currentEditName,
decoration: InputDecoration(
hintText: L10n.global().nameInputHint,
),
validator: (_) {
// use text in state here because the value might be wrong if user
// scrolled the app bar off screen
if (context.read<_Bloc>().state.currentEditName.isNotEmpty) {
return null;
} else {
return L10n.global().nameInputInvalidEmpty;
}
},
onChanged: (value) {
context.read<_Bloc>().add(_EditName(value));
},
style: TextStyle(
color: Theme.of(context).appBarTheme.foregroundColor,
),
),
),
leading: IconButton(
icon: const Icon(Icons.check),
color: Theme.of(context).colorScheme.primary,
tooltip: L10n.global().doneButtonTooltip,
onPressed: () {
context.read<_Bloc>().add(const _DoneEdit());
},
),
actions: [
if (capabilitiesAdapter.isPermitted(CollectionCapability.labelItem))
IconButton(
icon: const Icon(Icons.text_fields),
tooltip: L10n.global().albumAddTextTooltip,
onPressed: () => _onAddTextPressed(context),
),
if (capabilitiesAdapter.isPermitted(CollectionCapability.sort))
IconButton(
icon: const Icon(Icons.sort_by_alpha),
tooltip: L10n.global().sortTooltip,
onPressed: () => _onSortPressed(context),
),
],
);
}
Future<void> _onAddTextPressed(BuildContext context) async {
final result = await showDialog<String>(
context: context,
builder: (context) => SimpleInputDialog(
buttonText: MaterialLocalizations.of(context).saveButtonLabel,
),
);
if (result == null) {
return;
}
context.read<_Bloc>().add(_AddLabelToCollection(result));
}
Future<void> _onSortPressed(BuildContext context) async {
final current = context
.read<_Bloc>()
.state
.run((s) => s.editSort ?? s.collection.itemSort);
final result = await showDialog<CollectionItemSort>(
context: context,
builder: (context) => FancyOptionPicker(
title: Text(L10n.global().sortOptionDialogTitle),
items: [
_SortDialogParams(
L10n.global().sortOptionTimeDescendingLabel,
CollectionItemSort.dateDescending,
),
_SortDialogParams(
L10n.global().sortOptionTimeAscendingLabel,
CollectionItemSort.dateAscending,
),
_SortDialogParams(
L10n.global().sortOptionFilenameAscendingLabel,
CollectionItemSort.nameAscending,
),
_SortDialogParams(
L10n.global().sortOptionFilenameDescendingLabel,
CollectionItemSort.nameDescending,
),
if (current == CollectionItemSort.manual)
_SortDialogParams(
L10n.global().sortOptionManualLabel,
CollectionItemSort.manual,
),
]
.map((e) => FancyOptionPickerItem(
label: e.label,
isSelected: current == e.value,
onSelect: () {
Navigator.of(context).pop(e.value);
},
))
.toList(),
),
);
if (result == null) {
return;
}
context.read<_Bloc>().add(_EditSort(result));
}
}
enum _MenuOption {
edit,
unsetCover,
download,
export,
albumFixShare,
}
enum _SelectionMenuOption {
download,
removeFromAlbum,
archive,
delete,
}
class _SortDialogParams {
const _SortDialogParams(this.label, this.value);
final String label;
final CollectionItemSort value;
}