Android权限如何禁止以及友好提示用户开通必要权限详解
android权限
android安全架构规定:默认情况下,任何应用都没有权限执行对其他应用、操作系统或用户有不利影响的任何操作。这包括读写用户的私有数据(联系人,短信,相册,位置)、读写其他应用的文件、执行网络访问、使设备保持唤醒状态等等。
如果是一些正常的权限(非高危权限),比如网络访问等在应用清单文件(androidmanifest.xml)中配置,系统会自动授予,
但是如果有一些高危权限,位置,文件存储,短信等这个时候系统会要求用户授予权限,android 发出权限请求的方式取决于系统版本:
1、如果设备运行的是android 6.0(marshmallow,api 23)或更高版本,并且应用的targetsdkversion是23或更高版本,则应用将在运行时向用户请求权限(runtime permissions)。用户可随时撤销权限,因此应用每次运行时都应该检查自身是否具备所需的权限。
2、如果设备运行的是android 5.1(lollipop_mr1,api 22)或更低版本,并且应用的targetsdkversion是22或更低版本,则系统在用户安装应用时就要求用户授予权限。如果更新应用时又新增了权限,系统会在用户更新应用时要求授予该权限。用户一旦安装应用,他们撤销权限的唯一方式是卸载应用。
如果我们程序中某些功能,发布一些图片等操作,这个时候避免不了要访问用户设备的图片,但是用户开始的时候禁止了,用户并不知道,这个时候访问就会有问题了,那么怎样去提示用户并且引导用户去授予这些权限呢,看下面的效果(说的再多都不如图来的实际)
下面就开始撸码了
private void getpermission() { //两个日历权限和一个数据读写权限 string[] permissions = new string[]{manifest.permission.access_fine_location, manifest.permission.read_phone_state, manifest.permission.write_external_storage, manifest.permission.camera}; // permissionsutils.showsystemsetting = true;//是否支持显示系统设置权限设置窗口跳转 //这里的this不是上下文,是activity对象! permissionsutils.getinstance().chekpermissions(this, permissions, permissionsresult); }
首先在页面activity启动之后调用这个方法(注意:数组中的权限可以自己配置,这里我写了四个)
权限监听接口对象
//创建监听权限的接口对象 permissionsutils.ipermissionsresult permissionsresult = new permissionsutils.ipermissionsresult() { @override public void passpermissons() { //权限通过执行的方法 //权限通过验证 } @override public void forbitpermissons() { //这是没有通过权限的时候提示的内容,自定义即可 toast.maketext(mcontext, "您没有允许部分权限,可能会导致部分功能不能正常使用,如需正常使用 请允许权限", toast.length_short).show(); finish(); // tool.exitapp(); } };
还需要在activity中重写一个方法
@override public void onrequestpermissionsresult(int requestcode, @nonnull string[] permissions, @nonnull int[] grantresults) { super.onrequestpermissionsresult(requestcode, permissions, grantresults); //就多一个参数this permissionsutils.getinstance().onrequestpermissionsresult(this, requestcode, permissions, grantresults); }
权限以及弹窗设置权限工具类permissionsutils.java
/** * 权限工具类 */ public class permissionsutils { private final int mrequestcode = 100;//权限请求码 public static boolean showsystemsetting = true; private permissionsutils() { } private static permissionsutils permissionsutils; private ipermissionsresult mpermissionsresult; public static permissionsutils getinstance() { if (permissionsutils == null) { permissionsutils = new permissionsutils(); } return permissionsutils; } public void chekpermissions(activity context, string[] permissions, @nonnull ipermissionsresult permissionsresult) { mpermissionsresult = permissionsresult; if (build.version.sdk_int < 23) { //6.0才用动态权限 permissionsresult.passpermissons(); return; } //创建一个mpermissionlist,逐个判断哪些权限未授予,未授予的权限存储到mperrrmissionlist中 list<string> mpermissionlist = new arraylist<>(); //逐个判断你要的权限是否已经通过 for (int i = 0; i < permissions.length; i++) { if (contextcompat.checkselfpermission(context, permissions[i]) != packagemanager.permission_granted) { mpermissionlist.add(permissions[i]);//添加还未授予的权限 } } //申请权限 if (mpermissionlist.size() > 0) {//有权限没有通过,需要申请 activitycompat.requestpermissions(context, permissions, mrequestcode); } else { //说明权限都已经通过,可以做你想做的事情去 permissionsresult.passpermissons(); return; } } //请求权限后回调的方法 //参数: requestcode 是我们自己定义的权限请求码 // 参数: permissions 是我们请求的权限名称数组 // 参数: grantresults 是我们在弹出页面后是否允许权限的标识数组,数组的长度对应的是权限名称数组的长度,数组的数据0表示允许权限,-1表示我们点击了禁止权限 public void onrequestpermissionsresult(activity context, int requestcode, @nonnull string[] permissions, @nonnull int[] grantresults) { boolean haspermissiondismiss = false; //有权限没有通过 if (mrequestcode == requestcode) { for (int i = 0; i < grantresults.length; i++) { if (grantresults[i] == -1) { haspermissiondismiss = true; } } //如果有权限没有被允许 if (haspermissiondismiss) { if (showsystemsetting) { showsystempermissionssettingdialog(context);//跳转到系统设置权限页面,或者直接关闭页面,不让他继续访问 } else { mpermissionsresult.forbitpermissons(); } } else { //全部权限通过,可以进行下一步操作。。。 mpermissionsresult.passpermissons(); } } } /** * 不再提示权限时的展示对话框 */ alertdialog mpermissiondialog; private void showsystempermissionssettingdialog(final activity context) { final string mpackname = context.getpackagename(); if (mpermissiondialog == null) { mpermissiondialog = new alertdialog.builder(context).setmessage("已禁用权限,请手动授予").setpositivebutton("设置", new dialoginterface.onclicklistener() { @override public void onclick(dialoginterface dialog, int which) { cancelpermissiondialog(); uri packageuri = uri.parse("package:" + mpackname); intent intent = new intent(settings.action_application_details_settings, packageuri); context.startactivity(intent); context.finish(); } }).setnegativebutton("取消", new dialoginterface.onclicklistener() { @override public void onclick(dialoginterface dialog, int which) { //关闭页面或者做其他操作 cancelpermissiondialog(); //mcontext.finish(); mpermissionsresult.forbitpermissons(); } }).create(); } mpermissiondialog.show(); //放在show()之后,不然有些属性是没有效果的,比如height和width //以下代码设置解决弹窗不居中问题,一侧有边距,一侧没有 window dialogwindow = mpermissiondialog.getwindow(); windowmanager m = context.getwindowmanager(); display d = m.getdefaultdisplay(); // 获取屏幕宽、高 windowmanager.layoutparams p = dialogwindow.getattributes(); // 获取对话框当前的参数值 // 设置宽度 p.width = (int) (d.getwidth() * 0.95); // 宽度设置为屏幕的0.95 p.gravity = gravity.center;//设置位置 //p.alpha = 0.8f;//设置透明度 dialogwindow.setattributes(p); } //关闭对话框 private void cancelpermissiondialog() { if (mpermissiondialog != null) { mpermissiondialog.cancel(); mpermissiondialog = null; } } public interface ipermissionsresult { void passpermissons(); void forbitpermissons(); } }
以上这些代码就可以实现gif图的效果,当然可以自定义任意效果
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。