import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; import 'package:logging/logging.dart'; import 'package:page_transition/page_transition.dart'; import 'package:photos/core/configuration.dart'; import 'package:photos/db/files_db.dart'; import 'package:photos/models/collection.dart'; import 'package:photos/models/collection_items.dart'; import 'package:photos/models/file.dart'; import 'package:photos/models/selected_files.dart'; import 'package:photos/services/collections_service.dart'; import 'package:photos/ui/collection_page.dart'; import 'package:photos/ui/loading_widget.dart'; import 'package:photos/ui/thumbnail_widget.dart'; import 'package:photos/utils/dialog_util.dart'; import 'package:photos/utils/file_uploader.dart'; import 'package:photos/utils/toast_util.dart'; class CreateCollectionPage extends StatefulWidget { final SelectedFiles selectedFiles; const CreateCollectionPage(this.selectedFiles, {Key key}) : super(key: key); @override _CreateCollectionPageState createState() => _CreateCollectionPageState(); } class _CreateCollectionPageState extends State { final _logger = Logger("CreateCollectionPage"); String _albumName; @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text("add files"), ), body: _getBody(context), ); } Widget _getBody(BuildContext context) { return SingleChildScrollView( child: Column( mainAxisSize: MainAxisSize.min, children: [ Row( children: [ Expanded( child: Padding( padding: const EdgeInsets.only( top: 30, bottom: 12, left: 40, right: 40), child: OutlineButton.icon( padding: EdgeInsets.all(20), icon: Icon(Icons.create_new_folder_outlined), label: Text( "to a new album", style: Theme.of(context).textTheme.bodyText1, ), onPressed: () { _showNameAlbumDialog(); }, ), ), ), ], ), Padding( padding: const EdgeInsets.fromLTRB(40, 24, 40, 20), child: Align( alignment: Alignment.centerLeft, child: Text( "to an existing album", style: TextStyle( fontWeight: FontWeight.bold, color: Theme.of(context).primaryColorLight.withOpacity(0.8), ), ), ), ), Padding( padding: const EdgeInsets.fromLTRB(20, 4, 20, 0), child: _getExistingCollectionsWidget(), ), ], ), ); } Widget _getExistingCollectionsWidget() { return FutureBuilder>( future: _getCollectionsWithThumbnail(), builder: (context, snapshot) { if (snapshot.hasError) { return Text(snapshot.error.toString()); } else if (snapshot.hasData) { return ListView.builder( itemBuilder: (context, index) { return _buildCollectionItem(snapshot.data[index]); }, itemCount: snapshot.data.length, shrinkWrap: true, physics: NeverScrollableScrollPhysics(), ); } else { return loadWidget; } }, ); } Widget _buildCollectionItem(CollectionWithThumbnail item) { return Container( padding: EdgeInsets.only(left: 24, bottom: 16), child: GestureDetector( child: Row( children: [ ClipRRect( borderRadius: BorderRadius.circular(2.0), child: Container( child: ThumbnailWidget(item.thumbnail), height: 64, width: 64, ), ), Padding(padding: EdgeInsets.all(8)), Expanded( child: Text( item.collection.name, style: TextStyle( fontSize: 16, ), ), ), ], ), onTap: () async { if (await _addToCollection(item.collection.id)) { showToast("added successfully to '" + item.collection.name); _navigateToCollection(item.collection); } }, ), ); } Future> _getCollectionsWithThumbnail() async { final List collectionsWithThumbnail = []; final latestCollectionFiles = await FilesDB.instance.getLatestCollectionFiles(); for (final file in latestCollectionFiles) { final c = CollectionsService.instance.getCollectionByID(file.collectionID); if (c.owner.id == Configuration.instance.getUserID()) { collectionsWithThumbnail.add(CollectionWithThumbnail(c, file, await FilesDB.instance.getLastModifiedFileInCollection(c.id))); } } collectionsWithThumbnail.sort((first, second) { return second.lastUpdatedFile.updationTime .compareTo(first.lastUpdatedFile.updationTime); }); return collectionsWithThumbnail; } void _showNameAlbumDialog() async { AlertDialog alert = AlertDialog( title: Text("album title"), content: TextFormField( decoration: InputDecoration( hintText: "Christmas 2020 / Dinner at Alice's", contentPadding: EdgeInsets.all(8), ), onChanged: (value) { setState(() { _albumName = value; }); }, autofocus: true, keyboardType: TextInputType.text, textCapitalization: TextCapitalization.words, ), actions: [ FlatButton( child: Text("ok"), onPressed: () async { Navigator.of(context, rootNavigator: true).pop('dialog'); final collection = await _createAlbum(_albumName); if (collection != null) { if (await _addToCollection(collection.id)) { showToast("album '" + _albumName + "' created."); _navigateToCollection(collection); } } }, ), ], ); showDialog( context: context, builder: (BuildContext context) { return alert; }, ); } void _navigateToCollection(Collection collection) { Navigator.pop(context); Navigator.push( context, PageTransition( type: PageTransitionType.bottomToTop, child: CollectionPage( collection, ))); } Future _addToCollection(int collectionID) async { final dialog = createProgressDialog(context, "uploading files to album..."); await dialog.show(); try { final files = List(); for (final file in widget.selectedFiles.files) { final currentFile = await FilesDB.instance.getFile(file.generatedID); if (currentFile.uploadedFileID == null) { final uploadedFile = (await FileUploader.instance .forceUpload(currentFile, collectionID)); files.add(uploadedFile); } else { files.add(currentFile); } } await CollectionsService.instance.addToCollection(collectionID, files); await dialog.hide(); widget.selectedFiles.clearAll(); return true; } catch (e, s) { _logger.severe("Could not add to album", e, s); await dialog.hide(); showGenericErrorDialog(context); } return false; } Future _createAlbum(String albumName) async { var collection; final dialog = createProgressDialog(context, "creating album..."); await dialog.show(); try { collection = await CollectionsService.instance.createAlbum(albumName); } catch (e, s) { _logger.severe(e, s); await dialog.hide(); showGenericErrorDialog(context); } finally { await dialog.hide(); } return collection; } }