兩種方式封裝實現(xiàn)6.0上申請運行時權(quán)限

1.第一種 (抽取基類+回調(diào))

public class PermissionActivity extends AppCompatActivity {

    private PermissionsResultListener mListener;
    private int mRequestCode;


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

    protected void checkPermissions(@NonNull String[] permissions, int requestCode, PermissionsResultListener listener) {
        List<String> permissionsList = new ArrayList<>();
        //權(quán)限不能為空
        if (permissions != null || permissions.length != 0) {
            mListener = listener;
            mRequestCode = requestCode;
            for (int i = 0; i < permissions.length; i++) {
                if (!isHavePermissions(permissions[i])) {
                    permissionsList.add(permissions[i]);
                } else {
                    mListener.onPermissionGranted(permissions[i]);
                }
            }
            //遍歷完后申請
            requestPermissions(permissionsList);
        }
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == mRequestCode) {
            if (grantResults.length > 0) {
                for (int i = 0; i < grantResults.length; i++) {
                    if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
                        mListener.onPermissionGranted(permissions[i]);
                    } else {
                        //如果應用之前請求過此權(quán)限但用戶拒絕了請求掉分,ActivityCompat.shouldShowRequestPermissionRationale(this,permissions[i])方法將返回 true,
                        // 但實際使用過程中發(fā)現(xiàn)第一次出現(xiàn)權(quán)限對話框也會返回true百拓,如果用戶在過去拒絕了權(quán)限請求渠鸽,并在權(quán)限請求系統(tǒng)對話框中選擇了 Don’t ask again 選項嫂粟,
                        // 此方法將返回 false
                        if (!(ActivityCompat.shouldShowRequestPermissionRationale(this, permissions[i]))) {
                            mListener.onPermissionDeniedDontAskAgain(permissions[i]);
                        } else {
                            mListener.onPermissionDenied(permissions[i]);
                        }

                    }
                }
            } else {
                mListener.onPermissionFailure();
            }
        }
    }

    //判斷權(quán)限是否申請
    private boolean isHavePermissions(@NonNull String permissions) {
        if (ContextCompat.checkSelfPermission(this, permissions) != PackageManager.PERMISSION_GRANTED) {
            return false;
        }
        return true;
    }

    //申請權(quán)限
    private void requestPermissions(List<String> permissionsList) {
        ActivityCompat.requestPermissions(this, permissionsList.toArray(new String[permissionsList.size()]), mRequestCode);
    }

    public interface PermissionsResultListener {

        //拒絕權(quán)限
        void onPermissionDenied(String permissions);

        //獲取權(quán)限失敗
        void onPermissionFailure();

        //成功獲取權(quán)限
        void onPermissionGranted(String permissions);

        //拒絕了權(quán)限并且勾選了不再提示
        void onPermissionDeniedDontAskAgain(String permissions);
    }
}

在使用過程中需要將Activity繼承自PermissionActivity 读虏,并實現(xiàn)回調(diào)方法责静,如下所示

public class MainActivity extends PermissionActivity implements PermissionActivity.PermissionsResultListener {

    private String TAG = "MainActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        checkPermissions(new String[]{Manifest.permission.CALL_PHONE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1, this);


    }

    @Override
    public void onPermissionDenied(String permission) {
        switch (permission) {
            case Manifest.permission.CALL_PHONE:
                Log.e(TAG, "拒絕了撥打電話的權(quán)限");
                break;
            case Manifest.permission.WRITE_EXTERNAL_STORAGE:
                Log.e(TAG, "拒絕了寫存儲卡的權(quán)限");
                break;
        }
    }

    @Override
    public void onPermissionFailure() {
        Log.e(TAG, "申請權(quán)限失敗");
    }

    @Override
    public void onPermissionGranted(String permission) {
        switch (permission) {
            case Manifest.permission.CALL_PHONE:
                Log.e(TAG, "同意了撥打電話的權(quán)限");
                break;
            case Manifest.permission.WRITE_EXTERNAL_STORAGE:
                Log.e(TAG, "同意了寫存儲卡的權(quán)限");
                break;
        }

    }

    @Override
    public void onPermissionDeniedDontAskAgain(String permission) {
        switch (permission) {
            case Manifest.permission.CALL_PHONE:
                Log.e(TAG, "拒絕了撥打電話的權(quán)限并且不再詢問");
                break;
            case Manifest.permission.WRITE_EXTERNAL_STORAGE:
                Log.e(TAG, "拒絕了寫存儲卡的權(quán)限并且不再詢問");
                break;
        }
    }

}

