Android 6.0 - 動(dòng)態(tài)權(quán)限管理的解決方案

歡迎Follow我的GitHub, 關(guān)注我的簡(jiǎn)書(shū). 其余參考Android目錄.

Permissions

本文的合集已經(jīng)編著成書(shū),高級(jí)Android開(kāi)發(fā)強(qiáng)化實(shí)戰(zhàn),歡迎各位讀友的建議和指導(dǎo)。在京東即可購(gòu)買(mǎi):https://item.jd.com/12385680.html

Android

Android 6.0版本(Api 23)推出了很多新的特性, 大幅提升了用戶(hù)體驗(yàn), 同時(shí)也為程序員帶來(lái)新的負(fù)擔(dān). 動(dòng)態(tài)權(quán)限管理就是這樣, 一方面讓用戶(hù)更加容易的控制自己的隱私, 一方面需要重新適配應(yīng)用權(quán)限. 時(shí)代總是不斷發(fā)展, 程序總是以人為本, 讓我們?yōu)閼?yīng)用添加動(dòng)態(tài)權(quán)限管理吧! 這里提供了一個(gè)非常不錯(cuò)的解決方案, 提供源碼, 項(xiàng)目可以直接使用.

Android系統(tǒng)包含默認(rèn)的授權(quán)提示框, 但是我們?nèi)孕枰O(shè)置自己的頁(yè)面. 原因是系統(tǒng)提供的授權(quán)框, 會(huì)有不再提示的選項(xiàng). 如果用戶(hù)選擇, 則無(wú)法觸發(fā)授權(quán)提示. 使用自定義的提示頁(yè)面, 可以給予用戶(hù)手動(dòng)修改授權(quán)的指導(dǎo).

本文示例的GitHub下載地址

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

流程圖

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

流程圖

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


1. 權(quán)限

在AndroidManifest中, 添加兩個(gè)權(quán)限, 錄音修改音量.

    <!--危險(xiǎn)權(quán)限-->
    <uses-permission android:name="android.permission.RECORD_AUDIO"/>

    <!--一般權(quán)限-->
    <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>

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

檢測(cè)權(quán)限類(lèi)

/**
 * 檢查權(quán)限的工具類(lèi)
 * <p/>
 * 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. 首頁(yè)

假設(shè)首頁(yè)需要使用權(quán)限, 在頁(yè)面顯示前, 即onResume時(shí), 檢測(cè)權(quán)限,
如果缺少, 則進(jìn)入權(quán)限獲取頁(yè)面; 接收返回值, 拒絕權(quán)限時(shí), 直接關(guān)閉.

public class MainActivity extends AppCompatActivity {

    private static final int REQUEST_CODE = 0; // 請(qǐng)求碼

    // 所需的全部權(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)限檢測(cè)器

    @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)限時(shí), 進(jìn)入權(quán)限配置頁(yè)面
        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);
        // 拒絕時(shí), 關(guān)閉頁(yè)面, 缺少主要權(quán)限, 無(wú)法運(yùn)行
        if (requestCode == REQUEST_CODE && resultCode == PermissionsActivity.PERMISSIONS_DENIED) {
            finish();
        }
    }
}

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


3. 授權(quán)頁(yè)

授權(quán)頁(yè), 首先使用系統(tǒng)默認(rèn)的授權(quán)頁(yè), 當(dāng)用戶(hù)拒絕時(shí), 指導(dǎo)用戶(hù)手動(dòng)設(shè)置, 當(dāng)用戶(hù)再次操作失敗后, 返回繼續(xù)提示. 用戶(hù)手動(dòng)退出授權(quán)頁(yè)時(shí), 給使用頁(yè)發(fā)送授權(quán)失敗的通知.

/**
 * 權(quán)限獲取頁(yè)面
 * <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)限管理頁(yè)面的參數(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)限檢測(cè)器
    private boolean isRequireCheck; // 是否需要系統(tǒng)權(quán)限檢測(cè)

    // 啟動(dòng)當(dāng)前權(quán)限頁(yè)面的公開(kāi)接口
    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方法啟動(dòng)!");
        }
        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); // 請(qǐng)求權(quán)限
            } else {
                allPermissionsGranted(); // 全部權(quán)限都已獲取
            }
        } else {
            isRequireCheck = true;
        }
    }

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

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

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

    /**
     * 用戶(hù)權(quán)限處理,
     * 如果全部獲取, 則直接過(guò).
     * 如果權(quán)限缺失, 則提示Dialog.
     *
     * @param requestCode  請(qǐng)求碼
     * @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();
    }

    // 啟動(dòng)應(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兼容低版本.

效果


自定義授權(quán)

關(guān)鍵部分就這些了, 動(dòng)態(tài)權(quán)限授權(quán)雖然給程序員帶來(lái)了一些麻煩, 但是對(duì)用戶(hù)還是很有必要的, 我們也應(yīng)該歡迎, 畢竟每個(gè)程序員都是半個(gè)產(chǎn)品經(jīng)理.

危險(xiǎn)權(quán)限列表

危險(xiǎn)權(quán)限列表

參考1, 參考2.

OK, that's all! Enjoy it.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,443評(píng)論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,530評(píng)論 3 416
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 176,407評(píng)論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,981評(píng)論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,759評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,204評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,263評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,415評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,955評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,782評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,983評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,222評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,650評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,892評(píng)論 1 286
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,675評(píng)論 3 392
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,967評(píng)論 2 374

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