(轉(zhuǎn))Android 6.0: 動態(tài)權(quán)限管理的解決方案

Android 6.0版本(Api 23)推出了很多新的特性, 大幅提升了用戶體驗(yàn), 同時也為程序員帶來新的負(fù)擔(dān). 動態(tài)權(quán)限管理就是這樣, 一方面讓用戶更加容易的控制自己的隱私, 一方面需要重新適配應(yīng)用權(quán)限. 時代總是不斷發(fā)展, 程序總是以人為本, 讓我們?yōu)閼?yīng)用添加動態(tài)權(quán)限管理吧! 這里提供了一個非常不錯的解決方案, 提供源碼, 項目可以直接使用.
Android系統(tǒng)包含默認(rèn)的授權(quán)提示框, 但是我們?nèi)孕枰O(shè)置自己的頁面. 原因是系統(tǒng)提供的授權(quán)框, 會有不再提示的選項. 如果用戶選擇, 則無法觸發(fā)授權(quán)提示. 使用自定義的提示頁面, 可以給予用戶手動修改授權(quán)的指導(dǎo).

本文示例GitHub下載地址.

在Api 23中, 權(quán)限需要動態(tài)獲取, 核心權(quán)限必須滿足. 標(biāo)準(zhǔn)流程:


圖片.png

如果用戶點(diǎn)擊, 不再提示, 則系統(tǒng)授權(quán)彈窗將不會彈出. 流程變?yōu)?


圖片.png

流程就這些, 讓我們看看代碼吧.

  1. 權(quán)限
    在AndroidManifest中, 添加兩個權(quán)限, 錄音和修改音量.
    <!--危險權(quán)限-->
    <uses-permission android:name="android.permission.RECORD_AUDIO"/>
    <!--一般權(quán)限-->
    <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>

危險權(quán)限必須要授權(quán), 一般權(quán)限不需要.

檢測權(quán)限類
```

/**
 * 檢查權(quán)限的工具類
 * Created by wangchenlong on 16/1/26.
 */
public class PermissionsChecker {

    private final Context mContext;

    public PermissionsChecker(Context context) {
        mContext = context.getApplicationContext();
    }

    // 判斷權(quán)限集合
    public boolean lacksPermissions(String... permissions) {
        for (String permission : permissions) {
            if (lacksPermission(permission)) {
                return true;
            }
        }
        return false;
    }

    // 判斷是否缺少權(quán)限
    private boolean lacksPermission(String permission) {
        return ContextCompat.checkSelfPermission(mContext, permission) ==
                PackageManager.PERMISSION_DENIED;
    }
}
2. 首頁
假設(shè)首頁需要使用權(quán)限, 在頁面顯示前, 即onResume時, 檢測權(quán)限, 
如果缺少, 則進(jìn)入權(quán)限獲取頁面; 接收返回值, 拒絕權(quán)限時, 直接關(guān)閉.

public class MainActivity extends AppCompatActivity {

    private static final int REQUEST_CODE = 0; // 請求碼

    // 所需的全部權(quán)限
    static final String[] PERMISSIONS = new String[]{
            Manifest.permission.RECORD_AUDIO,
            Manifest.permission.MODIFY_AUDIO_SETTINGS
    };

    @Bind(R.id.main_t_toolbar) Toolbar mTToolbar;

    private PermissionsChecker mPermissionsChecker; // 權(quán)限檢測器

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);

        setSupportActionBar(mTToolbar);

        mPermissionsChecker = new PermissionsChecker(this);
    }

    @Override protected void onResume() {
        super.onResume();

        // 缺少權(quán)限時, 進(jìn)入權(quán)限配置頁面
        if (mPermissionsChecker.lacksPermissions(PERMISSIONS)) {
            startPermissionsActivity();
        }
    }

    private void startPermissionsActivity() {
        PermissionsActivity.startActivityForResult(this, REQUEST_CODE, PERMISSIONS);
    }

    @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        // 拒絕時, 關(guān)閉頁面, 缺少主要權(quán)限, 無法運(yùn)行
        if (requestCode == REQUEST_CODE && resultCode == PermissionsActivity.PERMISSIONS_DENIED) {
            finish();
        }
    }
}
```