上述方法看起來通俗易懂,但是在申請的權(quán)限較多時盖桥,需要進行多次的switch語句進行判斷灾螃。代碼稍顯多。

第二種(注解+反射)

1.添加兩個注解

//權(quán)限授予后執(zhí)行的方法注解
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PermissionGrantedMethod {
    public int requestCode();
}

//權(quán)限拒絕后執(zhí)行的方法注解
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PermissionDeniedMethod {
    public int requestCode();
}

2.增加權(quán)限的處理類PermissionHelper

public class PermissionHelper {
    private Object mObject;  //Fragment or Activity
    private int mRequestCode;  //請求碼
    private String[] mRequestPermissions;  //請求的權(quán)限

    public PermissionHelper(Object object) {
        this.mObject = object;
    }

    public static void requestPermission(FragmentActivity activity, int requestCode, String[] permissions) {
        PermissionHelper.with(activity).requestCode(requestCode).requestPermission(permissions).request();
    }

    //鏈式調(diào)用
    public PermissionHelper with(Fragment fragment) {
        return new PermissionHelper(fragment);
    }

    public static PermissionHelper with(FragmentActivity activity) {
        return new PermissionHelper(activity);
    }

    public PermissionHelper requestCode(int requestCode) {
        mRequestCode = requestCode;
        return this;
    }

    public PermissionHelper requestPermission(String... permissions) {
        mRequestPermissions = permissions;
        return this;
    }

    //真正實現(xiàn)和發(fā)起權(quán)限請求
    public void request() {
        //判斷當前版本是不是6.0及以上
        if (!PermissionUtil.isOverMarshmallow()) {
            //如果不是6.0以上則直接執(zhí)行方法揩徊,通過反射獲取方法執(zhí)行
            PermissionUtil.excuteGrantedMethod(mObject, mRequestCode);
            return;
        } else {
            //如果是6.0以上腰鬼,首先判斷權(quán)限是否已經(jīng)授予,獲取還沒有授予的權(quán)限
            List<String> deniedPermissionsList = PermissionUtil.getDeniedPermissionsList(mObject, mRequestPermissions);
            if (deniedPermissionsList.size() > 0) {
                //列表中含有沒有授予的權(quán)限塑荒,則需要去請求權(quán)限
                PermissionUtil.requestPermissions(mObject, deniedPermissionsList, mRequestCode);

            } else {
                //如果權(quán)限已經(jīng)授予則直接執(zhí)行
                PermissionUtil.excuteGrantedMethod(mObject, mRequestCode);
            }
        }

    }

    /**
     * 判斷原來沒有申請到的權(quán)限是否由用戶點擊授予
     *
     * @param object
     * @param requestCode
     * @param permissions
     */
    public static void requestPermissionResult(Object object, int requestCode, String[] permissions, @NonNull int[] grantResults) {
        List<String> deniedPermissionsList = new ArrayList<>();
        //表示第一次申請熄赡,或上次申請權(quán)限均拒絕
        if (permissions.length == grantResults.length) {
            deniedPermissionsList = Arrays.asList(permissions);
        } else {
            deniedPermissionsList = PermissionUtil.getDeniedPermissionsList(object, permissions);
        }


        if (deniedPermissionsList.size() == 0) {
            //用戶都同意授予權(quán)限,則執(zhí)行方法
            PermissionUtil.excuteGrantedMethod(object, requestCode);
        } else {
            //申請的權(quán)限中有用戶不同意的齿税,同意則執(zhí)行同意的代碼彼硫。需要對權(quán)限進行區(qū)分。
            List<String> grantPermissionsList = PermissionUtil.getGrantPermissionsList();
            for (int i = 0; i < grantPermissionsList.size(); i++) {
                PermissionUtil.excuteGrantedMethod(object, requestCode, grantPermissionsList.get(i));
            }
            for (int i = 0; i < deniedPermissionsList.size(); i++) {
                if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
                    PermissionUtil.excuteGrantedMethod(object, requestCode, deniedPermissionsList.get(i));
                } else {
                    PermissionUtil.excuteDeniedMethod(object, requestCode, deniedPermissionsList.get(i));
                }
            }

        }
    }


}

3.權(quán)限的輔助工具類

public class PermissionUtil {
    private static List<String> grantPermissionsList = new ArrayList<>();

    public PermissionUtil() {
        throw new UnsupportedOperationException("Can not instantiated");
    }

    //判斷當前版本是否在6.0及以上
    public static boolean isOverMarshmallow() {
        return Build.VERSION.SDK_INT >= Build.VERSION_CODES.M;
    }

