import 'package:aves/model/entry.dart'; import 'package:aves/model/filters/album.dart'; import 'package:aves/model/settings/settings.dart'; import 'package:aves/model/source/collection_source.dart'; import 'package:aves/services/common/services.dart'; import 'package:aves/utils/android_file_utils.dart'; import 'package:aves/utils/collection_utils.dart'; import 'package:aves/widgets/common/extensions/build_context.dart'; import 'package:collection/collection.dart'; import 'package:flutter/widgets.dart'; mixin AlbumMixin on SourceBase { final Set _directories = {}; final Set _newAlbums = {}; List get rawAlbums => List.unmodifiable(_directories); Set getNewAlbumFilters(BuildContext context) => Set.unmodifiable(_newAlbums.map((v) => AlbumFilter(v, getAlbumDisplayName(context, v)))); int compareAlbumsByName(String a, String b) { final ua = getAlbumDisplayName(null, a); final ub = getAlbumDisplayName(null, b); final c = compareAsciiUpperCaseNatural(ua, ub); if (c != 0) return c; final va = androidFileUtils.getStorageVolume(a)?.path ?? ''; final vb = androidFileUtils.getStorageVolume(b)?.path ?? ''; return compareAsciiUpperCaseNatural(va, vb); } void notifyAlbumsChanged() { eventBus.fire(AlbumsChangedEvent()); } void _onAlbumChanged({bool notify = true}) { invalidateAlbumDisplayNames(); if (notify) { notifyAlbumsChanged(); } } Map getAlbumEntries() { final entries = sortedEntriesByDate; final regularAlbums = [], appAlbums = [], specialAlbums = []; for (final album in rawAlbums) { switch (androidFileUtils.getAlbumType(album)) { case AlbumType.regular: regularAlbums.add(album); break; case AlbumType.app: appAlbums.add(album); break; default: specialAlbums.add(album); break; } } return Map.fromEntries([...specialAlbums, ...appAlbums, ...regularAlbums].map((album) => MapEntry( album, entries.firstWhereOrNull((entry) => entry.directory == album), ))); } void updateDirectories() { final visibleDirectories = visibleEntries.map((entry) => entry.directory).toSet(); addDirectories(albums: visibleDirectories); cleanEmptyAlbums(); } void addDirectories({required Set albums, bool notify = true}) { if (!_directories.containsAll(albums)) { _directories.addAll(albums.whereNotNull()); _onAlbumChanged(notify: notify); } } void cleanEmptyAlbums([Set? albums]) { final emptyAlbums = (albums ?? _directories).where((v) => _isEmptyAlbum(v) && !_newAlbums.contains(v)).toSet(); if (emptyAlbums.isNotEmpty) { _directories.removeAll(emptyAlbums); _onAlbumChanged(); invalidateAlbumFilterSummary(directories: emptyAlbums); final bookmarks = settings.drawerAlbumBookmarks; emptyAlbums.forEach((album) { bookmarks?.remove(album); }); settings.drawerAlbumBookmarks = bookmarks; } } bool _isEmptyAlbum(String? album) => !visibleEntries.any((entry) => entry.directory == album); // filter summary // by directory final Map _filterEntryCountMap = {}, _filterSizeMap = {}; final Map _filterRecentEntryMap = {}; void invalidateAlbumFilterSummary({ Set? entries, Set? directories, bool notify = true, }) { if (_filterEntryCountMap.isEmpty && _filterSizeMap.isEmpty && _filterRecentEntryMap.isEmpty) return; if (entries == null && directories == null) { _filterEntryCountMap.clear(); _filterSizeMap.clear(); _filterRecentEntryMap.clear(); } else { directories ??= {}; if (entries != null) { directories.addAll(entries.map((entry) => entry.directory).whereNotNull()); } directories.forEach((directory) { _filterEntryCountMap.remove(directory); _filterSizeMap.remove(directory); _filterRecentEntryMap.remove(directory); }); } if (notify) { eventBus.fire(AlbumSummaryInvalidatedEvent(directories)); } } int albumEntryCount(AlbumFilter filter) { return _filterEntryCountMap.putIfAbsent(filter.album, () => visibleEntries.where(filter.test).length); } int albumSize(AlbumFilter filter) { return _filterSizeMap.putIfAbsent(filter.album, () => visibleEntries.where(filter.test).map((v) => v.sizeBytes).sum); } AvesEntry? albumRecentEntry(AlbumFilter filter) { return _filterRecentEntryMap.putIfAbsent(filter.album, () => sortedEntriesByDate.firstWhereOrNull(filter.test)); } // new albums void createAlbum(String directory) { _newAlbums.add(directory); addDirectories(albums: {directory}); } void renameNewAlbum(String source, String destination) { if (_newAlbums.remove(source)) { cleanEmptyAlbums({source}); createAlbum(destination); } } void forgetNewAlbums(Set directories) { _newAlbums.removeAll(directories); } // display names final Map _albumDisplayNamesWithContext = {}, _albumDisplayNamesWithoutContext = {}; void invalidateAlbumDisplayNames() { _albumDisplayNamesWithContext.clear(); _albumDisplayNamesWithoutContext.clear(); } String _computeDisplayName(BuildContext? context, String dirPath) { final separator = pContext.separator; assert(!dirPath.endsWith(separator)); if (context != null) { final type = androidFileUtils.getAlbumType(dirPath); switch (type) { case AlbumType.camera: return context.l10n.albumCamera; case AlbumType.download: return context.l10n.albumDownload; case AlbumType.screenshots: return context.l10n.albumScreenshots; case AlbumType.screenRecordings: return context.l10n.albumScreenRecordings; case AlbumType.videoCaptures: return context.l10n.albumVideoCaptures; case AlbumType.regular: case AlbumType.app: break; } } final dir = VolumeRelativeDirectory.fromPath(dirPath); if (dir == null) return dirPath; final relativeDir = dir.relativeDir; if (relativeDir.isEmpty) { final volume = androidFileUtils.getStorageVolume(dirPath)!; return volume.getDescription(context); } String unique(String dirPath, Set others) { final parts = pContext.split(dirPath); for (var i = parts.length - 1; i > 0; i--) { final name = pContext.joinAll(['', ...parts.skip(i)]); final testName = '$separator$name'; if (others.every((item) => !item.endsWith(testName))) return name; } return dirPath; } final otherAlbumsOnDevice = _directories.whereNotNull().where((item) => item != dirPath).toSet(); final uniqueNameInDevice = unique(dirPath, otherAlbumsOnDevice); if (uniqueNameInDevice.length <= relativeDir.length) { return uniqueNameInDevice; } final volumePath = dir.volumePath; String trimVolumePath(String? path) => path!.substring(dir.volumePath.length); final otherAlbumsOnVolume = otherAlbumsOnDevice.where((path) => path.startsWith(volumePath)).map(trimVolumePath).toSet(); final uniqueNameInVolume = unique(trimVolumePath(dirPath), otherAlbumsOnVolume); final volume = androidFileUtils.getStorageVolume(dirPath)!; if (volume.isPrimary) { return uniqueNameInVolume; } else { return '$uniqueNameInVolume (${volume.getDescription(context)})'; } } String getAlbumDisplayName(BuildContext? context, String dirPath) { final names = (context != null ? _albumDisplayNamesWithContext : _albumDisplayNamesWithoutContext); return names.putIfAbsent(dirPath, () => _computeDisplayName(context, dirPath)); } } class AlbumsChangedEvent {} class AlbumSummaryInvalidatedEvent { final Set? directories; const AlbumSummaryInvalidatedEvent(this.directories); }