Android6.0運行時權限封裝(避免用戶選擇不再提示後無法獲取權限的問題)

轉載請標明出處:http://blog.csdn.net/donkor_/article/details/77326989

前言
Android 6.0 爲了保護用戶隱私,將一些權限的申請放在了應用運行的時候去申請, 比如以往的開發中,開發人員只需要將需要的權限在清單文件中配置即可,安裝後用戶可以在設置中的應用信息中看到:XX應用以獲取**權限。用戶點擊可以選擇給應用相應的權限。此前的應用權限用戶可以選擇允許、提醒和拒絕。在安裝的時候用戶是已經知道應用需要的權限的。但是這樣存在一個問題,就是用戶在安裝的時候,應用需要的權限十分的多(有些開發者爲了省事,會請求一些不必要的權限或者請求全部的權限),這個時候用戶在安裝應用的時候也許並沒有發現某些侵犯自己隱私的權限請求,安裝之後才發現自己的隱私數據被竊取。其實Android6.0 動態權限一方面是爲了廣大用戶考慮,另一方面其實是Google爲了避免一些不必要的官司。
目前針對Android6.0權限適配的邏輯代碼,網上已經很多了。這裏針對用戶拒絕了權限請求並且選擇了“□ 不在提示”時該如何處理。

權限流程
▲在Api 23中, 權限需要動態獲取, 核心權限必須滿足. 標準流程:

▲但這裏有個問題,那就是在系統授權彈窗環節,提醒框會有個不再提示的複選框,如果用戶點擊不再提示,並拒絕授權,那麼再下次授權的時候,系統授權彈窗的提示框就不會在提示,所以我們很有必要需要自定義權限彈窗提示框,那麼流程圖就變成如下了。

權限類型


▲ 在圖中,我們可以看到整個權限裏,可以分爲系統權限和特殊權限授權。系統權限中,又分爲normal和dangerous類型。
normal:這個權限類型並不直接威脅到用戶的隱私,可以直接在manifest清單裏註冊,系統會幫我們默認授權的。
dangerous:這個可以直接給app訪問用戶一些敏感的數據,不僅需要在manifest清單裏註冊,同時在使用的時候,需要向系統請求授權。
值得注意一點,這裏有特殊權限授權的區別,分別是SYSTEM_ALERT_WINDOW 和 WRITE_SETTINGS,雖然這兩個權限也是屬於dangerous權限類型,但是這兩個授權請求方式和其他dangerous權限是不一樣的,需要特殊處理 。



案例介紹
根據官方介紹,當發起首次權限請求時不會提供“不再提示”這一選擇,但如果用戶之前拒絕過權限請求並且再次發起權限請求時,權限請求對話框纔會提供“不在提示”這一選項供用戶選擇,當用戶選擇了“不再提示”之後就不會再彈出權限請求對話框。各大手機廠商的定製系統對該邏輯進行了一定的改動,例如小米MIUI系統當用戶首次選擇拒絕權限時之後就會“不在提示”了,華爲EMUI系統在首次彈出權限請求時就會提供“不在提示”的選項,三星部分系統始終不提供“不在提示”的選項。
當用戶選擇了“不在提示”時,開發者需要引導用戶去設置頁手動授權!
但是android 6.0系統並未提供對“不在提示”這一選項的監聽,那麼開發者該如何判斷用戶是否選擇了“不在提示”這一選項呢?
答案是通過shouldShowRequestPermissionRationale()這一方法。
這一方法的作用是判斷是否需要向用戶解釋爲何需要請求該權限。
當首次發起權限請求時該方法返回false。
當第二次權限請求時該方法返回true。
當發起第二次權限請求並且當用戶選擇了“不再提示”這一選項時該方法返回false。 通過這個邏輯我們可以反推出:當進行第二次權限請求被拒絕並且shouldShowRequestPermissionRationale()返回false時,那麼該用戶一定是選擇了“不再提示”這一選項。

▲以RECORD_AUDIO,CAMERA,READ_EXTERNAL_STORAGE爲例。全部權限請求成功,直接看妹砸。下面看圖 ヾ(◍°∇°◍)ノ゙ 沒圖我會亂說

▲當部分權限受限,且用戶未選中“不再提示”,彈出對話框說明,並進行請求權限

▲當部分權限受限,且用戶選中“不再提示”,彈出對話框說明,引導用戶到設置頁手動授權

▲主要代碼BaseActivity.java

package com.donkor.demo.permission;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Bundle;
import android.provider.Settings;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;

public class BaseActivity extends AppCompatActivity {

    private final int mRequestCode = 1024;
    private RequestPermissionCallBack mRequestPermissionCallBack;

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

