mirror of
https://github.com/immich-app/immich.git
synced 2025-01-21 00:52:43 -05:00
2b5cef156c
* Add i18n framework to mobile app and write simple translation generator * Replace all texts in login_form with i18n keys * Localization of sharing section * Localization of asset viewer section * Use JSON as base translation format * Add check for missing/unused translation keys * Add localizely * Remove i18n directory in favour of localizely * Backup Translation * More translations * Translate home page * Translation of search page * Translate new server version announcement * Reformat code * Fix typo in german translation * Update englisch translations * Change translation keys to match dart filenames * Add /api to translated endpoint_urls * Update localizely.yml * Add languages to ios plist * Remove unused keys * Added script to check outdated key in other translations * Add download key to localizely.yml Co-authored-by: Alex <alex.tran1502@gmail.com>
273 lines
9.2 KiB
Dart
273 lines
9.2 KiB
Dart
import 'package:auto_route/auto_route.dart';
|
|
import 'package:easy_localization/easy_localization.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_hooks/flutter_hooks.dart';
|
|
import 'package:fluttertoast/fluttertoast.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
|
|
import 'package:immich_mobile/modules/backup/ui/album_info_card.dart';
|
|
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
|
|
import 'package:immich_mobile/shared/ui/immich_toast.dart';
|
|
|
|
class BackupAlbumSelectionPage extends HookConsumerWidget {
|
|
const BackupAlbumSelectionPage({Key? key}) : super(key: key);
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
final availableAlbums = ref.watch(backupProvider).availableAlbums;
|
|
final selectedBackupAlbums = ref.watch(backupProvider).selectedBackupAlbums;
|
|
final excludedBackupAlbums = ref.watch(backupProvider).excludedBackupAlbums;
|
|
|
|
useEffect(() {
|
|
ref.read(backupProvider.notifier).getBackupInfo();
|
|
return null;
|
|
}, []);
|
|
|
|
_buildAlbumSelectionList() {
|
|
if (availableAlbums.isEmpty) {
|
|
return const Center(
|
|
child: ImmichLoadingIndicator(),
|
|
);
|
|
}
|
|
|
|
return SizedBox(
|
|
height: 265,
|
|
child: ListView.builder(
|
|
scrollDirection: Axis.horizontal,
|
|
itemCount: availableAlbums.length,
|
|
physics: const BouncingScrollPhysics(),
|
|
itemBuilder: ((context, index) {
|
|
var thumbnailData = availableAlbums[index].thumbnailData;
|
|
return Padding(
|
|
padding: index == 0
|
|
? const EdgeInsets.only(left: 16.00)
|
|
: const EdgeInsets.all(0),
|
|
child: AlbumInfoCard(
|
|
imageData: thumbnailData,
|
|
albumInfo: availableAlbums[index].albumEntity),
|
|
);
|
|
}),
|
|
),
|
|
);
|
|
}
|
|
|
|
_buildSelectedAlbumNameChip() {
|
|
return selectedBackupAlbums.map((album) {
|
|
void removeSelection() {
|
|
if (ref.watch(backupProvider).selectedBackupAlbums.length == 1) {
|
|
ImmichToast.show(
|
|
context: context,
|
|
msg: "backup_err_only_album".tr(),
|
|
toastType: ToastType.error,
|
|
gravity: ToastGravity.BOTTOM,
|
|
);
|
|
return;
|
|
}
|
|
|
|
ref.watch(backupProvider.notifier).removeAlbumForBackup(album);
|
|
}
|
|
|
|
return Padding(
|
|
padding: const EdgeInsets.only(right: 8.0),
|
|
child: GestureDetector(
|
|
onTap: removeSelection,
|
|
child: Chip(
|
|
visualDensity: VisualDensity.compact,
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(5)),
|
|
label: Text(
|
|
album.name,
|
|
style: const TextStyle(
|
|
fontSize: 10,
|
|
color: Colors.white,
|
|
fontWeight: FontWeight.bold),
|
|
),
|
|
backgroundColor: Theme.of(context).primaryColor,
|
|
deleteIconColor: Colors.white,
|
|
deleteIcon: const Icon(
|
|
Icons.cancel_rounded,
|
|
size: 15,
|
|
),
|
|
onDeleted: removeSelection,
|
|
),
|
|
),
|
|
);
|
|
}).toSet();
|
|
}
|
|
|
|
_buildExcludedAlbumNameChip() {
|
|
return excludedBackupAlbums.map((album) {
|
|
void removeSelection() {
|
|
ref
|
|
.watch(backupProvider.notifier)
|
|
.removeExcludedAlbumForBackup(album);
|
|
}
|
|
|
|
return GestureDetector(
|
|
onTap: removeSelection,
|
|
child: Padding(
|
|
padding: const EdgeInsets.only(right: 8.0),
|
|
child: Chip(
|
|
visualDensity: VisualDensity.compact,
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(5)),
|
|
label: Text(
|
|
album.name,
|
|
style: const TextStyle(
|
|
fontSize: 10,
|
|
color: Colors.white,
|
|
fontWeight: FontWeight.bold),
|
|
),
|
|
backgroundColor: Colors.red[300],
|
|
deleteIconColor: Colors.white,
|
|
deleteIcon: const Icon(
|
|
Icons.cancel_rounded,
|
|
size: 15,
|
|
),
|
|
onDeleted: removeSelection,
|
|
),
|
|
),
|
|
);
|
|
}).toSet();
|
|
}
|
|
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
leading: IconButton(
|
|
onPressed: () => AutoRouter.of(context).pop(),
|
|
icon: const Icon(Icons.arrow_back_ios_rounded),
|
|
),
|
|
title: const Text(
|
|
"backup_album_selection_page_select_albums",
|
|
style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
|
|
).tr(),
|
|
elevation: 0,
|
|
),
|
|
body: ListView(
|
|
physics: const ClampingScrollPhysics(),
|
|
children: [
|
|
Padding(
|
|
padding:
|
|
const EdgeInsets.symmetric(vertical: 8.0, horizontal: 16.0),
|
|
child: const Text(
|
|
"backup_album_selection_page_selection_info",
|
|
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14),
|
|
).tr(),
|
|
),
|
|
// Selected Album Chips
|
|
|
|
Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
|
child: Wrap(
|
|
children: [
|
|
..._buildSelectedAlbumNameChip(),
|
|
..._buildExcludedAlbumNameChip()
|
|
],
|
|
),
|
|
),
|
|
|
|
Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8),
|
|
child: Card(
|
|
margin: const EdgeInsets.all(0),
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(5), // if you need this
|
|
side: const BorderSide(
|
|
color: Color.fromARGB(255, 235, 235, 235),
|
|
width: 1,
|
|
),
|
|
),
|
|
elevation: 0,
|
|
borderOnForeground: false,
|
|
child: Column(
|
|
children: [
|
|
ListTile(
|
|
visualDensity: VisualDensity.compact,
|
|
title: Text(
|
|
"backup_album_selection_page_total_assets",
|
|
style: TextStyle(
|
|
fontWeight: FontWeight.bold,
|
|
fontSize: 14,
|
|
color: Colors.grey[700]),
|
|
).tr(),
|
|
trailing: Text(
|
|
ref
|
|
.watch(backupProvider)
|
|
.allUniqueAssets
|
|
.length
|
|
.toString(),
|
|
style: const TextStyle(fontWeight: FontWeight.bold),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
|
|
ListTile(
|
|
title: Text(
|
|
"backup_album_selection_page_albums_device"
|
|
.tr(args: [availableAlbums.length.toString()]),
|
|
style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 14),
|
|
),
|
|
subtitle: Padding(
|
|
padding: const EdgeInsets.symmetric(vertical: 8.0),
|
|
child: Text(
|
|
"backup_album_selection_page_albums_tap",
|
|
style: TextStyle(
|
|
fontSize: 12,
|
|
color: Theme.of(context).primaryColor,
|
|
fontWeight: FontWeight.bold,
|
|
),
|
|
).tr(),
|
|
),
|
|
trailing: IconButton(
|
|
splashRadius: 16,
|
|
icon: Icon(
|
|
Icons.info,
|
|
size: 20,
|
|
color: Theme.of(context).primaryColor,
|
|
),
|
|
onPressed: () {
|
|
// show the dialog
|
|
showDialog(
|
|
context: context,
|
|
builder: (BuildContext context) {
|
|
return AlertDialog(
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(12)),
|
|
elevation: 5,
|
|
title: Text(
|
|
'backup_album_selection_page_selection_info',
|
|
style: TextStyle(
|
|
fontSize: 16,
|
|
fontWeight: FontWeight.bold,
|
|
color: Theme.of(context).primaryColor,
|
|
),
|
|
).tr(),
|
|
content: SingleChildScrollView(
|
|
child: ListBody(
|
|
children: [
|
|
Text(
|
|
'backup_album_selection_page_assets_scatter',
|
|
style: TextStyle(
|
|
fontSize: 14, color: Colors.grey[700]),
|
|
).tr(),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
},
|
|
);
|
|
},
|
|
),
|
|
),
|
|
|
|
Padding(
|
|
padding: const EdgeInsets.only(bottom: 16.0),
|
|
child: _buildAlbumSelectionList(),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|