home_controller.dart 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. import 'dart:convert';
  2. import 'dart:io';
  3. import 'package:airbridge_flutter_sdk/airbridge_flutter_sdk.dart';
  4. import 'package:classify_photo/classify_photo.dart';
  5. import 'package:clean/base/base_controller.dart';
  6. import 'package:clean/data/consts/constants.dart';
  7. import 'package:clean/data/repositories/event_repository.dart';
  8. import 'package:clean/data/repositories/user_repository.dart';
  9. import 'package:clean/module/image_picker/image_picker_util.dart';
  10. import 'package:clean/module/locations_photo/locations_photo_view.dart';
  11. import 'package:clean/module/people_photo/people_photo_view.dart';
  12. import 'package:clean/module/screenshots_blurry/screenshots_view.dart';
  13. import 'package:clean/module/similar_photo/similar_photo_view.dart';
  14. import 'package:clean/router/app_pages.dart';
  15. import 'package:clean/utils/toast_util.dart';
  16. import 'package:get/get.dart';
  17. import 'package:permission_handler/permission_handler.dart';
  18. import 'package:wechat_assets_picker/wechat_assets_picker.dart';
  19. import '../../data/api/response/user_info_response.dart';
  20. import '../../data/consts/event_report_id.dart';
  21. import '../../handler/event_handler.dart';
  22. class HomeController extends BaseController {
  23. Rx<double> totalSpace = 0.0.obs;
  24. Rx<double> usedSpace = 0.0.obs;
  25. Rx<double> photoSpace = 0.0.obs;
  26. Rx<double> freeSpace = 0.0.obs;
  27. Rx<String> totalSpaceStr = "".obs;
  28. Rx<String> usedSpaceStr = "".obs;
  29. Rx<String> photoSpaceStr = "".obs;
  30. Rx<String> freeSpaceStr = "".obs;
  31. // 计算已用存储百分比
  32. double get usedSpacePercentage => (usedSpace.value / totalSpace.value) * 100;
  33. // 计算照片占用存储百分比
  34. double get photoSpacePercentage =>
  35. (photoSpace.value / totalSpace.value) * 100;
  36. // 计算可用存储百分比
  37. double get freeSpacePercentage => (freeSpace.value / totalSpace.value) * 100;
  38. RxList<String> similarImages =
  39. List.generate(4, (index) => 'iconHomeNoPhoto').obs;
  40. RxInt imageCount = 0.obs;
  41. // 相似图片
  42. RxList<AssetEntity> similarPhotos = <AssetEntity>[].obs;
  43. // 人物图片
  44. RxList<AssetEntity> peoplePhotos = <AssetEntity>[].obs;
  45. // 地点图片
  46. Rx<AssetEntity?> locationPhoto = Rx<AssetEntity?>(null);
  47. // 截图照片
  48. Rx<AssetEntity?> screenshotPhoto = Rx<AssetEntity?>(null);
  49. // 模糊照片
  50. Rx<AssetEntity?> blurryPhoto = Rx<AssetEntity?>(null);
  51. // 是否扫描完成
  52. RxBool isScanned = false.obs;
  53. // 存储是否扫描完成
  54. RxBool isStorageScanned = false.obs;
  55. UserInfoResponse? get userInfo => userRepository.userInfo.value;
  56. @override
  57. Future<void> onInit() async {
  58. // TODO: implement onInit
  59. super.onInit();
  60. if (Platform.isAndroid) {
  61. loadPhotosFromDirectory();
  62. }
  63. if (await Permission.photos.request().isGranted) {
  64. PhotoManager.clearFileCache();
  65. getStorageInfo();
  66. handlePhotos();
  67. } else {
  68. ToastUtil.show("Please enable the album permission");
  69. }
  70. await userRepository.getUserInfo();
  71. if (userRepository.userInfo.value != null) {
  72. Airbridge.setUserID(userRepository.userInfo.value!.ssid);
  73. // 接收归因结果
  74. Airbridge.setOnAttributionReceived((result) {
  75. print(result);
  76. Map<String, String> attr = <String, String>{};
  77. attr["attributedChannel"] = "Appstore";
  78. Airbridge.fetchDeviceUUID(onSuccess: (uuid) {
  79. eventRepository.attrPush(uuid, "airbridge", jsonEncode(result));
  80. });
  81. });
  82. }
  83. EventHandler.pushInstall();
  84. if (!isFirstIntoApp() && !userRepository.isVip()) {
  85. Get.toNamed(RoutePath.discount);
  86. }
  87. setFirstIntoApp(false);
  88. }
  89. @override
  90. void onReady() {
  91. super.onReady();
  92. // EventHandler.report(EventId.event_03000);
  93. }
  94. Future<void> loadPhotosFromDirectory() async {
  95. if (ImagePickerUtil.peoplePhotos.isEmpty ||
  96. ImagePickerUtil.similarPhotos.isEmpty ||
  97. ImagePickerUtil.locationPhotos.isEmpty ||
  98. ImagePickerUtil.screenshotPhotos.isEmpty) {
  99. try {
  100. final List<AssetEntity> result = await ImagePickerUtil.loadAssets();
  101. ImagePickerUtil.peoplePhotos.value = result ?? [];
  102. ImagePickerUtil.locationPhotos['location'] = result ?? [];
  103. ImagePickerUtil.screenshotPhotos.value = result ?? [];
  104. ImagePickerUtil.similarPhotos.add(result ?? []);
  105. ImagePickerUtil.blurryPhotos.value = result ?? [];
  106. } catch (e) {
  107. print('Error loading photos: $e');
  108. }
  109. }
  110. }
  111. Future<void> getStorageInfo() async {
  112. final classifyPhoto = ClassifyPhoto();
  113. try {
  114. final storageInfo = await classifyPhoto.getStorageInfo();
  115. // 转换为 GB
  116. final totalSpaceGB = storageInfo['totalSpace']! / (1000 * 1000 * 1000);
  117. final freeSpaceGB = storageInfo['freeSpace']! / (1024 * 1024 * 1024);
  118. final usedSpaceGB = storageInfo['usedSpace']! / (1024 * 1024 * 1024);
  119. final photoSpaceGB = storageInfo['photoSpace']! / (1024 * 1024 * 1024);
  120. totalSpaceStr.value = ImagePickerUtil.formatFileSize(
  121. storageInfo['totalSpace']!,
  122. decimals: 1);
  123. freeSpaceStr.value = ImagePickerUtil.formatFileSize(
  124. storageInfo['freeSpace']!,
  125. decimals: 1);
  126. usedSpaceStr.value = ImagePickerUtil.formatFileSize(
  127. storageInfo['usedSpace']!,
  128. decimals: 1);
  129. photoSpaceStr.value = ImagePickerUtil.formatFileSize(
  130. storageInfo['photoSpace']!,
  131. decimals: 1);
  132. totalSpace.value = totalSpaceGB.round().toDouble();
  133. freeSpace.value = freeSpaceGB;
  134. usedSpace.value = usedSpaceGB;
  135. photoSpace.value = photoSpaceGB;
  136. print('总容量: $totalSpaceStr');
  137. print('可用空间: $freeSpaceStr');
  138. print('已用空间: $usedSpaceStr');
  139. print('照片占用: $photoSpaceStr');
  140. isStorageScanned.value = true;
  141. } catch (e) {
  142. print('获取存储信息失败: $e');
  143. }
  144. }
  145. Future<void> handlePhotos() async {
  146. final photoClassify = ClassifyPhoto();
  147. try {
  148. print('开始获取照片');
  149. final photos = await photoClassify.getPhoto();
  150. print('获取照片完成: ${photos?.length ?? 0} 组照片');
  151. // 已完成扫描
  152. isScanned.value = true;
  153. if (photos != null) {
  154. await ImagePickerUtil.updatePhotos(photos);
  155. similarPhotos.clear();
  156. if (ImagePickerUtil.similarPhotos.isNotEmpty) {
  157. for (var group in ImagePickerUtil.similarPhotos) {
  158. for (var asset in group) {
  159. similarPhotos.add(asset);
  160. if (similarPhotos.length == 4) {
  161. break;
  162. }
  163. }
  164. }
  165. }
  166. // 处理地点照片
  167. locationPhoto.value = null;
  168. if (ImagePickerUtil.locationPhotos.isNotEmpty) {
  169. // 获取第一个地点的第一张照片
  170. final firstLocationPhotos =
  171. ImagePickerUtil.locationPhotos.values.first;
  172. if (firstLocationPhotos.isNotEmpty) {
  173. var asset = firstLocationPhotos.first;
  174. locationPhoto.value = asset;
  175. }
  176. }
  177. // 处理人物照片
  178. peoplePhotos.clear();
  179. if (ImagePickerUtil.peoplePhotos.isNotEmpty) {
  180. for (var personPhotos in ImagePickerUtil.peoplePhotos) {
  181. peoplePhotos.add(personPhotos);
  182. if (peoplePhotos.length == 2) {
  183. break;
  184. }
  185. }
  186. }
  187. if (ImagePickerUtil.screenshotPhotos.isNotEmpty) {
  188. var asset = ImagePickerUtil.screenshotPhotos.first;
  189. screenshotPhoto.value = asset;
  190. }
  191. if (ImagePickerUtil.blurryPhotos.isNotEmpty) {
  192. var asset = ImagePickerUtil.blurryPhotos.first;
  193. blurryPhoto.value = asset;
  194. }
  195. }
  196. } catch (e, stackTrace) {
  197. print('获取照片失败: $e');
  198. print('Stack trace: $stackTrace');
  199. }
  200. }
  201. void _navigateAndStartPage(Function pageStartFunction) {
  202. // if (isFirstClickHomeClean()) {
  203. // setFirstClickHomeClean(false);
  204. // Get.toNamed(RoutePath.discount)?.then((value) {
  205. // pageStartFunction();
  206. // });
  207. // } else {
  208. pageStartFunction();
  209. // }
  210. }
  211. similarCleanClick() {
  212. print('similarCleanClick');
  213. EventHandler.report(EventId.event_03001);
  214. _navigateAndStartPage(SimilarPhotoPage.start);
  215. }
  216. peopleCleanClick() {
  217. print('peopleCleanClick');
  218. EventHandler.report(EventId.event_03002);
  219. _navigateAndStartPage(PeoplePhotoPage.start);
  220. }
  221. locationCleanClick() {
  222. print('locationCleanClick');
  223. EventHandler.report(EventId.event_03003);
  224. _navigateAndStartPage(LocationsPhotoPage.start);
  225. }
  226. screenshotCleanClick() {
  227. print('screenshotCleanClick');
  228. EventHandler.report(EventId.event_03004);
  229. _navigateAndStartPage(() => ScreenshotsPage.start("Screenshots"));
  230. }
  231. blurryCleanClick() {
  232. print('blurCleanClick');
  233. EventHandler.report(EventId.event_03005);
  234. _navigateAndStartPage(() => ScreenshotsPage.start("Blurry"));
  235. }
  236. titleVipClick() {
  237. EventHandler.report(EventId.event_02000);
  238. Get.toNamed(RoutePath.store);
  239. }
  240. }