简介
感觉距离上次写文章已经过去很久很久啦,主要是一直没找到什么合适写的内容,而且抽空研究了一下Android以外的技术,写文章的事情就落下了。
最近要做一个app在线更新的功能,本想着自己写会不会有点小麻烦,因为涉及到一些7.0的apk文件下载后的获取,8.0的安装未知应用的权限,通知栏的设置什么的。然后突然发现有自带的DownloadManager可以用,现在想说真香~。
一
8.0安装未知应用权限申请
if (Build.VERSION.SDK_INT >= 26) {
boolean b = getPackageManager().canRequestPackageInstalls();
if (b) {
//下载的具体方法
downloadAPK();
} else {
//申请权限
Uri packageURI = Uri.parse("package:" + getPackageName());
Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, packageURI);
startActivityForResult(intent, REQUEST_CODE_UNKNOWN_APP);
}
} else {
downloadAPK();
}
申请权限之后回调
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
//安装位未知应用的权限申请成功
if (requestCode == REQUEST_CODE_UNKNOWN_APP) {
LogUtil.d(resultCode + "");
downloadAPK();
}
}
就是一个很常规的权限申请,downloadAPK是实际下载的方法,待会介绍。但是写到后面发现DownloadManager已经帮我把这部分的权限做好了,在下载完apk之后会去申请权限。所以第一步相当于没用。。。
二
downloadAPK的实现
private void downloadAPK() {
if (downloadId != 0) {
//防止重复下载
downloadManagerUtil.clearCurrentTask(downloadId);
}
downloadId = downloadManagerUtil.download(mDownloadUrl, "test.apk", "下载完成后,点击安装");
}
主要调用DownloadManager的工具类DownloadManagerUtil
public class DownloadManagerUtil {
private Context mContext;
public DownloadManagerUtil(Context context) {
mContext = context;
}
public long download(String url, String title, String desc) {
ToastUtil.toastShort("开始下载");
Uri uri = Uri.parse(url);
DownloadManager.Request req = new DownloadManager.Request(uri);
//设置WIFI下进行更新,默认是所有网络都支持
// req.setAllowedNetworkTypes(DownloadManager.Request.NETWORK_WIFI);
//下载中和下载完后都显示通知栏
req.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
//使用系统默认的下载路径 此处为应用内 /android/data/packages ,所以兼容7.0
req.setDestinationInExternalPublicDir("test", "test.apk");
//通知栏标题
req.setTitle(title);
//通知栏描述信息
req.setDescription(desc);
//设置类型为.apk
req.setMimeType("application/vnd.android.package-archive");
//获取下载任务ID
DownloadManager dm = (DownloadManager) mContext.getSystemService(Context.DOWNLOAD_SERVICE);
long requestId = dm.enqueue(req);
//查询下载信息
DownloadManager.Query query = new DownloadManager.Query();
query.setFilterById(requestId);
return requestId;
}
/**
* 下载前先移除前一个任务,防止重复下载
*
* @param downloadId
*/
public void clearCurrentTask(long downloadId) {
DownloadManager dm = (DownloadManager) mContext.getSystemService(Context.DOWNLOAD_SERVICE);
try {
dm.remove(downloadId);
} catch (IllegalArgumentException ex) {
ex.printStackTrace();
}
}
}
主要使用的DownloadManager.Request去设置一些下载的属性,主要的注释都标识出来了,比较要注意的问题是这个路径req.setDestinationInExternalPublicDir("test", "test.apk");待会用FileProvider去获取路径的时候,要对应上这个路径,切记切记!要不然会提示软件安装包解析错误,因为路径没对上,FileProvider没找到对应的路径下的apk。所以就会提示解析错误,我在这个问题上弄了很久。
三
在使用DownloadManager下载完成之后,会发送一个广播,所以需要注册一个广播接收。
public class DownloadApkReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {
//下载完成
long downloadId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, -1L);
// 根据获取到的ID,使用上面第3步的方法查询是否下载成功
DownloadManager manager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
DownloadManager.Query query = new DownloadManager.Query();
query.setFilterById(downloadId);
Cursor cursor = manager.query(query);
if (!cursor.moveToFirst()) {
cursor.close();
return;
}
int status = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS));
String localFilename;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
localFilename = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI));
} else {
localFilename = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_FILENAME));
}
if (status == DownloadManager.STATUS_SUCCESSFUL) {
// setPermission(localFilename);
installApk(context, localFilename);
}
} else if (intent.getAction().equals(DownloadManager.ACTION_NOTIFICATION_CLICKED)) {
//在下载过程中点击
Intent viewDownloadIntent = new Intent(DownloadManager.ACTION_VIEW_DOWNLOADS);
viewDownloadIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(viewDownloadIntent);
}
}
private void installApk(Context context, String fileName) {
File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath(), "test/test.apk");
Intent install = new Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
if (Build.VERSION.SDK_INT >= 24) {
//判读版本是否在7.0以上,7.0以上获取下载的apk文件要用FileProvider
Uri apkUri = FileProvider.getUriForFile(context.getApplicationContext(),
"你的FileProvider的authority", file);
install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件
install.setDataAndType(apkUri, "application/vnd.android.package-archive");
} else {
install.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
}
context.startActivity(install);
}
}
广播的创建就不介绍了。不过要注意的是清单文件里面的intent-filter。要写成下面这样子才可以
DownloadManager.ACTION_DOWNLOAD_COMPLETE和DownloadManager.ACTION_NOTIFICATION_CLICKED这两个action分别对应不同的事件。
在installApk这个方法里面,涉及到7.0的FileProvider权限适配,FileProvider详细用法就不介绍了,百度一下一大堆,值得一提的是File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath(), "test/test.apk");在第二个参数里面写的"test/test.apk"对应req.setDestinationInExternalPublicDir("test", "test.apk")这里的两个参数,一定要对应上,要不然找不到apk会报解析软件安装包错误安装失败的情况。
四
最后既然有广播,肯定要有注册的地方,
private void regist() {
receiver = new DownloadApkReceiver();
//register download success broadcast
registerReceiver(receiver, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));
}
最终效果图
效果还是挺好的,比自己写notification,service这些要简洁不少。