需求说明:使用MUI+Vue等技术并且通过HBuilderX打包开发移动app,在有版本更新时需要自动提示用户有新版本,并且可以点击下载自动安装。

思路说明:

  1. 应用打开时(使用Vue的生命周期mounted),获取自己的版本信息appinfo;

    (本身app的信息一般存放在mainfest.json中,直接获取即可)
mui.plusReady(function(){
mui.getJSON("manifest.json", null, function(manifest){
var version = data.version
var vercode_local = version.code;
var vername_local = version.name;
console.log("版本名称:"+vername_local+",版本code:"+vercode_local);
});
});
  1. 获取remote服务器上移动应用最新的版本信息(一般是存放一个json的数据信息)
mui.getJSON(【服务器的url + appInfo.version_json】, null, function(data) {
var verCode = data.verCode;
var verName = data.verName;
//服务器版本
//console.log("服务器版本名称:"+verName +",服务器版本code:"+verCode);
});
  1. 判断两个版本号是否相等,不相等则需要更新
if(vercode_local == verCode) {
mui.toast("当前已经是最新版本!");
} else {
var btnArray = ['是', '否'];
mui.confirm('当前版本是:' + vername_local + ', 最新版本是:' + verName + ', 是否立即更新?', '发现最新版本', btnArray, function(z) {
if(z.index == 0) {
console.log('确定');
installApk(BASEINFO.maxsvc + appInfo.version_apk);
} else {
console.log('不确定');
return;
}
});
}

1.在vue的mounted生命周期进行更新校验

mounted() {
mui.plusReady(function() {
//console.log("检查更新!!!!");
mui.init({statusBarBackground: '#EEEEEE'});
plus.screen.lockOrientation("landscape-primary");
if(mui.os.android) {
mui.getJSON("../../manifest.json", null, function(data) {
var version = data.version
var vercode_local = version.code;
var vername_local = version.name;
//当前版本
//console.log("版本名称:"+vername_local+",版本code:"+vercode_local);
mui.getJSON(BASEINFO.maxsvc + appInfo.version_json, null, function(data) {
var verCode = data.verCode;
//服务器版本
//console.log("服务器版本code:"+verCode);
var verName = data.verName;
if(vercode_local == verCode) {
mui.toast("当前已经是最新版本!");
} else {
var btnArray = ['是', '否'];
mui.confirm('当前版本是:' + vername_local + ', 最新版本是:' + verName + ', 是否立即更新?', '发现最新版本', btnArray, function(z) {
if(z.index == 0) {
console.log('确定');
installApk(BASEINFO.maxsvc + appInfo.version_apk);
} else {
console.log('不确定');
return;
}
});
}
});
});
}
});
}

2.其中需要用到application.js中声明的函数

3.application.js中的函数

