下载文件以及API23的权限申请
今天实现下载升级功能,遇到了权限申请的问题。我们知道Android6.0以后(API23以上)有些「危险权限」即便写到Manifest里面也是需要手动赋予的。比如WRITE_EXTERNAL_STORAGE
等等,今天在实现下载更新的时候遇到了这个问题,因为DownloadManager.Request
的setDestinationInExternalPublicDir
方法,只支持外部存储,比如把它的参数设置成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这个方法的实现。
但在调试的时候我还没有发现这个方法被调用过。这就要提到申请权限的步骤了:
- 用
checkSelfPermission
方法检查是否已经有权限。
int permission = ContextCompat.checkSelfPermission(RunningContext.sAppContext,
Manifest.permission.WRITE_EXTERNAL_STORAGE);
这个方法在所有手机上首次启动都会返回-1,也就是未授予。。
- 判断是否需要「解释」。
if (ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
很烦,这个方法在所有手机上都返回了false。于是我们根本不需弹出对话框要询问用户是否需要授予权限。
- 请求权限。
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();
}
}
关于下载文件
提到下载文件,有很多种方式,
- 比如用在AsyncTasj中用HttpConnection,然后用 mProgressDialog.setIndeterminate(false);更新进度条;
- Service中开新的线程配合Receiver;
- DownloadManager。
我发现DownloadManager
已经实现得非常好了,它直接帮你在通知栏下载了,而且有进度条。所以,它的源码也是可以读一读的。
Reference:
[1] http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1125/2057.html3