Android踩坑日记:android7.0动态相机权限

Android踩坑日记:android7.0动态相机权限

前提:

项目中使用的动态权限开源库github:https://github.com/yanzhenjie/AndPermission。

转载必须注明本文转自严振杰的博客:http://blog.csdn.net/yanzhenjie1003

添加依赖:

compile 'com.yanzhenjie:permission:1.0.3'

Android6.0:

众所周知,Android6.0时相机摄像头权限改成了动态权限申请。实际上在xml中加入CAMERA,WRITE_EXTERNAL_STORAGE全向后,直接调用摄像头。此时是没有“检查权限是否授予”,“没有授予再申请权限”的代码的。

但是(重点),我发现

1,在VIVO,华为等国产机会弹出对话框,

2,三星,sony等外国机不会有弹窗,调用摄像头直接崩溃,

3,魅族手机没有弹出,但是可以直接用摄像头。

我猜测是VIVO,华为定制系统帮助用户检查并申请了相机权限,外国机则没有,魅族可能直接授予权限。为统一,建议android6.0每次都检查并申请相机权限,如下.

 
  1. /**

  2. * 申请相机权限

  3. *

  4. * @param context

  5. * @param photoFromCamera 拍照保存图片路径

  6. *

  7. *

  8. * @see {https://github.com/yanzhenjie/AndPermission}

  9. * */

  10.  
  11. public static void requestCameraPermission(final Context context, final String photoFromCamera){

  12. //API >=23

  13. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){

  14.  
  15. AndPermission.with(context)

  16. .requestCode(PERMISSION_MEDIA_REQUEST_CODE)

  17. .permission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE,Manifest.permission.READ_EXTERNAL_STORAGE)

  18. .rationale(new RationaleListener() {

  19. @Override

  20. public void showRequestPermissionRationale(int requestCode, Rationale rationale) {

  21. // 此对话框可以自定义,调用rationale.resume()就可以继续申请。

  22. AndPermission.rationaleDialog(context, rationale).show();

  23. }

  24. })

  25. .callback(new PermissionListener() {

  26. @Override

  27. public void onSucceed(int requestCode, @NonNull List<String> grantPermissions) {

  28. // 权限申请成功回调。

  29. if(requestCode == PERMISSION_MEDIA_REQUEST_CODE) {

  30. UIRouter.JumpToCameraActivity(context,photoFromCamera);

  31. }

  32. }

  33.  
  34. @Override

  35. public void onFailed(int requestCode, @NonNull List<String> deniedPermissions) {

  36. // 权限申请失败回调。

  37. if(requestCode == PERMISSION_MEDIA_REQUEST_CODE) {

  38. ToastView.showToast(context,"拒绝授权");

  39. }

  40. }

  41. })

  42. .start();

  43. }

  44. }

 
  1. /**

  2. * 调用系统拍照

  3. * @param saveImagePathFromCamera 拍照图片保存路径

  4. * @param context

  5. */

  6. public static void JumpToCameraActivity(Context context, String saveImagePathFromCamera) {

  7. /*调用系统拍照*/

  8.  
  9. Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

  10.  
  11. Uri uri = null;

  12. File imageFile = FileUtil.getFile(saveImagePathFromCamera);//此路径可以为storage/mounted/0/DCIM或其他外部存储路径

  13. uri = Uri.fromFile(imageFile); intent.putExtra(MediaStore.EXTRA_OUTPUT, uri); ((Activity) context).startActivityForResult(intent, CAMERA_REQUEST_CODE); }

 

Android7.0以上:

