手把手教你:android调用系统相机、相册功能,适配6.0权限获取以及7.0以后获取URI(兼容多版本)

来源:互联网 发布:mysql数据库最大容量 编辑:程序博客网 时间:2024/06/06 04:45

  Android中调用系统相机来拍摄照片的代码,如下:

1、首先设置Uri获取判断以及相机请求Code

  public final int TYPE_TAKE_PHOTO = 1;//Uri获取类型判断  public final int CODE_TAKE_PHOTO = 1;//相机RequestCode

 2、调起系统相机

 Intent takeIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); Uri photoUri = getMediaFileUri(TYPE_TAKE_PHOTO); takeIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri); startActivityForResult(takeIntent, CODE_TAKE_PHOTO);
3、封装获取Uri代码

    public Uri getMediaFileUri(int type){        File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), "相册名字");        if (!mediaStorageDir.exists()) {            if (!mediaStorageDir.mkdirs()) {                return null;            }        }        //创建Media File        String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());        File mediaFile;        if (type == TYPE_TAKE_PHOTO) {            mediaFile = new File(mediaStorageDir.getPath() + File.separator + "IMG_" + timeStamp + ".jpg");        } else {            return null;        }        return Uri.fromFile(mediaFile);    }

4、相机拍照完毕后获取返回数据,并在页面显示照片

@Override    protected void onActivityResult(int requestCode, int resultCode, Intent data) {        super.onActivityResult(requestCode, resultCode, data);        switch (requestCode) {            case CODE_TAKE_PHOTO:                if (resultCode == RESULT_OK) {                    if (data != null) {                        if (data.hasExtra("data")) {                            Log.i("URI", "data is not null");                            Bitmap bitmap = data.getParcelableExtra("data");                            imageView.setImageBitmap(bitmap);//imageView即为当前页面需要展示照片的控件,可替换                        }                    } else {                        Log.i("URI", "Data is null");                        Bitmap bitmap = BitmapFactory.decodeFile(fileUri.getPath());                        imageView.setImageBitmap(bitmap);//imageView即为当前页面需要展示照片的控件,可替换                    }                }                break;        }    }
  特殊:

      一般情况,以上代码在Android7.0以下,也就是api<24时,运行是没有任何问题的。可是当targetSdkVersion变成24及其以上并且在android7.0(及以上版本)系统运行时,会抛出异常:FileUriExposedException。

  权限:

      由于已经写过一篇文章对android6.0后权限设置和6.0以前权限设置进行过讲述,在此就不再赘述如何配置权限了,对于6.0后权限设置不了解的,可以看看:通俗易懂,手把手教会你android 6.0后(兼容6.0之前版本)申请危险权限的方法。

  Android7.0及以上版本调用系统相机

1、为什么会报FileUriExposedException异常

  android N以后收回了访问文件的权限,按照android N的要求,若在应用间共享文件,需要发送Content://Uri,而不再是File://Uri,并且需要对此Uri进行临时访问授权。

2、解决办法:

  使用FileProvider,FileProvider是V4包下的类,继承自ContentProvider。

3、使用步骤:

  ①、首先在清单文件中进行FileProvider注册(与activity同级,四大组件没啥说的):

<provider            android:name="android.support.v4.content.FileProvider"            android:authorities="项目包名.fileprovider"            android:exported="false"            android:grantUriPermissions="true">            <meta-data                android:name="android.support.FILE_PROVIDER_PATHS"                android:resource="@xml/provider_path" />        </provider>
  补充:android:exported="false"  表示对其他应用不可用
        android:grantUriPermissions="true"  授予临时权限

  ②、在res目录下新建xml文件夹,即res/xml,在xml中新建一个provider_path文件,如下:

<?xml version="1.0" encoding="utf-8"?><paths xmlns:android="http://schemas.android.com/apk/res/android">    <external-path name="external_files" path=""/></paths>
     补充:<paths>下一般常用以下子节点:

        <files-path>:Context.getFilesDir()——指向内部存储要共享的目录 

        <cache-path>:Context.getCacheDir()——指向缓存要共享的目录

        <external-path>:Environment.getExternalStorageDirectory()——指向外部存储要共享的目录 

        <root-path>:尚未发现官方对其的说明,知道的童鞋欢迎补充。根据字面理解,为整个存储的根路径,针对诸如查找不到照片地址的Bug。

        name为自定义的名字,path为目录,path=""指的是全部目录|path="."即为当前的根目录

  ③、只用更改上面贴出来的getMediaFileUri()方法,因为牵涉多版本调用相机,所以再封装一个适用于7.0以上获取Uri的方法get24MediaFileUri(),代码如下:

/**     * 版本24以上     */    public Uri get24MediaFileUri(int type) {        File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), "相册名字");        if (!mediaStorageDir.exists()) {            if (!mediaStorageDir.mkdirs()) {                return null;            }        }        //创建Media File        String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());        File mediaFile;        if (type == TYPE_TAKE_PHOTO) {            mediaFile = new File(mediaStorageDir.getPath() + File.separator + "IMG_" + timeStamp + ".jpg");        } else {            return null;        }        return FileProvider.getUriForFile(this, getPackageName()+".fileprovider", mediaFile);    }

  注意:get24MediaFileUri()与getMediaFileUri()唯一的不同为:api24以下,使用的是Uri.fromFile(File)获取的Uri,api24及以上必须使用FileProvider,调用FileProvider.getUriForFile(this, getPackageName()+".fileprovider", File)来获取Uri。

  ④、7.0以上调用系统相机,如下:

 Intent takeIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); Uri photoUri = get24MediaFileUri(TYPE_TAKE_PHOTO); takeIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri); startActivityForResult(takeIntent, CODE_TAKE_PHOTO);

  只是将Uri获取的途径由getMediaFileUri()改为了get24MediaFileUri()。

  ⑤、多版本适配

1、权限获取的适配,详细请看 通俗易懂,手把手教会你android 6.0后(兼容6.0之前版本)申请危险权限的方法

2、在权限获取后,判断SDK版本,然后进行相应操作,如下:

if (Build.VERSION.SDK_INT >= 24) {        Intent takeIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);        Uri photoUri = get24MediaFileUri(TYPE_TAKE_PHOTO);        takeIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);        startActivityForResult(takeIntent, CODE_TAKE_PHOTO);    } else {        Intent takeIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);        Uri photoUri = getMediaFileUri(TYPE_TAKE_PHOTO);        takeIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);        startActivityForResult(takeIntent, CODE_TAKE_PHOTO);    }

  ⑥、拍照完毕后,获取返回数据显示照片,api24以后,需要转换输入流来获取Bitmap,而输入流的获取需要通过getContentResolvrer.openInputStream()来获取,具体代码如下:

