import 'package:aves/model/image_entry.dart'; import 'package:aves/model/source/collection_lens.dart'; import 'package:aves/services/android_app_service.dart'; import 'package:aves/services/image_file_service.dart'; import 'package:aves/utils/durations.dart'; import 'package:aves/widgets/common/action_delegates/feedback.dart'; import 'package:aves/widgets/common/action_delegates/permission_aware.dart'; import 'package:aves/widgets/common/action_delegates/rename_entry_dialog.dart'; import 'package:aves/widgets/common/aves_dialog.dart'; import 'package:aves/widgets/common/entry_actions.dart'; import 'package:aves/widgets/common/image_providers/uri_image_provider.dart'; import 'package:aves/widgets/fullscreen/fullscreen_debug_page.dart'; import 'package:aves/widgets/fullscreen/source_viewer_page.dart'; import 'package:flutter/material.dart'; import 'package:flutter/scheduler.dart'; import 'package:flutter/services.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:pdf/pdf.dart'; import 'package:pdf/widgets.dart' as pdf; import 'package:pedantic/pedantic.dart'; import 'package:printing/printing.dart'; class EntryActionDelegate with FeedbackMixin, PermissionAwareMixin { final CollectionLens collection; final VoidCallback showInfo; EntryActionDelegate({ @required this.collection, @required this.showInfo, }); bool get hasCollection => collection != null; void onActionSelected(BuildContext context, ImageEntry entry, EntryAction action) async { // wait for the popup menu to hide before proceeding with the action await Future.delayed(Durations.popupMenuAnimation * timeDilation); switch (action) { case EntryAction.toggleFavourite: entry.toggleFavourite(); break; case EntryAction.delete: unawaited(_showDeleteDialog(context, entry)); break; case EntryAction.edit: unawaited(AndroidAppService.edit(entry.uri, entry.mimeType)); break; case EntryAction.info: showInfo(); break; case EntryAction.rename: unawaited(_showRenameDialog(context, entry)); break; case EntryAction.open: unawaited(AndroidAppService.open(entry.uri, entry.mimeTypeAnySubtype)); break; case EntryAction.openMap: unawaited(AndroidAppService.openMap(entry.geoUri)); break; case EntryAction.print: unawaited(_print(entry)); break; case EntryAction.rotateCCW: unawaited(_rotate(context, entry, clockwise: false)); break; case EntryAction.rotateCW: unawaited(_rotate(context, entry, clockwise: true)); break; case EntryAction.flip: unawaited(_flip(context, entry)); break; case EntryAction.setAs: unawaited(AndroidAppService.setAs(entry.uri, entry.mimeType)); break; case EntryAction.share: unawaited(AndroidAppService.share({entry})); break; case EntryAction.viewSource: _goToSourceViewer(context, entry); break; case EntryAction.debug: _goToDebug(context, entry); break; } } Future _print(ImageEntry entry) async { final uri = entry.uri; final mimeType = entry.mimeType; final rotationDegrees = entry.rotationDegrees; final isFlipped = entry.isFlipped; final documentName = entry.bestTitle ?? 'Aves'; final doc = pdf.Document(title: documentName); PdfImage pdfImage; if (entry.isSvg) { final bytes = await ImageFileService.getImage(uri, mimeType, entry.rotationDegrees, entry.isFlipped); if (bytes != null && bytes.isNotEmpty) { final svgRoot = await svg.fromSvgBytes(bytes, uri); final viewBox = svgRoot.viewport.viewBox; // 1000 is arbitrary, but large enough to look ok in the print preview final targetSize = viewBox * 1000 / viewBox.longestSide; final picture = svgRoot.toPicture(size: targetSize); final uiImage = await picture.toImage(targetSize.width.ceil(), targetSize.height.ceil()); pdfImage = await pdfImageFromImage( pdf: doc.document, image: uiImage, ); } } else { pdfImage = await pdfImageFromImageProvider( pdf: doc.document, image: UriImage( uri: uri, mimeType: mimeType, rotationDegrees: rotationDegrees, isFlipped: isFlipped, ), ); } if (pdfImage != null) { doc.addPage(pdf.Page(build: (context) => pdf.Center(child: pdf.Image(pdfImage)))); // Page unawaited(Printing.layoutPdf( onLayout: (format) => doc.save(), name: documentName, )); } } Future _flip(BuildContext context, ImageEntry entry) async { if (!await checkStoragePermission(context, [entry])) return; final success = await entry.flip(); if (!success) showFeedback(context, 'Failed'); } Future _rotate(BuildContext context, ImageEntry entry, {@required bool clockwise}) async { if (!await checkStoragePermission(context, [entry])) return; final success = await entry.rotate(clockwise: clockwise); if (!success) showFeedback(context, 'Failed'); } Future _showDeleteDialog(BuildContext context, ImageEntry entry) async { final confirmed = await showDialog( context: context, builder: (context) { return AvesDialog( content: Text('Are you sure?'), actions: [ TextButton( onPressed: () => Navigator.pop(context), child: Text('Cancel'.toUpperCase()), ), TextButton( onPressed: () => Navigator.pop(context, true), child: Text('Delete'.toUpperCase()), ), ], ); }, ); if (confirmed == null || !confirmed) return; if (!await checkStoragePermission(context, [entry])) return; if (!await entry.delete()) { showFeedback(context, 'Failed'); } else if (hasCollection) { // update collection collection.source.removeEntries([entry]); if (collection.sortedEntries.isEmpty) { Navigator.pop(context); } } else { // leave viewer unawaited(SystemNavigator.pop()); } } Future _showRenameDialog(BuildContext context, ImageEntry entry) async { final newName = await showDialog( context: context, builder: (context) => RenameEntryDialog(entry), ); if (newName == null || newName.isEmpty) return; if (!await checkStoragePermission(context, [entry])) return; showFeedback(context, await entry.rename(newName) ? 'Done!' : 'Failed'); } void _goToSourceViewer(BuildContext context, ImageEntry entry) { Navigator.push( context, MaterialPageRoute( settings: RouteSettings(name: SourceViewerPage.routeName), builder: (context) => SourceViewerPage(entry: entry), ), ); } void _goToDebug(BuildContext context, ImageEntry entry) { Navigator.push( context, MaterialPageRoute( settings: RouteSettings(name: FullscreenDebugPage.routeName), builder: (context) => FullscreenDebugPage(entry: entry), ), ); } }