    public static void excuteGrantedMethod(Object mObject, int mRequestCode, String mRequestPermission) {
        Class clazz = mObject.getClass();
        Method[] methods = clazz.getDeclaredMethods();
        for (Method method : methods) {
            //反射獲取帶有PermissionGrantedMethod的方法
            PermissionGrantedMethod permissionGrantedMethod = method.getAnnotation(PermissionGrantedMethod.class);
            //找到了有該注解的方法
            if (permissionGrantedMethod != null) {
                //獲取注解中的請求碼
                int requestCode = permissionGrantedMethod.requestCode();
                String requestPermission = permissionGrantedMethod.requestPermission();
                //若注解中的請求碼和傳遞的請求碼一直凌箕,則執(zhí)行這個方法
                if (requestCode == mRequestCode && requestPermission == mRequestPermission) {
                    //反射執(zhí)行該方法拧篮,第一個參數(shù)是這個方法屬于哪個類羡洁,第二個參數(shù)是該方法的參數(shù)
                    excuteMethod(mObject, method);
                }
            }

        }
    }

    public static void excuteGrantedMethod(Object mObject, int mRequestCode) {
        Class clazz = mObject.getClass();
        Method[] methods = clazz.getDeclaredMethods();
        for (Method method : methods) {
            //反射獲取帶有PermissionGrantedMethod的方法
            PermissionGrantedMethod permissionGrantedMethod = method.getAnnotation(PermissionGrantedMethod.class);
            //找到了有該注解的方法
            if (permissionGrantedMethod != null) {
                //獲取注解中的請求碼
                int requestCode = permissionGrantedMethod.requestCode();
                //若注解中的請求碼和傳遞的請求碼一直负蠕,則執(zhí)行這個方法
                if (requestCode == mRequestCode) {
                    //反射執(zhí)行該方法,第一個參數(shù)是這個方法屬于哪個類枉圃,第二個參數(shù)是該方法的參數(shù)
                    excuteMethod(mObject, method);
                }
            }

        }
    }

    /**
     * 用戶拒絕了權(quán)限芜壁,則執(zhí)行拒絕的方法
     *
     * @param mObject
     * @param mRequestCode
     */
    public static void excuteDeniedMethod(Object mObject, int mRequestCode, String mRequestPermission) {
        Class clazz = mObject.getClass();
        Method[] methods = clazz.getDeclaredMethods();
        for (Method method : methods) {
            //反射獲取帶有PermissionDeniedMethod的方法
            PermissionDeniedMethod permissionDeniedMethod = method.getAnnotation(PermissionDeniedMethod.class);
            //找到了有該注解的方法
            if (permissionDeniedMethod != null) {
                //獲取注解中的請求碼
                int requestCode = permissionDeniedMethod.requestCode();
                String requestPermission = permissionDeniedMethod.requestPermission();
                //若注解中的請求碼和傳遞的請求碼一直礁凡,則執(zhí)行這個方法
                if (requestCode == mRequestCode && requestPermission == mRequestPermission) {
                    //反射執(zhí)行該方法,第一個參數(shù)是這個方法屬于哪個類沿盅,第二個參數(shù)是該方法的參數(shù)
                    excuteMethod(mObject, method);
                }
            }
        }
    }

    //反射執(zhí)行方法
    private static void excuteMethod(Object mObject, Method method) {
        try {
            method.setAccessible(true);
            method.invoke(mObject, new Object[]{});
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }

    /**
     * 獲取沒有授予的權(quán)限
     *
     * @param object
     * @param requestPermissions
     * @return
     */
    public static List<String> getDeniedPermissionsList(Object object, String[] requestPermissions) {
        grantPermissionsList.clear();
        List<String> deniedPermissionsList = new ArrayList<>();
        for (String requestPermission : requestPermissions) {
            if (!isHavePermissions(object, requestPermission)) {
                //沒有申請到這個權(quán)限則將其添加到列表中
                deniedPermissionsList.add(requestPermission);
            } else {
                grantPermissionsList.add(requestPermission);
            }
        }

        return deniedPermissionsList;
    }

    public static List<String> getGrantPermissionsList() {
        return grantPermissionsList;
    }


    //判斷權(quán)限是否申請
    private static boolean isHavePermissions(Object object, @NonNull String permission) {
        if (ContextCompat.checkSelfPermission(getActivity(object), permission) != PackageManager.PERMISSION_GRANTED) {
            return false;
        }
        return true;
    }

    private static FragmentActivity getActivity(Object object) {
        if (object instanceof Fragment) {
            return ((Fragment) object).getActivity();
        } else if (object instanceof FragmentActivity) {
            return (FragmentActivity) object;
        }
        return null;
    }

    //申請權(quán)限
    public static void requestPermissions(Object object, List<String> deniedPermissionsList, int requestCode) {
        ActivityCompat.requestPermissions(getActivity(object), deniedPermissionsList.toArray(new String[deniedPermissionsList.size()]), requestCode);
    }
}

4.使用框架實例

public class SecondActivity extends AppCompatActivity {

