photo_selected_preview_controller.dart 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. import 'package:clean/base/base_controller.dart';
  2. import 'package:clean/data/bean/photos_type.dart';
  3. import 'package:clean/dialog/photo_delete_finish_dialog.dart';
  4. import 'package:clean/dialog/photo_deleting_dialog.dart';
  5. import 'package:clean/module/image_picker/image_picker_util.dart';
  6. import 'package:clean/module/people_photo/photo_group.dart';
  7. import 'package:clean/module/photo_preview/photo_preview_controller.dart';
  8. import 'package:clean/utils/toast_util.dart';
  9. import 'package:flutter/Material.dart';
  10. import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
  11. import 'package:get/get.dart';
  12. import 'package:wechat_assets_picker/wechat_assets_picker.dart';
  13. class PhotoSelectedPreviewController extends BaseController {
  14. late PhotosType photosType;
  15. final RxList<PhotoGroup> photoGroups = <PhotoGroup>[].obs;
  16. final RxDouble selectedFilesSize = 0.0.obs;
  17. RxInt selectedFileCount = 0.obs;
  18. RxList<AssetEntity> selectedPhotos = <AssetEntity>[].obs;
  19. late final RxSet<String> selectedPhotosIds;
  20. PhotoPreviewController photoPreviewController =
  21. Get.find<PhotoPreviewController>();
  22. @override
  23. onInit() {
  24. super.onInit();
  25. _getArgs();
  26. loadPhotoSelectedPreview();
  27. }
  28. _getArgs() {
  29. photosType = parameters?['photosType'];
  30. selectedPhotosIds = parameters?['selectedIds'];
  31. }
  32. loadPhotoSelectedPreview() {
  33. photoGroups.clear();
  34. for (var assetsEntity in photoPreviewController.photoGroups) {
  35. if (selectedPhotosIds.contains(assetsEntity.id)) {
  36. selectedPhotos.add(assetsEntity);
  37. }
  38. }
  39. if (selectedPhotos.isNotEmpty) {
  40. photoGroups.add(PhotoGroup(
  41. isSelected: true,
  42. images: selectedPhotos,
  43. ));
  44. }
  45. updateSelectedFilesSize();
  46. selectedFileCount.value = selectedPhotosIds.length;
  47. }
  48. void restoreSelections() async {
  49. for (var group in photoGroups) {
  50. for (int i = 0; i < group.images.length; i++) {
  51. if (selectedPhotosIds.contains(group.images[i].id)) {
  52. group.selectedImages[i] = true;
  53. } else {
  54. group.selectedImages[i] = false;
  55. }
  56. }
  57. group.isSelected.value =
  58. group.selectedImages.every((selected) => selected);
  59. }
  60. updateSelectedFilesSize();
  61. selectedFileCount.value = selectedPhotosIds.length;
  62. }
  63. clickDelete() async {
  64. print('clickDelete');
  65. photoDeletingDialog();
  66. switch (photosType) {
  67. case PhotosType.peoplePhotos:
  68. ImagePickerUtil.selectedPeoplePhotosIds.assignAll(selectedPhotosIds);
  69. break;
  70. case PhotosType.screenshots:
  71. ImagePickerUtil.selectedScreenshotPhotosIds
  72. .assignAll(selectedPhotosIds);
  73. break;
  74. case PhotosType.similarPhotos:
  75. ImagePickerUtil.selectedSimilarPhotosIds.assignAll(selectedPhotosIds);
  76. break;
  77. case PhotosType.locationPhotos:
  78. ImagePickerUtil.selectedLocationPhotosIds.assignAll(selectedPhotosIds);
  79. break;
  80. case PhotosType.blurryPhotos:
  81. ImagePickerUtil.selectedBlurryPhotosIds.assignAll(selectedPhotosIds);
  82. break;
  83. }
  84. if (selectedPhotosIds.isNotEmpty) {
  85. final assetsToDelete = photoGroups
  86. .expand((group) => group.images
  87. .where((asset) => selectedPhotosIds.contains(asset.id)))
  88. .toList();
  89. final List<String> result = await PhotoManager.editor.deleteWithIds(
  90. assetsToDelete.map((e) => e.id).toList(),
  91. );
  92. // 比较result和selectedPhotosIds,如果result和selectedPhotosIds相等,说明删除成功,走下面的逻辑
  93. // 如果不相等,说明有删除失败的,走else逻辑
  94. if (result.length == selectedPhotosIds.length) {
  95. switch (photosType) {
  96. case PhotosType.peoplePhotos:
  97. ImagePickerUtil.updatePhotoGroupDate(
  98. photosType, ImagePickerUtil.selectedPeoplePhotosIds);
  99. break;
  100. case PhotosType.screenshots:
  101. ImagePickerUtil.updatePhotoGroupDate(
  102. photosType, ImagePickerUtil.selectedScreenshotPhotosIds);
  103. break;
  104. case PhotosType.similarPhotos:
  105. ImagePickerUtil.updatePhotoGroupDate(
  106. photosType, ImagePickerUtil.selectedSimilarPhotosIds);
  107. break;
  108. case PhotosType.locationPhotos:
  109. ImagePickerUtil.updatePhotoGroupDate(
  110. photosType, ImagePickerUtil.selectedLocationPhotosIds);
  111. break;
  112. case PhotosType.blurryPhotos:
  113. ImagePickerUtil.updatePhotoGroupDate(
  114. photosType, ImagePickerUtil.selectedBlurryPhotosIds);
  115. break;
  116. }
  117. photoGroups.removeWhere((element) => element.images
  118. .any((image) => selectedPhotosIds.contains(image.id)));
  119. photoGroups.removeWhere((element) => element.images.isEmpty);
  120. selectedPhotosIds.clear();
  121. restoreSelections();
  122. Future.delayed(Duration(seconds: 3), () {
  123. SmartDialog.dismiss(tag: 'photoDeletingDialog');
  124. photoDeleteFinishDialog();
  125. });
  126. } else {
  127. Future.delayed(Duration(seconds: 3), () {
  128. SmartDialog.dismiss(tag: 'photoDeletingDialog');
  129. });
  130. ToastUtil.show("Delete failed");
  131. }
  132. }
  133. }
  134. void toggleGroupSelection(List<AssetEntity> images) {
  135. final group = getGroupByImages(images);
  136. final newValue = !group.isSelected.value;
  137. group.toggleSelectAll(newValue);
  138. for (var image in group.images) {
  139. _updateSelectedPhotosIds(image.id, newValue);
  140. }
  141. updateSelectedFilesSize();
  142. selectedFileCount.value = selectedPhotosIds.length;
  143. }
  144. Future<void> updateSelectedFilesSize() async {
  145. double totalSize = 0;
  146. for (var group in photoGroups) {
  147. for (int i = 0; i < group.images.length; i++) {
  148. if (group.selectedImages[i]) {
  149. final file = await group.images[i].file;
  150. if (file != null) {
  151. totalSize += file.lengthSync();
  152. }
  153. }
  154. }
  155. }
  156. selectedFilesSize.value = totalSize / 1024; // Convert to KB
  157. // 如果超过1MB转成MB
  158. if (selectedFilesSize.value > 1024) {
  159. selectedFilesSize.value = selectedFilesSize.value / 1024;
  160. }
  161. // 如果超过1GB转成GB
  162. if (selectedFilesSize.value > 1024 * 1024) {
  163. selectedFilesSize.value = selectedFilesSize.value / 1024 / 1024;
  164. }
  165. }
  166. void clickImage(List<AssetEntity> images, int imageIndex) {
  167. final group = getGroupByImages(images);
  168. final image = group.images[imageIndex];
  169. }
  170. void toggleImageSelection(List<AssetEntity> images, int imageIndex) {
  171. final group = getGroupByImages(images);
  172. final image = group.images[imageIndex];
  173. final selected = !group.selectedImages[imageIndex];
  174. group.selectedImages[imageIndex] = selected;
  175. _updateSelectedPhotosIds(image.id, selected);
  176. group.isSelected.value = group.selectedImages.every((selected) => selected);
  177. updateSelectedFilesSize();
  178. selectedFileCount.value = selectedPhotosIds.length;
  179. }
  180. PhotoGroup getGroupByImages(List<AssetEntity> images) {
  181. final imageIds = images.map((img) => img.id).toSet();
  182. return photoGroups.firstWhere(
  183. (group) => group.images.every((image) => imageIds.contains(image.id)));
  184. }
  185. void _updateSelectedPhotosIds(String photoId, bool isSelected) {
  186. if (isSelected) {
  187. selectedPhotosIds.add(photoId);
  188. } else {
  189. selectedPhotosIds.remove(photoId);
  190. }
  191. }
  192. }