123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518 |
- import 'dart:async';
- import 'dart:math';
- import 'package:flutter/foundation.dart';
- import 'package:flutter/material.dart';
- import 'package:flutter/services.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/events/clear_selections_event.dart';
- import 'package:photos/events/files_updated_event.dart';
- import 'package:photos/extensions/string_ext.dart';
- import 'package:photos/models/file.dart';
- import 'package:photos/models/selected_files.dart';
- import 'package:photos/theme/ente_theme.dart';
- import 'package:photos/ui/huge_listview/place_holder_widget.dart';
- import 'package:photos/ui/viewer/file/detail_page.dart';
- import 'package:photos/ui/viewer/file/thumbnail_widget.dart';
- import 'package:photos/ui/viewer/gallery/gallery.dart';
- import 'package:photos/utils/date_time_util.dart';
- import 'package:photos/utils/navigation_util.dart';
- import 'package:visibility_detector/visibility_detector.dart';
- class LazyLoadingGallery extends StatefulWidget {
- final List<File> files;
- final int index;
- final Stream<FilesUpdatedEvent>? reloadEvent;
- final Set<EventType> removalEventTypes;
- final GalleryLoader asyncLoader;
- final SelectedFiles selectedFiles;
- final String tag;
- final String? logTag;
- final Stream<int> currentIndexStream;
- final int photoGirdSize;
- LazyLoadingGallery(
- this.files,
- this.index,
- this.reloadEvent,
- this.removalEventTypes,
- this.asyncLoader,
- this.selectedFiles,
- this.tag,
- this.currentIndexStream, {
- this.logTag = "",
- this.photoGirdSize = photoGridSizeDefault,
- Key? key,
- }) : super(key: key ?? UniqueKey());
- @override
- State<LazyLoadingGallery> createState() => _LazyLoadingGalleryState();
- }
- class _LazyLoadingGalleryState extends State<LazyLoadingGallery> {
- static const kRecycleLimit = 400;
- static const kNumberOfDaysToRenderBeforeAndAfter = 8;
- late Logger _logger;
- late List<File> _files;
- late StreamSubscription<FilesUpdatedEvent> _reloadEventSubscription;
- late StreamSubscription<int> _currentIndexSubscription;
- bool? _shouldRender;
- final ValueNotifier<bool> _toggleSelectAllFromDay = ValueNotifier(false);
- final ValueNotifier<bool> _showSelectAllButton = ValueNotifier(false);
- final ValueNotifier<bool> _areAllFromDaySelected = ValueNotifier(false);
- @override
- void initState() {
- //this is for removing the 'select all from day' icon on unselecting all files with 'cancel'
- widget.selectedFiles.addListener(_selectedFilesListener);
- super.initState();
- _init();
- }
- void _init() {
- _logger = Logger("LazyLoading_${widget.logTag}");
- _shouldRender = true;
- _files = widget.files;
- _reloadEventSubscription = widget.reloadEvent!.listen((e) => _onReload(e));
- _currentIndexSubscription =
- widget.currentIndexStream.listen((currentIndex) {
- final bool shouldRender = (currentIndex - widget.index).abs() <
- kNumberOfDaysToRenderBeforeAndAfter;
- if (mounted && shouldRender != _shouldRender) {
- setState(() {
- _shouldRender = shouldRender;
- });
- }
- });
- }
- Future _onReload(FilesUpdatedEvent event) async {
- final galleryDate =
- DateTime.fromMicrosecondsSinceEpoch(_files[0].creationTime!);
- final filesUpdatedThisDay = event.updatedFiles.where((file) {
- final fileDate = DateTime.fromMicrosecondsSinceEpoch(file.creationTime!);
- return fileDate.year == galleryDate.year &&
- fileDate.month == galleryDate.month &&
- fileDate.day == galleryDate.day;
- });
- if (filesUpdatedThisDay.isNotEmpty) {
- if (kDebugMode) {
- _logger.info(
- filesUpdatedThisDay.length.toString() +
- " files were updated due to ${event.reason} on " +
- getDayTitle(galleryDate.microsecondsSinceEpoch),
- );
- }
- if (event.type == EventType.addedOrUpdated) {
- final dayStartTime =
- DateTime(galleryDate.year, galleryDate.month, galleryDate.day);
- final result = await widget.asyncLoader(
- dayStartTime.microsecondsSinceEpoch,
- dayStartTime.microsecondsSinceEpoch + microSecondsInDay - 1,
- );
- if (mounted) {
- setState(() {
- _files = result.files;
- });
- }
- } else if (widget.removalEventTypes.contains(event.type)) {
- // Files were removed
- final generatedFileIDs = <int?>{};
- final uploadedFileIds = <int?>{};
- for (final file in filesUpdatedThisDay) {
- if (file.generatedID != null) {
- generatedFileIDs.add(file.generatedID);
- } else if (file.uploadedFileID != null) {
- uploadedFileIds.add(file.uploadedFileID);
- }
- }
- final List<File> files = [];
- files.addAll(_files);
- files.removeWhere(
- (file) =>
- generatedFileIDs.contains(file.generatedID) ||
- uploadedFileIds.contains(file.uploadedFileID),
- );
- if (kDebugMode) {
- _logger.finest(
- "removed ${_files.length - files.length} due to ${event.reason}",
- );
- }
- if (mounted) {
- setState(() {
- _files = files;
- });
- }
- } else {
- if (kDebugMode) {
- debugPrint("Unexpected event ${event.type.name}");
- }
- }
- }
- }
- @override
- void dispose() {
- _reloadEventSubscription.cancel();
- _currentIndexSubscription.cancel();
- widget.selectedFiles.removeListener(_selectedFilesListener);
- _toggleSelectAllFromDay.dispose();
- _showSelectAllButton.dispose();
- _areAllFromDaySelected.dispose();
- super.dispose();
- }
- @override
- void didUpdateWidget(LazyLoadingGallery oldWidget) {
- super.didUpdateWidget(oldWidget);
- if (!listEquals(_files, widget.files)) {
- _reloadEventSubscription.cancel();
- _init();
- }
- }
- @override
- Widget build(BuildContext context) {
- if (_files.isEmpty) {
- return const SizedBox.shrink();
- }
- return Column(
- children: [
- Row(
- mainAxisAlignment: MainAxisAlignment.spaceBetween,
- children: [
- getDayWidget(
- context,
- _files[0].creationTime!,
- widget.photoGirdSize,
- ),
- ValueListenableBuilder(
- valueListenable: _showSelectAllButton,
- builder: (context, dynamic value, _) {
- return !value
- ? const SizedBox.shrink()
- : GestureDetector(
- behavior: HitTestBehavior.translucent,
- child: SizedBox(
- width: 48,
- height: 44,
- child: ValueListenableBuilder(
- valueListenable: _areAllFromDaySelected,
- builder: (context, dynamic value, _) {
- return value
- ? const Icon(
- Icons.check_circle,
- size: 18,
- )
- : Icon(
- Icons.check_circle_outlined,
- color: getEnteColorScheme(context)
- .strokeMuted,
- size: 18,
- );
- },
- ),
- ),
- onTap: () {
- //this value has no significance
- //changing only to notify the listeners
- _toggleSelectAllFromDay.value =
- !_toggleSelectAllFromDay.value;
- },
- );
- },
- )
- ],
- ),
- _shouldRender!
- ? _getGallery()
- : PlaceHolderWidget(
- _files.length,
- widget.photoGirdSize,
- ),
- ],
- );
- }
- Widget _getGallery() {
- final List<Widget> childGalleries = [];
- final subGalleryItemLimit = widget.photoGirdSize * subGalleryMultiplier;
- for (int index = 0; index < _files.length; index += subGalleryItemLimit) {
- childGalleries.add(
- LazyLoadingGridView(
- widget.tag,
- _files.sublist(
- index,
- min(index + subGalleryItemLimit, _files.length),
- ),
- widget.asyncLoader,
- widget.selectedFiles,
- index == 0,
- _files.length > kRecycleLimit,
- _toggleSelectAllFromDay,
- _areAllFromDaySelected,
- widget.photoGirdSize,
- ),
- );
- }
- return Column(
- children: childGalleries,
- );
- }
- void _selectedFilesListener() {
- if (widget.selectedFiles.files.isEmpty) {
- _showSelectAllButton.value = false;
- } else {
- _showSelectAllButton.value = true;
- }
- }
- }
- class LazyLoadingGridView extends StatefulWidget {
- final String tag;
- final List<File> filesInDay;
- final GalleryLoader asyncLoader;
- final SelectedFiles selectedFiles;
- final bool shouldRender;
- final bool shouldRecycle;
- final ValueNotifier toggleSelectAllFromDay;
- final ValueNotifier areAllFilesSelected;
- final int? photoGridSize;
- LazyLoadingGridView(
- this.tag,
- this.filesInDay,
- this.asyncLoader,
- this.selectedFiles,
- this.shouldRender,
- this.shouldRecycle,
- this.toggleSelectAllFromDay,
- this.areAllFilesSelected,
- this.photoGridSize, {
- Key? key,
- }) : super(key: key ?? UniqueKey());
- @override
- State<LazyLoadingGridView> createState() => _LazyLoadingGridViewState();
- }
- class _LazyLoadingGridViewState extends State<LazyLoadingGridView> {
- bool? _shouldRender;
- int? _currentUserID;
- late StreamSubscription<ClearSelectionsEvent> _clearSelectionsEvent;
- @override
- void initState() {
- _shouldRender = widget.shouldRender;
- _currentUserID = Configuration.instance.getUserID();
- widget.selectedFiles.addListener(_selectedFilesListener);
- _clearSelectionsEvent =
- Bus.instance.on<ClearSelectionsEvent>().listen((event) {
- if (mounted) {
- setState(() {});
- }
- });
- widget.toggleSelectAllFromDay.addListener(_toggleSelectAllFromDayListener);
- super.initState();
- }
- @override
- void dispose() {
- widget.selectedFiles.removeListener(_selectedFilesListener);
- _clearSelectionsEvent.cancel();
- widget.toggleSelectAllFromDay
- .removeListener(_toggleSelectAllFromDayListener);
- super.dispose();
- }
- @override
- void didUpdateWidget(LazyLoadingGridView oldWidget) {
- super.didUpdateWidget(oldWidget);
- if (!listEquals(widget.filesInDay, oldWidget.filesInDay)) {
- _shouldRender = widget.shouldRender;
- }
- }
- @override
- Widget build(BuildContext context) {
- if (widget.shouldRecycle) {
- return _getRecyclableView();
- } else {
- return _getNonRecyclableView();
- }
- }
- Widget _getRecyclableView() {
- return VisibilityDetector(
- key: UniqueKey(),
- onVisibilityChanged: (visibility) {
- final shouldRender = visibility.visibleFraction > 0;
- if (mounted && shouldRender != _shouldRender) {
- setState(() {
- _shouldRender = shouldRender;
- });
- }
- },
- child: _shouldRender!
- ? _getGridView()
- : PlaceHolderWidget(widget.filesInDay.length, widget.photoGridSize!),
- );
- }
- Widget _getNonRecyclableView() {
- if (!_shouldRender!) {
- return VisibilityDetector(
- key: UniqueKey(),
- onVisibilityChanged: (visibility) {
- if (mounted && visibility.visibleFraction > 0 && !_shouldRender!) {
- setState(() {
- _shouldRender = true;
- });
- }
- },
- child:
- PlaceHolderWidget(widget.filesInDay.length, widget.photoGridSize!),
- );
- } else {
- return _getGridView();
- }
- }
- Widget _getGridView() {
- return GridView.builder(
- shrinkWrap: true,
- physics: const NeverScrollableScrollPhysics(),
- // to disable GridView's scrolling
- itemBuilder: (context, index) {
- return _buildFile(context, widget.filesInDay[index]);
- },
- itemCount: widget.filesInDay.length,
- gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
- crossAxisSpacing: 2,
- mainAxisSpacing: 2,
- crossAxisCount: widget.photoGridSize!,
- ),
- padding: const EdgeInsets.all(0),
- );
- }
- Widget _buildFile(BuildContext context, File file) {
- final isFileSelected = widget.selectedFiles.isFileSelected(file);
- Color selectionColor = Colors.white;
- if (isFileSelected &&
- file.isUploaded &&
- (file.ownerID != _currentUserID ||
- file.pubMagicMetadata!.uploaderName != null)) {
- final avatarColors = getEnteColorScheme(context).avatarColors;
- final int randomID = file.ownerID != _currentUserID
- ? file.ownerID!
- : file.pubMagicMetadata!.uploaderName.sumAsciiValues;
- selectionColor = avatarColors[(randomID).remainder(avatarColors.length)];
- }
- return GestureDetector(
- onTap: () {
- if (widget.selectedFiles.files.isNotEmpty) {
- _selectFile(file);
- } else {
- _routeToDetailPage(file, context);
- }
- },
- onLongPress: () {
- HapticFeedback.lightImpact();
- _selectFile(file);
- },
- child: ClipRRect(
- borderRadius: BorderRadius.circular(1),
- child: Stack(
- children: [
- Hero(
- tag: widget.tag + file.tag,
- child: ColorFiltered(
- colorFilter: ColorFilter.mode(
- Colors.black.withOpacity(
- isFileSelected ? 0.4 : 0,
- ),
- BlendMode.darken,
- ),
- child: ThumbnailWidget(
- file,
- diskLoadDeferDuration: thumbnailDiskLoadDeferDuration,
- serverLoadDeferDuration: thumbnailServerLoadDeferDuration,
- shouldShowLivePhotoOverlay: true,
- key: Key(widget.tag + file.tag),
- thumbnailSize: widget.photoGridSize! < photoGridSizeDefault
- ? thumbnailLargeSize
- : thumbnailSmallSize,
- shouldShowOwnerAvatar: !isFileSelected,
- ),
- ),
- ),
- Visibility(
- visible: isFileSelected,
- child: Positioned(
- right: 4,
- top: 4,
- child: Icon(
- Icons.check_circle_rounded,
- size: 20,
- color: selectionColor, //same for both themes
- ),
- ),
- )
- ],
- ),
- ),
- );
- }
- void _selectFile(File file) {
- widget.selectedFiles.toggleSelection(file);
- }
- void _routeToDetailPage(File file, BuildContext context) {
- final page = DetailPage(
- DetailPageConfiguration(
- List.unmodifiable(widget.filesInDay),
- widget.asyncLoader,
- widget.filesInDay.indexOf(file),
- widget.tag,
- ),
- );
- routeToPage(context, page, forceCustomPageRoute: true);
- }
- void _selectedFilesListener() {
- if (widget.selectedFiles.files.containsAll(widget.filesInDay.toSet())) {
- widget.areAllFilesSelected.value = true;
- } else {
- widget.areAllFilesSelected.value = false;
- }
- bool shouldRefresh = false;
- for (final file in widget.filesInDay) {
- if (widget.selectedFiles.isPartOfLastSelected(file)) {
- shouldRefresh = true;
- }
- }
- if (shouldRefresh && mounted) {
- setState(() {});
- }
- }
- void _toggleSelectAllFromDayListener() {
- if (widget.selectedFiles.files.containsAll(widget.filesInDay.toSet())) {
- setState(() {
- widget.selectedFiles.unSelectAll(widget.filesInDay.toSet());
- });
- } else {
- widget.selectedFiles.selectAll(widget.filesInDay.toSet());
- }
- }
- }
|