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

最近幾天測試說我們的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也不會崩潰,并且有友好的提示及引導.

關于作者

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

推薦閱讀更多精彩內容