import 'dart:io'; import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:like_button/like_button.dart'; import 'package:photo_manager/photo_manager.dart'; import 'package:photos/services/favorites_service.dart'; import 'package:photos/models/file_type.dart'; import 'package:photos/models/file.dart'; import 'package:photos/ui/video_widget.dart'; import 'package:photos/ui/zoomable_image.dart'; import 'package:photos/utils/date_time_util.dart'; import 'package:photos/utils/dialog_util.dart'; import 'package:photos/utils/file_util.dart'; import 'package:photos/utils/share_util.dart'; import 'package:logging/logging.dart'; import 'package:photos/utils/toast_util.dart'; class DetailPage extends StatefulWidget { final List files; final int selectedIndex; final String tagPrefix; DetailPage(this.files, this.selectedIndex, this.tagPrefix, {key}) : super(key: key); @override _DetailPageState createState() => _DetailPageState(); } class _DetailPageState extends State { final _logger = Logger("DetailPageState"); bool _shouldDisableScroll = false; List _files; PageController _pageController; int _selectedIndex = 0; bool _hasPageChanged = false; @override void initState() { _files = widget.files; _selectedIndex = widget.selectedIndex; super.initState(); } @override Widget build(BuildContext context) { _logger.info("Opening " + _files[_selectedIndex].toString() + ". " + _selectedIndex.toString() + " / " + _files.length.toString() + " files ."); return Scaffold( appBar: _buildAppBar(), extendBodyBehindAppBar: true, body: Center( child: Container( child: _buildPageView(), ), ), backgroundColor: Colors.black, ); } Widget _buildPageView() { _pageController = PageController(initialPage: _selectedIndex); return PageView.builder( itemBuilder: (context, index) { final file = _files[index]; Widget content; if (file.fileType == FileType.image) { content = ZoomableImage( file, shouldDisableScroll: (value) { setState(() { _shouldDisableScroll = value; }); }, tagPrefix: widget.tagPrefix, ); } else if (file.fileType == FileType.video) { content = VideoWidget( file, autoPlay: !_hasPageChanged, // Autoplay if it was opened directly tagPrefix: widget.tagPrefix, ); } else { content = Icon(Icons.error); } _preloadFiles(index); return content; }, onPageChanged: (index) { setState(() { _selectedIndex = index; _hasPageChanged = true; }); _preloadFiles(index); }, physics: _shouldDisableScroll ? NeverScrollableScrollPhysics() : PageScrollPhysics(), controller: _pageController, itemCount: _files.length, ); } void _preloadFiles(int index) { if (index > 0) { preloadFile(_files[index - 1]); } if (index < _files.length - 1) { preloadFile(_files[index + 1]); } } AppBar _buildAppBar() { final actions = List(); actions.add(_getFavoriteButton()); actions.add(PopupMenuButton( itemBuilder: (context) { return [ PopupMenuItem( value: 1, child: Row( children: [ Icon(Platform.isAndroid ? Icons.share_outlined : CupertinoIcons.share), Padding( padding: EdgeInsets.all(8), ), Text("share"), ], ), ), PopupMenuItem( value: 2, child: Row( children: [ Icon(Platform.isAndroid ? Icons.info_outline : CupertinoIcons.info), Padding( padding: EdgeInsets.all(8), ), Text("info"), ], ), ), PopupMenuItem( value: 3, child: Row( children: [ Icon(Platform.isAndroid ? Icons.delete_outline : CupertinoIcons.delete), Padding( padding: EdgeInsets.all(8), ), Text("delete"), ], ), ) ]; }, onSelected: (value) { if (value == 1) { share(context, [_files[_selectedIndex]]); } else if (value == 2) { _displayInfo(_files[_selectedIndex]); } else if (value == 3) { _showDeleteSheet(); } }, )); return AppBar( actions: actions, backgroundColor: Color(0x00000000), elevation: 0, ); } Widget _getFavoriteButton() { final file = _files[_selectedIndex]; return FutureBuilder( future: FavoritesService.instance.isFavorite(file), builder: (context, snapshot) { if (snapshot.hasData) { return _getLikeButton(file, snapshot.data); } else { return _getLikeButton(file, false); } }, ); } Widget _getLikeButton(File file, bool isLiked) { return LikeButton( isLiked: isLiked, onTap: (oldValue) async { final isLiked = !oldValue; bool hasError = false; if (isLiked) { final shouldBlockUser = file.uploadedFileID == null; var dialog; if (shouldBlockUser) { dialog = createProgressDialog(context, "adding to favorites..."); await dialog.show(); } try { await FavoritesService.instance.addToFavorites(file); } catch (e, s) { _logger.severe(e, s); hasError = true; showToast("sorry, could not add this to favorites!"); } finally { if (shouldBlockUser) { await dialog.hide(); } } } else { try { await FavoritesService.instance.removeFromFavorites(file); } catch (e, s) { _logger.severe(e, s); hasError = true; showToast("sorry, could not remove this from favorites!"); } } return hasError ? oldValue : isLiked; }, likeBuilder: (isLiked) { return Icon( Icons.favorite_border, color: isLiked ? Colors.pinkAccent : Colors.white, size: 30, ); }, ); } Future _displayInfo(File file) async { AssetEntity asset; int fileSize; final isLocalFile = file.localID != null; if (isLocalFile) { asset = await file.getAsset(); fileSize = await (await asset.originFile).length(); } return showDialog( context: context, builder: (BuildContext context) { var items = [ Row( children: [ Icon(Icons.calendar_today_outlined), Padding(padding: EdgeInsets.all(4)), Text(getFormattedTime( DateTime.fromMicrosecondsSinceEpoch(file.creationTime))), ], ), Padding(padding: EdgeInsets.all(4)), Row( children: [ Icon(Icons.folder_outlined), Padding(padding: EdgeInsets.all(4)), Text(file.deviceFolder), ], ), Padding(padding: EdgeInsets.all(4)), ]; if (isLocalFile) { items.add(Row( children: [ Icon(Icons.sd_storage_outlined), Padding(padding: EdgeInsets.all(4)), Text((fileSize / (1024 * 1024)).toStringAsFixed(2) + " MB"), ], )); items.add( Padding(padding: EdgeInsets.all(4)), ); if (file.fileType == FileType.image) { items.add(Row( children: [ Icon(Icons.photo_size_select_actual_outlined), Padding(padding: EdgeInsets.all(4)), Text(asset.width.toString() + " x " + asset.height.toString()), ], )); } else { items.add(Row( children: [ Icon(Icons.timer_outlined), Padding(padding: EdgeInsets.all(4)), Text(asset.videoDuration.toString().split(".")[0]), ], )); } items.add( Padding(padding: EdgeInsets.all(4)), ); } if (file.uploadedFileID != null) { items.add(Row( children: [ Icon(Icons.cloud_upload_outlined), Padding(padding: EdgeInsets.all(4)), Text(getFormattedTime( DateTime.fromMicrosecondsSinceEpoch(file.updationTime))), ], )); } return AlertDialog( title: Text(file.title), content: SingleChildScrollView( child: ListBody( children: items, ), ), actions: [ FlatButton( child: Text('ok'), onPressed: () { Navigator.of(context, rootNavigator: true).pop('dialog'); }, ), ], ); }, ); } void _showDeleteSheet() { final fileToBeDeleted = _files[_selectedIndex]; final actions = List(); if (fileToBeDeleted.uploadedFileID == null) { actions.add(CupertinoActionSheetAction( child: Text("everywhere"), isDestructiveAction: true, onPressed: () async { await deleteFilesFromEverywhere(context, [fileToBeDeleted]); _onFileDeleted(); }, )); } else { if (fileToBeDeleted.localID != null) { actions.add(CupertinoActionSheetAction( child: Text("on this device"), isDestructiveAction: true, onPressed: () async { await deleteFilesOnDeviceOnly(context, [fileToBeDeleted]); showToast("file deleted from device"); Navigator.of(context, rootNavigator: true).pop(); }, )); } actions.add(CupertinoActionSheetAction( child: Text("everywhere"), isDestructiveAction: true, onPressed: () async { await deleteFilesFromEverywhere(context, [fileToBeDeleted]); _onFileDeleted(); }, )); } final action = CupertinoActionSheet( title: Text("delete file?"), actions: actions, cancelButton: CupertinoActionSheetAction( child: Text("cancel"), onPressed: () { Navigator.of(context, rootNavigator: true).pop(); }, ), ); showCupertinoModalPopup(context: context, builder: (_) => action); } Future _onFileDeleted() async { final file = _files[_selectedIndex]; final totalFiles = _files.length; if (totalFiles == 1) { // Deleted the only file Navigator.of(context, rootNavigator: true).pop(); // Close pageview Navigator.of(context, rootNavigator: true).pop(); // Close gallery return; } if (_selectedIndex == totalFiles - 1) { // Deleted the last file await _pageController.previousPage( duration: Duration(milliseconds: 200), curve: Curves.easeInOut); setState(() { _files.remove(file); }); } else { await _pageController.nextPage( duration: Duration(milliseconds: 200), curve: Curves.easeInOut); setState(() { _selectedIndex--; _files.remove(file); }); } Navigator.of(context, rootNavigator: true).pop(); // Close dialog } }