核心權(quán)限必須滿足, 如攝像應(yīng)用, 攝像頭權(quán)限就是必須的, 如果用戶不予授權(quán), 則直接關(guān)閉.

  1. 授權(quán)頁
    授權(quán)頁, 首先使用系統(tǒng)默認(rèn)的授權(quán)頁, 當(dāng)用戶拒絕時, 指導(dǎo)用戶手動設(shè)置, 當(dāng)用戶再次操作失敗后, 返回繼續(xù)提示. 用戶手動退出授權(quán)頁時, 給使用頁發(fā)送授權(quán)失敗的通知.
    /**
     * 權(quán)限獲取頁面
     * <p/>
     * Created by wangchenlong on 16/1/26.
     */
    public class PermissionsActivity extends AppCompatActivity {

        public static final int PERMISSIONS_GRANTED = 0; // 權(quán)限授權(quán)
        public static final int PERMISSIONS_DENIED = 1; // 權(quán)限拒絕

        private static final int PERMISSION_REQUEST_CODE = 0; // 系統(tǒng)權(quán)限管理頁面的參數(shù)
        private static final String EXTRA_PERMISSIONS =
                "me.chunyu.clwang.permission.extra_permission"; // 權(quán)限參數(shù)
        private static final String PACKAGE_URL_SCHEME = "package:"; // 方案

        private PermissionsChecker mChecker; // 權(quán)限檢測器
        private boolean isRequireCheck; // 是否需要系統(tǒng)權(quán)限檢測

        // 啟動當(dāng)前權(quán)限頁面的公開接口
        public static void startActivityForResult(Activity activity, int requestCode, String... permissions) {
            Intent intent = new Intent(activity, PermissionsActivity.class);
            intent.putExtra(EXTRA_PERMISSIONS, permissions);
            ActivityCompat.startActivityForResult(activity, intent, requestCode, null);
        }

        @Override protected void onCreate(@Nullable Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            if (getIntent() == null || !getIntent().hasExtra(EXTRA_PERMISSIONS)) {
                throw new RuntimeException("PermissionsActivity需要使用靜態(tài)startActivityForResult方法啟動!");
            }
            setContentView(R.layout.activity_permissions);

            mChecker = new PermissionsChecker(this);
            isRequireCheck = true;
        }

        @Override protected void onResume() {
            super.onResume();
            if (isRequireCheck) {
                String[] permissions = getPermissions();
                if (mChecker.lacksPermissions(permissions)) {
                    requestPermissions(permissions); // 請求權(quán)限
                } else {
                    allPermissionsGranted(); // 全部權(quán)限都已獲取
                }
            } else {
                isRequireCheck = true;
            }
        }

        // 返回傳遞的權(quán)限參數(shù)
        private String[] getPermissions() {
            return getIntent().getStringArrayExtra(EXTRA_PERMISSIONS);
        }

        // 請求權(quán)限兼容低版本
        private void requestPermissions(String... permissions) {
            ActivityCompat.requestPermissions(this, permissions, PERMISSION_REQUEST_CODE);
        }

        // 全部權(quán)限均已獲取
        private void allPermissionsGranted() {
            setResult(PERMISSIONS_GRANTED);
            finish();
        }

        /**
         * 用戶權(quán)限處理,
         * 如果全部獲取, 則直接過.
         * 如果權(quán)限缺失, 則提示Dialog.
         *
         * @param requestCode  請求碼
         * @param permissions  權(quán)限
         * @param grantResults 結(jié)果
         */
        @Override
        public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
            if (requestCode == PERMISSION_REQUEST_CODE && hasAllPermissionsGranted(grantResults)) {
                isRequireCheck = true;
                allPermissionsGranted();
            } else {
                isRequireCheck = false;
                showMissingPermissionDialog();
            }
        }

        // 含有全部的權(quán)限
        private boolean hasAllPermissionsGranted(@NonNull int[] grantResults) {
            for (int grantResult : grantResults) {
                if (grantResult == PackageManager.PERMISSION_DENIED) {
                    return false;
                }
            }
            return true;
        }

        // 顯示缺失權(quán)限提示
        private void showMissingPermissionDialog() {
            AlertDialog.Builder builder = new AlertDialog.Builder(PermissionsActivity.this);
            builder.setTitle(R.string.help);
            builder.setMessage(R.string.string_help_text);

            // 拒絕, 退出應(yīng)用
            builder.setNegativeButton(R.string.quit, new DialogInterface.OnClickListener() {
                @Override public void onClick(DialogInterface dialog, int which) {
                    setResult(PERMISSIONS_DENIED);
                    finish();
                }
            });

            builder.setPositiveButton(R.string.settings, new DialogInterface.OnClickListener() {
                @Override public void onClick(DialogInterface dialog, int which) {
                    startAppSettings();
                }
            });

            builder.show();
        }

        // 啟動應(yīng)用的設(shè)置
        private void startAppSettings() {
            Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
            intent.setData(Uri.parse(PACKAGE_URL_SCHEME + getPackageName()));
            startActivity(intent);
        }
    }
