Android 6.0动态权限及跳转GPS设置界面的方法

1.动态权限申请

模糊的位置信息android.permission.ACCESS_COARSE_LOCATION权限为例

在AndroidManifest文件中加入权限

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

然后java代码中动态申请

//动态申请权限的测试方法
public void test() {
 // 要申请的权限 数组 可以同时申请多个权限
 String[] permissions = {Manifest.permission.ACCESS_COARSE_LOCATION};

 if (Build.VERSION.SDK_INT >= 23) {
  //如果超过6.0才需要动态权限,否则不需要动态权限
  //如果同时申请多个权限,可以for循环遍历
  int check = ContextCompat.checkSelfPermission(this,permissions[0]);
  // 权限是否已经 授权 GRANTED---授权 DINIED---拒绝
  if (check == PackageManager.PERMISSION_GRANTED) {
   //写入你需要权限才能使用的方法
   run();
  } else {
   //手动去请求用户打开权限(可以在数组中添加多个权限) 1 为请求码 一般设置为final静态变量
   requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, 1);
  }
 } else {
  //写入你需要权限才能使用的方法
  run();
 }
}

其中run()为你自己需要权限才能执行的方法

然后重写申请权限的回掉方法

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,
          @NonNull int[] grantResults) {
 super.onRequestPermissionsResult(requestCode, permissions, grantResults);
 //回调,判断用户到底点击是还是否。
 //如果同时申请多个权限,可以for循环遍历
 if (requestCode == 1 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
  //写入你需要权限才能使用的方法
  run();
 } else {
  // 没有获取 到权限,从新请求,或者关闭app
  Toast.makeText(this,"需要获得XXX权限",Toast.LENGTH_SHORT).show();
 }
}

2.跳转GPS设置界面

相关字符串

<string name="notifyTitle">提示</string>
<string name="notifyMsg">当前应用缺少必要权限。\n\n请点击\"设置\"-\"权限\"-打开所需权限。</string>
<string name="gpsNotifyMsg">当前应用需要打开定位功能。\n\n请点击\"设置\"-\"定位服务\"-打开定位功能。</string>
<string name="setting">设置</string>
<string name="cancel">取消</string>

java代码

private int GPS_REQUEST_CODE = 10;

/**
 * 检测GPS是否打开
 *
 * @return
 */
private boolean checkGPSIsOpen() {
 boolean isOpen;
 LocationManager locationManager = (LocationManager) this
   .getSystemService(Context.LOCATION_SERVICE);
 isOpen = locationManager.isProviderEnabled(android.location.LocationManager.GPS_PROVIDER);
 return isOpen;
}

/**
 * 跳转GPS设置
 */
private void openGPSSettings() {
 if (checkGPSIsOpen()) {
  initLocation(); //自己写的定位方法
 } else {
  //没有打开则弹出对话框
  new AlertDialog.Builder(this)
    .setTitle(R.string.notifyTitle)
    .setMessage(R.string.gpsNotifyMsg)
    // 拒绝, 退出应用
    .setNegativeButton(R.string.cancel,
      new DialogInterface.OnClickListener() {
       @Override
       public void onClick(DialogInterface dialog, int which) {
        finish();
       }
      })

    .setPositiveButton(R.string.setting,
      new DialogInterface.OnClickListener() {
       @Override
       public void onClick(DialogInterface dialog, int which) {
        //跳转GPS设置界面
        Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
        startActivityForResult(intent, GPS_REQUEST_CODE);
       }
      })

    .setCancelable(false)
    .show();

 }
}

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
 super.onActivityResult(requestCode, resultCode, data);
 if (requestCode == GPS_REQUEST_CODE) {
  //做需要做的事情,比如再次检测是否打开GPS了 或者定位
  openGPSSettings();
 }
}

最后在需要的地方调用 openGPSSettings()方法。

效果(是在相关权限是已经申请好的情况下的效果)

3.下面是我在高德地图Demo中看见的检测权限的Activity