var BASEINFO={
//接口url
maxsvc:"http://*.*.*.*:port/appUpdate",
maxbiz:"http://*.*.*.*:port/maxbiz",
download:"http://*.*.*.*:port/whatsup/releaseapk.jsp"
} function getWinWH() {
var temp = {};
var winWidth = "0";
var winHeight = "0";
if(window.innerWidth) {
winWidth = window.innerWidth;
} else if((document.body) && (document.body.clientWidth)) {
winWidth = document.body.clientWidth;
}
if(window.innerHeight) {
winHeight = window.innerHeight;
} else if((document.body) && (document.body.clientHeight)) {
winHeight = document.body.clientHeight;
}
if(document.documentElement && document.documentElement.clientHeight && document.documentElement.clientWidth) {
winHeight = document.documentElement.clientHeight;
winWidth = document.documentElement.clientWidth;
}
temp.winWidth = winWidth;
temp.winHeight = winHeight; return temp;
} function getAppInfo(appname) {
var prefix = "";
var path = "";
var name = "";
// MaxUA
if(appname == "mua") {
path = "com.maxnerva.maxua";
prefix = "";
name = path;
}
// 异常停线通知(L5)
if(appname == "men") {
path = "com.newpc.b2b.";
prefix = "asm_";
name = path + prefix + appname;
}
// E点名(L5)
if(appname == "mrc") {
path = "com.newpc.b2b.";
prefix = "asm_";
name = path + prefix + appname;
}
// 组装生产日报表(L5)
if(appname == "pdr") {
path = "com.newpc.b2b.";
prefix = "asm_";
name = path + prefix + appname;
}
// 仓库进料转仓(L6)
if(appname == "mgr") {
path = "com.foxconn.";
prefix = "";
name = path + prefix + appname;
}
// 发料单查询(L6)
if(appname == "mgi") {
path = "com.foxconn.max.activity";
prefix = "";
name = path;
}
// 仓库分批发料(L5)
if(appname == "mso") {
path = "com.newpc.b2b.";
prefix = "asm_";
name = path + prefix + appname;
}
// 上料表查询(L6)
if(appname == "mgf") {
path = "com.newpc.b2b.";
prefix = "asm_";
name = path + prefix + appname;
}
// IQC进料检验(L6)
if(appname == "mqi") {
path = "com.foxconn.iqc";
prefix = "";
name = path;
}
// 流程卡(L5)
if(appname == "wfc") {
path = "com.mts.";
prefix = "";
name = path + prefix + appname;
} // mqi_v
if(appname == "mqi_v") {
path = "com.foxconn.";
prefix = "";
name = path + prefix + appname;
} var appInfo = {};
appInfo.scheme = "open";
appInfo.host = name;
appInfo.package = name;
appInfo.version_json = "/resource/download/" + appname + ".json";
appInfo.version_apk = "/resource/download/" + appname + ".apk";
appInfo.version_desc = "/resource/download/" + appname + ".desc";
return appInfo;
} function installApk(url) {
var dtask = plus.downloader.createDownload(url, {}, function(d, status) {
if(status == 200) {
plus.nativeUI.toast("正在准备环境,请稍后!");
sleep(1000);
var path = d.filename;
console.log(d.filename);
plus.runtime.install(path);
} else {
alert('Download failed:' + status);
}
});
dtask.start();
} function sleep(numberMillis) {
var now = new Date();
var exitTime = now.getTime() + numberMillis;
while(true) {
now = new Date();
if(now.getTime() > exitTime)
return;
}
} function getAppVersion(packagename) {
var versionJson = {};
var main = plus.android.runtimeMainActivity();
var pm = main.getPackageManager();
var PackageManager = plus.android.importClass(pm);
var pi = pm.getPackageInfo(packagename, 0);
var PackageInfo = plus.android.importClass(pi);
var vercode_local = plus.android.getAttribute(pi, "versionCode");
var vername_local = plus.android.getAttribute(pi, "versionName");
versionJson.vercode_local = vercode_local;
versionJson.vername_local = vername_local;
return versionJson;
} function jumpAndroid(appname) {
var appInfo = getAppInfo(appname);
//
console.log("appInfo "+appInfo); var url = appInfo.scheme + "://" + appInfo.host +
"?BUNDLE_CAMP=" + localStorage.getItem("musercamp") +
"&BUNDLE_USERNAME=" + localStorage.getItem("musername") +
"&BUNDLE_USERPASSWRD=" + localStorage.getItem("mpassword");
//
console.log("appInfo "+appInfo); try {
if(plus.os.name == "Android") {
if(plus.runtime.isApplicationExist({
pname: appInfo.host,
action: appInfo.scheme + '://'
})) {
var versionJosn = getAppVersion(appInfo.package);
var vercode_local = versionJosn.vercode_local;
var vername_local = versionJosn.vername_local;
mui.getJSON(BASEINFO.maxsvc + appInfo.version_json, null, function(data) {
var verCode = data.verCode;
var verName = data.verName;
if(vercode_local == verCode) {
location.href = url;
} else {
var btnArray = ['是', '否'];
mui.confirm('当前版本是:' + vername_local + ', 最新版本是:' + verName + ', 是否立即更新?', '发现最新版本', btnArray, function(z) {
if(z.index == 0) {
installApk(BASEINFO.maxsvc + appInfo.version_apk);
} else {
location.href = url;
}
});
}
});
} else {
var btnArray = ['是', '否'];
mui.confirm('应用程式未安装, 是否立即安装?', '应用程式安装确认', btnArray, function(z) {
if(z.index == 0) {
installApk(BASEINFO.maxsvc + appInfo.version_apk);
} else {
return;
}
});
}
}
} catch(e) {
mui.toast("更新失败 "+e.toString());
}
} function myAjax(url, type, input, timeout, success, error) {
var xhr = new plus.net.XMLHttpRequest();
if(timeout && timeout > 0) xhr.timeout = timeout;
xhr.onreadystatechange = function() {
switch(xhr.readyState) {
case 0:
console.log("xhr请求已初始化");
break;
case 1:
console.log("xhr请求已打开");
break;
case 2:
console.log("xhr请求已发送");
break;
case 3:
console.log("xhr请求已响应");
break;
case 4:
if(xhr.status == 200) {
success(eval('('+xhr.responseText + ')'));
} else {
error(xhr.readyState, xhr);
}
break;
default:
break;
}
}
if(input) {
if(type == 'post' || type == 'get') {
xhr.open(type || "GET", url);
xhr.send(JSON.stringify(input));
} else {
throw new Error("type is undefined !")
}
} else {
if(type != 'post' && type != 'get') {
throw new Error("type is undefined !")
}
xhr.open(type || "GET", url);
xhr.send();
}
} function isSysAdmin(groups) {
var result = false;
for(var i = 0; i < groups.length; i++) {
var group = groups[i];
if(group.groupcode.toUpperCase() == "SYSADMIN") {
result = true;
break;
}
}
return result;
} function isReadonly() {
var result = false;
var muserStr=localStorage.getItem("muser");
var muserObj=eval('(' + muserStr + ')');
var groups=muserObj.group;
if(groups && groups.length>0 && groups[0].groupcode.toUpperCase().indexOf("_READ")>0) result = true;
return result;
} function getLongUsername(){
var muserStr=localStorage.getItem("muser");
var muserObj=eval('(' + muserStr + ')');
var muser=muserObj.user;
return muser.username+"("+muser.lastname+")";
} //其他方法检查版本
function svn(t) {
var xhr_svn = new plus.net.XMLHttpRequest();
xhr_svn.onreadystatechange = function() {
if (xhr_svn.readyState == 4) {
if (xhr_svn.status == 200) {
var res = JSON.parse(xhr_svn.responseText);
if (res.state == 'yes') {
if (res.mark != t) {
var upr;
plus.nativeUI.confirm( "有新版本发布了,是否件更新?", function(e){
upr=(e.index==0)?"Y":"N";
console.log(upr);
if(upr=="Y"){
var wt = plus.nativeUI.showWaiting('下载更新中,请勿关闭');
var url = res.url; // 下载文件地址
var dtask = plus.downloader.createDownload(url, {}, function(d, status) {
if (status == 200) { // 下载成功
var path = d.filename;
console.log(d.filename);
plus.runtime.install(path);
} else { //下载失败
alert("Download failed: " + status);
}
});
dtask.start();
}else{ }
}, "XXX系统", ["确认","取消"] );
} else {
console.log('最新');
}
}
} else {
plus.nativeUI.toast( "网络连接错误!");
}
}
}
xhr_svn.open("GET", "http:/XXX/APPobject/imes/update.json");//这里的地址是上面json文件的地址
xhr_svn.send();
}

