最近开发中需要用到相机扫码的功能,但是有的手机可以直接打开摄像头扫码,而有的手机点开则是黑屏并且没有权限请求。
原因:
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.效果图如下:
以上!共勉!