file_sync_util.dart 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. import 'dart:math';
  2. import 'package:computer/computer.dart';
  3. import 'package:logging/logging.dart';
  4. import 'package:photo_manager/photo_manager.dart';
  5. import 'package:photos/models/file.dart';
  6. final _logger = Logger("FileSyncUtil");
  7. final ignoreSizeConstraint = SizeConstraint(ignoreSize: true);
  8. final assetFetchPageSize = 2000;
  9. Future<List<File>> getDeviceFiles(
  10. int fromTime,
  11. int toTime,
  12. Computer computer,
  13. ) async {
  14. final pathEntities = await _getGalleryList(fromTime, toTime);
  15. List<File> files = [];
  16. AssetPathEntity recents;
  17. for (AssetPathEntity pathEntity in pathEntities) {
  18. if (pathEntity.name == "Recent" || pathEntity.name == "Recents") {
  19. recents = pathEntity;
  20. } else {
  21. files = await _computeFiles(pathEntity, fromTime, files, computer);
  22. }
  23. }
  24. if (recents != null) {
  25. files = await _computeFiles(recents, fromTime, files, computer);
  26. }
  27. files.sort(
  28. (first, second) => first.creationTime.compareTo(second.creationTime),
  29. );
  30. return files;
  31. }
  32. Future<List<LocalAsset>> getAllLocalAssets() async {
  33. final filterOptionGroup = FilterOptionGroup();
  34. filterOptionGroup.setOption(
  35. AssetType.image,
  36. FilterOption(sizeConstraint: ignoreSizeConstraint),
  37. );
  38. filterOptionGroup.setOption(
  39. AssetType.video,
  40. FilterOption(sizeConstraint: ignoreSizeConstraint),
  41. );
  42. filterOptionGroup.createTimeCond = DateTimeCond.def().copyWith(ignore: true);
  43. final assetPaths = await PhotoManager.getAssetPathList(
  44. hasAll: true,
  45. type: RequestType.common,
  46. filterOption: filterOptionGroup,
  47. );
  48. final List<LocalAsset> assets = [];
  49. for (final assetPath in assetPaths) {
  50. for (final asset in await _getAllAssetLists(assetPath)) {
  51. assets.add(LocalAsset(asset.id, assetPath.name));
  52. }
  53. }
  54. return assets;
  55. }
  56. Future<List<File>> getUnsyncedFiles(
  57. List<LocalAsset> assets,
  58. Set<String> existingIDs,
  59. Set<String> invalidIDs,
  60. Computer computer,
  61. ) async {
  62. final Map<String, dynamic> args = <String, dynamic>{};
  63. args['assets'] = assets;
  64. args['existingIDs'] = existingIDs;
  65. args['invalidIDs'] = invalidIDs;
  66. final unsyncedAssets = await computer.compute(_getUnsyncedAssets, param: args);
  67. if (unsyncedAssets.isEmpty) {
  68. return [];
  69. }
  70. return _convertToFiles(unsyncedAssets, computer);
  71. }
  72. List<LocalAsset> _getUnsyncedAssets(Map<String, dynamic> args) {
  73. final List<LocalAsset> assets = args['assets'];
  74. final Set<String> existingIDs = args['existingIDs'];
  75. final Set<String> invalidIDs = args['invalidIDs'];
  76. final List<LocalAsset> unsyncedAssets = [];
  77. for (final asset in assets) {
  78. if (!existingIDs.contains(asset.id) && !invalidIDs.contains(asset.id)) {
  79. unsyncedAssets.add(asset);
  80. }
  81. }
  82. return unsyncedAssets;
  83. }
  84. Future<List<File>> _convertToFiles(
  85. List<LocalAsset> assets,
  86. Computer computer,
  87. ) async {
  88. final List<LocalAsset> recents = [];
  89. final List<LocalAssetEntity> entities = [];
  90. for (final asset in assets) {
  91. if (asset.path == "Recent" || asset.path == "Recents") {
  92. recents.add(asset);
  93. } else {
  94. entities.add(
  95. LocalAssetEntity(await AssetEntity.fromId(asset.id), asset.path),
  96. );
  97. }
  98. }
  99. // Ignore duplicate items in recents
  100. for (final recent in recents) {
  101. bool presentInOthers = false;
  102. for (final entity in entities) {
  103. if (recent.id == entity.entity.id) {
  104. presentInOthers = true;
  105. break;
  106. }
  107. }
  108. if (!presentInOthers) {
  109. entities.add(
  110. LocalAssetEntity(await AssetEntity.fromId(recent.id), recent.path),
  111. );
  112. }
  113. }
  114. return await computer.compute(_getFilesFromAssets, param: entities);
  115. }
  116. Future<List<AssetPathEntity>> _getGalleryList(
  117. final int fromTime,
  118. final int toTime,
  119. ) async {
  120. final filterOptionGroup = FilterOptionGroup();
  121. filterOptionGroup.setOption(
  122. AssetType.image,
  123. FilterOption(needTitle: true, sizeConstraint: ignoreSizeConstraint),
  124. );
  125. filterOptionGroup.setOption(
  126. AssetType.video,
  127. FilterOption(needTitle: true, sizeConstraint: ignoreSizeConstraint),
  128. );
  129. filterOptionGroup.updateTimeCond = DateTimeCond(
  130. min: DateTime.fromMicrosecondsSinceEpoch(fromTime),
  131. max: DateTime.fromMicrosecondsSinceEpoch(toTime),
  132. );
  133. final galleryList = await PhotoManager.getAssetPathList(
  134. hasAll: true,
  135. type: RequestType.common,
  136. filterOption: filterOptionGroup,
  137. );
  138. galleryList.sort((s1, s2) {
  139. return s2.assetCount.compareTo(s1.assetCount);
  140. });
  141. return galleryList;
  142. }
  143. Future<List<File>> _computeFiles(
  144. AssetPathEntity pathEntity,
  145. int fromTime,
  146. List<File> files,
  147. Computer computer,
  148. ) async {
  149. final Map<String, dynamic> args = <String, dynamic>{};
  150. args["pathEntity"] = pathEntity;
  151. args["assetList"] = await _getAllAssetLists(pathEntity);
  152. args["fromTime"] = fromTime;
  153. args["files"] = files;
  154. return await computer.compute(_getFiles, param: args);
  155. }
  156. Future<List<AssetEntity>> _getAllAssetLists(AssetPathEntity pathEntity) async {
  157. List<AssetEntity> result = [];
  158. int currentPage = 0;
  159. List<AssetEntity> currentPageResult = [];
  160. do {
  161. currentPageResult = await pathEntity.getAssetListPaged(
  162. page: currentPage,
  163. size: assetFetchPageSize,
  164. );
  165. result.addAll(currentPageResult);
  166. currentPage = currentPage + 1;
  167. } while (currentPageResult.length >= assetFetchPageSize);
  168. return result;
  169. }
  170. Future<List<File>> _getFiles(Map<String, dynamic> args) async {
  171. final pathEntity = args["pathEntity"];
  172. final assetList = args["assetList"];
  173. final fromTime = args["fromTime"];
  174. final files = args["files"];
  175. for (AssetEntity entity in assetList) {
  176. if (max(
  177. entity.createDateTime.microsecondsSinceEpoch,
  178. entity.modifiedDateTime.microsecondsSinceEpoch,
  179. ) >
  180. fromTime) {
  181. try {
  182. final file = await File.fromAsset(pathEntity.name, entity);
  183. if (!files.contains(file)) {
  184. files.add(file);
  185. }
  186. } catch (e) {
  187. _logger.severe(e);
  188. }
  189. }
  190. }
  191. return files;
  192. }
  193. Future<List<File>> _getFilesFromAssets(List<LocalAssetEntity> assets) async {
  194. final List<File> files = [];
  195. for (final asset in assets) {
  196. files.add(
  197. await File.fromAsset(
  198. asset.path,
  199. asset.entity,
  200. ),
  201. );
  202. }
  203. return files;
  204. }
  205. class LocalAsset {
  206. final String id;
  207. final String path;
  208. LocalAsset(
  209. this.id,
  210. this.path,
  211. );
  212. }
  213. class LocalAssetEntity {
  214. final AssetEntity entity;
  215. final String path;
  216. LocalAssetEntity(this.entity, this.path);
  217. }