前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【Android 应用开发】动态权限管理示例 ( 使用原生代码实现 | 申请权限 | 判定权限申请结果 | 判定 “ 不再询问 “ 情况 )

【Android 应用开发】动态权限管理示例 ( 使用原生代码实现 | 申请权限 | 判定权限申请结果 | 判定 “ 不再询问 “ 情况 )

作者头像
韩曙亮
发布2023-03-29 14:20:47
2.2K0
发布2023-03-29 14:20:47
举报

文章目录

一、申请权限


首先 , 判定权限是否已经通过 , 如果没有通过再进行申请 ; 如果下面函数返回值为 PackageManager.PERMISSION_GRANTED , 说明权限申请通过 ; 如果返回值为 PackageManager.PERMISSION_DENIED , 说明权限没有被授予 ;

代码语言:javascript
复制
ContextCompat.checkSelfPermission(mActivity, mRequestPermissions[i]);

然后 , 申请没有通过的权限 ; 第

2

个参数是 String 数组 , 内容是权限字符串 ;

代码语言:javascript
复制
    /**
     * 需要申请的权限
     */
    protected String[] mRequestPermissions = new String[]{
            Manifest.permission.READ_EXTERNAL_STORAGE,
            Manifest.permission.WRITE_EXTERNAL_STORAGE
    };
    
	ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);

完整过程 : 在 Build.VERSION_CODES.M , Android 6.0 ( API 23 ) 才启用动态权限申请 ; 只要有

1

个权限没有通过 , 就需要权限申请 ;

代码语言:javascript
复制
    /**
     * 请求动态权限
     *
     * @return
     */
    public boolean requestPermission() {
        // Android 6.0 ( API 23 ) 才启用动态权限申请
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            // 用户是否不同意权限, 只要有 1 个不同意, 则为 true, 默认 false
            boolean isDisagree = false;

            // 判定是否有权限未获取
            for (int i = 0; i < mRequestPermissions.length; i++) {
                if (ContextCompat.checkSelfPermission(
                        mActivity,
                        mRequestPermissions[i]) != PackageManager.PERMISSION_GRANTED) {
                    isDisagree = true;
                }
            }

            if (isDisagree) {
                // 存在权限没有通过,需要申请
                ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);
                return false;
            } else {
                // 所有权限都已同意
                return true;
            }
        } else {
            // 6.0 以下默认有动态权限
            return true;
        }
    }

二、判定权限申请结果


在 Activity 的 onRequestPermissionsResult 回调方法中 , 可以获取到权限是否授予 ;

代码语言:javascript
复制
    @Override
    public void onRequestPermissionsResult(
            int requestCode,
            @NonNull String[] permissions,
            @NonNull int[] grantResults) {
	}

遍历第

3

个参数 grantResults 数组 , 如果指定索引的元素为

-1

, 说明 permissions 数组中指定的权限没有授予通过 , 被用户拒绝了 ;

如果 grantResults 数组中所有的值都为

0

, 说明所有权限授予通过 , 可以继续执行后续操作 ;

代码示例 :

代码语言:javascript
复制
    public void onRequestPermissionsResult(
            int requestCode,
            @NonNull String[] permissions,
            @NonNull int[] grantResults) {

        if (REQUEST_CODE != requestCode) {
            return;
        }

        // 权限是否赋予完毕, 如果有任意一个没有同意, 则判定权限申请失败
        boolean allAgree = true;

        // 遍历 grantResults 数组, 判定哪个权限被拒绝了
        for (int i = 0; i < grantResults.length; i++) {
            if (grantResults[i] == -1) {
                if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permissions[i])) {
                    // 被用户拒绝了, 但是还可以申请, 说明没有设置 "不再询问" 选项
                } else {
                    // 被用户拒绝了, 不能弹出, 说明用户设置了 "不再询问" 选项
                    showDialog();
                }
                allAgree = false;
            }
        }

        // 如果都同意, 则执行相关操作
        if (allAgree) {
            Toast.makeText(mActivity, "权限设置完毕, 执行相关操作", Toast.LENGTH_LONG).show();
        }
    }

三、判定 " 不再询问 " 情况


使用 ActivityCompat.shouldShowRequestPermissionRationale 方法判定用户是否选择了 " 不再询问 " 选项 ;

代码语言:javascript
复制
ActivityCompat.shouldShowRequestPermissionRationale(mActivity, 权限字符串) ; 

shouldShowRequestPermissionRationale 方法的含义是当前是否 提示用户进行权限申请 , 指的是显示给用户申请权限的理由 ;

是否显示申请权限的原理 , 也就是 显示给用户 " 为什么应用需要你授予这个权限 " , 要想方设法劝用户授予这个权限 , 以及说明不授予权限不能使用的哪些功能 ;

如果该方法 返回 true , 则显示 ; 如果该方法返回 false , 则不显示 ;

分为四种情况 :

① 首次申请 : 由于是第

1

次申请权限 , 直接申请即可 , 不需要给用户显示申请权限的理由 , 返回 false ;

② 用户拒绝了申请 : 如果用户拒绝了权限的申请 , 开发者需要给用户显示 " 为什么申请该权限 , 要使用权限做那些事 " , 因此需要给用户提示 , 返回 true ;

③ 用户拒绝申请并选择 " 不再询问 " : 用户已经明确拒绝 , 就不要再骚扰用户了 , 不用给出进一步的提示信息 , 返回 false ;

④ 用户同意权限申请 : 用户已经同意了 , 也不用给用户进行原理提示 , 返回 false ;

