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

2024-04-26 11:58

本文主要是介绍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每次都检查并申请相机权限,如下.

 /*** 申请相机权限** @param context* @param photoFromCamera  拍照保存图片路径*** @see {https://github.com/yanzhenjie/AndPermission}* */public static void requestCameraPermission(final Context context, final String photoFromCamera){//API >=23if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){AndPermission.with(context).requestCode(PERMISSION_MEDIA_REQUEST_CODE).permission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE,Manifest.permission.READ_EXTERNAL_STORAGE).rationale(new RationaleListener() {@Overridepublic void showRequestPermissionRationale(int requestCode, Rationale rationale) {// 此对话框可以自定义,调用rationale.resume()就可以继续申请。AndPermission.rationaleDialog(context, rationale).show();}}).callback(new PermissionListener() {@Overridepublic void onSucceed(int requestCode, @NonNull List<String> grantPermissions) {// 权限申请成功回调。if(requestCode == PERMISSION_MEDIA_REQUEST_CODE) {UIRouter.JumpToCameraActivity(context,photoFromCamera);}}@Overridepublic void onFailed(int requestCode, @NonNull List<String> deniedPermissions) {// 权限申请失败回调。if(requestCode == PERMISSION_MEDIA_REQUEST_CODE) {ToastView.showToast(context,"拒绝授权");}}}).start();}}

   /*** 调用系统拍照* @param saveImagePathFromCamera 拍照图片保存路径* @param context*/public static void JumpToCameraActivity(Context context, String saveImagePathFromCamera) {/*调用系统拍照*/Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);Uri uri = null; File imageFile = FileUtil.getFile(saveImagePathFromCamera);//此路径可以为storage/mounted/0/DCIM或其他外部存储路径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

 /*** 申请相机权限** @param context* @param photoFromCamera  拍照保存图片路径*** @see {https://github.com/yanzhenjie/AndPermission}* */public static void requestCameraPermission(final Context context, final String photoFromCamera){//API <23if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M){UIRouter.JumpToCameraActivity(context,photoFromCamera);}else {//API >=23AndPermission.with(context).requestCode(PERMISSION_MEDIA_REQUEST_CODE).permission(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE,Manifest.permission.READ_EXTERNAL_STORAGE).rationale(new RationaleListener() {@Overridepublic void showRequestPermissionRationale(int requestCode, Rationale rationale) {// 此对话框可以自定义,调用rationale.resume()就可以继续申请。AndPermission.rationaleDialog(context, rationale).show();}}).callback(new PermissionListener() {@Overridepublic void onSucceed(int requestCode, @NonNull List<String> grantPermissions) {// 权限申请成功回调。if(requestCode == .PERMISSION_MEDIA_REQUEST_CODE) {UIRouter.JumpToCameraActivity(context,photoFromCamera);}}@Overridepublic void onFailed(int requestCode, @NonNull List<String> deniedPermissions) {// 权限申请失败回调。if(requestCode ==PERMISSION_MEDIA_REQUEST_CODE) {ToastView.showToast(context,"拒绝授权");}}}).start();}}

    /*** 调用系统拍照** @param context*/public static void JumpToCameraActivity(Context context, String saveImagePathFromCamera) {/*调用系统拍照*/Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);Uri uri = null;try {File imageFile = FileUtil.getFile(saveImagePathFromCamera);//API>=24 android 7.0if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){if (intent.resolveActivity(context.getPackageManager()) != null){String imageName = imageFile.getName();//7.0以上 的拍照文件必须在storage/emulated/0/Android/data/包名/files/pictures文件夹File storageDir = context.getExternalFilesDir(Environment.DIRECTORY_PICTURES);File file = FileUtil.getFile(storageDir+"/"+imageName);intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件uri = FileProvider.getUriForFile(context,"包名.fileprovider",file);}}else {//<24 uri = Uri.fromFile(imageFile);}intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);((Activity) context).startActivityForResult(intent, CAMERA_REQUEST_CODE);} catch (IOException e) {e.printStackTrace();}}

1,在manifest.xml中加入:

 <providerandroid:name="android.support.v4.content.FileProvider"android:authorities="包名.fileprovider"android:exported="false"android:grantUriPermissions="true"><meta-dataandroid:name="android.support.FILE_PROVIDER_PATHS"android:resource="@xml/file_paths" /></provider>

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

<?xml version="1.0" encoding="utf-8"?>
<resources><paths><external-pathname="camera_photos"<!--任意-->path="Android/data/包名/files/Pictures" /><!--相机图片保存图片路径,属于APP的存储空间--></paths></resources>




这篇关于Android踩坑日记:android7.0动态相机权限的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/937531

相关文章

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

springboot如何通过http动态操作xxl-job任务

《springboot如何通过http动态操作xxl-job任务》:本文主要介绍springboot如何通过http动态操作xxl-job任务的问题,具有很好的参考价值,希望对大家有所帮助,如有错... 目录springboot通过http动态操作xxl-job任务一、maven依赖二、配置文件三、xxl-

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局

Android ViewBinding使用流程

《AndroidViewBinding使用流程》AndroidViewBinding是Jetpack组件,替代findViewById,提供类型安全、空安全和编译时检查,代码简洁且性能优化,相比Da... 目录一、核心概念二、ViewBinding优点三、使用流程1. 启用 ViewBinding (模块级

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

如何搭建并配置HTTPD文件服务及访问权限控制

《如何搭建并配置HTTPD文件服务及访问权限控制》:本文主要介绍如何搭建并配置HTTPD文件服务及访问权限控制的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、安装HTTPD服务二、HTTPD服务目录结构三、配置修改四、服务启动五、基于用户访问权限控制六、

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio