user_collections_tab.dart 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. import 'dart:async';
  2. import 'package:flutter/material.dart';
  3. import 'package:logging/logging.dart';
  4. import "package:photos/core/configuration.dart";
  5. import 'package:photos/core/event_bus.dart';
  6. import 'package:photos/events/collection_updated_event.dart';
  7. import 'package:photos/events/local_photos_updated_event.dart';
  8. import 'package:photos/events/user_logged_out_event.dart';
  9. import "package:photos/generated/l10n.dart";
  10. import 'package:photos/models/collection.dart';
  11. import 'package:photos/services/collections_service.dart';
  12. import "package:photos/ui/collections/button/archived_button.dart";
  13. import "package:photos/ui/collections/button/hidden_button.dart";
  14. import "package:photos/ui/collections/button/trash_button.dart";
  15. import "package:photos/ui/collections/button/uncategorized_button.dart";
  16. import "package:photos/ui/collections/collection_list_page.dart";
  17. import "package:photos/ui/collections/create_new_album_widget.dart";
  18. import "package:photos/ui/collections/device/device_folders_grid_view.dart";
  19. import "package:photos/ui/collections/device/device_folders_vertical_grid_view.dart";
  20. import "package:photos/ui/collections/flex_grid_view.dart";
  21. import 'package:photos/ui/common/loading_widget.dart';
  22. import 'package:photos/ui/components/buttons/icon_button_widget.dart';
  23. import "package:photos/ui/tabs/section_title.dart";
  24. import "package:photos/ui/viewer/actions/delete_empty_albums.dart";
  25. import "package:photos/ui/viewer/gallery/empty_state.dart";
  26. import 'package:photos/utils/local_settings.dart';
  27. import "package:photos/utils/navigation_util.dart";
  28. class UserCollectionsTab extends StatefulWidget {
  29. const UserCollectionsTab({Key? key}) : super(key: key);
  30. @override
  31. State<UserCollectionsTab> createState() => _UserCollectionsTabState();
  32. }
  33. class _UserCollectionsTabState extends State<UserCollectionsTab>
  34. with AutomaticKeepAliveClientMixin {
  35. final _logger = Logger((_UserCollectionsTabState).toString());
  36. late StreamSubscription<LocalPhotosUpdatedEvent> _localFilesSubscription;
  37. late StreamSubscription<CollectionUpdatedEvent>
  38. _collectionUpdatesSubscription;
  39. late StreamSubscription<UserLoggedOutEvent> _loggedOutEvent;
  40. AlbumSortKey? sortKey;
  41. String _loadReason = "init";
  42. final _scrollController = ScrollController();
  43. static const int _kOnEnteItemLimitCount = 10;
  44. @override
  45. void initState() {
  46. _localFilesSubscription =
  47. Bus.instance.on<LocalPhotosUpdatedEvent>().listen((event) {
  48. _loadReason = event.reason;
  49. setState(() {});
  50. });
  51. _collectionUpdatesSubscription =
  52. Bus.instance.on<CollectionUpdatedEvent>().listen((event) {
  53. _loadReason = event.reason;
  54. setState(() {});
  55. });
  56. _loggedOutEvent = Bus.instance.on<UserLoggedOutEvent>().listen((event) {
  57. _loadReason = event.reason;
  58. setState(() {});
  59. });
  60. sortKey = LocalSettings.instance.albumSortKey();
  61. super.initState();
  62. }
  63. @override
  64. Widget build(BuildContext context) {
  65. super.build(context);
  66. _logger.info("Building, trigger: $_loadReason");
  67. return FutureBuilder<List<Collection>>(
  68. future: CollectionsService.instance.getCollectionForOnEnteSection(),
  69. builder: (context, snapshot) {
  70. if (snapshot.hasData) {
  71. return _getCollectionsGalleryWidget(snapshot.data!);
  72. } else if (snapshot.hasError) {
  73. return Text(snapshot.error.toString());
  74. } else {
  75. return const EnteLoadingWidget();
  76. }
  77. },
  78. );
  79. }
  80. Widget _getCollectionsGalleryWidget(List<Collection> collections) {
  81. final TextStyle trashAndHiddenTextStyle =
  82. Theme.of(context).textTheme.titleMedium!.copyWith(
  83. color: Theme.of(context)
  84. .textTheme
  85. .titleMedium!
  86. .color!
  87. .withOpacity(0.5),
  88. );
  89. return CustomScrollView(
  90. controller: _scrollController,
  91. slivers: [
  92. SliverToBoxAdapter(
  93. child: SectionOptions(
  94. Hero(
  95. tag: "OnDeviceAppTitle",
  96. child: SectionTitle(title: S.of(context).onDevice),
  97. ),
  98. trailingWidget: IconButtonWidget(
  99. icon: Icons.chevron_right,
  100. iconButtonType: IconButtonType.secondary,
  101. onTap: () {
  102. unawaited(
  103. routeToPage(
  104. context,
  105. DeviceFolderVerticalGridView(
  106. appTitle: SectionTitle(
  107. title: S.of(context).onDevice,
  108. ),
  109. tag: "OnDeviceAppTitle",
  110. ),
  111. ),
  112. );
  113. },
  114. ),
  115. ),
  116. ),
  117. const SliverToBoxAdapter(child: DeviceFoldersGridView()),
  118. SliverToBoxAdapter(
  119. child: SectionOptions(
  120. SectionTitle(titleWithBrand: getOnEnteSection(context)),
  121. trailingWidget: _sortMenu(collections),
  122. padding: const EdgeInsets.only(left: 12, right: 6),
  123. ),
  124. ),
  125. SliverToBoxAdapter(child: DeleteEmptyAlbums(collections ?? [])),
  126. Configuration.instance.hasConfiguredAccount()
  127. ? CollectionsFlexiGridViewWidget(
  128. collections,
  129. displayLimitCount: _kOnEnteItemLimitCount,
  130. shrinkWrap: true,
  131. )
  132. : const SliverToBoxAdapter(child: EmptyState()),
  133. collections.length > _kOnEnteItemLimitCount
  134. ? SliverToBoxAdapter(
  135. child: GestureDetector(
  136. behavior: HitTestBehavior.opaque,
  137. onTap: () {
  138. unawaited(
  139. routeToPage(
  140. context,
  141. CollectionListPage(
  142. collections,
  143. sectionType: UISectionType.homeCollections,
  144. appTitle: SectionTitle(
  145. titleWithBrand: getOnEnteSection(context),
  146. ),
  147. initialScrollOffset: _scrollController.offset,
  148. ),
  149. ),
  150. );
  151. },
  152. child: SectionOptions(
  153. SectionTitle(
  154. title: S.of(context).viewAll,
  155. mutedTitle: true,
  156. ),
  157. trailingWidget: const IconButtonWidget(
  158. icon: Icons.chevron_right,
  159. iconButtonType: IconButtonType.secondary,
  160. ),
  161. ),
  162. ),
  163. )
  164. : const SliverToBoxAdapter(child: SizedBox.shrink()),
  165. const SliverToBoxAdapter(child: Divider()),
  166. const SliverToBoxAdapter(child: SizedBox(height: 12)),
  167. SliverToBoxAdapter(
  168. child: Padding(
  169. padding: const EdgeInsets.symmetric(horizontal: 12),
  170. child: Column(
  171. children: [
  172. UnCategorizedCollections(trashAndHiddenTextStyle),
  173. const SizedBox(height: 12),
  174. ArchivedCollectionsButton(trashAndHiddenTextStyle),
  175. const SizedBox(height: 12),
  176. HiddenCollectionsButtonWidget(trashAndHiddenTextStyle),
  177. const SizedBox(height: 12),
  178. TrashSectionButton(trashAndHiddenTextStyle),
  179. ],
  180. ),
  181. ),
  182. ),
  183. SliverToBoxAdapter(
  184. child: SizedBox(height: 64 + MediaQuery.of(context).padding.bottom),
  185. ),
  186. ],
  187. );
  188. }
  189. Widget _sortMenu(List<Collection> collections) {
  190. Text sortOptionText(AlbumSortKey key) {
  191. String text = key.toString();
  192. switch (key) {
  193. case AlbumSortKey.albumName:
  194. text = S.of(context).name;
  195. break;
  196. case AlbumSortKey.newestPhoto:
  197. text = S.of(context).newest;
  198. break;
  199. case AlbumSortKey.lastUpdated:
  200. text = S.of(context).lastUpdated;
  201. }
  202. return Text(
  203. text,
  204. style: Theme.of(context).textTheme.titleMedium!.copyWith(
  205. fontSize: 14,
  206. color: Theme.of(context).iconTheme.color!.withOpacity(0.7),
  207. ),
  208. );
  209. }
  210. return Theme(
  211. data: Theme.of(context).copyWith(
  212. highlightColor: Colors.transparent,
  213. splashColor: Colors.transparent,
  214. ),
  215. child: Row(
  216. children: [
  217. const CreateNewAlbumIcon(),
  218. GestureDetector(
  219. onTapDown: (TapDownDetails details) async {
  220. final int? selectedValue = await showMenu<int>(
  221. context: context,
  222. position: RelativeRect.fromLTRB(
  223. details.globalPosition.dx,
  224. details.globalPosition.dy,
  225. details.globalPosition.dx,
  226. details.globalPosition.dy + 50,
  227. ),
  228. items: List.generate(AlbumSortKey.values.length, (index) {
  229. return PopupMenuItem(
  230. value: index,
  231. child: sortOptionText(AlbumSortKey.values[index]),
  232. );
  233. }),
  234. );
  235. if (selectedValue != null) {
  236. sortKey = AlbumSortKey.values[selectedValue];
  237. await LocalSettings.instance.setAlbumSortKey(sortKey!);
  238. setState(() {});
  239. }
  240. },
  241. child: const IconButtonWidget(
  242. icon: Icons.sort_outlined,
  243. iconButtonType: IconButtonType.secondary,
  244. ),
  245. ),
  246. ],
  247. ),
  248. );
  249. }
  250. @override
  251. void dispose() {
  252. _localFilesSubscription.cancel();
  253. _collectionUpdatesSubscription.cancel();
  254. _loggedOutEvent.cancel();
  255. _scrollController.dispose();
  256. super.dispose();
  257. }
  258. @override
  259. bool get wantKeepAlive => true;
  260. }