EP27-下载文件以及API23的权限申请

下载文件以及API23的权限申请

今天实现下载升级功能,遇到了权限申请的问题。我们知道Android6.0以后(API23以上)有些「危险权限」即便写到Manifest里面也是需要手动赋予的。比如WRITE_EXTERNAL_STORAGE等等,今天在实现下载更新的时候遇到了这个问题,因为DownloadManager.RequestsetDestinationInExternalPublicDir方法,只支持外部存储,比如把它的参数设置成Environment.DIRECTORY_DOWNLOADS,这样,下载的文件会保存到系统公共的下载区域,是的,你可以使用原生的「下载」APP看到这个路径下下载的内容。

到底怎么申请权限

怎么申请权限?很多网页都介绍得很复杂,而且我发现很多ROM对这个的处理都不尽相同。比如我手上三台6.0手机,同样的代码,有的会弹出申请权限的对话框,有的就不会。

主要是对shouldShowRequestPermissionRationale这个函数的处理,这个函数的意思是,是否要展示申请权限的那个「理由(Rationale)」,比如你做了一个拍照应用,你请求拍照权限就不需要显示理由吧,但是你想记录照片的拍摄位置所以你请求Location权限,这样的话,不懂的兄弟就可能疑惑了,所以你就需要弹一个对话框来解释一下理由你为什么需要定位。但是系统怎么知道你是否需要弹出对话框来向用户解释呢?也就是shouldShowRequestPermissionRationale这个函数什么时候返回true/false?我们发现shouldShowRequestPermissionRationale这个函数最终指向:

//Activity.java
public boolean shouldShowRequestPermissionRationale(@NonNull String permission) {
        return getPackageManager().shouldShowRequestPermissionRationale(permission);
    }

但是我们发现PackageManager是个抽象类:

//PackageManager.java
public abstract boolean shouldShowRequestPermissionRationale(String permission);

getPackageManager指向了:

//ContextWrapper.java
    @Override
    public PackageManager getPackageManager() {
        return mBase.getPackageManager();
    }

也就是说mBase一定实现了shouldShowRequestPermissionRationale这个方法。我又搜到:

Context mBase;  //该属性指向一个ContextIml实例,一般在创建Application、Service、Activity时赋值  

是在是跟不到shouldShowRequestPermissionRationale这个方法的实现。
但在调试的时候我还没有发现这个方法被调用过。这就要提到申请权限的步骤了:

  1. checkSelfPermission方法检查是否已经有权限。
        int permission = ContextCompat.checkSelfPermission(RunningContext.sAppContext,
                Manifest.permission.WRITE_EXTERNAL_STORAGE);

这个方法在所有手机上首次启动都会返回-1,也就是未授予。。

  1. 判断是否需要「解释」。
            if (ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {

很烦,这个方法在所有手机上都返回了false。于是我们根本不需弹出对话框要询问用户是否需要授予权限。

  1. 请求权限。
        ActivityCompat.requestPermissions(activity,
                new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                REQUEST_WRITE_STORAGE);

这个方法,在Nexus 5上执行后会弹出一个系统默认对话框询问是否给权限。HTC,魅族之类的手机上就没弹出,而且执行后直接授予了权限。

我写了个简单的PermissionUtil,Fragment中请求权限之后可以弹出下载对话框:

package com.jd.wallet.indonesia.utils;

import android.Manifest;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.PackageManager;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;

import com.jd.wallet.appframe.RunningContext;
import com.jd.wallet.indonesia.main.ui.home.HomeFragment;

/**
 * Permission Utility.
 * Created by DrunkPiano on 2016/12/28.
 */

public class PermissionUtil {
    public static final int REQUEST_WRITE_STORAGE = 112;

    public static void checkPermissionAndDownload(final Activity activity, final HomeFragment.ShowDialogCallback showDialogCallback) {
        //首次启动APP,checkSelfPermission会返回-1,代表还没有对应权限
        int permission = ContextCompat.checkSelfPermission(RunningContext.sAppContext,
                Manifest.permission.WRITE_EXTERNAL_STORAGE);
        if (permission != PackageManager.PERMISSION_GRANTED) {
            if (ActivityCompat.shouldShowRequestPermissionRationale(activity,
                    Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
                AlertDialog.Builder builder = new AlertDialog.Builder(RunningContext.sAppContext);
                builder.setMessage("Permission to access the SD-CARD is required for this app to Download APK.")
                        .setTitle("Permission required");
                builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {

                    public void onClick(DialogInterface dialog, int id) {
                        makeRequest(activity, showDialogCallback);
                    }
                });
                AlertDialog dialog = builder.create();
                dialog.show();
            } else {
                makeRequest(activity, showDialogCallback);
                return;
            }
        }
        //已经获取权限的话,直接更新
        if (permission == PackageManager.PERMISSION_GRANTED) {
            showDialogCallback.showDialog();
        }
    }

    private static void makeRequest(Activity activity, HomeFragment.ShowDialogCallback showDialogCallback) {
        ActivityCompat.requestPermissions(activity,
                new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                REQUEST_WRITE_STORAGE);
        showDialogCallback.showDialog();
    }
}

关于下载文件

提到下载文件,有很多种方式,

  1. 比如用在AsyncTasj中用HttpConnection,然后用 mProgressDialog.setIndeterminate(false);更新进度条;
  2. Service中开新的线程配合Receiver;
  3. DownloadManager

我发现DownloadManager已经实现得非常好了,它直接帮你在通知栏下载了,而且有进度条。所以,它的源码也是可以读一读的。

Reference:
[1] http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1125/2057.html3

你可能感兴趣的:(EP27-下载文件以及API23的权限申请)