Permission(使用与框架使用)

目录
   一、引言
   二、效果
   三、Normal Permission和Dangerous Permission
   四、实现支持运行时权限
   五、permissionsdispatcher框架的使用
一、引言

  在Android 6.0以前(targetSdkVersion 23)以前,我们安装APP时会列出所有该APP在Manifest中列出的访问权限,而且只会出现一次,一旦我们同意安装此APP并同意所有权限,APP就可以在用户不知情的情况下访问手机中的数据,包括(通讯录、定位)等隐私,在6.0以后,APP在运行时会一个一个询问用户授予权限。


二、效果

正常允许权限:
在这里插入图片描述

拒绝允许权限的处理
在这里插入图片描述


三、Normal Permission和Dangerous Permission

Google给出的关于权限分类的官方文档,
https://developer.android.google.cn/guide/topics/security/permissions.html#normal-dangerous

Dangero Permission以分组形式给出
在这里插入图片描述

在Android6.0(API 23) 到8.0(API 26)之间,同一组的任何权限被授权了,其他权限也自动授权,此外对于申请时弹出的提示框上面的文本说明也是对整个权限组的说明,而不是单个权限的说明。但是Android 8.0(APi 26)以后对于危险权限需要一个一个的申请。


四、实现支持运行时权限

1、首先是我们拨打电话的点击事件:

 dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                call();
            }
        });

2、这时进行权限的检查,是否拥有拨打电话的权限,如果有则进行拨号,如果没有进行权限的申请:

 private void call() {
        //检查APP是否有权限
        if(ActivityCompat.checkSelfPermission(this,Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED){
            //如果没有
            ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CALL_PHONE},PERMISSION_REQURST_CALL_PHONE);
        }else{
            callPhone();
        }
    }
    
    /**
     * 拨打电话
     */
    private void callPhone() {
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:"+ 10086);
        intent.setData(data);
        startActivity(intent);
    }

3、权限申请回调的处理:

 @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if(requestCode == PERMISSION_REQURST_CALL_PHONE){
            if(grantResults[0] != 0  && grantResults[0] == PackageManager.PERMISSION_GRANTED){
            //允许申请的权限
                callPhone();
            }else{
            //申请权限被拒绝
            Toast.makeText(this,"申请权限被拒绝",Toast.SHORTTIME).show();
                
            }
            return;
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }

4、处理“不再询问”选项
  在申请权限时,如果我们点击“拒绝”,则下一次还会继续弹出权限申请框,只不过这一次会多出来一项“不再询问”,如果我们勾选了这个选项,则下一次不会再弹出此权限申请框,用户需要去设置—》应用权限列表中手动打开权限。

在这里插入图片描述

用户对于已经勾选“不再询问”的权限,每次调用该权限API时都会失效,用户无法使用该功能,这显然不会带来好的用户体验,这个时候我们需要使用shouldShowRequestPermissionRationale()方法向用户解释权限使用情况,并设置进入应用的设置权限所在:
在这里插入图片描述

 @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        if(requestCode == PERMISSION_REQURST_CALL_PHONE){
            if(grantResults[0] != 0  && grantResults[0] == PackageManager.PERMISSION_GRANTED){
                callPhone();
            }else{
                if(!ActivityCompat.shouldShowRequestPermissionRationale(MainActivity.this,Manifest.permission.CALL_PHONE)){
                    final AlertDialog dialog = new AlertDialog.Builder(MainActivity.this)
                            .setTitle("提示")
                            .setMessage("该功能需要访问电话的权限,否则无法使用")
                            .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                                @Override
                                public void onClick(DialogInterface dialogInterface, int i) {
                                    dialogInterface.dismiss();
                                }
                            })
                            .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                                @Override
                                public void onClick(DialogInterface dialogInterface, int i) {
                                    //跳转到当前APP的设置页面
                                    Intent intent = new Intent();
                                    intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                                    intent.addCategory(Intent.CATEGORY_DEFAULT);
                                    intent.setData(Uri.parse("package:" + getPackageName()));
                                    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                                    intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                                    intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                                    startActivity(intent);
                                }
                            }).create();
                    dialog.show();
                }
            }
            return;
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }
五、permissionsdispatcher框架的使用

