ente/lib/ui/viewer/gallery/device_folder_page.dart

200 lines
7.4 KiB
Dart
Raw Normal View History

2020-04-18 18:46:38 +00:00
import 'package:flutter/material.dart';
import 'package:logging/logging.dart';
import 'package:photos/core/configuration.dart';
import 'package:photos/core/constants.dart';
import 'package:photos/core/event_bus.dart';
import 'package:photos/db/device_files_db.dart';
import 'package:photos/db/files_db.dart';
import 'package:photos/events/files_updated_event.dart';
import 'package:photos/events/local_photos_updated_event.dart';
import 'package:photos/models/device_collection.dart';
import 'package:photos/models/file.dart';
2022-07-03 10:09:01 +00:00
import 'package:photos/models/gallery_type.dart';
import 'package:photos/models/selected_files.dart';
import 'package:photos/services/ignored_files_service.dart';
import 'package:photos/services/remote_sync_service.dart';
import 'package:photos/theme/ente_theme.dart';
import 'package:photos/ui/components/captioned_text_widget.dart';
import 'package:photos/ui/components/menu_item_widget.dart';
import 'package:photos/ui/components/menu_section_description_widget.dart';
import 'package:photos/ui/components/toggle_switch_widget.dart';
import 'package:photos/ui/viewer/actions/file_selection_overlay_bar.dart';
import 'package:photos/ui/viewer/gallery/gallery.dart';
import 'package:photos/ui/viewer/gallery/gallery_app_bar_widget.dart';
2020-04-18 18:46:38 +00:00
class DeviceFolderPage extends StatelessWidget {
final DeviceCollection deviceCollection;
final _selectedFiles = SelectedFiles();
2020-04-18 18:46:38 +00:00
DeviceFolderPage(this.deviceCollection, {Key? key}) : super(key: key);
2020-04-18 18:46:38 +00:00
@override
Widget build(Object context) {
final gallery = Gallery(
asyncLoader: (creationStartTime, creationEndTime, {limit, asc}) {
return FilesDB.instance.getFilesInDeviceCollection(
deviceCollection,
2022-06-11 08:23:52 +00:00
creationStartTime,
creationEndTime,
limit: limit,
asc: asc,
);
2021-04-20 20:11:39 +00:00
},
reloadEvent: Bus.instance.on<LocalPhotosUpdatedEvent>(),
removalEventTypes: const {
2021-10-29 23:56:27 +00:00
EventType.deletedFromDevice,
EventType.deletedFromEverywhere,
EventType.hide,
},
tagPrefix: "device_folder:" + deviceCollection.name,
selectedFiles: _selectedFiles,
2022-07-03 09:49:33 +00:00
header: Configuration.instance.hasConfiguredAccount()
? BackupHeaderWidget(deviceCollection)
2022-08-23 12:23:17 +00:00
: const SizedBox.shrink(),
initialFiles: [deviceCollection.thumbnail!],
);
2020-04-18 18:46:38 +00:00
return Scaffold(
appBar: PreferredSize(
2022-07-04 06:02:17 +00:00
preferredSize: const Size.fromHeight(50.0),
child: GalleryAppBarWidget(
2022-07-03 07:47:15 +00:00
GalleryType.localFolder,
deviceCollection.name,
_selectedFiles,
deviceCollection: deviceCollection,
),
2020-04-18 18:46:38 +00:00
),
body: Stack(
alignment: Alignment.bottomCenter,
children: [
gallery,
FileSelectionOverlayBar(
2022-07-03 07:47:15 +00:00
GalleryType.localFolder,
_selectedFiles,
)
],
),
2020-04-18 18:46:38 +00:00
);
}
2021-03-25 16:58:30 +00:00
}
class BackupHeaderWidget extends StatelessWidget {
final DeviceCollection deviceCollection;
2021-03-25 16:58:30 +00:00
const BackupHeaderWidget(this.deviceCollection, {super.key});
2022-08-23 12:23:17 +00:00
2021-03-25 16:58:30 +00:00
@override
Widget build(BuildContext context) {
final Future<List<File>> filesInDeviceCollection =
_filesInDeviceCollection();
final ValueNotifier<bool> shouldBackup =
ValueNotifier(deviceCollection.shouldBackup);
final colorScheme = getEnteColorScheme(context);
return Padding(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 20),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Column(
mainAxisSize: MainAxisSize.min,
children: [
MenuItemWidget(
captionedTextWidget: const CaptionedTextWidget(title: "Backup"),
borderRadius: 8.0,
menuItemColor: colorScheme.fillFaint,
alignCaptionedTextToLeft: true,
trailingWidget: ToggleSwitchWidget(
value: () => shouldBackup.value,
onChanged: () async {
await RemoteSyncService.instance
.updateDeviceFolderSyncStatus(
{deviceCollection.id: !shouldBackup.value},
).then(
(val) => shouldBackup.value = !shouldBackup.value,
onError: (e) {
Logger("BackupHeaderWidget").severe(
"Could not update device folder sync status",
);
},
);
},
),
),
ValueListenableBuilder(
valueListenable: shouldBackup,
builder: (BuildContext context, bool value, _) {
return MenuSectionDescriptionWidget(
content: value
? "Files added to this device album will automatically get uploaded to ente."
: "Turn on backup to automatically upload files added to this device folder to ente.",
);
},
),
FutureBuilder(
future: _hasIgnoredFiles(filesInDeviceCollection),
builder: (context, snapshot) {
if (snapshot.hasData && snapshot.data as bool) {
return Column(
children: [
const SizedBox(height: 24),
MenuItemWidget(
captionedTextWidget: const CaptionedTextWidget(
title: "Reset ignored files",
),
borderRadius: 8.0,
menuItemColor: colorScheme.fillFaint,
leadingIcon: Icons.cloud_off_outlined,
onTap: () async {
await _removeFilesFromIgnoredFiles(
filesInDeviceCollection,
);
RemoteSyncService.instance.sync(silently: true);
},
),
const MenuSectionDescriptionWidget(
content:
"Some files in this album are ignored from upload because they had previously been deleted from ente.",
),
],
);
} else {
return const SizedBox.shrink();
}
},
)
],
),
],
),
);
}
Future<List<File>> _filesInDeviceCollection() async {
return (await FilesDB.instance.getFilesInDeviceCollection(
deviceCollection,
galleryLoadStartTime,
galleryLoadEndTime,
))
.files;
}
Future<bool> _hasIgnoredFiles(
Future<List<File>> filesInDeviceCollection) async {
final List<File> deviceCollectionFiles = await filesInDeviceCollection;
final localIDsOfFiles = <String>{};
for (File file in deviceCollectionFiles) {
localIDsOfFiles.add(file.localID!);
}
final ignoredFiles = await IgnoredFilesService.instance.ignoredIDs;
return ignoredFiles.intersection(localIDsOfFiles).isNotEmpty;
}
Future<void> _removeFilesFromIgnoredFiles(
Future<List<File>> filesInDeviceCollection,
) async {
final List<File> deviceCollectionFiles = await filesInDeviceCollection;
await IgnoredFilesService.instance
.removeIgnoredMappings(deviceCollectionFiles);
}
2020-04-18 18:46:38 +00:00
}