最近幾天測試說我們的app在6.0的機型上經常崩潰,發現是6.0新的運行時權限的問題,這里給大家分享下我們適配6.0運行時權限的方案
6.0之前的權限模型
- 在
AndroidManifest
中聲明可能用到的所有權限 - 用戶在安裝時,系統展示所有權限,用戶安裝即授予所有權限,取消則拒絕安裝
6.0新的運行時權限
- 將權限分為一般權限和危險權限兩種,一般權限跟以前一樣在
AndroidManifest
聲明,危險權限需要開發者在代碼中手動的動態申請 - 動態申請權限,系統彈出對話框,用戶可點擊確定或拒絕,系統提供用戶的選擇回調,從而開發者處理相應的邏輯
- 用戶可以選擇拒絕并不再提醒
運行時權限分類
- 危險權限
危險權限即需要動態申請的權限,一共9組,取得一組中某一個權限的授權,則自動擁有該組的所有授權
身體傳感器
日歷
攝像頭
通訊錄
地理位置
麥克風
電話
短信
存儲空間
具體為:
- 一般權限
除上面的危險權限之外的權限即為一般權限(普通權限)
暫時不做運行時權限適配的方案(不推薦)
如果暫時還不想適配6.0運行時權限,但是又想要app可以在6.0及以上機型運行,那我們可以將目標版本改為23以下,如:targetSdkVersion 22
,這樣做的效果:
- 不管是6.0以下還是以上的機型都可以運行,跟之前一樣權限在安裝時一次性授予
- 由于6.0機型在設置中可以進行權限管理,用戶可以取消該應用的某個權限,但是app并不知道該權限被取消,此時app會崩潰(合理的try可以避免)
適配6.0運行時權限推薦()
ok,這才是這篇文章的重點,前面的都是廢話呢,23333,下面使用開源庫PermissionsDispatcher
來適配運行時權限.
開源庫地址:https://github.com/hotchemi/PermissionsDispatcher
- 效果圖:
1,將targetSdkVersion
升級為23及以上,如targetSdkVersion 23
2,依賴開源庫PermissionsDispatcher
project.gradle
buildscript {
dependencies {
classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
}
}
app.gradle
apply plugin: 'android-apt'
dependencies {
compile 'com.github.hotchemi:permissionsdispatcher:${latest.version}'
apt 'com.github.hotchemi:permissionsdispatcher-processor:${latest.version}'
}
3,在需要運行時權限的Activity或Fragment中動態申請權限
提供四種注解:
@RuntimePermissions 標記需要運行時判斷的類(用于動態生成代理類)
@NeedsPermission 標記需要檢查權限的方法
@OnShowRationale 授權提示回調
@OnPermissionDenied 授權被拒絕回調
@OnNeverAskAgain 授權被拒絕并不再提醒回調
具體代碼如下,關鍵代碼都寫了注釋
@RuntimePermissions
public class PremissionsActivity extends IBaseActivity {
private static final int CALL_CAMERA_REQUEST_CODE = 1;
@Bind(R.id.btn_camera)
Button mBtnCamera;
@Bind(R.id.iv_logo)
CircleImageView mIvLogo;
@Override
protected void init() {
}
@Override
public int initLayout() {
return R.layout.activity_pre;
}
@OnClick(R.id.btn_camera)
public void onClick() {
PremissionsActivityPermissionsDispatcher.callPhoneWithCheck(this);//發起權限申請
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[]
grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
PremissionsActivityPermissionsDispatcher.onRequestPermissionsResult(this, requestCode, grantResults);//將回調交給代理類處理
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode) {
case CALL_CAMERA_REQUEST_CODE:
if (resultCode == Activity.RESULT_OK) {
Bitmap bmap = data.getParcelableExtra("data");
mIvLogo.setImageBitmap(bmap);
IToast.show("頭像保存成功");
}
break;
}
}
@NeedsPermission(Manifest.permission.CAMERA)//權限申請成功
void callPhone() {
Intent openCameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(openCameraIntent, CALL_CAMERA_REQUEST_CODE);
}
@OnShowRationale(Manifest.permission.CAMERA)//申請前告知用戶為什么需要該權限
void showRationaleForCamera(PermissionRequest request) {
showRationaleDialog("使用此功能需要打開照相機的權限", request);
}
@OnPermissionDenied(Manifest.permission.CAMERA)//被拒絕
void onCameraDenied() {
IToast.show("你拒絕了權限,該功能不可用");
}
@OnNeverAskAgain(Manifest.permission.CAMERA)//被拒絕并且勾選了不再提醒
void onCameraNeverAskAgain() {
AskForPermission();
}
/**
* 告知用戶具體需要權限的原因
* @param messageResId
* @param request
*/
private void showRationaleDialog(String messageResId, final PermissionRequest request) {
new AlertDialog.Builder(this)
.setPositiveButton("確定", new DialogInterface.OnClickListener() {
@Override
public void onClick(@NonNull DialogInterface dialog, int which) {
request.proceed();//請求權限
}
})
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(@NonNull DialogInterface dialog, int which) {
request.cancel();
}
})
.setCancelable(false)
.setMessage(messageResId)
.show();
}
/**
* 被拒絕并且不再提醒,提示用戶去設置界面重新打開權限
*/
private void AskForPermission() {
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("當前應用缺少拍照權限,請去設置界面打開\n打開之后按兩次返回鍵可回到該應用哦");
builder.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
}
});
builder.setPositiveButton("設置", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse("package:" + getPackageName())); // 根據包名打開對應的設置界面
startActivity(intent);
}
});
builder.create().show();
}
}
關于上面的代碼:
-
OnShowRationale
注解方法里的代碼不是必須的,可以直接在該位置使用request.proceed()
發起權限申請,我這里申請之前彈出對話框告知用戶具體需要權限的原因,點擊確定再發起權限申請 -
OnNeverAskAgain
注解方法里的代碼也不是必須的,可以簡單的處理,如彈土司告知用戶"需要權限才可以完成拍照".我這里是引導用戶去權限設置界面手動打開拍照權限
經過以上的處理,我們的app可以在6.0以下和6.0及以上的手機上順利運行,并且更加符合動態權限的思想(需要的時候才去申請權限),用戶在設置里取消授權,app也不會崩潰,并且有友好的提示及引導.
關于作者
- 簡 書:uncochen
- github:ChenZhen
- 新浪微博:@Chen丶振
- Email:18620156376@163.com