在早期版本的Android中,权限的申请十分简单,在AndroidManifest.xml中请求对应的权限就可以了。
Google发现,这样子的权限申请方式,问题好大!!因为用户在安装的时候,只需要同意一次即会同意所有权限,没办法单个拒绝权限,造成用户信息的泄露。(直接安装apk,同意一次即可;如果在GooglePlay安装,也是同意权限就直接下载了)一个相对合理的权限方案,千呼万唤终于来了。为了不对早期版本的App造成伤害,在早期的Build版本的Apk,直接在Manifest里申请权限,是没有问题了,不过发布Google Play的时候,会提示你要用最新的BuildTool(国内环境跟不上没办法)早期的方案就不记录了,这里看看最新的方案要怎么弄!(其实官方教程和网上资料都很多,官方教程很详细,网上的资料就比较乱,甚至有些把简单的权限方式弄的太复杂了,或者说对应一般的情况,可以有更简单的解决方法)
这里先大致贴一下思路(假设获取位置的FINE_LOCATION权限):
1.检查是否已经有这个权限(有权限,直接就去执行定位吧)
2.如果没有这个权限,就向用户申请这个权限
3.用户如果允许,执行定位,用户如果拒绝,执行其他操作(千万别像某信,某宝,某度那样,不给权限不运行App哦!!!)
那么对应的代码就是这样:
// RequestCode是获取返回用的!
private final int PERMISSION_REQUEST_CODE = 1;
// 方法,获取位置权限
protected void requestFineLocationPermission() {
String permission = Manifest.permission.ACCESS_FINE_LOCATION;
if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {
// 此时还没有权限!
ActivityCompat.requestPermissions(this, new String[]{permission}, PERMISSION_REQUEST_CODE);
}
else {
// 有权限了,去做该做的事情吧!
}
}
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
if(requestCode == PERMISSION_REQUEST_CODE) {
if(grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//权限通过
}
}else{
//权限拒绝
}
return;
}
super.onRequestPermissionsResult(requestCode,permissions,grantResults);
}
是的,就是这么简单。不过,如果你就这样子用在了你的项目里,会有坑!!!
如果你本身就是Android用户,细心的你一定会发现,App在向你申请权限的时候,你会有几个选项:
1.允许
2.禁止
3.还可以禁止后不再询问!(坑就出现在这里了)
如果按上面的代码,你会永远得到一个结果:用户拒绝!!(如果用户是不小心点错,或者出于好奇心尝试,那很容易就懵逼了)而且你的requestPermissions方法,在用户面前完全不会显示权限对话框!
所以我们要修改一下第二段:
// 方法,获取位置权限
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
if (requestCode == PERMISSION_REQUEST_CODE) {
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// 权限通过
} else {
// 权限拒绝
if (!ActivityCompat.shouldShowRequestPermissionRationale(this, permissions[0])) {
// 禁止后不再询问了!
} else {
// 用户此次选择了禁止权限
}
}
return;
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
这个方法,shouldShowRequestPermissionRationale,翻译过来就是:是否应该显示请求权限的说明。我们就可以用来检查用户是否选择了“禁止后不再询问”。
这里就简单说一下返回值Boolean的意思,True表示此时应该显示说明,False表示用户选择了不要再提示了,证明用户不想将该权限赋予你的App,不用解释了……
至此,这个功能算是可用了!!!但是,还不完美!!!必须稍微封装一下让自己用起来更方便嘛!思路如下:
1.一个Class MyBaseActivity,这个相信你会有
2.一个Interface PermissionListener(可选,不然就在MyBaseActivity里添加可复写方法)
这里用Interface的方法贴一下代码:(其实在MyBaseActivity里会更简单哈)
private final int PERMISSION_REQUEST_CODE = 1;
private int mPermissionRequestId;
private PermissionListener mPermissionListener;
protected void permissionRequest(int requestId, String permission, PermissionListener listener) {
mPermissionListener = listener;
mPermissionRequestId = requestId;
if (ContextCompat.checkSelfPermission(this, permission) == PackageManager.PERMISSION_GRANTED) {
mPermissionListener.onPermissionConfirmed(mPermissionRequestId);
} else {
// 尝试请求权限
ActivityCompat.requestPermissions(this, new String[]{permission}, PERMISSION_REQUEST_CODE);
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
if (requestCode == PERMISSION_REQUEST_CODE) {
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//权限通过
mPermissionListener.onPermissionConfirmed(mPermissionRequestId);
} else {
//权限拒绝
if (!ActivityCompat.shouldShowRequestPermissionRationale(this, permissions[0])) {
// 用户直接拒绝了(不再显示)
mPermissionListener.onPermissionDeniedForever(mPermissionRequestId);
} else {
// 用户此次选择了禁止权限
mPermissionListener.onPermissionDenied(mPermissionRequestId);
}
}
return;
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
public interface PermissionListener {
void onPermissionConfirmed(int requestId);
void onPermissionDenied(int requestId);
void onPermissionDeniedForever(int requestId);
}
然后,在你的Activity里,extend你的MyBaseActivity,implements刚才那个Interface,然后通过刚才写的这个方法:permissionRequest(YOUR_REQUEST_ID, Manifest.permission.PERMISSION_YOU_WANT, YourActivity.this);就可以在接口回调里知道权限的情况咯!
(个人觉得在MyBaseActivity里加模板方法的方案会更优,这里就不贴出来了,自己尝试吧!)