测试过程中,发现在android7.0以上的设备上使用摄像头时,直接崩溃掉了。原因是android7.0开始,相机拍照的图像保存路径必须在此应用的内部存储文件夹(storage/mounted/0/Android/data/包名//files/pictures文件夹)。需要使用FileProvider获取内部文件的uri

 
  1. /**

  2. * 申请相机权限

  3. *

  4. * @param context

  5. * @param photoFromCamera 拍照保存图片路径

  6. *

  7. *

  8. * @see {https://github.com/yanzhenjie/AndPermission}

  9. * */

  10.  
  11. public static void requestCameraPermission(final Context context, final String photoFromCamera){

  12. //API <23

  13. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M){

  14. UIRouter.JumpToCameraActivity(context,photoFromCamera);

  15. }else {

  16. //API >=23

  17. AndPermission.with(context)

  18. .requestCode(PERMISSION_MEDIA_REQUEST_CODE)

  19. .permission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE,Manifest.permission.READ_EXTERNAL_STORAGE)

  20. .rationale(new RationaleListener() {

  21. @Override

  22. public void showRequestPermissionRationale(int requestCode, Rationale rationale) {

  23. // 此对话框可以自定义,调用rationale.resume()就可以继续申请。

  24. AndPermission.rationaleDialog(context, rationale).show();

  25. }

  26. })

  27. .callback(new PermissionListener() {

  28. @Override

  29. public void onSucceed(int requestCode, @NonNull List<String> grantPermissions) {

  30. // 权限申请成功回调。

  31. if(requestCode == .PERMISSION_MEDIA_REQUEST_CODE) {

  32. UIRouter.JumpToCameraActivity(context,photoFromCamera);

  33. }

  34. }

  35.  
  36. @Override

  37. public void onFailed(int requestCode, @NonNull List<String> deniedPermissions) {

  38. // 权限申请失败回调。

  39. if(requestCode ==PERMISSION_MEDIA_REQUEST_CODE) {

  40. ToastView.showToast(context,"拒绝授权");

  41. }

  42. }

  43. })

  44. .start();

  45. }

  46. }

 
  1. /**

  2. * 调用系统拍照

  3. *

  4. * @param context

  5. */

  6. public static void JumpToCameraActivity(Context context, String saveImagePathFromCamera) {

  7. /*调用系统拍照*/

  8.  
  9. Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

  10.  
  11. Uri uri = null;

  12. try {

  13.  
  14. File imageFile = FileUtil.getFile(saveImagePathFromCamera);

  15. //API>=24 android 7.0

  16. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){

  17. if (intent.resolveActivity(context.getPackageManager()) != null){

  18. String imageName = imageFile.getName();

  19. //7.0以上 的拍照文件必须在storage/emulated/0/Android/data/包名/files/pictures文件夹

  20. File storageDir = context.getExternalFilesDir(Environment.DIRECTORY_PICTURES);

  21. File file = FileUtil.getFile(storageDir+"/"+imageName);

  22.  
  23. intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件

  24. uri = FileProvider.getUriForFile(context,"包名.fileprovider",file);

  25. }

  26. }else {//<24

  27.  
  28. uri = Uri.fromFile(imageFile);

  29. }

  30.  
  31. intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);

  32. ((Activity) context).startActivityForResult(intent, CAMERA_REQUEST_CODE);

  33.  
  34. } catch (IOException e) {

  35. e.printStackTrace();

  36. }

  37.  
  38. }


1,在manifest.xml中加入:

 
  1. <provider

  2. android:name="android.support.v4.content.FileProvider"

  3. android:authorities="包名.fileprovider"

  4. android:exported="false"

  5. android:grantUriPermissions="true">

  6. <meta-data

  7. android:name="android.support.FILE_PROVIDER_PATHS"

  8. android:resource="@xml/file_paths" />

  9. </provider>

2,在res中新建xml文件夹,创建file_paths.xml文件

 
  1. <?xml version="1.0" encoding="utf-8"?>

  2. <resources>

  3.  
  4. <paths>

  5. <external-path

  6. name="camera_photos"<!--任意-->

  7. path="Android/data/包名/files/Pictures" /><!--相机图片保存图片路径,属于APP的存储空间-->

  8. </paths>

  9.  
  10.  

猜你喜欢

转载自blog.csdn.net/zhangbijun1230/article/details/82986178