zoukankan      html  css  js  c++  java
  • Android调用系统相机并解决两大问题

    前言

    源码Demo:请点击此处
    Android调用系统相机会遇到的两大问题:

    • 1.指定存储图片路径,Android7.0及之后的机型调用系统相机会抛出android.os.FileUriExposedException异常
    • 2.指定存储图片路径,调用系统相机返回 intent 为:null

    问题《一》

    • Android 7.0后系统禁止应用向外部公开file://URI ,因此需要FileProvider来向外界传递URI。所以针对安卓7.0及其之后的系统需要做一个适配。
    • 实际开发中,推荐该方式。知道文件路径,可以根据需求执行相应压缩处理。

    开始代码示例(Android Studio, SdkVersion 29)

    • 1️⃣AndroidManifest.xml 清单文件中添加所需权限
    <!--相机权限-->
    <uses-permission android:name="android.permission.CAMERA" />
    <!--SD卡权限-->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    
    • 2️⃣ activity_play_photo(PlayPhotoActivity的xml界面)
    <?xml version="1.0" encoding="utf-8"?>
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
        <ImageView
            android:id="@+id/ivMyPhoto"
            android:layout_width="300dp"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true" />
        <Button
            android:layout_width="match_parent"
            android:layout_height="56dp"
            android:layout_alignParentBottom="true"
            android:gravity="center"
            android:onClick="playPhoto"
            android:padding="16dp"
            android:text="拍照(原图-路径获取)"
            android:textColor="#FF212121"
            android:textSize="16sp"
            android:textStyle="bold" />
    </RelativeLayout>
    
    • 3️⃣ PlayPhotoActivity(activity中调用相机拍照并返回展示图片)
    public class PlayPhotoActivity extends BaseActivity {
        //定义一个文件夹路径
        private String localPath = MyApplication.localPath + File.separator + "123";
        private ImageView ivMyPhoto;
        private File photoFile;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_play_photo);
            ivMyPhoto = findViewById(R.id.ivMyPhoto);
    
            photoFile = new File(localPath, "temp.png");
            if ((photoFile.getParentFile() != null) && (!photoFile.getParentFile().exists())) {
                photoFile.getParentFile().mkdirs();
            }
            Log.e("相机", "路径-localPath:" + localPath);
        }
    
        //相机点击事件:打开照相机(该方式获取到的图片是原图)
        public void playPhoto(View view) {
            //创建打开本地相机的意图对象
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            //设置图片的保存位置(兼容Android7.0)
            Uri fileUri = getUriForFile(this, photoFile);
            //指定图片保存位置
            intent.putExtra(MediaStore.EXTRA_OUTPUT, fileUri);
            //开启意图
            startActivityForResult(intent, 100);
        }
    
        @Override
        protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent intent) {
            super.onActivityResult(requestCode, resultCode, intent);
            //拍照完成后返回调用
            if (resultCode == RESULT_OK) {
                if (requestCode == 100) {
                    //该方式获取到的图片是原图
                    FileInputStream fis = null;
                    try {
                        fis = new FileInputStream(photoFile);
                        Bitmap bitmap = BitmapFactory.decodeStream(fis);
                        ivMyPhoto.setImageBitmap(bitmap);
                    } catch (Exception e) {
                        e.printStackTrace();
                    } finally {
                        try {
                            if (fis != null)
                                fis.close();
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
                } 
            }
        }
    
        private Uri getUriForFile(Context context, File file) {
            Uri fileUri;
            if (Build.VERSION.SDK_INT >= 24) {
                //参数:authority 需要和清单文件中配置的保持完全一致:${applicationId}.xxx
                fileUri = FileProvider.getUriForFile(context, context.getPackageName() + ".xxx", file);
            } else {
                fileUri = Uri.fromFile(file);
            }
            return fileUri;
        }
    }
    
    • 4️⃣ 清单文件配置
    1. SdkVersion 29之前使用:android.support.v4(下述)
      android:name="android.support.v4.content.FileProvider"
    2. SdkVersion 29开始使用:androidx(下述)
      android:name="androidx.core.content.FileProvider"
    3. authorities可以随意定义(默认规程:采用本应用包名+定义串)
      android:authorities="包名.xxx"
            <provider
                android:name="androidx.core.content.FileProvider"
                android:authorities="${applicationId}.xxx"
                android:exported="false"
                android:grantUriPermissions="true">
                <meta-data
                    android:name="android.support.FILE_PROVIDER_PATHS"
                    android:resource="@xml/file_paths" />
            </provider>
    
    • 5️⃣在 res 目录下创建 xml 目录,并在res/xml目录下创建文件:file_paths(代码如示)
    <?xml version="1.0" encoding="utf-8"?>
    <paths xmlns:android="http://schemas.android.com/apk/res/android">
        <root-path
            name="root"
            path="" />
        <!--files-path  相当于 getFilesDir()-->
        <files-path
            name="files"
            path="path" />
        <!--cache-path  相当于 getCacheDir()-->
        <cache-path
            name="cache"
            path="path" />
        <!--external-path  相当于 Environment.getExternalStorageDirectory()-->
        <external-path
            name="external"
            path="path" />
        <!--external-files-path  相当于 getExternalFilesDir("") -->
        <external-files-path
            name="external-files"
            path="path" />
        <!--external-cache-path  相当于 getExternalCacheDir() -->
        <external-cache-path
            name="external-cache"
            path="path" />
    </paths>
    

    问题《二》

    • 在调用系统相机的时候,如果传入了:指定的路径(文件保存地址),那么在activity的回调方法:onActivityResult 中,intent对象会是null。
    • 如问题一的示例代码:onActivityResult的intent对象亦是null
    • 如何解决呢?可以参考下述代码(但实际开发中,不推荐该方式,该方式获取到的图片数据是Android系统压缩后的图片。)

    开始代码示例(Android Studio, SdkVersion 29)

    • 1️⃣ 参考《问题一》第一步
    • 2️⃣ 参考《问题一》第二步
    • 3️⃣ PlayPhotoActivity(activity中调用相机拍照并返回展示图片)
    public class PlayPhotoActivity extends BaseActivity {
        private ImageView ivMyPhoto;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_play_photo);
            ivMyPhoto = findViewById(R.id.ivMyPhoto);
        }
        //相机点击事件:打开照相机(该方式获取到的图片是缩略图)
        public void playPhoto(View view) {
            //创建打开本地相机的意图对象
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            //如果intent指定了存储图片的路径,那么onActivityResult回调中Intent对象就会为null
            //intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
            //开启意图
            startActivityForResult(intent, 200);
        }
        @Override
        protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent intent) {
            super.onActivityResult(requestCode, resultCode, intent);
            //拍照完成后返回调用
            if (resultCode == RESULT_OK) {
                if (requestCode == 200) {
                    //该方式获取到的图片是缩略图
                    Bundle bundle = intent.getExtras();
                    Bitmap bitmap = (Bitmap) bundle.get("data");
                    ivMyPhoto.setImageBitmap(bitmap);
                }
            }
        }
    }
    
  • 相关阅读:
    SpringMVC的各种注解
    ThreadLocal的一些总结
    HashTable和HashMap
    SaxReader读取xml
    Java多线程(七):ReentrantLock
    Java多线程(六):wait(),notify()和notifyAll()
    Java多线程(五):死锁
    Java多线程(四):volatile
    Java多线程(三):synchronized
    每日一题,每日一字
  • 原文地址:https://www.cnblogs.com/io1024/p/11590382.html
Copyright © 2011-2022 走看看