Flutter 相机、相册图片裁剪问题处置惩罚与封装使用(image_picker、image_ ...

打印 上一主题 下一主题

主题 992|帖子 992|积分 2976

一、安装



  • 安装插件,留意:裁剪库 必要额外配置
    1. # 相机、相册库
    2. $ flutter pub add image_picker
    3. # 裁剪库
    4. $ flutter pub add image_cropper
    复制代码
  • 每次修改了配置,必要 clean 在 get 重新安装,要不然会使用编译好的缓存运行,所以避免问题,从根本解决问题:
    1. # 先清空
    2. $ flutter clean
    3. # 再安装
    4. $ flutter pub get
    5. # 运行
    6. $ flutter run
    复制代码
  • Android 权限配置,文件路径 android/app/src/main/AndroidManifest.xml
    1. <!-- 请求相机权限 -->
    2. <uses-permission android:name="android.permission.CAMERA"/>
    3. <!-- 请求读取存储权限(读取相册) -->
    4. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    5. <!-- 请求写入存储权限(选择和保存相册中的图片)-->
    6. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    7. <!-- 如果使用了访问相册,且是 Android 10(API 29)及以上版本,还需要配置 -->
    8. <uses-permission android:name="android.permission.ACCESS_MEDIA_LOCATION" />
    复制代码

  • iOS 权限配置,文件路径 ios/Runner/Info.plist
    1. <key>Localization native development region</key>
    2. <string>zh_CN</string>
    3. <key>NSCameraUsageDescription</key>
    4. <string>我们需要访问您的相机</string>
    5. <key>NSPhotoLibraryUsageDescription</key>
    6. <string>我们需要访问您的照片库</string>
    复制代码
    Localization native development region 会使相机、相册打开后表现中文。附:# Flutter iOS 调起相机、相册表现英文,必要改成中文。
二、image_cropper 安卓问题处置惩罚



  • 不必要裁剪功能,就不消看这段了。
  • image_cropper 在 iOS 上没问题,但是在安卓上必要将 compileSdk 修改为 35,文件路径 android/app/build.gradle
    将来最新的版本可能不是 35,可能更高,按着报错改就行。

    不修改会报错:
    1. Your project is configured to compile against Android SDK 34, but the following plugin(s) require to be compiled against a higher Android SDK version:
    2. - image_cropper compiles against Android SDK 35
    3. Fix this issue by compiling against the highest Android SDK version (they are backward compatible).
    4. Add the following to /Users/dengzemiao/Desktop/Project/flutter/flutter-edu-app/android/app/build.gradle:
    5.     android {
    6.         compileSdk = 35
    7.         ...
    8.     }
    9. FAILURE: Build failed with an exception.
    10. * What went wrong:
    11. Execution failed for task ':app:processDebugResources'.
    12. > A failure occurred while executing com.android.build.gradle.internal.res.LinkApplicationAndroidResourcesTask$TaskAction
    13.    > Android resource linking failed
    14.      com.example.testbaseproject.app-mergeDebugResources-42:/values-v35/values-v35.xml:4: error: style attribute 'android:attr/windowOptOutEdgeToEdgeEnforcement' not found.
    15.      error: failed linking references.
    16. * Try:
    17. > Run with --stacktrace option to get the stack trace.
    18. > Run with --info or --debug option to get more log output.
    19. > Run with --scan to get full insights.
    20. > Get more help at https://help.gradle.org.
    21. BUILD FAILED in 9s
    复制代码
    修改后,还不行,必要确定 Android Studio 安装好了 35 的 sdk,留意安装包支持的 CPU 类型,安装好了就行了。

  • nonono,再次运行,可以正常使用了,但是在拍好照片进行裁剪的还会报错,可能还会报错,一大串的,可以搜搜里面有没有 com.yalantis.ucrop.UCropActivity,就是缺少了 activity 的申明,所以还必要配置下,文件路径 android/app/src/main/AndroidManifest.xml
    在 application 里面再加一个 activity 申明:
    1. <activity android:name="com.yalantis.ucrop.UCropActivity"
    2.           android:theme="@style/Theme.AppCompat.Light.NoActionBar"/>
    复制代码

  • 到这里就真的 ok 了,可以放心使用裁剪了,如果不必要裁剪反面这些配置可以不消配置,只要配置权限即可。
三、裁剪封装



  • 使用
    1. // 导入
    2. import 'package:base_project/utils/camera_utils.dart';
    3. // 使用
    4. CustomBottomSheet.show(
    5.   context: context,
    6.   list: ['拍照', '打开相册'],
    7.   onConfirm: (index) async {
    8.     if (index == 0) {
    9.       // 打开相机
    10.       await camera.showCamera(
    11.         cropping: true,
    12.         onChanged: (value) => {
    13.           print('图片路径: $value')
    14.         }
    15.       );
    16.     } else {
    17.       // 打开相册
    18.       await camera.showAlbum(
    19.         cropping: true,
    20.         onChanged: (value) => {
    21.           print('图片路径: $value')
    22.         }
    23.       );
    24.     }
    25.   },
    26. );
    复制代码
  • camera_utils.dart
    1. import 'package:flutter/material.dart';
    2. import 'package:image_picker/image_picker.dart';
    3. import 'package:image_cropper/image_cropper.dart';
    4. class CameraUtils {
    5.   // 静态变量存储单例
    6.   static final CameraUtils _instance = CameraUtils._internal();
    7.   // 静态方法获取单例实例
    8.   factory CameraUtils() => _instance;
    9.   // 私有构造函数,确保只能通过工厂方法获取实例
    10.   CameraUtils._internal();
    11.   // 静态的 _picker 变量,确保只会初始化一次
    12.   final ImagePicker _picker = ImagePicker();
    13.   // 打开相机
    14.   Future<void> showCamera({
    15.     bool? cropping,
    16.     ValueChanged<String>? onChanged
    17.   }) async {
    18.     try {
    19.       final XFile? file = await _picker.pickImage(source: ImageSource.camera);
    20.       if (file != null) {
    21.         if (cropping == true) {
    22.           await _cropImage(
    23.             path: file.path,
    24.             onChanged: onChanged
    25.           );
    26.         } else {
    27.           if (onChanged != null) { onChanged(file.path); }
    28.         }
    29.       } else {
    30.         if (onChanged != null) { onChanged(''); }
    31.       }
    32.     } catch (e) {
    33.       if (onChanged != null) { onChanged(''); }
    34.       // print('图片选择出错: $e');
    35.     }
    36.   }
    37.   // 打开相册
    38.   Future<void> showAlbum({
    39.     bool? cropping,
    40.     ValueChanged<String>? onChanged
    41.   }) async {
    42.     try {
    43.       final XFile? file = await _picker.pickImage(source: ImageSource.gallery);
    44.       if (file != null) {
    45.         if (cropping == true) {
    46.           await _cropImage(
    47.             path: file.path,
    48.             onChanged: onChanged
    49.           );
    50.         } else {
    51.           if (onChanged != null) { onChanged(file.path); }
    52.         }
    53.       } else {
    54.         if (onChanged != null) { onChanged(''); }
    55.       }
    56.     } catch (e) {
    57.       // print('图片选择出错: $e');
    58.       if (onChanged != null) { onChanged(''); }
    59.     }
    60.   }
    61.   // 图片裁剪方法
    62.   Future<void> _cropImage({
    63.     required String path,
    64.     ValueChanged<String>? onChanged
    65.   }) async {
    66.     try {
    67.       // 使用 image_cropper 进行裁剪
    68.       final file = await ImageCropper().cropImage(
    69.         sourcePath: path,
    70.         uiSettings: [
    71.           AndroidUiSettings(
    72.             toolbarTitle: '裁剪图片',
    73.             toolbarColor: Colors.deepOrange,
    74.             toolbarWidgetColor: Colors.white,
    75.             initAspectRatio: CropAspectRatioPreset.square,
    76.             lockAspectRatio: false,
    77.           ),
    78.           IOSUiSettings(
    79.             minimumAspectRatio: 1.0,
    80.             aspectRatioLockEnabled: false
    81.           ),
    82.         ]
    83.       );
    84.       // 如果裁剪成功
    85.       if (file != null && onChanged != null) {
    86.         // 图片裁剪成功,返回裁剪后的图片
    87.         onChanged(file.path);
    88.       }
    89.     } catch (e) {
    90.       // print('裁剪出错: $e');
    91.       if (onChanged != null) { onChanged(''); }
    92.     }
    93.   }
    94. }
    复制代码

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

tsx81429

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表