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
这个方案目前还需要完善,如果用户在打开应用的情况下,去设置里边修改权限,我不清楚怎么监听这块权限的修改,微信是直接重新打开应用,这样就会重新获取权限,如果有知道的大神可以讨论一下 。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍详解Android6.0运行时权限管理,包括了详解Android6.0运行时权限管理的使用技巧和注意事项,需要的朋友参考一下 自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验,不会再强迫用户因拒绝不该拥有的权限而导致的无法安装的事情,也不会再不征求用户授权的情况下,就可以任意的访问用户
本文向大家介绍android6.0运行时权限完美封装方法,包括了android6.0运行时权限完美封装方法的使用技巧和注意事项,需要的朋友参考一下 前几天看了郭大神的运行时权限的专讲,深受启发,由于现在基于目前项目中的运行时权限封装的还不是那么完美,趁着郭神建议的还是历历在目。于是把它完整的敲了下来。并在此基础上添加上自己的一些见解,封装成一个完整的demo,希望与大家进行交流与。 在这里我进行了
仿微信基本功能。基于XMPP服务器的即时通信以及交友客户端。第一期代码的功能如下: ----功能介绍----- 1.新用户注册 2.登陆并连接XMPP服务器 握手 保持在线 2.查找最新注册用户:(下拉刷新 上拉翻页,做的比较匆忙,没操作提示) 3.点击某个用户即可对他发送消息(聊天记录下一期更新) 4.首页显示收到的和发送的所有消息(排序和分组下一期更新) 由于作者没有做屏幕适配,所以请在Ret
高仿微信的打飞机游戏。仿得十分惟妙惟肖,不信就去看视频!Code4App 独家首发! [Code4App.com]
本文向大家介绍Android仿微信语音聊天界面设计,包括了Android仿微信语音聊天界面设计的使用技巧和注意事项,需要的朋友参考一下 有段时间没有看视频了,昨天晚上抽了点空时间,又看了下鸿洋大神的视频教程,又抽时间写了个学习记录。代码和老师讲的基本一样,网上也有很多相同的博客。我只是在AndroidStudio环境下写的。 —-主界面代码—— —自定义Button——- —-对话框管理工具类——
本文向大家介绍android6.0权限动态申请框架permissiondispatcher的方法,包括了android6.0权限动态申请框架permissiondispatcher的方法的使用技巧和注意事项,需要的朋友参考一下 1,添加依赖 在project的build.gradle文件中添加 在module的build.gradle中添加 2,添加注解 注释说明 RuntimePermissio
本文向大家介绍微信开发 微信授权详解,包括了微信开发 微信授权详解的使用技巧和注意事项,需要的朋友参考一下 最近有机会做到一个微信项目;把其中自己整理的笔记分享给大家,有不足或错误的地方望大家指正! 1关于微信授权这块的流程图,如下 一些代码碎片仅供参考: 本地存储的运用: 1. 做微信开发,建议先造一些数据,取到对应的openId ,方便在浏览器里调试; 2.稍微麻烦的就是每次调整完成,需要发布