0. 背景
Android 6.0 (API level 23)中几于,將權(quán)限分成了兩類坯辩。
- 一類是Install權(quán)限(稱之為安裝時(shí)權(quán)限)
- 另一類是Runtime權(quán)限(稱之為運(yùn)行時(shí)權(quán)限)吩坝。
Install權(quán)限是什么? Install權(quán)限:安裝時(shí)權(quán)限,顧名思義着憨,是在安裝app時(shí),就賦予該app的權(quán)限务嫡,即安裝后立即獲取到的權(quán)限甲抖。normal和signature級(jí)別的權(quán)限都是安裝時(shí)權(quán)限漆改。賦予app normal和signature權(quán)限時(shí),不會(huì)給用戶提示界面准谚,系統(tǒng)自動(dòng)決定權(quán)限的賦予挫剑。對(duì)于signature權(quán)限,如果使用權(quán)限的app與聲明權(quán)限的app的簽名不一致柱衔,則系統(tǒng)拒絕賦予該signature權(quán)限樊破。
Runtime權(quán)限是什么? Runtime權(quán)限:運(yùn)行時(shí)權(quán)限唆铐,是指在app運(yùn)行過程中哲戚,賦予app的權(quán)限。這個(gè)過程中艾岂,會(huì)顯示明顯的權(quán)限授予界面顺少,讓用戶決定是否授予權(quán)限。如果app的targetSdkVersion
是22(Lollipop MR1)及以下王浴,dangerous權(quán)限是安裝時(shí)權(quán)限祈纯,否則dangerous權(quán)限是運(yùn)行時(shí)權(quán)限。如果一個(gè)app的targetSdkVersion
是23(或者23以上)叼耙,那么該app所申請(qǐng)的所有dangerous權(quán)限都是運(yùn)行時(shí)權(quán)限腕窥。如果運(yùn)行在Android 6.0的環(huán)境中,該app在運(yùn)行時(shí)必須主動(dòng)申請(qǐng)這些dangerous權(quán)限(調(diào)用requestPermissions()
)筛婉,否則該app沒有獲取到dangerous權(quán)限簇爆。
1. RxPermissions的好處
RxPermissions是幫助開發(fā)者簡(jiǎn)化requestPermissions()
相關(guān)的處理。
(1) 開發(fā)者不用擔(dān)心Android運(yùn)行環(huán)境的版本爽撒,如果系統(tǒng)是Android 6.0之前的版本入蛆,RxPermissions返回的結(jié)果是,app請(qǐng)求的每個(gè)權(quán)限都被允許(granted)硕勿。 RxPermissions內(nèi)部已經(jīng)對(duì)Android版本進(jìn)行了判斷:
boolean isMarshmallow() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.M;
}
public boolean isGranted(String permission){
// 如果是Android 6.0 (Api 23)之前哨毁,則權(quán)限被允許使用。
return !isMarshmallow() || mRxPermissionsFragment.isGranted(permission);
}
(2) 將權(quán)限申請(qǐng)的代碼(requestPermissions())和請(qǐng)求結(jié)果的代碼(onRequestPermissionsResult())放在一起管理源武,避免了代碼的分散扼褪。
(3) RxPermissions具備Rx(RxJava)的特性,例如可以使用鏈?zhǔn)讲僮髁黄埽梢詧?zhí)行filter操作话浇,transform操作,等等闹究。
2. RxPermissions的版本
RxPermissions是基于RxJava的幔崖,RxJava現(xiàn)在有2個(gè)大版本,一個(gè)RxJava 1.x(通常說RxJava),另一個(gè)是RxJava2赏寇。所以RxPermissions有2個(gè)版本吉嫩,分別支持RxJava和RxJava2。
RxPermissions的源碼在https://github.com/tbruyelle/RxPermissions嗅定。 其中master
分支是支持RxJava 1.x的版本率挣,包名為com.tbruyelle.rxpermissions。2.x分支是支持RxJava2的版本露戒,包名為com.tbruyelle.rxpermissions2椒功。 默認(rèn)看到的是master分支,所以看到的代碼是支持RxJava 1.x的RxPermissions智什。
分支的選擇(如下圖):
點(diǎn)擊『Branch: master』之后动漾,可以看到目前有3個(gè)分支:2.x,fix46,和master荠锭。
1
查看支持RxJava2的RxPermissions旱眯,將分支切換到2.x,如下:
2
3. RxPermissions代碼下載
下載包名為com.tbruyelle.rxpermissions
的代碼(支持RxJava1.x):
下載包名為com.tbruyelle.rxpermissions2
的代碼(支持RxJava2):
git clone https://github.com/tbruyelle/RxPermissions RxPermissions2 -b 2.x
4. RxPermissions使用的注意事項(xiàng)
參考 https://github.com/tbruyelle/RxPermissions 中的README证九。
(1) minSdkVersion必須 >= 11删豺。
(2) 使用RxPermissions申請(qǐng)權(quán)限,必須在Activity.onCreate()或者View.onFinishInflate()
中處理愧怜。不能在onResume()中處理呀页,因?yàn)閛nResume()在App的生命周期中可能執(zhí)行的很頻繁。如果在請(qǐng)求權(quán)限的時(shí)候拥坛,App重新啟動(dòng)了(例如屏幕旋轉(zhuǎn)導(dǎo)致的App關(guān)閉再重新創(chuàng)建)蓬蝶,那么用戶的選擇(允許 或者 拒絕)將無法發(fā)給App。 更多討論猜惋,請(qǐng)參考:https://github.com/tbruyelle/RxPermissions/issues/69
5. RxPermissions使用
基于RxJava2丸氛,使用包名為com.tbruyelle.rxpermissions2
的RxPermissions。
5.1 App module的build.gradle
dependencies {
...
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile 'io.reactivex.rxjava2:rxjava:2.0.5'
compile 'com.tbruyelle.rxpermissions2:rxpermissions:0.9.3@aar'
...
}
5.2 AndroidManifest.xml中使用權(quán)限
<!-- protection level is dangerous, need request runtime permission -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.READ_CALL_LOG"/>
<uses-permission android:name="android.permission.RECORD_AUDIO"/>
<uses-permission android:name="android.permission.CAMERA"/> <uses-permission android:name="android.permission.GET_ACCOUNTS"/>
<uses-permission android:name="android.permission.READ_CONTACTS"/>
<uses-permission android:name="android.permission.READ_CALENDAR"/>
<uses-permission android:name="android.permission.SEND_SMS"/>
<uses-permission android:name="android.permission.READ_SMS"/>
<uses-permission android:name="android.permission.CALL_PHONE"/>
5.3 在Activity的onCreate()中申請(qǐng)權(quán)限
package sysshare.lq.com.rxpermissiontest;
import android.Manifest;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import com.tbruyelle.rxpermissions2.Permission;
import com.tbruyelle.rxpermissions2.RxPermissions;
import io.reactivex.functions.Consumer;
public class MainActivity extends AppCompatActivity {
private static final String TAG = "RxPermissionTest";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
requestPermissions();
}
private void requestPermissions() {
RxPermissions rxPermission = new RxPermissions(MainActivity.this);
rxPermission
.requestEach(Manifest.permission.ACCESS_FINE_LOCATION,
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.READ_CALENDAR,
Manifest.permission.READ_CALL_LOG,
Manifest.permission.READ_CONTACTS,
Manifest.permission.READ_PHONE_STATE,
Manifest.permission.READ_SMS,
Manifest.permission.RECORD_AUDIO,
Manifest.permission.CAMERA,
Manifest.permission.CALL_PHONE,
Manifest.permission.SEND_SMS)
.subscribe(new Consumer<Permission>() {
@Override
public void accept(Permission permission) throws Exception {
if (permission.granted) {
// 用戶已經(jīng)同意該權(quán)限
Log.d(TAG, permission.name + " is granted.");
} else if (permission.shouldShowRequestPermissionRationale) {
// 用戶拒絕了該權(quán)限著摔,沒有選中『不再詢問』(Never ask again),那么下次再次啟動(dòng)時(shí)缓窜,還會(huì)提示請(qǐng)求權(quán)限的對(duì)話框
Log.d(TAG, permission.name + " is denied. More info should be provided.");
} else {
// 用戶拒絕了該權(quán)限,并且選中『不再詢問』
Log.d(TAG, permission.name + " is denied.");
}
}
});
}
}
5.4 界面顯示及l(fā)og
[圖片上傳中谍咆。禾锤。。(3)]
如果點(diǎn)擊『拒絕』卧波,不選中『不再詢問』时肿,log為:
D/RxPermissionTest:android.permission.ACCESS_FINE_LOCATION is denied. More info should be provided.
如果點(diǎn)擊『拒絕』,選中了『不再詢問』港粱,則log為:
D/RxPermissionTest:android.permission.ACCESS_FINE_LOCATION is denied.
如果點(diǎn)擊『允許』,log為:
D/RxPermissionTest:android.permission.ACCESS_FINE_LOCATION is granted.
如果訪問被拒絕了
當(dāng)?shù)谝淮握?qǐng)求權(quán)限申請(qǐng)被拒絕后再進(jìn)行第二次申請(qǐng)時(shí),對(duì)話框中會(huì)多出一個(gè) 不再詢問 的復(fù)選框查坪。如果勾選了該復(fù)選框并且拒絕請(qǐng)求寸宏,那么以后將無法再申請(qǐng)?jiān)摍?quán)限。也就是說在調(diào)用 requestPermissions() 后偿曙,onRequestPermissionsResult() 會(huì)立刻被調(diào)用并且申請(qǐng)結(jié)果為 PERMISSION_DENIED 氮凝。 其實(shí)這個(gè)時(shí)候還是有一根救命稻草的。
判斷是否勾選不再詢問
首先需要判斷用戶是否勾選了不再詢問望忆。
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch (requestCode){
case MY_PERMISSIONS_APPLY:
for (int i = 0; i < grantResults.length; i++) {
//判斷權(quán)限的結(jié)果罩阵,如果有被拒絕,就return
if (grantResults[i] == PackageManager.PERMISSION_DENIED){
if (!ActivityCompat.shouldShowRequestPermissionRationale(this,
permissions[i])){
showToast("點(diǎn)擊權(quán)限启摄,并打開全部權(quán)限");
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivityForResult(intent, REQUEST_PERMISSION_SETTING);
}
ActivityCompat 位于 support.v7 包中稿壁,因?yàn)檫\(yùn)行時(shí)權(quán)限是 6.0 的新特性,使用該類可以省略對(duì)版本的判斷當(dāng)權(quán)限申請(qǐng)被拒絕并且shouldShowRequestPermissionRationale() 返回 false 就表示勾選了不再詢問歉备。轉(zhuǎn)到設(shè)置界面現(xiàn)在我們唯一能做的就是跳轉(zhuǎn)到我們 App 的設(shè)置界面傅是,讓用戶手動(dòng)開啟權(quán)限了。
也就是這幾行代碼蕾羊,上面已經(jīng)貼出喧笔,方便大家參考:
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivityForResult(intent, REQUEST_PERMISSION_SETTING);
·當(dāng)然,你也可以先彈出對(duì)話框詢問用戶是否要授予權(quán)限龟再,然后再跳轉(zhuǎn)书闸。
跳轉(zhuǎn)到設(shè)置界面調(diào)用的是 startActivityForResult() 而不是 startActivity() 。
在 onActivityResult() 中沒有必要對(duì) resultCode 進(jìn)行判斷利凑,因?yàn)橛脩糁荒芡ㄟ^返回鍵才能回到我們的 App 中梗劫,所以 resultCode 總是為 RESULT_CANCEL。
在 onActivityResult() 中還需要對(duì)權(quán)限進(jìn)行判斷截碴,因?yàn)橛脩粲锌赡軟]有授權(quán)就返回了梳侨!