Android6.0仿微信权限设置

Android 6.0版本对于程序员兄弟来说最不友好的就是权限的问题,动态权限的设置曾经让我很苦恼,目前大部分关于6.0权限设置的框架基本都是一次性访问多个权限(EasyPermissions),这样导致的问题就是如果我们申请了三种权限,而用户只同意了其中一种,下次再申请权限又是一次性申请三种,很不方便对于用户来说很不友好,偶然情况下发现了安卓猴的这篇文章,
http://sunjiajia.com/2016/04/19/android-m-permissions/
在此基础上做了修改,就实现了想要的那种效果(仿照微信获取权限设置,在启动页每次只访问一个权限,用户同意则继续访问下一个权限,如果用户选择拒绝,不管用户选择的是“不再询问”还是“拒绝”都视为拒绝,就弹出提示框提示该权限的必要性,指引用户去打开权限)
下面我们以存储空间、电话、相机权限为例,
图片做的不太好,见谅见谅~~

添加权限

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

权限处理工具类

package com.fly.permissiondemo; /* * * * * * * =================================== * * * Copyright (c) 2016. * * * 作者:安卓猴 * * * 微博:@安卓猴 * * * 博客:http://sunjiajia.com * * * Github:https://github.com/opengit * * * * * * 注意**:如果您使用或者修改该代码,请务必保留此版权信息。 * * * =================================== * * * * * */ import android.Manifest; import android.app.Activity; import android.content.Context; import android.content.SharedPreferences; import android.content.pm.PackageManager; import android.os.Build; import android.support.v4.app.ActivityCompat; import java.util.ArrayList; import java.util.List; /** * 权限控制工具类: * 为了适配API23,即Android M 在清单文件中配置use permissions后,还要在程序运行的时候进行申请。 * <p/> * ***整个权限的申请与处理的过程是这样的: * *****1.进入主Activity,首先申请所有的权限; * *****2.用户对权限进行授权,有2种情况: * ********1).用户Allow了权限,则表示该权限已经被授权,无须其它操作; * ********2).用户Deny了权限,则下次启动Activity会再次弹出系统的Permisssions申请授权对话框。 * *****3.如果用户Deny了权限,那么下次再次进入Activity,会再次申请权限,这次的权限对话框上,会有一个选项“dont ask me again”: * ********1).如果用户勾选了“dont ask me again”的checkbox,下次启动时就必须自己写Dialog或者Snackbar引导用户到应用设置里面去手动授予权限; * ********2).如果用户未勾选上面的选项,若选择了Allow,则表示该权限已经被授权,无须其它操作; * ********3).如果用户未勾选上面的选项,若选择了Deny,则下次启动Activity会再次弹出系统的Permisssions申请授权对话框。 */ public class PermissionsUtil { // 状态码、标志位 public static final int REQUEST_STATUS_CODE = 0x001; public static final int REQUEST_PERMISSION_SETTING = 0x002; //常量字符串数组,将需要申请的权限写进去,同时必须要在Androidmanifest.xml中声明。 public static String[] PERMISSIONS_GROUP_SORT = { Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.CALL_PHONE, Manifest.permission.CAMERA }; private static PermissionCallbacks callbacks; public interface PermissionCallbacks { void onPermissionsGranted();//权限都有 void onPermissionsDenied(int requestCode, List<String> perms); } public static void checkAndRequestPermissions(final Activity activity, PermissionCallbacks callback) { if (Build.VERSION.SDK_INT >= 23) { callbacks = callback; // 一个list,用来存放没有被授权的权限 ArrayList<String> denidArray = new ArrayList<>(); // 遍历PERMISSIONS_GROUP,将没有被授权的权限存放进denidArray for (String permission : PERMISSIONS_GROUP_SORT) { int grantCode = ActivityCompat.checkSelfPermission(activity, permission); if (grantCode == PackageManager.PERMISSION_DENIED) { denidArray.add(permission); } } // 如果该字符串数组长度大于0,说明有未被授权的权限 if (denidArray.size() > 0) { //循环处理所有未授权的权限,每次只添加一个权限进行获取 ArrayList<String> denidArrayNew = new ArrayList<>(); denidArrayNew.add(denidArray.get(0)); // 将denidArray转化为字符串数组,方便下面调用requestPermissions来请求授权 String[] denidPermissions = denidArrayNew.toArray(new String[denidArrayNew.size()]); requestPermissions(activity, denidPermissions); } else { //已授权 callbacks.onPermissionsGranted(); } } } /** * 关于shouldShowRequestPermissionRationale函数的一点儿注意事项: * ***1).应用安装后第一次访问,则直接返回false; * ***2).第一次请求权限时,用户Deny了,再次调用shouldShowRequestPermissionRationale(),则返回true; * ***3).第二次请求权限时,用户Deny了,并选择了“dont ask me again”的选项时,再次调用shouldShowRequestPermissionRationale()时,返回false; * ***4).设备的系统设置中,禁止了应用获取这个权限的授权,则调用shouldShowRequestPermissionRationale(),返回false。 */ public static boolean showRationaleUI(Activity activity, String permission) { return ActivityCompat.shouldShowRequestPermissionRationale(activity, permission); } /** * 对权限字符串数组中的所有权限进行申请授权,如果用户选择了“dont ask me again”,则不会弹出系统的Permission申请授权对话框 */ public static void requestPermissions(Activity activity, String[] permissions) { ActivityCompat.requestPermissions(activity, permissions, REQUEST_STATUS_CODE); } /** * 用来判断,App是否是首次启动: * ***由于每次调用shouldShowRequestPermissionRationale得到的结果因情况而变,因此必须判断一下App是否首次启动,才能控制好出现Dialog和SnackBar的时机 */ public static boolean isAppFirstRun(Activity activity) { SharedPreferences sp = activity.getSharedPreferences("config", Context.MODE_PRIVATE); SharedPreferences.Editor editor = sp.edit(); if (sp.getBoolean("first_run", true)) { editor.putBoolean("first_run", false); editor.commit(); return true; } else { editor.putBoolean("first_run", false); editor.commit(); return false; } } }

使用方法:
在启动页AppStart跳转首页的时候,调用

PermissionsUtil.checkAndRequestPermissions(AppStart.this, new PermissionsUtil.PermissionCallbacks() { @Override public void onPermissionsGranted() { //所有权限都已经获取到跳转 toMainActivity(); } @Override public void onPermissionsDenied(int requestCode, List<String> perms) { } });

这个是在AppStart中的回调,现在的处理办法是,根据每一次提出的权限申请的回调结果来处理对应权限,并且是每一次处理完都会遍历一次
“PERMISSIONS_GROUP_SORT”,循环处理所有的权限,直到每个权限都获取到,在“onPermissionsGranted()”中进行跳转。这样处理就可以在下次启动时直接询问没有获得的权限。

@Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode == PermissionsUtil.REQUEST_STATUS_CODE) { if (permissions[0].equals(Manifest.permission.READ_EXTERNAL_STORAGE)) {//读写权限 if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {//同意 PermissionsUtil.checkAndRequestPermissions(this, new PermissionsUtil.PermissionCallbacks() { @Override public void onPermissionsGranted() { toMainActivity(); } @Override public void onPermissionsDenied(int requestCode, List<String> perms) { } });//请求 } else {//不同意-提示信息 createLoadedAlertDialog("在设置-应用-"+ getString(R.string.app_name) +"-权限中开启存储空间权限,以正常使用App功能"); } } if (permissions[0].equals(Manifest.permission.CALL_PHONE)) {//电话权限 if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {//同意 PermissionsUtil.checkAndRequestPermissions(this, new PermissionsUtil.PermissionCallbacks() { @Override public void onPermissionsGranted() { toMainActivity(); } @Override public void onPermissionsDenied(int requestCode, List<String> perms) { } }); } else {//不同意-提示信息 createLoadedAlertDialog("在设置-应用-" + getString(R.string.app_name) + "-权限中开启电话权限,以正常使用App功能"); } } if (permissions[0].equals(Manifest.permission.CAMERA)) {//电话权限 if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {//同意 //所有权限均获取 toMainActivity(); } else {//不同意-提示信息 createLoadedAlertDialog("在设置-应用-"+ getString(R.string.app_name) +"-权限中开启照相机权限,以正常使用App功能"); } } } }

在设置的权限组里边有几个权限就需要在这个回调中写几个判断来处理对应的友好提示信息,单对单处理,这种方式避免了跟用户不断扯犊子,简单粗暴提示用户获取权限,一旦用户不从,直接跳设置,负责就退出应用。
下面是git地址 https://git.oschina.net/feiyangwei/PermissionDemo.git
这个方案目前还需要完善,如果用户在打开应用的情况下,去设置里边修改权限,我不清楚怎么监听这块权限的修改,微信是直接重新打开应用,这样就会重新获取权限,如果有知道的大神可以讨论一下 。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

时间: 2024-10-25 03:44:32

Android6.0仿微信权限设置的相关文章

Android6.0仿微信权限设置_Android

Android 6.0版本对于程序员兄弟来说最不友好的就是权限的问题,动态权限的设置曾经让我很苦恼,目前大部分关于6.0权限设置的框架基本都是一次性访问多个权限(EasyPermissions),这样导致的问题就是如果我们申请了三种权限,而用户只同意了其中一种,下次再申请权限又是一次性申请三种,很不方便对于用户来说很不友好,偶然情况下发现了安卓猴的这篇文章, http://sunjiajia.com/2016/04/19/android-m-permissions/ 在此基础上做了修改,就实现了

详解Android6.0运行时权限管理_Android

自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验,不会再强迫用户因拒绝不该拥有的权限而导致的无法安装的事情,也不会再不征求用户授权的情况下,就可以任意的访问用户隐私,而且即使在授权之后也可以及时的更改权限.这就是6.0版本做出的更拥护和注重用户的一大体现. 一.认知 今天我们就来学习下Android6.0的权限管理. Android6.0系统把权限分为两个级别: 一个是Normal Permiss

Android仿微信QQ设置图形头像裁剪功能_Android

最近在做毕业设计,想有一个功能和QQ一样可以裁剪头像并设置圆形头像,额,这是设计狮的一种潮流. 而纵观现在主流的APP,只要有用户系统这个功能,这个需求一般都是在(bu)劫(de)难(bu)逃(xue)! 图片裁剪实现方式有两种,一种是利用系统自带的裁剪工具,一种是使用开源工具Cropper.本节就为大家带来如何使用系统自带的裁剪工具进行图片裁剪~ 还是先来个简单的运行图. 额,简单说下,我待会会把代码写成小demo分享给大家,在文章末尾会附上github链接,需要的可以自行下载~ 下面来简单分

Android6.0运行时权限解析,RxPermissions的使用,自己封装一套权限框架

Android6.0运行时权限解析,RxPermissions的使用,自己封装一套权限框架 在Android6.0中,新增加了一个运行时的权限,我相信很多人都已经知道了,估计也知道怎么用了,这篇博客很简单,就是告诉大家如何去申请运行时权限和RxPermission这个权限框架的使用,同时根据现有的技术封装思想,去封装一个自己可用的权限框架,好的,我们继续往下看 一.Android M 运行时权限介绍 关于Android M的更新变化,我就不啰嗦了,有兴趣的可以看下Android M更新 而我们的

详解Android6.0运行时权限管理

自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验,不会再强迫用户因拒绝不该拥有的权限而导致的无法安装的事情,也不会再不征求用户授权的情况下,就可以任意的访问用户隐私,而且即使在授权之后也可以及时的更改权限.这就是6.0版本做出的更拥护和注重用户的一大体现. 一.认知 今天我们就来学习下Android6.0的权限管理. Android6.0系统把权限分为两个级别: 一个是Normal Permiss

Android仿微信QQ设置图形头像裁剪功能

最近在做毕业设计,想有一个功能和QQ一样可以裁剪头像并设置圆形头像,额,这是设计狮的一种潮流. 而纵观现在主流的APP,只要有用户系统这个功能,这个需求一般都是在(bu)劫(de)难(bu)逃(xue)! 图片裁剪实现方式有两种,一种是利用系统自带的裁剪工具,一种是使用开源工具Cropper.本节就为大家带来如何使用系统自带的裁剪工具进行图片裁剪~ 还是先来个简单的运行图. 额,简单说下,我待会会把代码写成小demo分享给大家,在文章末尾会附上github链接,需要的可以自行下载~ 下面来简单分

Android6.0动态申请权限所遇到的问题小结_Android

白天在做SDK23版本的适配,遇到了不少坑,现在抽空记下来,以此为戒. 首先要知道哪些坑,就得先了解一些定义和基本使用方式. 那么先介绍一下动态申请的权限分组情况. 下面的权限组是由谷歌官方定义的,目的是在申请权限时,只要用户允许同一权限组的任意一条权限,那么该组的其他权限也就默认是允许的.不过据高人介绍,在使用时最好是用到哪个权限就具体的请求该权限,因为保不齐哪天谷歌一高兴就把权限组换了甚至删了 group:android.permission-group.CONTACTS permissio

Android6.0动态申请权限所遇到的问题小结

白天在做SDK23版本的适配,遇到了不少坑,现在抽空记下来,以此为戒. 首先要知道哪些坑,就得先了解一些定义和基本使用方式. 那么先介绍一下动态申请的权限分组情况. 下面的权限组是由谷歌官方定义的,目的是在申请权限时,只要用户允许同一权限组的任意一条权限,那么该组的其他权限也就默认是允许的.不过据高人介绍,在使用时最好是用到哪个权限就具体的请求该权限,因为保不齐哪天谷歌一高兴就把权限组换了甚至删了 group:android.permission-group.CONTACTS permissio

windows下IIS6.0网站最小权限设置详解

先来个大概备忘录安全策略,很多细节来不及写了. 分区,使用NTFS格式化 事先规划好分区,及目录,以及设置各文件夹权限,每根目录只保留Administrators组的.系统组权限. C:\Documents and Settings删除除了Administrators组的其他组权限,需手工重置子文件和目录权限. 管理员账号 密码设置为强壮交叉密码,10位到16位 删除c:\inetpub目录中的所属文件夹 删除C:\WINNT\system32\inetsrv中的iisadmpwd目录. 在本地