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

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

Permissions

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

Android

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)限管理吧! 這里提供了一個非常不錯的解決方案, 提供源碼, 項(xiàng)目可以直接使用.

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

本文示例的GitHub下載地址

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

流程圖

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

流程圖

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


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)限的工具類
 * <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. 首頁

假設(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)閉.


3. 授權(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兼容低版本.

效果


自定義授權(quán)

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

危險權(quán)限列表

危險權(quán)限列表

參考1, 參考2.

OK, that's all! Enjoy it.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末闷旧,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子城舞,更是在濱河造成了極大的恐慌轩触,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件家夺,死亡現(xiàn)場離奇詭異脱柱,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)拉馋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門榨为,熙熙樓的掌柜王于貴愁眉苦臉地迎上來惨好,“玉大人,你說我怎么就攤上這事随闺∪沾ǎ” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵矩乐,是天一觀的道長龄句。 經(jīng)常有香客問我,道長散罕,這世上最難降的妖魔是什么分歇? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮欧漱,結(jié)果婚禮上职抡,老公的妹妹穿的比我還像新娘。我一直安慰自己硫椰,他們只是感情好繁调,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著靶草,像睡著了一般蹄胰。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上奕翔,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天裕寨,我揣著相機(jī)與錄音,去河邊找鬼派继。 笑死宾袜,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的驾窟。 我是一名探鬼主播庆猫,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼绅络!你這毒婦竟也來了月培?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤恩急,失蹤者是張志新(化名)和其女友劉穎杉畜,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體衷恭,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡此叠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了随珠。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片灭袁。...
    茶點(diǎn)故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡猬错,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出茸歧,到底是詐尸還是另有隱情兔魂,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布举娩,位于F島的核電站,受9級特大地震影響构罗,放射性物質(zhì)發(fā)生泄漏铜涉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一遂唧、第九天 我趴在偏房一處隱蔽的房頂上張望芙代。 院中可真熱鬧,春花似錦盖彭、人聲如沸纹烹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽铺呵。三九已至,卻和暖如春隧熙,著一層夾襖步出監(jiān)牢的瞬間片挂,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工贞盯, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留音念,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓躏敢,卻偏偏與公主長得像闷愤,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子件余,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評論 2 344

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