image_util.dart 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. import 'dart:io';
  2. import 'dart:typed_data';
  3. import 'package:clean/model/asset_info.dart';
  4. import 'package:clean/module/image_picker/image_picker_util.dart';
  5. import 'package:clean/utils/toast_util.dart';
  6. import 'package:intl/intl.dart';
  7. import 'package:wechat_assets_picker/wechat_assets_picker.dart';
  8. import 'file_utils.dart';
  9. import 'package:exif/exif.dart';
  10. class ImageUtil {
  11. // 生成月份 key (用于内部存储)
  12. static String getMonthKey(DateTime date) {
  13. return '${date.year}-${date.month.toString().padLeft(2, '0')}';
  14. }
  15. // 格式化月份显示 (例如: Jan 2025)
  16. static String formatMonthKey(String monthKey) {
  17. final parts = monthKey.split('-');
  18. if (parts.length == 2) {
  19. final date = DateTime(int.parse(parts[0]), int.parse(parts[1]));
  20. return DateFormat('MMM yyyy').format(date);
  21. }
  22. return monthKey;
  23. }
  24. // 获取指定索引的月份显示文本
  25. static String getMonthText(Map<String, List<AssetInfo>> assets, int index) {
  26. final monthKeys = assets.keys.toList()
  27. ..sort((a, b) => b.compareTo(a)); // 最新的月份在前
  28. if (index < monthKeys.length) {
  29. return formatMonthKey(monthKeys[index]);
  30. }
  31. return '';
  32. }
  33. // 获取指定月份的图片
  34. static List<AssetInfo> getMonthAssets(
  35. Map<String, List<AssetInfo>> assets, int index) {
  36. final monthKeys = assets.keys.toList()
  37. ..sort((a, b) => b.compareTo(a)); // 最新的月份在前
  38. if (index < monthKeys.length) {
  39. return assets[monthKeys[index]] ?? [];
  40. }
  41. return [];
  42. }
  43. // 获取缩略图数据
  44. static Future<Uint8List?> getImageThumbnail(
  45. FileType type, AssetInfo asset) async {
  46. try {
  47. // 先尝试从本地读取缩略图
  48. final localThumb = await FileUtils.getThumbData(
  49. type, Platform.isIOS ? asset.id.substring(0, 36) : asset.id);
  50. if (localThumb != null) {
  51. return localThumb;
  52. }
  53. // 如果本地没有,则从 AssetEntity 获取
  54. return await asset.thumbnailData;
  55. } catch (e) {
  56. print('获取缩略图失败: $e');
  57. return null;
  58. }
  59. }
  60. // 获取原始图片文件
  61. static Future<File?> getImageFile(FileType type, AssetInfo asset) async {
  62. try {
  63. // 先尝试从本地读取
  64. final assetPath = await FileUtils.getAssetPath(type);
  65. final localFile = File(
  66. '$assetPath/${Platform.isIOS ? asset.id.substring(0, 36) : asset.id}.PNG');
  67. if (await localFile.exists()) {
  68. return localFile;
  69. }
  70. // 如果本地没有,则从 AssetEntity 获取
  71. return asset.file;
  72. } catch (e) {
  73. print('获取图片文件失败: $e');
  74. return null;
  75. }
  76. }
  77. static Future<Map<String, dynamic>> getPhotoDetails(AssetInfo asset) async {
  78. try {
  79. final Map<String, dynamic> details = {};
  80. var title = "";
  81. if (asset.originalPath != null) {
  82. // 查找 "o_" 的位置
  83. final int index = asset.originalPath!.lastIndexOf('_o_');
  84. if (index != -1) {
  85. // 截取 "o_" 后面的字符串
  86. title = asset.originalPath!.substring(index + 3);
  87. } else {
  88. // 如果没有找到 "o_",返回原始文件名
  89. title = asset.originalPath!.split('/').last;
  90. }
  91. }
  92. // 基本信息
  93. details['fileName'] = title;
  94. // 创建格式化器
  95. final formatter = DateFormat('yyyy-MM-dd EEE HH:mm');
  96. details['createDate'] = formatter.format(asset.createDateTime);
  97. details['width'] = asset.width;
  98. details['height'] = asset.height;
  99. details['size'] = asset.size;
  100. // 获取文件
  101. final path = asset.originalPath;
  102. // final file = await asset.file;
  103. if (path != null && path.isNotEmpty) {
  104. try {
  105. final file = File(path);
  106. // 读取 EXIF 数据
  107. final bytes = await file.readAsBytes();
  108. final exifData = await readExifFromBytes(bytes);
  109. if (exifData.isNotEmpty) {
  110. // 相机信息
  111. details['make'] = exifData['Image Make']?.printable;
  112. details['model'] = exifData['Image Model']?.printable;
  113. // 拍摄参数
  114. // 光圈值
  115. details['aperture'] = exifData['EXIF FNumber']?.printable;
  116. // 快门速度/曝光时间
  117. details['exposureTime'] = exifData['EXIF ExposureTime']?.printable;
  118. // ISO感光度
  119. details['iso'] = exifData['EXIF ISOSpeedRatings']?.printable;
  120. // 焦距
  121. details['focalLength'] =
  122. exifData['EXIF FocalLengthIn35mmFilm']?.printable;
  123. // GPS 信息
  124. if (exifData.containsKey('GPS GPSLatitude') &&
  125. exifData.containsKey('GPS GPSLongitude')) {
  126. details['latitude'] = exifData['GPS GPSLatitude']?.printable;
  127. details['longitude'] = exifData['GPS GPSLongitude']?.printable;
  128. }
  129. }
  130. } catch (e) {
  131. print('读取 EXIF 数据失败: $e');
  132. ToastUtil.show('Failed to get photo details');
  133. }
  134. }
  135. print(details);
  136. return details;
  137. } catch (e) {
  138. print('Failed to get photo details: $e');
  139. return {};
  140. }
  141. }
  142. /// 格式化图片尺寸
  143. static String formatResolution(int width, int height) {
  144. final megapixels = (width * height) / 1000000.0;
  145. if (megapixels >= 1) {
  146. return '${megapixels.toStringAsFixed(1)} MP ($width × $height)';
  147. } else {
  148. return '$width × $height';
  149. }
  150. }
  151. /// 格式化文件大小
  152. static String formatFileSize(int bytes) {
  153. if (bytes < 1024) return '$bytes B';
  154. if (bytes < 1024 * 1024) return '${(bytes / 1024).toStringAsFixed(1)} KB';
  155. if (bytes < 1024 * 1024 * 1024) {
  156. return '${(bytes / (1024 * 1024)).toStringAsFixed(1)} MB';
  157. }
  158. return '${(bytes / (1024 * 1024 * 1024)).toStringAsFixed(1)} GB';
  159. }
  160. /// 格式化光圈值
  161. static String formatAperture(String? value) {
  162. if (value == null) return 'Unknown';
  163. return 'f/$value';
  164. }
  165. /// 格式化曝光时间
  166. static String formatExposureTime(String? value) {
  167. if (value == null) return 'Unknown';
  168. // 将分数转换为更易读的格式
  169. if (value.contains('/')) {
  170. final parts = value.split('/');
  171. if (parts.length == 2) {
  172. final numerator = int.parse(parts[0]);
  173. final denominator = int.parse(parts[1]);
  174. if (numerator == 1) {
  175. return '1/${denominator}s';
  176. }
  177. }
  178. }
  179. return '${value}s';
  180. }
  181. }