ente/lib/ui/viewer/file/detail_page.dart

405 lines
12 KiB
Dart
Raw Normal View History

2023-08-29 22:30:04 +00:00
import "dart:math";
2021-06-02 13:54:31 +00:00
import 'package:extended_image/extended_image.dart';
2023-08-29 14:05:41 +00:00
import "package:flutter/foundation.dart";
2020-03-24 19:59:36 +00:00
import 'package:flutter/material.dart';
2021-07-05 18:28:57 +00:00
import 'package:flutter/services.dart';
2021-06-09 13:38:27 +00:00
import 'package:logging/logging.dart';
2021-08-09 13:26:06 +00:00
import 'package:photos/core/configuration.dart';
import 'package:photos/core/constants.dart';
2021-08-09 13:26:06 +00:00
import 'package:photos/core/errors.dart';
2023-04-07 05:41:42 +00:00
import "package:photos/generated/l10n.dart";
2023-08-25 04:39:30 +00:00
import 'package:photos/models/file/file.dart';
import "package:photos/models/file/file_type.dart";
import "package:photos/ui/common/fast_scroll_physics.dart";
2022-07-01 14:39:02 +00:00
import 'package:photos/ui/tools/editor/image_editor_page.dart';
import "package:photos/ui/viewer/file/file_app_bar.dart";
import "package:photos/ui/viewer/file/file_bottom_bar.dart";
import 'package:photos/ui/viewer/file/file_widget.dart';
import 'package:photos/ui/viewer/gallery/gallery.dart';
2020-10-23 15:20:51 +00:00
import 'package:photos/utils/dialog_util.dart';
import 'package:photos/utils/file_util.dart';
2021-06-02 13:54:31 +00:00
import 'package:photos/utils/navigation_util.dart';
2023-01-05 07:29:47 +00:00
import 'package:photos/utils/toast_util.dart';
2020-03-24 19:59:36 +00:00
2021-09-15 20:40:08 +00:00
enum DetailPageMode {
minimalistic,
full,
}
2021-06-09 13:38:27 +00:00
class DetailPageConfiguration {
2023-08-24 16:56:24 +00:00
final List<EnteFile> files;
final GalleryLoader? asyncLoader;
final int selectedIndex;
final String tagPrefix;
2021-09-15 20:40:08 +00:00
final DetailPageMode mode;
final bool sortOrderAsc;
2020-03-24 19:59:36 +00:00
2021-06-09 13:38:27 +00:00
DetailPageConfiguration(
this.files,
this.asyncLoader,
this.selectedIndex,
2021-09-15 20:40:08 +00:00
this.tagPrefix, {
this.mode = DetailPageMode.full,
this.sortOrderAsc = false,
2021-09-15 20:40:08 +00:00
});
2021-06-09 13:38:27 +00:00
DetailPageConfiguration copyWith({
2023-08-24 16:56:24 +00:00
List<EnteFile>? files,
GalleryLoader? asyncLoader,
int? selectedIndex,
String? tagPrefix,
bool? sortOrderAsc,
2021-06-09 13:38:27 +00:00
}) {
return DetailPageConfiguration(
files ?? this.files,
asyncLoader ?? this.asyncLoader,
selectedIndex ?? this.selectedIndex,
tagPrefix ?? this.tagPrefix,
2023-06-13 13:51:13 +00:00
sortOrderAsc: sortOrderAsc ?? this.sortOrderAsc,
2021-06-09 13:38:27 +00:00
);
}
}
class DetailPage extends StatefulWidget {
final DetailPageConfiguration config;
const DetailPage(this.config, {key}) : super(key: key);
2020-04-17 08:17:37 +00:00
@override
2022-07-03 09:45:00 +00:00
State<DetailPage> createState() => _DetailPageState();
2020-04-17 08:17:37 +00:00
}
class _DetailPageState extends State<DetailPage> {
static const kLoadLimit = 100;
2020-05-06 16:43:03 +00:00
final _logger = Logger("DetailPageState");
2020-04-17 08:17:37 +00:00
bool _shouldDisableScroll = false;
2023-08-24 16:56:24 +00:00
List<EnteFile>? _files;
late PageController _pageController;
final _selectedIndexNotifier = ValueNotifier(0);
bool _hasLoadedTillStart = false;
bool _hasLoadedTillEnd = false;
final _enableFullScreenNotifier = ValueNotifier(false);
bool _isFirstOpened = true;
2020-04-23 20:00:20 +00:00
2020-03-24 19:59:36 +00:00
@override
2020-04-25 22:57:43 +00:00
void initState() {
2023-05-18 18:19:20 +00:00
super.initState();
_files = [
2023-08-19 11:39:56 +00:00
...widget.config.files,
]; // Make a copy since we append preceding and succeeding entries to this
_selectedIndexNotifier.value = widget.config.selectedIndex;
2021-07-07 01:10:02 +00:00
_preloadEntries();
_pageController = PageController(initialPage: _selectedIndexNotifier.value);
2020-04-25 22:57:43 +00:00
}
2020-04-23 20:00:20 +00:00
@override
void dispose() {
_pageController.dispose();
2023-07-31 23:24:14 +00:00
_enableFullScreenNotifier.dispose();
_selectedIndexNotifier.dispose();
SystemChrome.setEnabledSystemUIMode(
SystemUiMode.manual,
overlays: SystemUiOverlay.values,
);
super.dispose();
}
2020-04-25 22:57:43 +00:00
@override
Widget build(BuildContext context) {
try {
_files![_selectedIndexNotifier.value];
} catch (e) {
_logger.severe(e);
Navigator.pop(context);
}
2022-06-11 08:23:52 +00:00
_logger.info(
"Opening " +
_files![_selectedIndexNotifier.value].toString() +
2022-06-11 08:23:52 +00:00
". " +
(_selectedIndexNotifier.value + 1).toString() +
2022-06-11 08:23:52 +00:00
" / " +
_files!.length.toString() +
2022-06-11 08:23:52 +00:00
" files .",
);
return Scaffold(
appBar: PreferredSize(
preferredSize: const Size.fromHeight(80),
child: ValueListenableBuilder(
builder: (BuildContext context, int selectedIndex, _) {
return FileAppBar(
_files![selectedIndex],
_onFileRemoved,
100,
widget.config.mode == DetailPageMode.full,
enableFullScreenNotifier: _enableFullScreenNotifier,
);
},
valueListenable: _selectedIndexNotifier,
),
),
extendBodyBehindAppBar: true,
resizeToAvoidBottomInset: false,
body: Center(
child: Stack(
children: [
_buildPageView(context),
ValueListenableBuilder(
builder: (BuildContext context, int selectedIndex, _) {
return FileBottomBar(
2023-08-29 14:05:41 +00:00
_files![selectedIndex],
_onEditFileRequested,
widget.config.mode == DetailPageMode.minimalistic,
onFileRemoved: _onFileRemoved,
userID: Configuration.instance.getUserID(),
enableFullScreenNotifier: _enableFullScreenNotifier,
);
},
valueListenable: _selectedIndexNotifier,
),
],
2020-03-24 19:59:36 +00:00
),
),
2020-03-24 19:59:36 +00:00
);
}
Widget _buildPageView(BuildContext context) {
2020-06-17 15:09:47 +00:00
return PageView.builder(
2020-04-25 22:57:43 +00:00
itemBuilder: (context, index) {
final file = _files![index];
2020-06-19 23:03:26 +00:00
_preloadFiles(index);
2023-08-29 14:05:41 +00:00
final Widget fileContent = FileWidget(
file,
autoPlay: shouldAutoPlay(),
tagPrefix: widget.config.tagPrefix,
shouldDisableScroll: (value) {
if (_shouldDisableScroll != value) {
setState(() {
2023-08-29 22:30:04 +00:00
_logger.fine('setState $_shouldDisableScroll to $value');
2023-08-29 14:05:41 +00:00
_shouldDisableScroll = value;
});
}
},
playbackCallback: (isPlaying) {
Future.delayed(Duration.zero, () {
_toggleFullScreen(shouldEnable: isPlaying);
});
},
2023-08-29 14:05:41 +00:00
backgroundDecoration: const BoxDecoration(color: Colors.black),
);
2021-07-05 17:34:45 +00:00
return GestureDetector(
onTap: () {
file.fileType != FileType.video ? _toggleFullScreen() : null;
2021-07-05 17:34:45 +00:00
},
child: kDebugMode
? Stack(
children: [
fileContent,
Positioned(
top: 80,
right: 80,
child: Text(
file.generatedID?.toString() ?? 'null',
style: const TextStyle(color: Colors.white),
),
),
],
)
: fileContent,
2021-07-05 17:34:45 +00:00
);
2020-04-25 22:57:43 +00:00
},
2020-06-17 15:09:47 +00:00
onPageChanged: (index) {
if (_selectedIndexNotifier.value == index) {
if (kDebugMode) {
2023-08-29 22:30:04 +00:00
debugPrint("onPageChanged called with same index $index");
}
// always notify listeners when the index is the same because
// the total number of files might have changed
2023-09-22 07:00:23 +00:00
// ignore: invalid_use_of_protected_member, invalid_use_of_visible_for_testing_member
2023-08-29 22:30:04 +00:00
_selectedIndexNotifier.notifyListeners();
} else {
_selectedIndexNotifier.value = index;
}
2021-07-07 01:10:02 +00:00
_preloadEntries();
2020-04-25 22:57:43 +00:00
},
physics: _shouldDisableScroll
2022-07-04 06:02:17 +00:00
? const NeverScrollableScrollPhysics()
: const FastScrollPhysics(speedFactor: 4.0),
controller: _pageController,
itemCount: _files!.length,
2020-04-25 22:57:43 +00:00
);
}
bool shouldAutoPlay() {
if (_isFirstOpened) {
_isFirstOpened = false;
return true;
}
return false;
}
void _toggleFullScreen({bool? shouldEnable}) {
if (shouldEnable != null) {
if (_enableFullScreenNotifier.value == shouldEnable) return;
}
_enableFullScreenNotifier.value = !_enableFullScreenNotifier.value;
Future.delayed(const Duration(milliseconds: 125), () {
SystemChrome.setEnabledSystemUIMode(
//to hide status bar?
SystemUiMode.manual,
2023-07-04 06:09:05 +00:00
overlays: _enableFullScreenNotifier.value ? [] : SystemUiOverlay.values,
2021-07-06 23:24:19 +00:00
);
});
}
2023-06-07 11:52:23 +00:00
Future<void> _preloadEntries() async {
final isSortOrderAsc = widget.config.sortOrderAsc;
2023-06-07 11:52:23 +00:00
if (widget.config.asyncLoader == null) return;
if (_selectedIndexNotifier.value == 0 && !_hasLoadedTillStart) {
2023-06-07 11:52:23 +00:00
await _loadStartEntries(isSortOrderAsc);
}
2023-06-07 11:52:23 +00:00
if (_selectedIndexNotifier.value == _files!.length - 1 &&
!_hasLoadedTillEnd) {
2023-06-07 11:52:23 +00:00
await _loadEndEntries(isSortOrderAsc);
}
}
2023-06-07 11:52:23 +00:00
Future<void> _loadStartEntries(bool isSortOrderAsc) async {
final result = isSortOrderAsc
? await widget.config.asyncLoader!(
galleryLoadStartTime,
_files![_selectedIndexNotifier.value].creationTime! - 1,
2023-06-07 11:52:23 +00:00
limit: kLoadLimit,
)
: await widget.config.asyncLoader!(
_files![_selectedIndexNotifier.value].creationTime! + 1,
2023-06-07 11:52:23 +00:00
DateTime.now().microsecondsSinceEpoch,
limit: kLoadLimit,
asc: true,
);
setState(() {
2023-08-29 22:30:04 +00:00
_logger.fine('setState loadStartEntries');
2023-06-07 11:52:23 +00:00
// Returned result could be a subtype of File
// ignore: unnecessary_cast
2023-08-24 16:56:24 +00:00
final files = result.files.reversed.map((e) => e as EnteFile).toList();
2023-06-07 11:52:23 +00:00
if (!result.hasMore) {
_hasLoadedTillStart = true;
}
final length = files.length;
files.addAll(_files!);
_files = files;
_pageController.jumpToPage(length);
_selectedIndexNotifier.value = length;
2023-06-07 11:52:23 +00:00
});
}
Future<void> _loadEndEntries(bool isSortOrderAsc) async {
final result = isSortOrderAsc
? await widget.config.asyncLoader!(
_files![_selectedIndexNotifier.value].creationTime! + 1,
2023-06-07 11:52:23 +00:00
DateTime.now().microsecondsSinceEpoch,
limit: kLoadLimit,
asc: true,
)
: await widget.config.asyncLoader!(
galleryLoadStartTime,
_files![_selectedIndexNotifier.value].creationTime! - 1,
2023-06-07 11:52:23 +00:00
limit: kLoadLimit,
);
setState(() {
if (!result.hasMore) {
_hasLoadedTillEnd = true;
}
2023-08-29 22:30:04 +00:00
_logger.fine('setState loadEndEntries hasMore ${result.hasMore}');
2023-06-07 11:52:23 +00:00
_files!.addAll(result.files);
});
}
2020-06-19 23:03:26 +00:00
void _preloadFiles(int index) {
2020-06-17 15:09:47 +00:00
if (index > 0) {
preloadFile(_files![index - 1]);
2020-06-17 15:09:47 +00:00
}
if (index < _files!.length - 1) {
preloadFile(_files![index + 1]);
2020-06-17 15:09:47 +00:00
}
}
2023-08-24 16:56:24 +00:00
Future<void> _onFileRemoved(EnteFile file) async {
final totalFiles = _files!.length;
if (totalFiles == 1) {
// Deleted the only file
Navigator.of(context).pop(); // Close pageview
return;
}
2023-08-29 22:30:04 +00:00
setState(() {
_files!.remove(file);
_selectedIndexNotifier.value = min(
_selectedIndexNotifier.value,
totalFiles - 2,
);
2023-08-29 22:30:04 +00:00
});
final currentPageIndex = _pageController.page!.round();
final int targetPageIndex = _files!.length > currentPageIndex
? currentPageIndex
: currentPageIndex - 1;
if (_files!.isNotEmpty) {
_pageController.animateToPage(
targetPageIndex,
2022-07-04 06:02:17 +00:00
duration: const Duration(milliseconds: 200),
2022-06-11 08:23:52 +00:00
curve: Curves.easeInOut,
);
}
}
2021-06-12 21:29:04 +00:00
2023-08-24 16:56:24 +00:00
Future<void> _onEditFileRequested(EnteFile file) async {
2021-08-09 13:26:06 +00:00
if (file.uploadedFileID != null &&
file.ownerID != Configuration.instance.getUserID()) {
2022-06-11 08:23:52 +00:00
_logger.severe(
"Attempt to edit unowned file",
UnauthorizedEditError(),
StackTrace.current,
);
showErrorDialog(
context,
2023-04-07 05:41:42 +00:00
S.of(context).sorry,
S.of(context).weDontSupportEditingPhotosAndAlbumsThatYouDont,
2022-06-11 08:23:52 +00:00
);
2021-08-09 13:26:06 +00:00
return;
}
2023-04-07 05:41:42 +00:00
final dialog = createProgressDialog(context, S.of(context).pleaseWait);
2021-06-12 21:29:04 +00:00
await dialog.show();
2023-01-05 07:29:47 +00:00
try {
final ioFile = await getFile(file);
if (ioFile == null) {
2023-04-07 05:41:42 +00:00
showShortToast(context, S.of(context).failedToFetchOriginalForEdit);
2023-01-05 07:29:47 +00:00
await dialog.hide();
return;
}
final imageProvider =
2023-01-05 08:42:55 +00:00
ExtendedFileImageProvider(ioFile, cacheRawData: true);
2023-01-05 07:29:47 +00:00
await precacheImage(imageProvider, context);
await dialog.hide();
replacePage(
context,
ImageEditorPage(
imageProvider,
file,
widget.config.copyWith(
files: _files,
selectedIndex: _selectedIndexNotifier.value,
2023-01-05 07:29:47 +00:00
),
),
2023-01-05 07:29:47 +00:00
);
} catch (e) {
await dialog.hide();
_logger.warning("Failed to initiate edit", e);
}
2021-07-06 09:57:30 +00:00
}
2020-04-23 20:00:20 +00:00
}