4.服务器端添加配置信息

1.主要添加json(新版本信息),apk(新版本安装包),desc(描述文件)

HBuilderX开发app实现自动更新版本的更多相关文章

  1. 如何实现已发布app的自动更新

    要实现app的自动更新,做两件事情就可以搞定 1.获取当前手机中的app版本号 我们可以通过查询mainbundle中的获取CFBundleVersion NSDictionary *infoDict ...

  2. H5+app,自动更新后自动删除安装包

    H5+app 自动删除安装包 一.前言 之前做好的app自动更新,遗留下了一个问题,就是自动更新后安装包没有自行删除掉. 好像现在的手机的系统是有安装完自动清理安装包的.想我这个H5+的app安装完后 ...

  3. iOS企业版使用第三方实现自动更新版本

    1.获取本地版本和互联网版本          NSDictionary *infoDictionary = [[NSBundle mainBundle] infoDictionary];     N ...

  4. nvidia驱动自动更新版本后问题解决 -- failed to initialize nvml: driver/library version mismatch

    因为必须关闭桌面窗口, 建议另外一台电脑ssh连接操作 1. 卸载旧版本并关闭图形界面 sudo apt-get remove --purge nvidia-\* sudo service light ...

  5. app的自动更新(调用DownloadManager)

    具体思路为:调用接口与服务器版本对比,当服务器版本号大于本地的,调用DownloadManager进行下载,之前也试过很多方法,但是兼容性都不是很好,还有一点要注意的是,在这里我并没有设置固定的下载路 ...

  6. 长期支持版本(即不自动更新版本) - Flash Player 18.0.0.268

    无意中发现,适合不喜欢折腾的朋友. 下载链接:(官方:http://www.adobe.com/cn/products/flashplayer/distribution3.html) (分流:http ...

  7. winform应用程序自动更新版本

    http://blog.csdn.net/gxxloveszj/article/details/8278187 http://www.cnblogs.com/x369/articles/105656. ...

  8. vb脚本自动更新版本信息

    使用的串口显示软件为secureCrt,支持脚本功能,今天写了一个简单的软件升级脚本(VB脚本). 如下: # $language = "VBScript" # $interfac ...

  9. Web APP自动更新

    我们的手机软件每天都要经营,经常需要更新,比如程序的Bug,好的功能,好的洁面... ... 这就需要我们的用户打开web app时候自动更新客户端程序,而不是再去应用程序商店从新下载.今天的笔记就是 ...

随机推荐

  1. redis中hash数据类型

    remoteSelf:1>hset website google "www.google.com" "1" remoteSelf:1>hget we ...

  2. (好题)POJ3057

    二分+二分图匹配+BFS 题意: 墙壁“X”,空区域(都是人)“.”, 门“D”. 人向门移动通过时视为逃脱,门每秒能出去一个人,人可以上下左右移动,墙阻止移动. 求最优移动方案下,最后一个人逃脱的最 ...

  3. pip 自己的源 搭建

    1  安装工具 pip install  pip2pi 2  下载 所需要的包 pip2tgz /application/nginx/html/yum/python/ apscheduler (172 ...

  4. vim 进化 编码问题

    " 解决菜单乱码 source $VIMRUNTIME/delmenu.vim source $VIMRUNTIME/menu.vim " 防止文件显示乱码 set fileenc ...

  5. 《趣谈 Linux 操作系统》学习笔记(一):为什么要学 Linux 及学习路径

    前言:学习的课程来自极客时间的专栏<趣谈 Linux 操作系统>,作者用形象化的比喻和丰富的图片让课程变得比较易懂,为了避免知识看过就忘,打算通过写学习笔记的形式记录自己的学习过程. Li ...

  6. itest(爱测试) 4.4.0 发布,开源BUG 跟踪管理 & 敏捷测试管理软件

    itest 简介 test 开源敏捷测试管理,testOps 践行者.可按测试包分配测试用例执行,也可建测试迭代(含任务,测试包,BUG)来组织测试工作,也有测试环境管理,还有很常用的测试度量:对于发 ...

  7. opencv —— erode、dilate 腐蚀与膨胀

    腐蚀与膨胀是形态学滤波.其中,腐蚀是最小值滤波,膨胀是最大值滤波,即分别选取内核中的最小值与最大值赋值给锚点.若内核为 N×1 或 1×N 形状,可用于横纵方向直线检测. 膨胀:dilate 函数 v ...

  8. 疑问:Iterator 遍历器和数据集合各种遍历方法的区别

    https://es6.ruanyifeng.com/#docs/iterator Iterator(遍历器)的概念 Iterator 接口主要供for...of消费 Iterator 的遍历过程是: ...

  9. mysql 表分区操作

    //不支持动态创建分区CREATE TABLE `rpt_exp_event_bucket_creative_d_across` ( `bucket_id` VARCHAR(200) NOT NULL ...

  10. 【笔记】机器学习 - 李宏毅 - 13 - Why Deep

    当参数一样多的时候,神经网络变得更高比变宽更有效果.为什么会这样呢? 其实和软件行业的模块化思想是一致的. 比如,如果直接对这四种分类进行训练,长发的男孩数据较少,那么这一类训练得到的classifi ...