美文网首页Android适配android技术
android 11 拍照裁剪适配

android 11 拍照裁剪适配

作者: 慢慢前进 | 来源:发表于2021-06-22 10:15 被阅读0次

    项目用到了app拍照相册裁剪,因为测试机是android10的,一顿操作猛如虎,功能完美呀。然后测试小姐姐用android11测试,直接GG了。
    看了之后才发现android11强制分区处理。
    android 适配拍照主要经历了 6.0 7.0 10和11
    6.0主要是处理运行时权限 7.0为FileProvider 文件共享 10和11则是分区处理。
    主要说下7.0和10,11的适配方法。

    参考:鸿洋_: Android 7.0 行为变更 通过FileProvider在应用间共享文件吧
    官方文档:Android 7.0 行为变更
    android7.0 主要是适配其中文件访问权限

    在官方7.0的以上的系统中,尝试传递 file://URI可能会触发FileUriExposedException。

    清单文件配置provider ,因为它是ContentProvider的子类。

          <provider
                android:name="android.support.v4.content.FileProvider"
                android:authorities="com.edu.lzdx.liangjianpro.fileProvider"
                android:exported="false"
                android:grantUriPermissions="true">
                <meta-data
                    android:name="android.support.FILE_PROVIDER_PATHS"
                    android:resource="@xml/provider_paths" />
            </provider>
    <?xml version="1.0" encoding="utf-8"?>
    

    android:authorities 表示授权者,这里的格式一般是[appId].fileprovider
    android:exported 只能为false
    android:grantUriPermissions="true" 表示授权Uri权限 ,且必须为true
    meta-data里设置指定的文件目录,为引用provider_paths文件

    provider_paths文件

    <paths xmlns:android="http://schemas.android.com/apk/res/android">
        <root-path name="root" path="" />
        <files-path name="files" path="" />
        <cache-path name="cache" path="" />
        <external-path name="external" path="" />
        <external-files-path name="name" path="path" />
         <external-cache-path name="name" path="path" />
    </paths>
    

    <root-path/> 代表设备的根目录new File("/");
    <files-path/> 代表context.getFilesDir()
    <cache-path/> 代表context.getCacheDir()
    <external-path/> 代表Environment.getExternalStorageDirectory()
    <external-files-path>代表context.getExternalFilesDirs()
    <external-cache-path>代表getExternalCacheDirs()
    常用到的 为 external-path 和external-files-path name和path按照自己需求编写。

    private void openCamera() {
            Intent captureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            // 判断是否有相机
            if (captureIntent.resolveActivity(getPackageManager()) != null) {
                File photoFile = null;
                try {
                   photoFile =  new File(getExternalCacheDir(), "head_image.jpg");
                } catch (IOException e) {
                    e.printStackTrace();
                 }
                if (photoFile != null) {
                   //根据判断版本 4.4也能安全运行 或者 授权
                    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                        //适配Android 7.0文件权限,通过FileProvider创建一个content类型的Uri  主要是这段代码
                        photoUri = FileProvider.getUriForFile(this, getPackageName() + ".fileProvider", photoFile);
                    } else {
                        photoUri = Uri.fromFile(photoFile);
                    }
                }
                if (photoUri != null) {
                    Log.e("TAG", "openCamera: 拍照初始路径===" + photoUri);
                    captureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
                    captureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                    startActivityForResult(captureIntent, CODE_CAMERA_REQUEST);
                }
            }
        }
    

    7.0大致就是这样。
    android10 如果 targetSdkVersion<=29 在清单文件中添加

      android:requestLegacyExternalStorage="true"
    

    可以让android10忽略分区,按照之前的方式。但是运行在android11的机型就会强制执行分区处理,所以这个只是过度方式,还是得适配。
    我遇到得问题就是一个拍照裁剪------加载图片失败,追其原因,为系统裁剪不能访问到app得私有目录,所以我们
    主要进行一个裁剪公域得处理。

    public static Intent getCropIntent(Uri uri,Context context,String path){
            Log.d("裁剪的Url", "cropRawPhoto: " + uri.toString());
            Intent intent = new Intent("com.android.camera.action.CROP");
            intent.setDataAndType(uri, "image/*");
            intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
            // 设置裁剪
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", output_X);
            intent.putExtra("outputY", output_Y);
            intent.putExtra("return-data", false);
            context.getExternalFilesDir(Environment.DIRECTORY_PICTURES);
            //android11 分区存储
            if (Build.VERSION.SDK_INT >= 29) {
                Log.e("TAG", "裁剪公域::"+path);
                File mOnputFile11 = new File(path, SpUtils.getInstance(context).getString("token","") + ".png");
                intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + mOnputFile11.getAbsolutePath()));
            } else {
                intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(
                        new File(context.getExternalCacheDir(), "face-cropped.jpg")));
            }
            return intent;
        }
    

    //等会儿。

    相关文章

      网友评论

        本文标题:android 11 拍照裁剪适配

        本文链接:https://www.haomeiwen.com/subject/cvspyltx.html