home_controller.dart 11 KB

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