    private String TAG = "MainActivity";
    private static final int CALL_PHONE_REQUEST_CODE = 0x11;
    private static final String CALL_PHONE_PERMISSION = Manifest.permission.CALL_PHONE;
    private static final String WRITE_EXTERNAL_STORAGE_PERMISSION = Manifest.permission.WRITE_EXTERNAL_STORAGE;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //checkPermissions(new String[]{Manifest.permission.CALL_PHONE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1, this);
        PermissionHelper.requestPermission(this, CALL_PHONE_REQUEST_CODE, new String[]{CALL_PHONE_PERMISSION, WRITE_EXTERNAL_STORAGE_PERMISSION});
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == CALL_PHONE_REQUEST_CODE)
            PermissionHelper.requestPermissionResult(this, CALL_PHONE_REQUEST_CODE, new String[]{CALL_PHONE_PERMISSION, WRITE_EXTERNAL_STORAGE_PERMISSION}, grantResults);
    }

    @PermissionGrantedMethod(requestCode = CALL_PHONE_REQUEST_CODE, requestPermission = CALL_PHONE_PERMISSION)
    public void Call() {
        Log.e(TAG, "撥打電話");
    }

    @PermissionDeniedMethod(requestCode = CALL_PHONE_REQUEST_CODE, requestPermission = CALL_PHONE_PERMISSION)
    private void error() {
        Log.e(TAG, "撥打電話權(quán)限取消");
    }

    @PermissionGrantedMethod(requestCode = CALL_PHONE_REQUEST_CODE, requestPermission = WRITE_EXTERNAL_STORAGE_PERMISSION)
    private void write() {
        Log.e(TAG, "寫存儲卡");
    }

    @PermissionDeniedMethod(requestCode = CALL_PHONE_REQUEST_CODE, requestPermission = WRITE_EXTERNAL_STORAGE_PERMISSION)
    private void writeQuit() {
        Log.e(TAG, "寫存儲卡權(quán)限取消");
    }

}

目前框架還不夠完善把篓,如果我在申請權(quán)限中一個申請碼對應兩種以上權(quán)限纫溃,如果拒絕一種腰涧,則全部拒絕的問題。后序?qū)⑿薷摹?/p>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末紊浩,一起剝皮案震驚了整個濱河市窖铡,隨后出現(xiàn)的幾起案子疗锐,更是在濱河造成了極大的恐慌,老刑警劉巖费彼,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件滑臊,死亡現(xiàn)場離奇詭異,居然都是意外死亡箍铲,警方通過查閱死者的電腦和手機雇卷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來颠猴,“玉大人关划,你說我怎么就攤上這事∏涛停” “怎么了贮折?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長资盅。 經(jīng)常有香客問我调榄,道長,這世上最難降的妖魔是什么呵扛? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任每庆,我火速辦了婚禮,結(jié)果婚禮上今穿,老公的妹妹穿的比我還像新娘扣孟。我一直安慰自己,他們只是感情好荣赶,可當我...
    茶點故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布凤价。 她就那樣靜靜地躺著,像睡著了一般拔创。 火紅的嫁衣襯著肌膚如雪利诺。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天剩燥,我揣著相機與錄音慢逾,去河邊找鬼。 笑死灭红,一個胖子當著我的面吹牛侣滩,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播变擒,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼君珠,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了娇斑?” 一聲冷哼從身側(cè)響起策添,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤材部,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后唯竹,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體乐导,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年浸颓,在試婚紗的時候發(fā)現(xiàn)自己被綠了物臂。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡产上,死狀恐怖鹦聪,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蒂秘,我是刑警寧澤泽本,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站姻僧,受9級特大地震影響规丽,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜撇贺,卻給世界環(huán)境...
    茶點故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一赌莺、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧松嘶,春花似錦艘狭、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至尽超,卻和暖如春官撼,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背似谁。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工傲绣, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人巩踏。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓秃诵,卻偏偏與公主長得像,于是被迫代替她去往敵國和親塞琼。 傳聞我的和親對象是個殘疾皇子菠净,可洞房花燭夜當晚...
    茶點故事閱讀 44,573評論 2 353

推薦閱讀更多精彩內(nèi)容