1、依赖
在Project的build.gradle 中添加如下代码:

dependencies {
        ...
        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
     
    }

在app的build.gradle中添加依赖:

dependencies {
    implementation 'com.github.hotchemi:permissionsdispatcher:2.1.3'
    annotationProcessor 'com.github.hotchemi:permissionsdispatcher-processor:2.1.3'
}

permissionsdispatcher以注释的方式进行权限的处理,其中几个重要的注释:

注释名称注释解释
RuntimePermissions必要的注释,用来注册一个Activity或者一个Fragment,使他们可以处理权限
NeedsPermission必要的注释,在需要获取权限的地方进行注释,用来获取权限
OnShowRationale提示用户为何需要开启此权限,在用户选择拒绝后,再次需要访问该权限时调用
OnPermissionDenied用户选择拒绝时的提示
OnNeverAskAgain用户选择不再询问后的提示

代码如下:


import android.Manifest;
import android.content.DialogInterface;
import android.content.Intent;
import android.net.Uri;
import android.provider.Settings;
import android.support.annotation.NonNull;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import permissions.dispatcher.NeedsPermission;
import permissions.dispatcher.OnNeverAskAgain;
import permissions.dispatcher.OnPermissionDenied;
import permissions.dispatcher.OnShowRationale;
import permissions.dispatcher.PermissionRequest;
import permissions.dispatcher.RuntimePermissions;

@RuntimePermissions
public class SecondActivity extends AppCompatActivity {

    private Button dialBtn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        initView();
    }

    private void initView() {
        dialBtn = (Button) findViewById(R.id.dial_btn);
        dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
              
            }
        });
    }

    @NeedsPermission(Manifest.permission.CALL_PHONE)
    //在需要获取权限的地方注释
    void call(){
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:" + 10086);
        intent.setData(data);
        startActivity(intent);
    }

    @OnShowRationale(Manifest.permission.CALL_PHONE)
    //提示用户为何要开启权限
    void showWhy(final PermissionRequest request){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("提示用户为何要开启权限")
                .setPositiveButton("知道了", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        request.proceed(); //再次执行权限请求
                    }
                })
                .show();
    }

    @OnPermissionDenied(Manifest.permission.CALL_PHONE)
    void showDenied(){
        Toast.makeText(SecondActivity.this,"用户选择拒绝时的提示",Toast.LENGTH_SHORT).show();
    }

    @OnNeverAskAgain(Manifest.permission.CALL_PHONE)
    void showNotAsk(){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("该功能需要访问电话的权限,不开启将无法使用该功能")
                .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        //跳转到当前APP的设置页面
                        Intent intent = new Intent();
                        intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                        intent.addCategory(Intent.CATEGORY_DEFAULT);
                        intent.setData(Uri.parse("package:" + getPackageName()));
                        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                        intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                        intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                        startActivity(intent);
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        dialogInterface.dismiss();
                    }
                })
                .show();
    }

}

这个时候我们编译程序,会生成一个辅助类SecondActivityPermissionsDispatcher,下面的事情由它进行解决,完整代码:

@RuntimePermissions
public class SecondActivity extends AppCompatActivity {

