之前调试的时候,出现了一个问题,就是当我打开二维码扫描界面的时候,对于一部分手机一直不会出现那个扫描框,这点我也很是郁闷,这不好整啊,毕竟二维码界面是用的别人的,怎么改啊?这个时候我分析了一下原因,最后知道只有部分6.0的手机才会出现这种情况,那么这就简单了。下面我就对关于6.0手机动态申请安全权限做一下讲解:
对于6.0以下的权限及在安装的时候,根据权限声明产生一个权限列表,用户只有在同意之后才能完成app的安装,造成了我们想要使用某个app,就只能默认接受其一些不必要的权限,而在6.0以后,当app需要我们授予不恰当的权限的时候,我们可以予以拒绝。但是这些权限也是有限制的,比如说只是针对一些安全权限做动态授权处理,如打开手机摄像头、打开联系人、打开录音等等,这些涉及到用户安全权限的时候,就要我们开发者手动去请求用户打开权限。废话就到这里,下面看具体实现。
1、检查权限:
if (ContextCompat.checkSelfPermission(thisActivity,
Manifest.permission.READ_CONTACTS)
!= PackageManager.PERMISSION_GRANTED) {
}else{
//
}
ContextCompat.checkSelfPermission,主要用于检测某个权限是否已经被授予,方法返回值为PackageManager.PERMISSION_DENIED或者PackageManager.PERMISSION_GRANTED。当返回DENIED就需要进行申请授权了。
2、申请权限:
ActivityCompat.requestPermissions(thisActivity,
new String[]{Manifest.permission.READ_CONTACTS},
MY_PERMISSIONS_REQUEST_READ_CONTACTS);
第二个参数是需要申请的权限的字符串数组,第三个参数为请求码,主要用于回调的时候检测。可以从方法名requestPermissions以及第二个参数看出,是支持一次性申请多个权限的,系统会通过弹出对话框一个个的询问用户是否授权。
3、权限的申请回调:
@Override
public void onRequestPermissionsResult(int requestCode,
String permissions[], int[] grantResults) {
switch (requestCode) {
case MY_PERMISSIONS_REQUEST_READ_CONTACTS: {
// If request is cancelled, the result arrays are empty.
if (grantResults.length > 0
&& grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// permission was granted, yay! Do the
// contacts-related task you need to do.
} else {
// permission denied, boo! Disable the
// functionality that depends on this permission.
}
return;
}
}
}
首先验证请求码并结合你的申请,然后验证grantResults对应的申请的结果,如果你申请的权限数组有两个权限,那么grantResults的length就为2
还有一点:
if (ActivityCompat.shouldShowRequestPermissionRationale(thisActivity,
Manifest.permission.READ_CONTACTS))
// Show an expanation to the userasynchronously– don’t block
// this thread waiting for the user’s response! After the user
// sees the explanation, try again to request the permission.
}
这个方法就是给用户一个权限申请作出必要的解释,比如用户第一次拒绝过你的权限申请,现在你又点击拍照,那么久需要调用这个方法,作出一个解释,解释为什么需要拍照权限。
好了,到这里估计你会说,怎么感觉有点小麻烦呢! 没错,劳资也感觉是很麻烦啊,我的理念就是统统一条方法解决一个问题,那么,这时候上面的都不要再去看了,什么鬼东西,只需要在你的BaseActivity或者BaseFragment中添加几行我的下面的代码,然后调用的时候,就是一行代码搞定。(上面的部分统统略去,下面的才是今天的主要)
在你的BaseActivity或者BaseFragment中添加几行下面的代码就行了:
private int mPermissionIdx = 0x10;//请求权限索引
private SparseArray mPermissions = new SparseArray<>();//请求权限运行列表
@SuppressLint("Override")
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
GrantedResult runnable = mPermissions.get(requestCode);
if (runnable == null) {
return;
}
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
runnable.mGranted = true;
}
runOnUiThread(runnable);
}
public void requestPermission(String[] permissions, String reason, GrantedResult runnable) {
if(runnable == null){
return;
}
runnable.mGranted = false;
if (Build.VERSION.SDK_INT < 23 || permissions == null || permissions.length == 0) {
runnable.mGranted = true;//新添加
runOnUiThread(runnable);
return;
}
final int requestCode = mPermissionIdx++;
mPermissions.put(requestCode, runnable);
/*
是否需要请求权限
*/
boolean granted = true;
for (String permission : permissions) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
granted = granted && checkSelfPermission(permission) == PackageManager.PERMISSION_GRANTED;
}
}
if (granted) {
runnable.mGranted = true;
runOnUiThread(runnable);
return;
}
/*
是否需要请求弹出窗
*/
boolean request = true;
for (String permission : permissions) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
request = request && !shouldShowRequestPermissionRationale(permission);
}
}
if (!request) {
final String[] permissionTemp = permissions;
AlertDialog dialog = new AlertDialog.Builder(this)
.setMessage(reason)
.setPositiveButton("确定", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
requestPermissions(permissionTemp, requestCode);
}
}
})
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
dialog.dismiss();
GrantedResult runnable = mPermissions.get(requestCode);
if (runnable == null) {
return;
}
runnable.mGranted = false;
runOnUiThread(runnable);
}
}).create();
dialog.show();
} else {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
requestPermissions(permissions, requestCode);
}
}
}
public static abstract class GrantedResult implements Runnable{
private boolean mGranted;
public abstract void onResult(boolean granted);
@Override
public void run(){
onResult(mGranted);
}
}
那么最最重要的来了,那就是用法咯:(往下接着看)
比如说你需要点击一下拍照按钮就打开系统的摄像头进行拍照,那么就在你打开按钮的监听事件中这么写:(比如说是在CarmerActivity中,继承BaseActivity)
openCarmer.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
requestPermission(new String[]{Manifest.permission.CAMERA}, “请求设备相机权限”, new GrantedResult() {
@Override
public void onResult(boolean granted) {
if(granted){//表示用户允许
createLocalStream();
}else {//用户拒绝
Toast.makeText(MainActivity.this,”权限拒绝”,Toast.LENGTH_SHORT).show();
}
}
});
}
});
同样的在fragment中一样可以使用,即((AbsBaseActivity)getActivity()).requestPermission(...)这种方式即可,当首次弹出系统授权弹出框,如果用户点击拒绝的话,那么第二次再次需要该权限的话,会打开自定义的提示框且提示自定义的提示描述,所以动态授权还是非常的人性化的。最后提示一下,如果你觉得在应用内每次都是用到的时候才去授权比较麻烦,那么你也可以在用户刚进入应用后的启动页面一次性开启所有的安全权限,这种方式更加的简便。
ok,大功告成!!!