Android6.0運行時權限解決方案

vn167544 8年前發布 | 13K 次閱讀 開源 安卓開發 Android開發 移動開發

最近幾天測試說我們的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 來適配運行時權限.

  • 效果圖:

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也不會崩潰,并且有友好的提示及引導.

 

來自:http://www.jianshu.com/p/d4a9855e92d3

 

 本文由用戶 vn167544 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!