//注意isRequireCheck參數(shù)的使用, 防止和系統(tǒng)提示框重疊.
//系統(tǒng)授權(quán)提示: ActivityCompat.requestPermissions, ActivityCompat兼容低版本.

效果


圖片.png

轉(zhuǎn)載:https://blog.csdn.net/caroline_wendy/article/details/50587230

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末肯腕,一起剝皮案震驚了整個濱河市允蜈,隨后出現(xiàn)的幾起案子挽懦,更是在濱河造成了極大的恐慌,老刑警劉巖麦箍,帶你破解...
    沈念sama閱讀 210,914評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異啥辨,居然都是意外死亡焕妙,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評論 2 383
  • 文/潘曉璐 我一進(jìn)店門沃斤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來圣蝎,“玉大人,你說我怎么就攤上這事衡瓶∨枪” “怎么了?”我有些...
    開封第一講書人閱讀 156,531評論 0 345
  • 文/不壞的土叔 我叫張陵哮针,是天一觀的道長关面。 經(jīng)常有香客問我坦袍,道長,這世上最難降的妖魔是什么等太? 我笑而不...
    開封第一講書人閱讀 56,309評論 1 282
  • 正文 為了忘掉前任捂齐,我火速辦了婚禮,結(jié)果婚禮上缩抡,老公的妹妹穿的比我還像新娘奠宜。我一直安慰自己,他們只是感情好瞻想,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,381評論 5 384
  • 文/花漫 我一把揭開白布压真。 她就那樣靜靜地躺著,像睡著了一般内边。 火紅的嫁衣襯著肌膚如雪榴都。 梳的紋絲不亂的頭發(fā)上待锈,一...
    開封第一講書人閱讀 49,730評論 1 289
  • 那天漠其,我揣著相機(jī)與錄音,去河邊找鬼竿音。 笑死和屎,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的春瞬。 我是一名探鬼主播柴信,決...
    沈念sama閱讀 38,882評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼宽气!你這毒婦竟也來了随常?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,643評論 0 266
  • 序言:老撾萬榮一對情侶失蹤萄涯,失蹤者是張志新(化名)和其女友劉穎绪氛,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體涝影,經(jīng)...
    沈念sama閱讀 44,095評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡枣察,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,448評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了燃逻。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片序目。...
    茶點(diǎn)故事閱讀 38,566評論 1 339
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖伯襟,靈堂內(nèi)的尸體忽然破棺而出猿涨,到底是詐尸還是另有隱情,我是刑警寧澤姆怪,帶...
    沈念sama閱讀 34,253評論 4 328
  • 正文 年R本政府宣布叛赚,位于F島的核電站舆瘪,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏红伦。R本人自食惡果不足惜英古,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,829評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望昙读。 院中可真熱鬧召调,春花似錦、人聲如沸蛮浑。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,715評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽沮稚。三九已至艺沼,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蕴掏,已是汗流浹背障般。 一陣腳步聲響...
    開封第一講書人閱讀 31,945評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留盛杰,地道東北人挽荡。 一個月前我還...
    沈念sama閱讀 46,248評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像即供,于是被迫代替她去往敵國和親定拟。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,440評論 2 348

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