调用系统相机、相册、剪裁图片,适配Android 12

第一步:FileProvider相关准备工作

在AndroidManifest.xml中增加provider节点:

<provider
            android:name="androidx.core.content.FileProvider"
            android:authorities="com.choosecrop.fileprovider"
            android:grantUriPermissions="true"
            android:exported="false">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/filepaths" />
  </provider>
       

其中:

android:authorities 表示授权列表,填写你的应用包名,当有多个授权时,用分号隔开。

android:exported 表示该内容提供器(FileProvider)是否能被第三方程序组件使用,必须为false,否则会报异常: java.lang.SecurityException: Provider must not be exported。

android:grantUriPermissions=“true” 表示授予 URI 临时访问权限。

接着,需要在res目录下建立一个xml目录,

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <external-files-path name="DCIM" path="." />
</paths>

说明:

files-path name=“name” path=“path” 对应 Context.getFilesDir()

cache-path name=“name” path=“path” 对应 getCacheDir()

external-path name=“name” path=“path” 对应Environment.getExternalStorageDirectory()

external-files-path name=“name” path=“path” 对应 Context.getExternalFilesDir()

external-cache-path name=“name” path=“path” 对应 Context.getExternalCacheDir()
external-media-path name=“name” path=“path” 对应 Context.getExternalMediaDirs()

注意:ExternalStorageDirectory在29后废弃,官方推荐使用Context.getExternalFilesDir()

可被使用的特定目录:
public static String DIRECTORY_ALARMS = “Alarms”;
public static String DIRECTORY_AUDIOBOOKS = “Audiobooks”;
public static String DIRECTORY_DCIM = “DCIM”;
public static String DIRECTORY_DOCUMENTS = “Documents”;
public static String DIRECTORY_DOWNLOADS = “Download”;
public static String DIRECTORY_MOVIES = “Movies”;
public static String DIRECTORY_MUSIC = “Music”;
public static String DIRECTORY_NOTIFICATIONS = “Notifications”;
public static String DIRECTORY_PICTURES = “Pictures”;
public static String DIRECTORY_PODCASTS = “Podcasts”;

第二步:使用FileProvider

在这之前,我们需要在AndroidManifest.xml中增加必要的读写权限:

 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

1. 通过相机获取图片

在通过Intent跳转系统相机前,我们需要对版本进行判断,如果在Android7.0以上,使用FileProvider获取Uri,代码如下:

    /**
     * 从相机获取图片
     */
    private void getImgFromCamera() {
        //用于保存调用相机拍照后所生成的文件
        tempFile = new File(this.getExternalFilesDir(DIRECTORY_DCIM), System.currentTimeMillis() + ".png");
        //跳转到调用系统相机
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        //判断版本
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {   //如果在Android7.0以上,使用FileProvider获取Uri
            intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.choosecrop.fileprovider", tempFile);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);

        } else {    //否则使用Uri.fromFile(file)方法获取Uri
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile));
        }
        startActivityForResult(intent, CAMERA_REQUEST_CODE);
    }

2.通过相册获取图片

    /**
     * 从相册获取图片
     */
    private void getImgFromDicm() {
        Intent photo = new Intent(Intent.ACTION_PICK);
        photo.setType("image/*");
        photo(photoPickerIntent, DICM_REQUEST_CODE);
    }

3.剪裁图片

    /**
     * 裁剪图片
     */
    private void cropPhoto(Uri uri) {
        Intent intent = new Intent("com.android.camera.action.CROP");
        // Intent intent = new Intent("android.intent.action.EDIT");
        // intent.setAction("android.intent.action.EDIT");
        //  intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        intent.setDataAndType(uri, "image/*");

        //  intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        intent.putExtra("crop", "true");
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);

        intent.putExtra("outputX", 100);
        intent.putExtra("outputY", 100);
        intent.putExtra("return-data", false);
        File cropTemp = this.getExternalFilesDir(DIRECTORY_DCIM);
        File cropTempName = new File(cropTemp, System.currentTimeMillis() + "_crop_temp.png");
        Log.e("getPath", cropTempName.getAbsolutePath());

        Uri uriForFile = FileProvider.getUriForFile(this, "com.choosecrop.fileprovider", cropTempName);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForFile);

        grantPermissionFix(intent, uriForFile);

        startActivityForResult(intent, CROP_REQUEST_CODE);

    }

此处应设置return-data为false。
如果设置为true,是直接返回bitmap格式的数据,不仅耗费内存,最重要的是在Android 12,会报错:
src:content://com.miui.gallery.open/raw/%2Fstorage%2Femulated%2F0%2FPictures%2FWeiXin%2Fmmexport1662718600424.jpg, des: null。

所以设置为false。然后,设置裁剪完之后保存的路径,在这里使用URI方式传递数据,即:intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForFile);
同时需要给系统裁剪APP赋予临时权限,