@Override    protected void onActivityResult(int requestCode, int resultCode, Intent data) {        super.onActivityResult(requestCode, resultCode, data);        switch (requestCode) {            case CODE_TAKE_PHOTO:                if (resultCode == RESULT_OK) {                    if (data != null) {                        if (data.hasExtra("data")) {                            Log.i("URI", "data is not null");                            Bitmap bitmap = data.getParcelableExtra("data");                            imageView.setImageBitmap(bitmap);//imageView即为当前页面需要展示照片的控件,可替换                        }                    } else {                        Log.i("URI", "Data is null");                         if (Build.VERSION.SDK_INT >= 24){                            Bitmap bitmap = null;                            try {                                bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(photoUri));                            } catch (FileNotFoundException e) {                                e.printStackTrace();                            }           imageView.setImageBitmap(bitmap);                        }else {                            Bitmap bitmap = BitmapFactory.decodeFile(photoUri.getPath());                            imageView.setImageBitmap(bitmap);                        }                    }                }                break;        }    }
通过以上步骤,即可在多版本系统中调用系统相机并显示了。

  系统相册选择图片

1、设置相册请求Code

 public final int CODE_SELECT_IMAGE = 2;//相册RequestCode

2、调用系统相册

Intent albumIntent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);                        startActivityForResult(albumIntent, CODE_SELECT_IMAGE);

注意:相册在6.0后会用到访问存储设备这个危险权限,所以也得做权限适配

3、选择图片后,获取返回数据,并显示图片

@Override    protected void onActivityResult(int requestCode, int resultCode, Intent data) {        super.onActivityResult(requestCode, resultCode, data);        switch (requestCode) {            case CODE_SELECT_IMAGE:                if (resultCode == RESULT_OK) {                    selectPic(data);                }                break;        }    }    

//选择照片    private void selectPic(Intent intent) {        Uri selectImageUri = intent.getData();        String[] filePathColumn = {MediaStore.Images.Media.DATA};        Cursor cursor = getContentResolver().query(selectImageUri, filePathColumn, null, null, null);        cursor.moveToFirst();        int columnIndex = cursor.getColumnIndex(filePathColumn[0]);        String picturePath = cursor.getString(columnIndex);        cursor.close();        imageView.setImageBitmap(BitmapFactory.decodeFile(picturePath));    }

以上即可调用系统相册选择照片,并显示。





5 0
原创粉丝点击