    private Button dialBtn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        initView();
    }

    private void initView() {
        dialBtn = (Button) findViewById(R.id.dial_btn);
        dialBtn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
           //*****************重点*****************
           SecondActivityPermissionsDispatcher.callWithCheck(SecondActivity.this);
            }
        });
    }

    @NeedsPermission(Manifest.permission.CALL_PHONE)
    //在需要获取权限的地方注释
    void call(){
        Intent intent = new Intent(Intent.ACTION_CALL);
        Uri data = Uri.parse("tel:" + 10086);
        intent.setData(data);
        startActivity(intent);
    }

    @OnShowRationale(Manifest.permission.CALL_PHONE)
    //提示用户为何要开启权限
    void showWhy(final PermissionRequest request){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("提示用户为何要开启权限")
                .setPositiveButton("知道了", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        request.proceed(); //再次执行权限请求
                    }
                })
                .show();
    }

    @OnPermissionDenied(Manifest.permission.CALL_PHONE)
    void showDenied(){
        Toast.makeText(SecondActivity.this,"用户选择拒绝时的提示",Toast.LENGTH_SHORT).show();
    }

    @OnNeverAskAgain(Manifest.permission.CALL_PHONE)
    void showNotAsk(){
        new AlertDialog.Builder(SecondActivity.this)
                .setMessage("该功能需要访问电话的权限,不开启将无法使用该功能")
                .setPositiveButton("前去设置", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        //跳转到当前APP的设置页面
                        Intent intent = new Intent();
                        intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
                        intent.addCategory(Intent.CATEGORY_DEFAULT);
                        intent.setData(Uri.parse("package:" + getPackageName()));
                        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                        intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                        intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
                        startActivity(intent);
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        dialogInterface.dismiss();
                    }
                })
                .show();
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
            //*****************重点*****************
            SecondActivityPermissionsDispatcher.onRequestPermissionsResult(SecondActivity.this,requestCode,grantResults);
    }
}

OK ,这是权限里面的内容了,希望能帮到你

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
FastBLE是一个基于Android的蓝牙BLE库,使用它可以简化开发蓝牙BLE相关的应用程序。下面是使用FastBLE进行蓝牙BLE开发的步骤: 1. 在Android Studio中创建一个新的项目,并在build.gradle文件中添加以下依赖: ```groovy dependencies { implementation 'com.polidea.rxandroidble:rxandroidble:1.11.1' } ``` 2. 打开AndroidManifest.xml文件,在文件中添加以下权限: ```xml <uses-permission android:name="android.permission.BLUETOOTH"/> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/> ``` 3. 在你的Activity中创建一个RxBluetooth对象: ```java RxBluetooth rxBluetooth = new RxBluetooth(this); ``` 4. 启动蓝牙: ```java rxBluetooth.enableBluetoothIfDisabled() .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Action() { @Override public void run() throws Exception { // Bluetooth enabled successfully } }, new Consumer<Throwable>() { @Override public void accept(Throwable throwable) throws Exception { // An error occurred while enabling Bluetooth } }); ``` 5. 扫描蓝牙设备: ```java rxBluetooth.scanBleDevices() .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Consumer<ScanResult>() { @Override public void accept(ScanResult scanResult) throws Exception { // A new BLE device has been found } }, new Consumer<Throwable>() { @Override public void accept(Throwable throwable) throws Exception { // An error occurred while scanning for BLE devices } }); ``` 6. 连接蓝牙设备: ```java rxBluetooth .connectBleDevice(device, autoConnect) .flatMapSingle(rxBleConnection -> rxBleConnection.readCharacteristic(characteristicUuid)) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Consumer<byte[]>() { @Override public void accept(byte[] bytes) throws Exception { // Characteristic value has been successfully read } }, new Consumer<Throwable>() { @Override public void accept(Throwable throwable) throws Exception { // An error occurred while reading the characteristic value } }); ``` 7. 断开蓝牙设备: ```java rxBluetooth .disconnectBleDevice(device) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Action() { @Override public void run() throws Exception { // Device disconnected successfully } }, new Consumer<Throwable>() { @Override public void accept(Throwable throwable) throws Exception { // An error occurred while disconnecting the device } }); ``` 以上就是使用FastBLE框架进行蓝牙BLE开发的基本步骤。具体操作可以参考官方文档:https://github.com/Polidea/RxAndroidBle。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值