base_photo_controller.dart 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. import 'package:clean/base/base_controller.dart';
  2. import 'package:clean/data/repositories/user_repository.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/store/store_view.dart';
  6. import 'package:flutter/cupertino.dart';
  7. import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
  8. import 'package:get/get.dart';
  9. import 'package:wechat_assets_picker/wechat_assets_picker.dart';
  10. import 'package:clean/module/image_picker/image_picker_util.dart';
  11. import 'package:clean/module/people_photo/photo_group.dart';
  12. import 'package:clean/utils/toast_util.dart';
  13. import 'package:clean/data/bean/photos_type.dart';
  14. import 'package:clean/module/photo_preview/photo_preview_view.dart';
  15. import '../utils/file_size_calculator_util.dart';
  16. abstract class BasePhotoController extends BaseController {
  17. final RxList<PhotoGroup> photoGroups = <PhotoGroup>[].obs;
  18. final RxDouble selectedFilesSize = 0.0.obs;
  19. RxInt selectedFileCount = 0.obs;
  20. final RxSet<String> selectedPhotosIds = <String>{}.obs;
  21. // 将selectedFilesSize转成String类型,然后单位转换,如果超过1MB,则转成MB,超过1GB,则转成GB,否则KB
  22. String get selectedFilesSizeString {
  23. debugPrint(
  24. "BasePhotoController selectedFilesSize.value ${selectedFilesSize.value}");
  25. final double sizeInKB = selectedFilesSize.value;
  26. if (sizeInKB >= 1024 * 1024) {
  27. // 先检查最大单位(GB)
  28. return "${(sizeInKB / (1024 * 1024)).toStringAsFixed(2)}GB";
  29. } else if (sizeInKB >= 1024) {
  30. // 然后检查MB
  31. return "${(sizeInKB / 1024).toStringAsFixed(2)}MB";
  32. } else {
  33. // 最后是KB
  34. return "${sizeInKB.toStringAsFixed(2)}KB";
  35. }
  36. }
  37. Future<void> updateSelectedFilesSize() async {
  38. if (selectedPhotosIds.isEmpty) {
  39. selectedFilesSize.value = 0;
  40. return;
  41. }
  42. double totalSize = 0;
  43. final uncasedIds = selectedPhotosIds.where((id) => !FileSizeCalculatorUtil.fileSizeCache.containsKey(id)).toSet();
  44. // **1️⃣ 先处理缓存中的文件**
  45. totalSize = selectedPhotosIds.fold(0, (prev, id) => prev + (FileSizeCalculatorUtil.fileSizeCache[id] ?? 0));
  46. // **2️⃣ 分批处理未缓存的文件**
  47. const batchSize = 50;
  48. for (int i = 0; i < uncasedIds.length; i += batchSize) {
  49. if (selectedPhotosIds.isEmpty) {
  50. selectedFilesSize.value = 0;
  51. return;
  52. }
  53. final batch = uncasedIds.skip(i).take(batchSize);
  54. final sizes = await Future.wait(batch.map(FileSizeCalculatorUtil.getFileSize));
  55. totalSize += sizes.fold(0, (sum, size) => sum + size);
  56. // **再检查一次是否被清空,避免无意义计算**
  57. if (selectedPhotosIds.isEmpty) {
  58. selectedFilesSize.value = 0;
  59. return;
  60. }
  61. // **减少 UI 更新频率**
  62. if (i % (batchSize * 2) == 0 || i + batchSize >= uncasedIds.length) {
  63. selectedFilesSize.value = totalSize;
  64. }
  65. await Future.delayed(Duration.zero);
  66. }
  67. selectedFilesSize.value = totalSize; // 确保最终更新总大小
  68. PhotoManager.clearFileCache();
  69. }
  70. // 切换图片选中状态
  71. Future<void> toggleImageSelection(
  72. List<AssetEntity> groupTitle, int imageIndex) async {
  73. print("BasePhotoController toggleImageSelection");
  74. final group = getGroupByImages(groupTitle);
  75. final image = group.images[imageIndex];
  76. final selected = !group.selectedImages[imageIndex];
  77. group.selectedImages[imageIndex] = selected;
  78. updateSelectedPhotosIds(image.id, selected);
  79. group.isSelected.value = group.selectedImages.every((selected) => selected);
  80. selectedFileCount.value = selectedPhotosIds.length;
  81. // 更新文件大小
  82. if (selected) {
  83. // 如果选中,增加文件大小
  84. if (FileSizeCalculatorUtil.fileSizeCache.containsKey(image.id)) {
  85. selectedFilesSize.value += FileSizeCalculatorUtil.fileSizeCache[image.id]!;
  86. } else {
  87. final file = await image.file;
  88. if (file != null) {
  89. selectedFilesSize.value += (await file.length()) / 1024; // 转换为KB
  90. }
  91. }
  92. } else {
  93. // 如果取消选中,减少文件大小
  94. if (FileSizeCalculatorUtil.fileSizeCache.containsKey(image.id)) {
  95. selectedFilesSize.value -= FileSizeCalculatorUtil.fileSizeCache[image.id]!;
  96. } else {
  97. final file = await image.file;
  98. if (file != null) {
  99. selectedFilesSize.value -= (await file.length()) / 1024; // 转换为KB
  100. }
  101. }
  102. }
  103. PhotoManager.clearFileCache();
  104. }
  105. void clickImage(List<AssetEntity> images, int imageIndex, PhotosType type) {
  106. print("BasePhotoController clickImage");
  107. final group = getGroupByImages(images);
  108. final image = group.images[imageIndex];
  109. PhotoPreviewPage.start(type, image.id);
  110. }
  111. // 切换图片组选中状态
  112. void toggleGroupSelection(List<AssetEntity> imagesList) {
  113. final group = getGroupByImages(imagesList);
  114. final newValue = !group.isSelected.value;
  115. group.toggleSelectAll(newValue);
  116. for (var image in group.images) {
  117. updateSelectedPhotosIds(image.id, newValue);
  118. }
  119. selectedFileCount.value = selectedPhotosIds.length;
  120. updateSelectedFilesSize();
  121. }
  122. PhotoGroup getGroupByImages(List<AssetEntity> images) {
  123. final imageIds = images.map((img) => img.id).toSet();
  124. return photoGroups.firstWhere(
  125. (group) => group.images.every((image) => imageIds.contains(image.id)));
  126. }
  127. void updateSelectedPhotosIds(String photoId, bool isSelected) {
  128. if (isSelected) {
  129. selectedPhotosIds.add(photoId);
  130. } else {
  131. selectedPhotosIds.remove(photoId);
  132. }
  133. }
  134. // 恢复选中状态
  135. void restoreSelections() async {
  136. final selectedIds = selectedPhotosIds.toSet();
  137. for (var group in photoGroups) {
  138. for (int i = 0; i < group.images.length; i++) {
  139. group.selectedImages[i] = selectedIds.contains(group.images[i].id);
  140. }
  141. group.isSelected.value =
  142. group.selectedImages.every((selected) => selected);
  143. }
  144. selectedFileCount.value = selectedIds.length;
  145. if (selectedIds.isEmpty) {
  146. selectedFilesSize.value = 0;
  147. return;
  148. }
  149. updateSelectedFilesSize();
  150. }
  151. @override
  152. void onInit() {
  153. print("BasePhotoController onInit");
  154. super.onInit();
  155. loadPhotos();
  156. restoreSelections();
  157. }
  158. void loadPhotos();
  159. // 点击删除
  160. void clickDelete() async {
  161. if (userRepository.isVip()) {
  162. if (selectedPhotosIds.isNotEmpty) {
  163. photoDeletingDialog();
  164. final assetsToDelete = photoGroups
  165. .expand((group) => group.images
  166. .where((asset) => selectedPhotosIds.contains(asset.id)))
  167. .toList();
  168. final List<String> result = await PhotoManager.editor.deleteWithIds(
  169. assetsToDelete.map((e) => e.id).toList(),
  170. );
  171. if (result.length == selectedPhotosIds.length) {
  172. for (var group in photoGroups) {
  173. group.images.removeWhere(
  174. (element) => selectedPhotosIds.contains(element.id));
  175. }
  176. ImagePickerUtil.updatePhotoGroupDate(
  177. getPhotosType(), selectedPhotosIds);
  178. selectedPhotosIds.clear();
  179. selectedFileCount.value = selectedPhotosIds.length;
  180. print(
  181. "BasePhotoController clickDelete selectedPhotosIds $selectedPhotosIds");
  182. updateSelectedFilesSize();
  183. ToastUtil.show("Delete success");
  184. Future.delayed(Duration(seconds: 2), () {
  185. SmartDialog.dismiss(tag: 'photoDeletingDialog');
  186. photoDeleteFinishDialog();
  187. });
  188. } else {
  189. SmartDialog.dismiss(tag: 'photoDeletingDialog');
  190. ToastUtil.show("Delete failed");
  191. }
  192. }
  193. } else {
  194. StorePage.start();
  195. }
  196. }
  197. void updateSelections(Set<String> selectedIds) {
  198. print(
  199. "BasePhotoController updateSelections selectedIds $selectedIds, selectedPhotosIds $selectedPhotosIds getPhotosType() ${getPhotosType()}");
  200. // selectedId如果是selectedPhotosIds对象,那么selectedPhotosIds.assignAll(selectedIds)会清空
  201. // selectedPhotosIds.assignAll(selectedIds);
  202. selectedPhotosIds.assignAll(Set.from(selectedIds));
  203. print(
  204. "BasePhotoController updateSelections selectedIds $selectedIds, selectedPhotosIds $selectedPhotosIds getPhotosType() ${getPhotosType()}");
  205. switch (getPhotosType()) {
  206. case PhotosType.peoplePhotos:
  207. ImagePickerUtil.selectedPeoplePhotosIds.assignAll(selectedPhotosIds);
  208. break;
  209. case PhotosType.screenshots:
  210. ImagePickerUtil.selectedScreenshotPhotosIds
  211. .assignAll(selectedPhotosIds);
  212. break;
  213. case PhotosType.similarPhotos:
  214. ImagePickerUtil.selectedSimilarPhotosIds.assignAll(selectedPhotosIds);
  215. break;
  216. case PhotosType.locationPhotos:
  217. ImagePickerUtil.selectedLocationPhotosIds.assignAll(selectedPhotosIds);
  218. break;
  219. case PhotosType.blurryPhotos:
  220. ImagePickerUtil.selectedBlurryPhotosIds.assignAll(selectedPhotosIds);
  221. break;
  222. }
  223. }
  224. PhotosType getPhotosType();
  225. // 将photoGroups中所有的图片返回
  226. List<AssetEntity> getAllPhotos() {
  227. return photoGroups.expand((group) => group.images).toList();
  228. }
  229. @override
  230. void onClose() {
  231. super.onClose();
  232. PhotoManager.clearFileCache();
  233. }
  234. }