最近开发中需要用到相机扫码的功能,但是有的手机可以直接打开摄像头扫码,而有的手机点开则是黑屏并且没有权限请求。

 

原因:

android6.0后,推出了动态权限管理。android6.0后,对于一些特别敏感的权限,开发者必须在程序中进行声明。拍照和从相册选择图片都是涉及到用户隐私的敏感权限,必须在程序中进行声明。

 

解决方法:

只需要在需要用到相机时添加以下代码:

//6.0以上都需要手动添加动态权限
	ActivityCompat.requestPermissions(getActivity(),
	new String[]{Manifest.permission.CAMERA, Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);

 

或者使用以下方法,虽然没有上面的简单粗暴,但是代码更明了,更严谨

第一步:

调用ContextCompat.checkSelfPermission(Context context, String permission)检查是否有WRITE_EXTERNAL_STORAGE权限,如果没有就申请权限,否则直接拍照。

if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
                //申请权限,CEMERA_OK是自定义的常量
                ActivityCompat.requestPermissions(this,
                        new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                        CEMERA_OK);
            }
        } else {
            //有权限,直接拍照
            //执行拍照的动作
        }

第二步:

重写onRequestPermissionsResult(int requestCode, String[] permissions,int[] grantResults)方法,检查权限申请是否成功。

@Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if (requestCode == REQUEST_TAKE_PHOTO_PERMISSION) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            //申请成功,可以拍照
            //进行拍照的方法
            } else {
                Toast.makeText(this, "CAMERA PERMISSION DENIED", Toast.LENGTH_SHORT).show();
            }
            return;
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }

 

并且,一定要记得在Menifest文件中添加对应的权限:

 

<uses-permission android:name="android.permission.CAMERA"/>

这样,再调用手机的相机时,会弹出一个授权请求,允许后,便可以使用相机了。

 

举一反三

当我们需要进行其他的需要动态权限的敏感操作时,也可以照葫芦画瓢,只要把权限关键词CAMRE改成其他权限名称就好,下面列一下权限的名称:

 

权限组

权限

CALENDAR

READ_CALENDAR

CALENDAR

WRITE_CALENDAR

CAMERA

CAMERA

CONTACTS

READ_CONTACTS

CONTACTS

WRITE_CONTACTS

CONTACTS

GET_ACCOUNTS

LOCATION

ACCESS_FINE_LOCATION

LOCATION

ACCESS_COARSE_LOCATION

MICROPHONE

RECORD_AUDIO

PHONE

READ_PHONE_STATE

PHONE

CALL_PHONE

PHONE

READ_CALL_LOG

PHONE

WRITE_CALL_LOG

PHONE

ADD_VOICEMAIL

PHONE

USE_SIP

PHONE

PROCESS_OUTGOING_CALLS

SENSORS

BODY_SENSORS

SMS

SEND_SMS

SMS

RECEIVE_SMS

SMS

READ_SMS

SMS

RECEIVE_WAP_PUSH

SMS

RECEIVE_MMS

STORAGE

READ_EXTERNAL_STORAGE

STORAGE

WRITE_EXTERNAL_STORAGE

 

同样,不要忘了Menifest里面的权限添加。

 

-------------------------------------------------------------我是分隔线--------------------------------------------------------------

 

这里贴出使用dialog进行动态权限请求的代码(这里以文件读写和录音权限为例子):

1.在要进行判断的Activity的onCreate()方法内,判断是否已经给予相应的权限

//6.0动态权限(文件)
		 if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
				 != PackageManager.PERMISSION_GRANTED) {
			 showMissingPermissionDialog();//弹出dialog方法
		 }
		 //6.0动态权限(录音)
		 if (ActivityCompat.checkSelfPermission(this, Manifest.permission.RECORD_AUDIO)
				 != PackageManager.PERMISSION_GRANTED) {
			 showMissingPermissionDialog();//弹出dialog方法
		 }

 

2.弹出dialog及其点击事件方法本体

/**
	 * 提示用户的 dialog(6.0动态权限)
	 */
	protected void showMissingPermissionDialog() {
		AlertDialog.Builder builder = new AlertDialog.Builder(this);
		builder.setTitle("提示");
		builder.setMessage("该应用缺少权限,可能会导致应用无法正常使用。\n\n请点击\"设置\"→\"权限管理\"打开所需权限,并重启应用。");
		// 拒绝, 退出应用
		builder.setNegativeButton(R.string.cancel,
				new DialogInterface.OnClickListener() {
					@Override
					public void onClick(DialogInterface dialog, int which) {
						Log.i("info", "8--权限被拒绝,此时不会再回调onRequestPermissionsResult方法");
					}
				});
		builder.setPositiveButton(R.string.setting,
				new DialogInterface.OnClickListener() {
					@Override
					public void onClick(DialogInterface dialog, int which) {
						Log.i("info", "4,需要用户手动设置,开启当前app设置界面");
						startAppSettings();
					}
				});
		builder.setCancelable(false);
		builder.show();
	}

	/**
	 * 打开     App设置界面(6.0动态权限)
	 */
	private void startAppSettings() {
		Intent intent = new Intent(
				Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
		intent.setData(Uri.parse("package:" + getPackageName()));
		startActivity(intent);
	}

 

3.效果图如下:

Android 摄像头权限 手机中摄像头权限管理_ide

 

Android 摄像头权限 手机中摄像头权限管理_权限管理_02

 

Android 摄像头权限 手机中摄像头权限管理_android_03

 

 

以上!共勉!