-----------------------------------uni-app--------------------------------------
checkAppVersion
// 版本更新
export function checkAppVersion(_t, isShowWarn = false) {
// #ifdef APP-PLUS
// 若是正在更新不执行
if(_t.isShowModal || _t.isUpdate) return;
const os_type = plus.os.name.toLowerCase();
const app_version = plus.runtime.version;
plus.runtime.getProperty(plus.runtime.appid, function(info) {
uni.setStorageSync('VERSION', info.version);
const res_version = info.version;
const requestData = {
osType: os_type,
appVersion: app_version,
resVersion: res_version
}
_t.$api.getAppVersion(requestData).then(res => {
Log.d("版本信息:" + JSON.stringify(res));
if (res.statusCode != 200 || !res.data.success) return;
const data = res.data.result;
if (data.isUpdate === 0) return;
const app_version_db = data.appVersion;
const res_version_db = data.resVersion;
const app_url = data.appUrl;
const res_url = data.resUrl;
const app_version_desc = data.content;
//大版本更新(涉及安卓壳子或壳子改动)
if (app_version_db !== app_version && (parseInt(app_version.replace(/\./g, "")) <
parseInt(app_version_db.replace(/\./g, "")))) {
// 定义安卓apk存放路径
const filename = "_downloads/park.apk";
// 处理安装提示在本应用上方弹出导致下载弹窗重复弹出问题
nativeJs("bridge", "getFileApkVersion", {
path: plus.io.convertLocalFileSystemURL(filename)
}, originAppVersion => {
Log.d("原来存在的apk包版本:" + originAppVersion)
if(parseInt(originAppVersion.replace(/\./g, "")) === parseInt(app_version_db.replace(/\./g, ""))) {
// 如果原来存在的apk包版本比app版本高直接提示安装
if(parseInt(originAppVersion.replace(/\./g, "")) > parseInt(app_version.replace(/\./g, ""))) {
// 决定是否弹出安装,解决部分机型在应用界面弹出安装提示而重复提示问题
nativeJs("bridge", "getFileCreatedTime", {
path: plus.io.convertLocalFileSystemURL(filename)
}, createdTime => {
if(createdTime) {
Log.d("现在时间" + new Date().getTime())
Log.d("文件创建时间" + parseInt(createdTime))
Log.d("现在时间与文件创建时间差" + (new Date().getTime() - parseInt(createdTime)))
// 超过1分钟提示安装
if(new Date().getTime() - parseInt(createdTime) > 60 * 1 * 1000) {
_t.isShowModal = true;
uni.showModal({ //提醒用户更新
title: "版本更新",
content: "是否立即更新",
showCancel: false,
complete: () => {
_t.isShowModal = false;
},
success: res => {
if(res.confirm) {
plus.runtime.install(plus.io.convertLocalFileSystemURL(filename), //安装APP
{
force: true
},
function() {
Log.d('安装旧包准备成功');
},
function() {
Log.d('安装旧包准备失败');
});
}
}
})
}
}
})
}else {
// 删除原来安装包缩小体积
nativeJs("bridge", "deleteFile", {
path: plus.io.convertLocalFileSystemURL(filename)
});
}
return;
}
_t.isShowModal = true;
uni.showModal({ //提醒用户更新
title: "版本更新",
content: app_version_desc,
showCancel: false,
complete: () => {
_t.isShowModal = false;
},
success: res => {
if(res.confirm) {
// plus.downloader.clear();
Log.d("全量更新地址:" + app_url)
// 清除下载任务
if(_t.dtask) _t.dtask.abort();
// 下载前清除以前下载的缓存,避免apk包累积应用变大
nativeJs("bridge", "deleteFile", {
path: plus.io.convertLocalFileSystemURL(filename)
});
_t.dtask = plus.downloader.createDownload(app_url, {
method: "GET",
filename //利用保存路径,实现下载文件的重命名
},
function(d, status){
Log.d("createDownloaded下载完成" + JSON.stringify(d))
if(status !== 200) {
_t.isUpdate = false;
// 清除下载任务
_t.dtask.abort();
}
});
_t.dtask.addEventListener("statechanged", function(task, status) {
switch(task.state) {
case 1: // 开始
break;
case 2: //已连接到服务器
break;
case 3: // 已接收到数据
const current = parseInt(100 * task.downloadedSize / task.totalSize);
nativeJs("notification", "setProgress", {
current
});
break;
case 4: // 下载完成
nativeJs("notification", "compProgressNotification", {
title: '下载完成'
});
plus.runtime.install(plus.io.convertLocalFileSystemURL(task.filename), //安装APP
{
force: true
},
function() {
Log.d('安装准备成功');
_t.isUpdate = false;
},
function() {
Log.d('安装准备失败');
_t.isUpdate = false;
});
break;
}
});
nativeJs("notification", "init");
_t.isUpdate = true;
_t.dtask.start();
}
}
})
});
} else {
//小版本更新(uniapp中的改动)
if (res_version_db !== res_version && (parseInt(res_version.replace(/\./g, "")) <
parseInt(res_version_db.replace(/\./g, "")))) {
uni.showLoading({
title: '更新中...',
mask: true
});
Log.d("热更新地址:" + res_url)
_t.isUpdate = true;
uni.downloadFile({
url: res_url,
success: (downloadResult) => {
Log.d("补丁信息:" + JSON.stringify(downloadResult))
if (downloadResult.statusCode === 200) {
plus.runtime.install(downloadResult.tempFilePath, {
force: true
}, function() {
Log.d('install success...');
uni.hideLoading();
_t.isUpdate = false;
plus.runtime.restart();
// 删除安装包
nativeJs("bridge", "deleteFile", {
path: plus.io.convertLocalFileSystemURL(downloadResult.tempFilePath)
});
}, function(e) {
uni.hideLoading();
_t.isUpdate = false;
Log.d('install fail...' + JSON.stringify(e));
// 删除安装包
nativeJs("bridge", "deleteFile", {
path: plus.io.convertLocalFileSystemURL(downloadResult.tempFilePath)
});
});
} else {
uni.hideLoading();
_t.isUpdate = false;
}
},
fail() {
uni.hideLoading();
_t.isUpdate = false;
}
});
}
if (isShowWarn && res_version_db === res_version && app_version_db ===
app_version) {
uni.showToast({
title: '已是最新版本',
icon: 'none',
duration: 1000,
mask: false
});
}
if (isShowWarn && (parseInt(res_version.replace(/\./g, "")) > parseInt(
res_version_db.replace(/\./g, "")) || parseInt(app_version.replace(
/\./g, "")) > parseInt(app_version_db.replace(/\./g, "")))) {
uni.showToast({
title: '暂无新版本',
icon: 'none',
duration: 1000,
mask: false
});
}
}
})
});
// #endif
}
nativeJs
// 通过uniapp自带一套执行
export function nativeJs(targetClass, method, params = {}, successCallback, errorCallback) {
// #ifdef APP-PLUS
return plus.bridge.exec(targetClass, method, [plus.bridge.callbackId(typeof successCallback ===
'function' ? successCallback : null, typeof errorCallback === 'function' ? errorCallback :
null), JSON.stringify(params)]);
// #endif
}
-----------------------------------android--------------------------------------
BridgeForUniApp
package com.park;
import android.content.Intent;
import android.util.Log;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.park.util.AppUtil;
import org.json.JSONArray;
import java.io.File;
import io.dcloud.common.DHInterface.IWebview;
import io.dcloud.common.DHInterface.StandardFeature;
import io.dcloud.common.util.JSUtil;
public class BridgeForUniApp extends StandardFeature {
/**
* h5调此跳转
* @param pWebView
* @param arr
*/
public void goTargetActivity(IWebview pWebView, JSONArray arr){
Intent intent = new Intent(pWebView.getContext(), RawActivity.class);
intent.putExtra("CallBackID", arr.optString(0));
JSONObject params = JSON.parseObject(arr.optString(1));
intent.putExtra("access_token", params.getString("token"));
Log.d("params", params.toJSONString());
pWebView.getContext().startActivity(intent);
}
/**
* 删除文件
* @param pWebView
* @param arr
* @return
*/
public Boolean deleteFile(IWebview pWebView, JSONArray arr){
JSONObject params = JSON.parseObject(arr.optString(1));
File file = new File(params.getString("path"));
if(file.exists()) file.delete();
Log.d("h5","删除文件成功" + params.getString("path"));
return true;
}
/**
* 获取apk版本号
* @param pWebView
* @param arr
* @return
*/
public void getFileApkVersion(IWebview pWebView, JSONArray arr){
JSONObject params = JSON.parseObject(arr.optString(1));
JSUtil.execCallback(pWebView, arr.optString(0), AppUtil.getFileApkVersion(pWebView.getActivity(), params.getString("path")), JSUtil.OK,false);
}
/**
* 根据文件路径获取文件创建时间
* @param pWebView
* @param arr
* @return
*/
public void getFileCreatedTime(IWebview pWebView, JSONArray arr){
JSONObject params = JSON.parseObject(arr.optString(1));
File file = new File(params.getString("path"));
if(!file.exists()) return;
JSUtil.execCallback(pWebView, arr.optString(0), file.lastModified(), JSUtil.OK,false);
}
}
DownloadProgress
package com.park;
import org.json.JSONArray;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.android.park.R;
import com.park.util.NotificationUtils;
import io.dcloud.common.DHInterface.IWebview;
import io.dcloud.common.DHInterface.StandardFeature;
/**
* app下载进度
* @author 蓝之静云
*/
public class DownloadProgress extends StandardFeature {
static final int ids = 1000;
// 通知工具栏
NotificationUtils notificationUtils;
public void init(IWebview pWebView, JSONArray array){
if(null != notificationUtils) notificationUtils.clearNotification();
notificationUtils = new NotificationUtils(pWebView.getActivity());
}
/**
* 初始设置
*
* @param pWebView
* @param array
*/
public void setNotification(IWebview pWebView, JSONArray array) {
JSONObject params = JSON.parseObject(array.optString(1));
String title = params.getString("title");
String content = params.getString("content");
notificationUtils.sendNotification(ids, title, content, R.mipmap.ic_launcher);
}
/**
* 进度显示
*
* @param pWebView
* @param array
*/
public void setProgress(IWebview pWebView, final JSONArray array) {
JSONObject params = JSON.parseObject(array.optString(1));
notificationUtils.setNotification(Integer.parseInt(params.getString("current")));
}
/**
* 下载完成后
*
* @param pWebView
* @param array
*/
public void compProgressNotification(IWebview pWebView, JSONArray array) {
JSONObject params = JSON.parseObject(array.optString(1));
String title = params.getString("title");
notificationUtils = new NotificationUtils(pWebView.getActivity());
notificationUtils.sendNotification(ids, title, "请去安装体验吧", R.mipmap.ic_launcher);
}
}
AppUtil
package com.park.util;
import android.content.Context;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.util.Log;
import java.io.File;
/**
* Created by lenovo on 2019/10/17.
*/
public class AppUtil {
/**
* 获取apk文件的包信息
* @param context
* @param apkPath
* @return
*/
public static String getFileApkVersion(Context context, String apkPath) {
File apkFile = new File(apkPath);
if(!apkFile.exists()) return "";
PackageManager pm = context.getPackageManager();
PackageInfo info = pm.getPackageArchiveInfo(apkPath, PackageManager.GET_ACTIVITIES);
if (info != null) {
ApplicationInfo appInfo = info.applicationInfo;
String packageName = appInfo.packageName; //得到安装包名称
String version = info.versionName;//获取安装包的版本号
int versionCode = info.versionCode;
try {
return version;
} catch (OutOfMemoryError e) {
Log.d("yue-tag", "GetApkInfo: " + e);
}
}
return "";
}
}
NotificationUtils
package com.park.util;
import android.annotation.TargetApi;
import android.app.Notification;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.ContextWrapper;
import android.content.Intent;
import android.graphics.Color;
import android.net.Uri;
import android.os.Build;
import android.widget.RemoteViews;
import androidx.annotation.RequiresApi;
import androidx.core.app.NotificationCompat;
import com.android.park.R;
import static android.app.Notification.VISIBILITY_SECRET;
import static androidx.core.app.NotificationCompat.PRIORITY_DEFAULT;
/**
* <pre>
* @author WTY
* desc : 通知栏工具类
* revise:
* </pre>
*/
public class NotificationUtils extends ContextWrapper {
public static final String CHANNEL_ID = "default";
private static final String CHANNEL_NAME = "Default_Channel";
private NotificationManager mManager;
private int[] flags;
public NotificationUtils(Context base) {
super(base);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
//android 8.0以上需要特殊处理,也就是targetSDKVersion为26以上
createNotificationChannel();
}
}
@TargetApi(Build.VERSION_CODES.O)
private void createNotificationChannel() {
//第一个参数:channel_id
//第二个参数:channel_name
//第三个参数:设置通知重要性级别
//注意:该级别必须要在 NotificationChannel 的构造函数中指定,总共要五个级别;
//范围是从 NotificationManager.IMPORTANCE_NONE(0) ~ NotificationManager.IMPORTANCE_HIGH(4)
NotificationChannel channel = new NotificationChannel(CHANNEL_ID, CHANNEL_NAME,
NotificationManager.IMPORTANCE_LOW);
channel.canBypassDnd();//是否绕过请勿打扰模式
channel.enableLights(true);//是否在桌面icon右上角展示小红点
channel.setLockscreenVisibility(VISIBILITY_SECRET);//锁屏显示通知
channel.setLightColor(Color.RED);//闪关灯的灯光颜色
channel.canShowBadge();//桌面launcher的消息角标
channel.enableVibration(true);//是否允许震动
channel.getAudioAttributes();//获取系统通知响铃声音的配置
channel.getGroup();//获取通知取到组
channel.setBypassDnd(true);//设置可绕过 请勿打扰模式
channel.setSound(null, null);
channel.setVibrationPattern(new long[]{100, 100, 200});//设置震动模式
channel.shouldShowLights();//是否会有灯光
channel.setShowBadge(true); //是否在久按桌面图标时显示此渠道的通知
getManager().createNotificationChannel(channel);
}
/**
* 获取创建一个NotificationManager的对象
* @return NotificationManager对象
*/
public NotificationManager getManager() {
if (mManager == null) {
mManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
}
return mManager;
}
/**
* 清空所有的通知
*/
public void clearNotification(){
getManager().cancelAll();
}
/**
* 获取Notification
* @param title title
* @param content content
*/
public Notification getNotification(String title, String content , int icon){
Notification build;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
//android 8.0以上需要特殊处理,也就是targetSDKVersion为26以上
//通知用到NotificationCompat()这个V4库中的方法。但是在实际使用时发现书上的代码已经过时并且Android8.0已经不支持这种写法
Notification.Builder builder = getChannelNotification(title, content, icon);
build = builder.build();
} else {
NotificationCompat.Builder builder = getNotificationCompat(title, content, icon);
build = builder.build();
}
if (flags!=null && flags.length>0){
for (int a=0 ; a<flags.length ; a++){
build.flags |= flags[a];
}
}
return build;
}
/**
* 建议使用这个发送通知
* 调用该方法可以发送通知
* @param notifyId notifyId
* @param title title
* @param content content
*/
public void sendNotification(int notifyId, String title, String content , int icon) {
Notification build;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
//android 8.0以上需要特殊处理,也就是targetSDKVersion为26以上
//通知用到NotificationCompat()这个V4库中的方法。但是在实际使用时发现书上的代码已经过时并且Android8.0已经不支持这种写法
Notification.Builder builder = getChannelNotification(title, content, icon);
build = builder.build();
} else {
NotificationCompat.Builder builder = getNotificationCompat(title, content, icon);
build = builder.build();
}
if (flags!=null && flags.length>0){
for (int a=0 ; a<flags.length ; a++){
build.flags |= flags[a];
}
}
getManager().notify(notifyId, build);
}
/**
* 调用该方法可以发送通知
* @param notifyId notifyId
* @param title title
* @param content content
*/
public void sendNotificationCompat(int notifyId, String title, String content , int icon) {
NotificationCompat.Builder builder = getNotificationCompat(title, content, icon);
Notification build = builder.build();
if (flags!=null && flags.length>0){
for (int a=0 ; a<flags.length ; a++){
build.flags |= flags[a];
}
}
getManager().notify(notifyId, build);
}
private NotificationCompat.Builder getNotificationCompat(String title, String content, int icon) {
NotificationCompat.Builder builder;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
builder = new NotificationCompat.Builder(getApplicationContext(), CHANNEL_ID);
} else {
//注意用下面这个方法,在8.0以上无法出现通知栏。8.0之前是正常的。这里需要增强判断逻辑
builder = new NotificationCompat.Builder(getApplicationContext());
builder.setPriority(PRIORITY_DEFAULT);
}
builder.setContentTitle(title);
builder.setContentText(content);
builder.setSmallIcon(icon);
builder.setPriority(priority);
builder.setOnlyAlertOnce(onlyAlertOnce);
builder.setOngoing(ongoing);
if (remoteViews!=null){
builder.setContent(remoteViews);
}
if (intent!=null){
builder.setContentIntent(intent);
}
if (ticker!=null && ticker.length()>0){
builder.setTicker(ticker);
}
if (when!=0){
builder.setWhen(when);
}
if (sound!=null){
builder.setSound(sound);
}
if (defaults!=0){
builder.setDefaults(defaults);
}
//点击自动删除通知
builder.setAutoCancel(true);
return builder;
}
@RequiresApi(api = Build.VERSION_CODES.O)
private Notification.Builder getChannelNotification(String title, String content, int icon){
Notification.Builder builder = new Notification.Builder(getApplicationContext(), CHANNEL_ID);
Notification.Builder notificationBuilder = builder
//设置标题
.setContentTitle(title)
//消息内容
.setContentText(content)
//设置通知的图标
.setSmallIcon(icon)
//让通知左右滑的时候是否可以取消通知
.setOngoing(ongoing)
//设置优先级
.setPriority(priority)
//是否提示一次.true - 如果Notification已经存在状态栏即使在调用notify函数也不会更新
.setOnlyAlertOnce(onlyAlertOnce)
.setAutoCancel(true);
if (remoteViews!=null){
//设置自定义view通知栏
notificationBuilder.setContent(remoteViews);
}
if (intent!=null){
notificationBuilder.setContentIntent(intent);
}
if (ticker!=null && ticker.length()>0){
//设置状态栏的标题
notificationBuilder.setTicker(ticker);
}
if (when!=0){
//设置通知时间,默认为系统发出通知的时间,通常不用设置
notificationBuilder.setWhen(when);
}
if (sound!=null){
//设置sound
notificationBuilder.setSound(sound);
}
if (defaults!=0){
//设置默认的提示音
notificationBuilder.setDefaults(defaults);
}
if (pattern!=null){
//自定义震动效果
notificationBuilder.setVibrate(pattern);
}
return notificationBuilder;
}
private boolean ongoing = false;
private RemoteViews remoteViews = null;
private PendingIntent intent = null;
private String ticker = "";
private int priority = Notification.PRIORITY_DEFAULT;
private boolean onlyAlertOnce = false;
private long when = 0;
private Uri sound = null;
private int defaults = 0;
private long[] pattern = null;
/**
* 让通知左右滑的时候是否可以取消通知
* @param ongoing 是否可以取消通知
* @return
*/
public NotificationUtils setOngoing(boolean ongoing){
this.ongoing = ongoing;
return this;
}
/**
* 设置自定义view通知栏布局
* @param remoteViews view
* @return
*/
public NotificationUtils setContent(RemoteViews remoteViews){
this.remoteViews = remoteViews;
return this;
}
/**
* 设置内容点击
* @param intent intent
* @return
*/
public NotificationUtils setContentIntent(PendingIntent intent){
this.intent = intent;
return this;
}
/**
* 设置状态栏的标题
* @param ticker 状态栏的标题
* @return
*/
public NotificationUtils setTicker(String ticker){
this.ticker = ticker;
return this;
}
/**
* 设置优先级
* 注意:
* Android 8.0以及上,在 NotificationChannel 的构造函数中指定,总共要五个级别;
* Android 7.1(API 25)及以下的设备,还得调用NotificationCompat 的 setPriority方法来设置
*
* @param priority 优先级,默认是Notification.PRIORITY_DEFAULT
* @return
*/
public NotificationUtils setPriority(int priority){
this.priority = priority;
return this;
}
/**
* 是否提示一次.true - 如果Notification已经存在状态栏即使在调用notify函数也不会更新
* @param onlyAlertOnce 是否只提示一次,默认是false
* @return
*/
public NotificationUtils setOnlyAlertOnce(boolean onlyAlertOnce){
this.onlyAlertOnce = onlyAlertOnce;
return this;
}
/**
* 设置通知时间,默认为系统发出通知的时间,通常不用设置
* @param when when
* @return
*/
public NotificationUtils setWhen(long when){
this.when = when;
return this;
}
/**
* 设置sound
* @param sound sound
* @return
*/
public NotificationUtils setSound(Uri sound){
this.sound = sound;
return this;
}
/**
* 设置默认的提示音
* @param defaults defaults
* @return
*/
public NotificationUtils setDefaults(int defaults){
this.defaults = defaults;
return this;
}
/**
* 自定义震动效果
* @param pattern pattern
* @return
*/
public NotificationUtils setVibrate(long[] pattern){
this.pattern = pattern;
return this;
}
/**
* 设置flag标签
* @param flags flags
* @return
*/
public NotificationUtils setFlags(int... flags){
this.flags = flags;
return this;
}
public void setNotification(int progress) {
if (this==null){
return;
}
Intent intent = new Intent();
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
RemoteViews remoteViews = new RemoteViews(this.getPackageName(), R.layout.remote_notification_view);
remoteViews.setTextViewText(R.id.tvTitle, "正在下载:");
remoteViews.setProgressBar(R.id.pb, 100, progress, false);
NotificationManager manager = this.getManager();
Notification notification = this.setContentIntent(pendingIntent)
.setContent(remoteViews)
.setFlags(Notification.FLAG_AUTO_CANCEL)
.setOnlyAlertOnce(true)
.getNotification("新消息", "应用开始更新", R.mipmap.ic_launcher);
//下载成功或者失败
if (progress == 100 || progress == -1) {
this.clearNotification();
} else {
manager.notify(1, notification);
}
}
}