public class CheckPermissionsActivity extends Activity implements
  ActivityCompat.OnRequestPermissionsResultCallback {
 /**
  * 需要进行检测的权限数组
  */
 protected String[] needPermissions = {
   Manifest.permission.ACCESS_COARSE_LOCATION,
   Manifest.permission.ACCESS_FINE_LOCATION,
   Manifest.permission.WRITE_EXTERNAL_STORAGE,
   Manifest.permission.READ_EXTERNAL_STORAGE,
   Manifest.permission.READ_PHONE_STATE
 };

 private static final int PERMISSON_REQUESTCODE = 0;
 private static final int SETTING_REQUESTCODE = 1;

 /**
  * 判断是否需要检测,防止不停的弹框
  */
 private boolean isNeedCheck = true;

 @Override
 protected void onResume() {
  super.onResume();
  if (isNeedCheck) {
   checkPermissions(needPermissions);
  }
 }

 /**
  * @param permissions
  * @since 2.5.0
  */
 private void checkPermissions(String... permissions) {
  List<String> needRequestPermissonList = findDeniedPermissions(permissions);
  if (null != needRequestPermissonList
    && needRequestPermissonList.size() > 0) {
   ActivityCompat.requestPermissions(this,
     needRequestPermissonList.toArray(
       new String[needRequestPermissonList.size()]),
     PERMISSON_REQUESTCODE);
  }
 }

 /**
  * 获取权限集中需要申请权限的列表
  *
  * @param permissions
  * @return
  * @since 2.5.0
  */
 private List<String> findDeniedPermissions(String[] permissions) {
  List<String> needRequestPermissonList = new ArrayList<String>();
  for (String perm : permissions) {
   if (ContextCompat.checkSelfPermission(this,
     perm) != PackageManager.PERMISSION_GRANTED
     || ActivityCompat.shouldShowRequestPermissionRationale(
     this, perm)) {
    needRequestPermissonList.add(perm);
   }
  }
  return needRequestPermissonList;
 }

 /**
  * 检测是否说有的权限都已经授权
  *
  * @param grantResults
  * @return
  * @since 2.5.0
  */
 private boolean verifyPermissions(int[] grantResults) {
  for (int result : grantResults) {
   if (result != PackageManager.PERMISSION_GRANTED) {
    return false;
   }
  }
  return true;
 }

 @Override
 public void onRequestPermissionsResult(int requestCode,
           String[] permissions, int[] paramArrayOfInt) {
  if (requestCode == PERMISSON_REQUESTCODE) {
   if (!verifyPermissions(paramArrayOfInt)) {
    showMissingPermissionDialog();
    isNeedCheck = false;
   }
  }
 }

 /**
  * 显示提示信息
  *
  * @since 2.5.0
  */
 private void showMissingPermissionDialog() {
  AlertDialog.Builder builder = new AlertDialog.Builder(this);
  builder.setTitle(R.string.notifyTitle);
  builder.setMessage(R.string.notifyMsg);

  // 拒绝, 退出应用
  builder.setNegativeButton(R.string.cancel,
    new DialogInterface.OnClickListener() {
     @Override
     public void onClick(DialogInterface dialog, int which) {
      finish();
     }
    });

  builder.setPositiveButton(R.string.setting,
    new DialogInterface.OnClickListener() {
     @Override
     public void onClick(DialogInterface dialog, int which) {
      startAppSettings();
     }
    });

  builder.setCancelable(false);

  builder.show();
 }

 /**
  * 启动应用的设置
  *
  * @since 2.5.0
  */
 private void startAppSettings() {
  Intent intent = new Intent(
    Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
  intent.setData(Uri.parse("package:" + getPackageName()));
  startActivityForResult(intent, SETTING_REQUESTCODE);
 }

 @Override
 public boolean onKeyDown(int keyCode, KeyEvent event) {
  if (keyCode == KeyEvent.KEYCODE_BACK) {
   this.finish();
   return true;
  }
  return super.onKeyDown(keyCode, event);
 }

 @Override
 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  super.onActivityResult(requestCode, resultCode, data);
  if (requestCode == SETTING_REQUESTCODE) {
   checkPermissions(needPermissions);
  }
 }
}

继承于CheckPermissionsActivity即可

不允许就跳转系统设置界面,若没设置再次检测权限并申请,直到允许为止。可和GPS设置搭配使用

