import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/api/api_util.dart' as api_util; import 'package:nc_photos/app_localizations.dart'; import 'package:nc_photos/entity/album.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/pref.dart'; import 'package:nc_photos/theme.dart'; import 'package:nc_photos/widget/album_viewer_app_bar.dart'; import 'package:nc_photos/widget/popup_menu_zoom.dart'; import 'package:nc_photos/widget/selectable_item_stream_list_mixin.dart'; mixin AlbumViewerMixin on SelectableItemStreamListMixin { @override initState() { super.initState(); _thumbZoomLevel = Pref.inst().getAlbumViewerZoomLevelOr(0); } @protected void initCover(Account account, List backingFiles) { try { final coverFile = backingFiles.firstWhere((element) => element.hasPreview ?? false); _coverPreviewUrl = api_util.getFilePreviewUrl(account, coverFile, width: 1024, height: 600); } catch (_) {} } @protected Widget buildNormalAppBar( BuildContext context, Account account, Album album, { List? actions, List> Function(BuildContext)? menuItemBuilder, void Function(int)? onSelectedMenuItem, }) { return AlbumViewerAppBar( account: account, album: album, coverPreviewUrl: _coverPreviewUrl, actions: [ PopupMenuButton( icon: const Icon(Icons.photo_size_select_large), tooltip: L10n.of(context).zoomTooltip, itemBuilder: (context) => [ PopupMenuZoom( initialValue: _thumbZoomLevel, minValue: 0, maxValue: 2, onChanged: (value) { setState(() { _thumbZoomLevel = value.round(); }); Pref.inst().setAlbumViewerZoomLevel(_thumbZoomLevel); }, ), ], ), ...(actions ?? []), PopupMenuButton( tooltip: MaterialLocalizations.of(context).moreButtonTooltip, itemBuilder: (context) => [ PopupMenuItem( value: -1, child: Text(L10n.of(context).editAlbumMenuLabel), ), ...(menuItemBuilder?.call(context) ?? []), ], onSelected: (option) { if (option >= 0) { onSelectedMenuItem?.call(option); } else { switch (option) { case _menuValueEdit: _onAppBarEditPressed(context, album); break; default: _log.shout("[buildNormalAppBar] Unknown value: $option"); break; } } }, ), ], ); } @protected Widget buildSelectionAppBar(BuildContext context, List actions) { return Theme( data: Theme.of(context).copyWith( appBarTheme: AppTheme.getContextualAppBarTheme(context), ), child: SliverAppBar( pinned: true, leading: IconButton( icon: const Icon(Icons.close), tooltip: MaterialLocalizations.of(context).closeButtonTooltip, onPressed: () { setState(() { clearSelectedItems(); }); }, ), title: Text( L10n.of(context).selectionAppBarTitle(selectedListItems.length)), actions: actions, ), ); } @protected Widget buildEditAppBar( BuildContext context, Account account, Album album, { List? actions, }) { return AlbumViewerEditAppBar( account: account, album: album, coverPreviewUrl: _coverPreviewUrl, actions: actions, onDonePressed: () { if (validateEditMode()) { setState(() { _isEditMode = false; }); doneEditMode(); } }, onAlbumNameSaved: (value) { _editFormValue.name = value; }, ); } @protected get isEditMode => _isEditMode; @protected @mustCallSuper void enterEditMode() {} /// Validates the pending modifications @protected bool validateEditMode(); @protected void doneEditMode(); /// Return a new album with the edits @protected Album makeEdited(Album album) { return album.copyWith( name: _editFormValue.name, ); } @protected int get thumbSize { switch (_thumbZoomLevel) { case 1: return 176; case 2: return 256; case 0: default: return 112; } } void _onAppBarEditPressed(BuildContext context, Album album) { setState(() { _isEditMode = true; enterEditMode(); _editFormValue = _EditFormValue(); }); } String? _coverPreviewUrl; var _thumbZoomLevel = 0; var _isEditMode = false; var _editFormValue = _EditFormValue(); static final _log = Logger("widget.album_viewer_mixin.AlbumViewerMixin"); static const _menuValueEdit = -1; } class _EditFormValue { late String name; }