private void grantPermissionFix(Intent intent, Uri uri) {
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        List<ResolveInfo> resolveInfos = getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
        for (ResolveInfo resolveInfo : resolveInfos) {
            String packageName = resolveInfo.activityInfo.packageName;
            grantUriPermission(packageName, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            intent.setAction(null);
            intent.setComponent(new ComponentName(resolveInfo.activityInfo.packageName, resolveInfo.activityInfo.name));
            break;
        }
    }

第三步:接收图片信息

在onActivityResult方法中获得返回的图片信息,
先调用剪裁去剪裁图片,然后对剪裁返回的图片进行设置、保存、上传等操作。

 
  @Override
   protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
       switch (requestCode) {

           case CAMERA_REQUEST_CODE:   //调用相机后返回
               if (resultCode == RESULT_OK) {
                   //用相机返回的照片去调用剪裁也需要对Uri进行处理
                   if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                       Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.choosecrop.fileprovider", tempFile);
                       cropPhoto(contentUri);
                   } else {
                       cropPhoto(Uri.fromFile(tempFile));
                   }
               }
               break;
           case DICM_REQUEST_CODE:    //调用相册后返回
               if (resultCode == RESULT_OK) {
                   Uri uri = intent.getData();
                   if (null != uri)
                       cropPhoto(uri);
                   else {
                       Log.e("e", "null");
                   }
               }
               break;
           case CROP_REQUEST_CODE:     //调用剪裁后返回
               Log.e("d", "--------------222222222-------");
               if (null != intent) {
                   Log.e("d", "---------------------not null");
                   Uri data = intent.getData();
                   mHeader_iv.setImageURI(data);
//                    Bundle bundle = intent.getExtras();
//                    if (bundle != null) {
//                        //在这里获得了剪裁后的Bitmap对象,可以用于上传
//                        Bitmap image = bundle.getParcelable("data");
//                        //设置到ImageView上
//                        mHeader_iv.setImageBitmap(image);
//                        //也可以进行一些保存、压缩等操作后上传
                    String path = saveImage("crop", image);
//                    }
               } else {
                   Log.e("d", "---------null");
               }
               break;
       }
   }

因为使用uri方式传递的文件,所以这样获取数据,
Uri data = intent.getData();

第四步:保存

    public String save(String name, Bitmap bmp) {
        File dir = new File(this.getExternalFilesDir(DIRECTORY_DCIM).getPath());
        if (!dir.exists()) {
            dir.mkdir();
        }
        String fileName = name + ".png";
        File file = new File(dir, fileName);
        try {
            FileOutputStream fos = new FileOutputStream(file);
            bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
            fos.flush();
            fos.close();
            return file.getAbsolutePath();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }

至此,Android 12 的兼容就OK了。

完整代码,可以到Github下载,别忘给star哦:

https://github.com/AlbertShen0211/ChooseCrop

  • 3
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Android中,要调用系统相机相册裁剪图片,需要使用一些系统提供的Intent和API。下面是一个示例的源码,实现了这个功能。 首先,在AndroidManifest.xml文件中添加如下权限: ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> ``` 然后,在你的Activity中添加如下代码: ```java private static final int REQUEST_IMAGE_CAPTURE = 1; // 调用相机的请求码 private static final int REQUEST_IMAGE_PICK = 2; // 调用相册的请求码 private static final int REQUEST_IMAGE_CROP = 3; // 调用裁剪的请求码 private Uri imageUri; // 保存相机拍照或相册选择的照片Uri // 调用系统相机拍照 private void takePicture() { Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); if (intent.resolveActivity(getPackageManager()) != null) { // 创建临时文件保存拍照的图片 File imageFile = createImageFile(); if (imageFile != null) { imageUri = FileProvider.getUriForFile(this, "com.example.fileprovider", imageFile); intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, REQUEST_IMAGE_CAPTURE); } } } // 调用系统相册选择照片 private void pickPicture() { Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI); intent.setType("image/*"); startActivityForResult(Intent.createChooser(intent, "选择图片"), REQUEST_IMAGE_PICK); } // 创建临时文件保存拍照的图片 private File createImageFile() { String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(new Date()); String imageFileName = "IMG_" + timeStamp; File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES); try { File imageFile = File.createTempFile(imageFileName, ".jpg", storageDir); return imageFile; } catch (IOException e) { e.printStackTrace(); } return null; } // 调用系统裁剪图片 private void cropPicture(Uri sourceUri) { Intent intent = new Intent("com.android.camera.action.CROP"); intent.setDataAndType(sourceUri, "image/*"); intent.putExtra("crop", "true"); intent.putExtra("aspectX", 1); intent.putExtra("aspectY", 1); intent.putExtra("outputX", 500); intent.putExtra("outputY", 500); intent.putExtra("return-data", true); startActivityForResult(intent, REQUEST_IMAGE_CROP); } // 处理相机相册返回的结果 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (resultCode == RESULT_OK) { switch (requestCode) { case REQUEST_IMAGE_CAPTURE: cropPicture(imageUri); // 拍照后裁剪图片 break; case REQUEST_IMAGE_PICK: cropPicture(data.getData()); // 选择照片后裁剪图片 break; case REQUEST_IMAGE_CROP: Bundle extras = data.getExtras(); if (extras != null) { Bitmap bitmap = extras.getParcelable("data"); // 在这里处理裁剪后的图片 } break; } } } ``` 以上就是调用系统相机相册裁剪图片的简单示例代码。你可以根据自己的需求进行修改和扩展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值