以上这篇Android 6.0动态权限及跳转GPS设置界面的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Android 中使用 dlib+opencv 实现动态人脸检测功能

    1 概述 完成 Android 相机预览功能以后,在此基础上我使用 dlib 与 opencv 库做了一个关于人脸检测的 demo.该 demo 在相机预览过程中对人脸进行实时检测,并将检测到的人脸用矩形框描绘出来.具体实现原理如下: 采用双层 View,底层的 TextureView 用于预览,程序从 TextureView 中获取预览帧数据,然后调用 dlib 库对帧数据进行处理,最后将检测结果绘制在顶层的 SurfaceView 中. 2 项目配置 由于项目中用到了 dlib 与 open

  • Android自定义动态壁纸开发详解

    看到有些手机酷炫的动态壁纸,有没有好奇过他们是如何实现的,其实我们自己也可以实现. 一.动态壁纸原理 如果你了解使用过SurfaceView的话,那么开发一款动态壁纸对你来说其实非常简单. 动态壁纸的本质其实就是一个服务在维护一个动态壁纸引擎Engine,所以我们看到的动态效果其实是通过这个引擎画出来的.而维护这个引擎的服务,就是WallpaperService.本篇文章并不讨论内部实现原理,只是让大家知道如何去实现动态壁纸,所以就不详细说了. 二.实现动态壁纸 大体上可分为三个步骤: 创建自定

  • Android底部导航栏的动态替换方案

    Android底部导航栏的动态替换方案,供大家参考,具体内容如下 1.通常来说,一般情况下,我们的app的BottomTab会有下面几种实现方式. 1).自定义view,然后自己写逻辑去实现互斥. 2).使用RadioGroup+RadioButton去实现底部的Tab. 自由度比极高,如果想实现搞复杂度的话可以重写 RadioButton. 3).使用google design包里面的 TabLayout去实现. 可上.可下.可以滑动 偷懒的话可以根据已有api来设置一些资源,也可以 setC

  • Android自定义动态壁纸开发(时钟)

    看到有些手机酷炫的动态壁纸,有没有好奇过他们是如何实现的,其实我们自己也可以实现. 先看效果 上图是动态壁纸钟的一个时钟. 我们先来看看 Livewallpaper(即动态墙纸)的实现,Android的动态墙纸并不是GIF图片,而是一个标准的Android应用程序,也就是APK.既然是应用程序,当然意味着天生具有GIF图片不具备的功能--能与用户发生交互,而且动态的背景变化绝不仅仅局限于GIF图片那般只能是固定的几张图片的循环播放.但是我们在这里没有加入与用户交互的动作,只是加入一个时钟(当然时

  • Android绘制动态折线图

    所谓动态折线图,就是折线图能随着手指的滑动进行动态绘制,这里很定会产生动画效果.基于这个效果,这里使用SurfaceView进行制图. 实现步奏如下: (1): 这里新建一个绘图ChartView,继承SurfaceView并实现SurfaceHolder.Callback , Runnable接口,主要绘图工作在子线程中完成. (2):现实 SurfaceHolder.Callback接口的三个方法,并在 surfaceCreated中开启子线程进行绘图. (3):重写onTouchEvent

  • Android实现动态添加标签及其点击事件

    在做Android开发的时候,会遇到动态添加标签让用户选择的功能,所以自己写了个例子,运行效果图如下. 标签可以左右滑动进行选择,点击的时候,会弹出toast提示选择或者取消选择了哪个标签.通过动态添加TextView作为标签,并给TextView设置背景,通过selector选择器改变其背景颜色,来确定是否处于选中状态. 代码如下所示: 1.标签的布局文件,我在标签里只设置了一个TextView <?xml version="1.0" encoding="utf-8&

  • Android动态修改应用图标与名称的方法实例

    遇到的坑 这里我把做这个功能中遇到的一些问题写在前面,是为了大家能先了解有什么问题存在,遇到这些问题的时候就不慌了,这里我把应用图标和名称先统一使用icon代替进行说明. 1.动态替换icon,只能替换内置的icon,无法从服务器端获取来更新icon: 2.动态替换icon以后,应用内更新的时候必须要切换到原始icon),否则可能导致更新安装失败(AS上表现为adb运行会失败),或者升级后应用图标出现多个甚至应用图标都不显示的情况(这些问题都可以通过下面我推荐的开发规则解决掉,所以这是一个坑点,

  • Android如何动态调整应用字体大小详解

    前言 为什么要动态设置字体大小?由于项目面对的是中老年客户项目中自带的字体无法满足客户需求. Android应用字体大小默认随系统设置的字体大小而变化,但您可能不希望您的应用字体大小随系统设置变化,想要自己控制,例如微信.本文简单介绍一下如何实现应用字体大小动态调整而不是依赖系统设置 字体大小变化是由android.content.res.Configuration.class类中的fontScale控制的,因此,若想我们的应用字体大小变化不随系统变化而是由我们自主控制,就需要我们修改fontS

  • android动态壁纸调用的简单实例

    调用后动态壁纸其实是显示在Activity的后面,而Activity则是透明显示,这样就可以看到下面的动态壁纸,如果Activity不是透明的则什么也看不到. 代码中有用到两个接口 IWallpaperService mService; IWallpaperEngine mEngine; 我们可以看到该目录下面有三个aidl接口,分别是 复制代码 代码如下: interface IWallpaperConnection { void attachEngine(IWallpaperEngine e

  • Android编程之动态壁纸实例分析

    本文实例讲述了Android编程之动态壁纸.分享给大家供大家参考,具体如下: 从android 2.1版本起引入了动态壁纸的概念,熟悉android的人一定不会陌生.这里解释一个动态壁纸是怎么形成又是怎么工作的. 首先动态桌面的动态体现出这个组件是实时变化的,也就是说有一个后台在不停的刷新这个组件.联想到后台组件首先想到的就是service,从代码角度看,果然如此.每一个动态桌面都继承自WallpaperService,其中必须实现的抽象方法onCreateEngine,返回一个Engine对象

随机推荐