nc-photos/app/lib/widget/album_importer.dart

268 lines
8.1 KiB
Dart
Raw Normal View History

2021-07-01 13:32:22 +02:00
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:kiwi/kiwi.dart';
2021-07-01 13:32:22 +02:00
import 'package:logging/logging.dart';
import 'package:nc_photos/account.dart';
2021-11-01 10:50:13 +01:00
import 'package:nc_photos/app_db.dart';
2021-07-25 07:00:38 +02:00
import 'package:nc_photos/app_localizations.dart';
2021-07-01 13:32:22 +02:00
import 'package:nc_photos/bloc/list_importable_album.dart';
import 'package:nc_photos/di_container.dart';
2021-07-01 13:32:22 +02:00
import 'package:nc_photos/entity/album.dart';
import 'package:nc_photos/entity/album/cover_provider.dart';
import 'package:nc_photos/entity/album/provider.dart';
2021-07-07 20:40:43 +02:00
import 'package:nc_photos/entity/album/sort_provider.dart';
2021-07-01 13:32:22 +02:00
import 'package:nc_photos/entity/file.dart';
import 'package:nc_photos/entity/file_util.dart' as file_util;
2021-07-01 13:32:22 +02:00
import 'package:nc_photos/exception_util.dart' as exception_util;
import 'package:nc_photos/iterable_extension.dart';
import 'package:nc_photos/k.dart' as k;
import 'package:nc_photos/snack_bar_manager.dart';
import 'package:nc_photos/theme.dart';
import 'package:nc_photos/use_case/create_album.dart';
2021-09-29 13:00:00 +02:00
import 'package:nc_photos/use_case/preprocess_album.dart';
import 'package:nc_photos/use_case/update_album_with_actual_items.dart';
2021-07-02 21:44:05 +02:00
import 'package:nc_photos/widget/processing_dialog.dart';
2021-07-01 13:32:22 +02:00
class AlbumImporterArguments {
AlbumImporterArguments(this.account);
final Account account;
}
class AlbumImporter extends StatefulWidget {
static const routeName = "/album-importer";
2021-07-23 22:05:57 +02:00
static Route buildRoute(AlbumImporterArguments args) => MaterialPageRoute(
builder: (context) => AlbumImporter.fromArgs(args),
);
2021-09-15 08:58:06 +02:00
const AlbumImporter({
2021-07-23 22:05:57 +02:00
Key? key,
required this.account,
2021-07-01 13:32:22 +02:00
}) : super(key: key);
2021-07-23 22:05:57 +02:00
AlbumImporter.fromArgs(AlbumImporterArguments args, {Key? key})
2021-07-01 13:32:22 +02:00
: this(
key: key,
account: args.account,
);
@override
createState() => _AlbumImporterState();
final Account account;
}
class _AlbumImporterState extends State<AlbumImporter> {
@override
initState() {
super.initState();
_initBloc();
}
@override
build(BuildContext context) {
return AppTheme(
child: Scaffold(
body:
BlocListener<ListImportableAlbumBloc, ListImportableAlbumBlocState>(
bloc: _bloc,
listener: (context, state) => _onStateChange(context, state),
child: BlocBuilder<ListImportableAlbumBloc,
ListImportableAlbumBlocState>(
bloc: _bloc,
builder: (context, state) => _buildContent(context, state),
),
),
),
);
}
void _initBloc() {
_log.info("[_initBloc] Initialize bloc");
_bloc = ListImportableAlbumBloc(KiwiContainer().resolve<DiContainer>());
2021-07-01 13:32:22 +02:00
_bloc.add(ListImportableAlbumBlocQuery(
widget.account,
widget.account.roots
.map((e) => File(path: file_util.unstripPath(widget.account, e)))
2021-07-01 13:32:22 +02:00
.toList()));
}
Widget _buildContent(
BuildContext context, ListImportableAlbumBlocState state) {
return SafeArea(
child: Column(
children: [
Padding(
padding: const EdgeInsets.all(24),
child: Column(
children: [
Text(
L10n.global().albumImporterHeaderText,
2021-07-01 13:32:22 +02:00
style: Theme.of(context).textTheme.headline5,
textAlign: TextAlign.center,
),
const SizedBox(height: 8),
Align(
alignment: AlignmentDirectional.topStart,
child: Text(
L10n.global().albumImporterSubHeaderText,
2021-07-01 13:32:22 +02:00
),
),
],
),
),
Expanded(
child: Stack(
children: [
Align(
alignment: Alignment.center,
child: _buildList(context, state),
),
if (state is ListImportableAlbumBlocLoading)
2021-09-15 08:58:06 +02:00
const Align(
2021-07-01 13:32:22 +02:00
alignment: Alignment.topCenter,
2021-09-15 08:58:06 +02:00
child: LinearProgressIndicator(),
2021-07-01 13:32:22 +02:00
),
],
),
),
Padding(
padding: const EdgeInsets.all(16),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
TextButton(
onPressed: () {
Navigator.of(context).pop();
},
child:
Text(MaterialLocalizations.of(context).cancelButtonLabel),
),
ElevatedButton(
onPressed: () => _onImportPressed(context),
child: Text(L10n.global().importButtonLabel),
2021-07-01 13:32:22 +02:00
),
],
),
),
],
),
);
}
Widget _buildList(BuildContext context, ListImportableAlbumBlocState state) {
return ListView.separated(
itemBuilder: (context, index) =>
_buildItem(context, _backingFiles[index]),
separatorBuilder: (context, index) => const Divider(),
itemCount: _backingFiles.length,
2021-07-01 13:32:22 +02:00
);
}
Widget _buildItem(BuildContext context, File file) {
final isPicked = _picks.containsIdentical(file);
2021-09-15 08:58:06 +02:00
onTap() {
2021-07-01 13:32:22 +02:00
setState(() {
if (isPicked) {
_picks.removeWhere((p) => identical(p, file));
} else {
_picks.add(file);
}
});
2021-09-15 08:58:06 +02:00
}
2021-07-01 13:32:22 +02:00
return ListTile(
dense: true,
leading: IconButton(
icon: AnimatedSwitcher(
duration: k.animationDurationShort,
transitionBuilder: (child, animation) =>
ScaleTransition(child: child, scale: animation),
child: Icon(
isPicked ? Icons.check_box : Icons.check_box_outline_blank,
key: ValueKey(isPicked),
),
),
onPressed: onTap,
),
title: Text(file.filename),
2021-07-01 13:32:22 +02:00
subtitle: Text(file.strippedPath),
onTap: onTap,
);
}
void _onStateChange(
BuildContext context, ListImportableAlbumBlocState state) {
if (state is ListImportableAlbumBlocSuccess ||
state is ListImportableAlbumBlocLoading) {
_transformItems(state.items);
} else if (state is ListImportableAlbumBlocFailure) {
SnackBarManager().showSnackBar(SnackBar(
content: Text(exception_util.toUserString(state.exception)),
2021-07-01 13:32:22 +02:00
duration: k.snackBarDurationNormal,
));
}
}
void _onImportPressed(BuildContext context) async {
showDialog(
barrierDismissible: false,
context: context,
2021-07-25 07:00:38 +02:00
builder: (context) =>
ProcessingDialog(text: L10n.global().albumImporterProgressText),
2021-07-01 13:32:22 +02:00
);
try {
await _createAllAlbums(context);
} finally {
// make sure we dismiss the dialog in any cases
Navigator.of(context).pop();
}
Navigator.of(context).pop();
}
Future<void> _createAllAlbums(BuildContext context) async {
for (final p in _picks) {
try {
var album = Album(
name: p.filename,
2021-07-01 13:32:22 +02:00
provider: AlbumDirProvider(
dirs: [p],
),
coverProvider: AlbumAutoCoverProvider(),
2021-09-15 08:58:06 +02:00
sortProvider: const AlbumTimeSortProvider(isAscending: false),
2021-07-01 13:32:22 +02:00
);
2021-09-25 09:32:15 +02:00
_log.info("[_createAllAlbums] Creating dir album: $album");
2021-07-01 13:32:22 +02:00
2021-11-01 10:50:13 +01:00
final items = await PreProcessAlbum(AppDb())(widget.account, album);
album = await UpdateAlbumWithActualItems(null)(
widget.account, album, items);
2021-07-01 13:32:22 +02:00
2021-11-01 10:50:13 +01:00
final albumRepo = AlbumRepo(AlbumCachedDataSource(AppDb()));
2021-07-01 13:32:22 +02:00
await CreateAlbum(albumRepo)(widget.account, album);
} catch (e, stacktrace) {
_log.shout(
"[_createAllAlbums] Failed creating dir album", e, stacktrace);
SnackBarManager().showSnackBar(SnackBar(
content: Text(exception_util.toUserString(e)),
2021-07-01 13:32:22 +02:00
duration: k.snackBarDurationNormal,
));
}
}
}
void _transformItems(List<ListImportableAlbumBlocItem> items) {
_backingFiles = items
.sorted((a, b) => b.photoCount - a.photoCount)
.map((e) => e.file)
.toList();
}
2021-07-23 22:05:57 +02:00
late ListImportableAlbumBloc _bloc;
2021-07-01 13:32:22 +02:00
var _backingFiles = <File>[];
final _picks = <File>[];
static final _log = Logger("widget.album_importer._AlbumImporterState");
}