    /**
     * 權限請求結果回調
     *
     * @param requestCode
     * @param permissions
     * @param grantResults
     */
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        boolean hasAllGranted = true;
        StringBuilder permissionName = new StringBuilder();
        for (String s : permissions) {
            permissionName = permissionName.append(s + "\r\n");
        }
        switch (requestCode) {
            case mRequestCode: {
                for (int i = 0; i < grantResults.length; ++i) {
                    if (grantResults[i] == PackageManager.PERMISSION_DENIED) {
                        hasAllGranted = false;
                        //在用戶已經拒絕授權的情況下,如果shouldShowRequestPermissionRationale返回false則
                        // 可以推斷出用戶選擇了“不在提示”選項,在這種情況下需要引導用戶至設置頁手動授權
                        if (!ActivityCompat.shouldShowRequestPermissionRationale(this, permissions[i])) {
                            new AlertDialog.Builder(BaseActivity.this).setTitle("PermissionTest")//設置對話框標題
                                    .setMessage("【用戶選擇了不再提示按鈕,或者系統默認不在提示(如MIUI)。" +
                                            "引導用戶到應用設置頁去手動授權,注意提示用戶具體需要哪些權限】" +
                                            "獲取相關權限失敗:" + permissionName +
                                            "將導致部分功能無法正常使用,需要到設置頁面手動授權")//設置顯示的內容
                                    .setPositiveButton("去授權", new DialogInterface.OnClickListener() {//添加確定按鈕
                                        @Override
                                        public void onClick(DialogInterface dialog, int which) {//確定按鈕的響應事件
                                            //TODO Auto-generated method stub
                                            Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                                            Uri uri = Uri.fromParts("package", getApplicationContext().getPackageName(), null);
                                            intent.setData(uri);
                                            startActivity(intent);
                                            dialog.dismiss();
                                        }
                                    }).setNegativeButton("取消", new DialogInterface.OnClickListener() {//添加返回按鈕
                                @Override
                                public void onClick(DialogInterface dialog, int which) {//響應事件
                                    // TODO Auto-generated method stub
                                    dialog.dismiss();
                                }
                            }).setOnCancelListener(new DialogInterface.OnCancelListener() {
                                @Override
                                public void onCancel(DialogInterface dialog) {
                                    mRequestPermissionCallBack.denied();
                                }
                            }).show();//在按鍵響應事件中顯示此對話框
                        } else {
                            //用戶拒絕權限請求,但未選中“不再提示”選項
                            mRequestPermissionCallBack.denied();
                        }
                        break;
                    }
                }
                if (hasAllGranted) {
                    mRequestPermissionCallBack.granted();
                }
            }
        }
    }

    /**
     * 發起權限請求
     *
     * @param context
     * @param permissions
     * @param callback
     */
    public void requestPermissions(final Context context, final String[] permissions,
                                   RequestPermissionCallBack callback) {
        this.mRequestPermissionCallBack = callback;
        StringBuilder permissionNames = new StringBuilder();
        for (String s : permissions) {
            permissionNames = permissionNames.append(s + "\r\n");
        }
        //如果所有權限都已授權,則直接返回授權成功,只要有一項未授權,則發起權限請求
        boolean isAllGranted = true;
        for (String permission : permissions) {
            if (ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_DENIED) {
                isAllGranted = false;
                if (ActivityCompat.shouldShowRequestPermissionRationale((Activity) context, permission)) {
                    new AlertDialog.Builder(BaseActivity.this).setTitle("PermissionTest")//設置對話框標題
                            .setMessage("【用戶曾經拒絕過你的請求,所以這次發起請求時解釋一下】" +
                                    "您好,需要如下權限:" + permissionNames +
                                    " 請允許,否則將影響部分功能的正常使用。")//設置顯示的內容
                            .setPositiveButton("確定", new DialogInterface.OnClickListener() {//添加確定按鈕
                                @Override
                                public void onClick(DialogInterface dialog, int which) {//確定按鈕的響應事件
                                    //TODO Auto-generated method stub
                                    ActivityCompat.requestPermissions(((Activity) context), permissions, mRequestCode);
                                }
                            }).show();//在按鍵響應事件中顯示此對話框
                } else {
                    ActivityCompat.requestPermissions(((Activity) context), permissions, mRequestCode);
                }
                break;
            }
        }
        if (isAllGranted) {
            mRequestPermissionCallBack.granted();
            return;
        }
    }

    /**
     * 權限請求結果回調接口
     */
    interface RequestPermissionCallBack {
        /**
         * 同意授權
         */
        void granted();

        /**
         * 取消授權
         */
        void denied();
    }
}

▲SplashActivity.java中Button請求權限主要代碼

requestPermissions(this, new String[]{Manifest.permission.RECORD_AUDIO,
                        Manifest.permission.CAMERA,
                        Manifest.permission.READ_EXTERNAL_STORAGE},
                new RequestPermissionCallBack() {
                    @Override
                    public void granted() {
                        Toast.makeText(SplashActivity.this, "權限獲取成功,執行下一步操作", Toast.LENGTH_SHORT).show();
                        Intent intent = new Intent(SplashActivity.this, MainActivity.class);
                        startActivity(intent);
                        SplashActivity.this.finish();
                    }

                    @Override
                    public void denied() {
                        Toast.makeText(SplashActivity.this, "部分權限獲取失敗,正常功能受到影響,2秒鐘之後自動退出", Toast.LENGTH_LONG).show();
                        //2秒鐘之後自動退出
                        mHandler.postDelayed(new Runnable() {
                            @Override
                            public void run() {
                                SplashActivity.this.finish();
                            }
                        }, 2000);
                    }
                });

資料與參考:
http://www.cnblogs.com/cr330326/p/5181283.html
http://blog.csdn.net/caroline_wendy/article/details/50587230


妹砸照片及源碼 CSDN下載地址 : http://download.csdn.net/download/donkor_/9938151
妹砸照片及源碼Gayhub下載地址: https://github.com/ChenYXin/PermissionDemo


About me
Email :[email protected]
Android開發交流QQ羣 : 537891203
Android開發交流羣

發佈了89 篇原創文章 · 獲贊 316 · 訪問量 41萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章