Google 的意思是如果用户选择了 " 不再询问 " , 那开发者就不能在提及与该权限相关的事了 ; 但是我们开发时总想把用户引导到权限设置界面 , 让用户自己设置 , 因此这里就有了这个 " 不再询问 " 判定问题 ;

有点反直觉 ;

判定 " 不再询问 " 情况 :

在 Activity 的 onRequestPermissionsResult 方法中 ,

代码语言:javascript
复制
    public void onRequestPermissionsResult(
            int requestCode,
            @NonNull String[] permissions,
            @NonNull int[] grantResults) {

判定 grantResults 数组 , 查看用户是否拒绝该权限 ;

在用户拒绝权限的前提下 , 如果

代码语言:javascript
复制
ActivityCompat.shouldShowRequestPermissionRationale(mActivity, 权限字符串)

方法返回 false , 此时就是用户点击了 " 不再询问 " 选项 ;

只能在上述情况下判定 , 其它情况都判定不了 ;

四、完整代码示例


1、权限管理代码

权限管理代码 :

代码语言:javascript
复制
package com.example.permission;

import android.Manifest;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.provider.Settings;
import android.widget.Toast;

import androidx.annotation.NonNull;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

public class PermissionManager {

    /**
     * 申请权限的 Activity 界面
     */
    private Activity mActivity;

    /**
     * "不再询问" 后的引导对话框
     */
    private AlertDialog mAlertDialog;

    /**
     * 申请权限的请求码, 要求必须 >0
     */
    public final int REQUEST_CODE = 100;

    /**
     * 需要申请的权限
     */
    protected String[] mRequestPermissions = new String[]{
            Manifest.permission.READ_EXTERNAL_STORAGE,
            Manifest.permission.WRITE_EXTERNAL_STORAGE
    };

    public PermissionManager(Activity activity) {
        this.mActivity = activity;
    }

    /**
     * 请求动态权限
     *
     * @return
     */
    public boolean requestPermission() {
        // Android 6.0 ( API 23 ) 才启用动态权限申请
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            // 用户是否不同意权限, 只要有 1 个不同意, 则为 true, 默认 false
            boolean isDisagree = false;

            // 判定是否有权限未获取
            for (int i = 0; i < mRequestPermissions.length; i++) {
                if (ContextCompat.checkSelfPermission(
                        mActivity,
                        mRequestPermissions[i]) != PackageManager.PERMISSION_GRANTED) {
                    isDisagree = true;
                }
            }

            if (isDisagree) {
                // 存在权限没有通过,需要申请
                ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);
                return false;
            } else {
                // 所有权限都已同意
                return true;
            }
        } else {
            // 6.0 以下默认有动态权限
            return true;
        }
    }

    public void onRequestPermissionsResult(
            int requestCode,
            @NonNull String[] permissions,
            @NonNull int[] grantResults) {

        if (REQUEST_CODE != requestCode) {
            return;
        }

        // 权限是否赋予完毕, 如果有任意一个没有同意, 则判定权限申请失败
        boolean allAgree = true;

        // 遍历 grantResults 数组, 判定哪个权限被拒绝了
        for (int i = 0; i < grantResults.length; i++) {
            if (grantResults[i] == -1) {
                if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permissions[i])) {
                    // 被用户拒绝了, 但是还可以申请, 说明没有设置 "不再询问" 选项
                } else {
                    // 被用户拒绝了, 不能弹出, 说明用户设置了 "不再询问" 选项
                    showDialog();
                }
                allAgree = false;
            }
        }

        // 如果都同意, 则执行相关操作
        if (allAgree) {
            Toast.makeText(mActivity, "权限设置完毕, 执行相关操作", Toast.LENGTH_LONG).show();
        }
    }

    /**
     * 用户选择 "不再询问" 后的提示方案
     */
    protected void showDialog() {
        // 不管同意/拒绝 , 只弹出一次
        if (mAlertDialog != null){
            return;
        }

        mAlertDialog = new AlertDialog.Builder(mActivity)
                .setMessage("手动设置权限")
                .setPositiveButton("设置", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        // 跳转到设置界面
                        Intent intent = new Intent(
                                Settings.ACTION_APPLICATION_DETAILS_SETTINGS,
                                Uri.parse("package:com.example.permission")
                        );
                        mActivity.startActivity(intent);

                        mAlertDialog.cancel();
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        mAlertDialog.cancel();
                    }
                })
                .create();
        mAlertDialog.show();
    }
}

2、主界面代码

代码语言:javascript
复制
package com.example.permission;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.view.View;

import com.example.permission.databinding.ActivityMainBinding;

public class MainActivity extends AppCompatActivity {

    /**
     * 权限管理
     */
    private PermissionManager mPermissionManager;

    /**
     * 视图绑定
     */
    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        binding = ActivityMainBinding.inflate(getLayoutInflater());
        setContentView(binding.getRoot());

        binding.button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mPermissionManager = new PermissionManager(MainActivity.this);
                mPermissionManager.requestPermission();
            }
        });
    }

    @Override
    public void onRequestPermissionsResult(
            int requestCode,
            @NonNull String[] permissions,
            @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        mPermissionManager.onRequestPermissionsResult(requestCode,
                permissions, grantResults);
    }
}

3、执行结果

在这里插入图片描述
在这里插入图片描述

五、博客资源

博客源码 :

本文参与?腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2021-06-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客?前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与?腾讯云自媒体分享计划? ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 文章目录
  • 一、申请权限
  • 二、判定权限申请结果
  • 三、判定 " 不再询问 " 情况
  • 四、完整代码示例
    • 1、权限管理代码
      • 2、主界面代码
        • 3、执行结果
        • 五、博客资源
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
        http://www.vxiaotou.com