diff --git a/App.vue b/App.vue index 08a81ad..fc58122 100644 --- a/App.vue +++ b/App.vue @@ -76,6 +76,8 @@ onLaunch: function(option) { this.globalData.statusBarHeight = uni.getSystemInfoSync().statusBarHeight + 'px'; this.globalData.uid = this.$store.state.app.uid; + + this.$store.dispatch('INIT_CONFIG'); // #ifdef APP-PLUS //监听uni小程序发送的事件 diff --git a/pages.json b/pages.json index 45db4e9..a6f4b73 100644 --- a/pages.json +++ b/pages.json @@ -102,7 +102,25 @@ } } - ], + ,{ + "path": "uni_modules/guyue-updater/pages/updater", + "style": { + "app-plus": { + "animationDuration": 200, + "animationType": "fade-in", + "background": "transparent", + "backgroundColorTop": "transparent", + "bounce": "none", + "popGesture": "none", + "scrollIndicator": false, + "titleNView": false + }, + "backgroundColor": "transparent", + "disableScroll": true, + "navigationStyle": "custom" + } +} +], "subPackages": [{ "root": "pages/goods_cate", "name": "goods_cate", diff --git a/store/modules/app.js b/store/modules/app.js index 9c5e7c0..b0b862e 100644 --- a/store/modules/app.js +++ b/store/modules/app.js @@ -9,6 +9,9 @@ import Cache from '../../utils/cache'; import { USER_INFO } from '../../config/cache'; +// #ifdef APP-PLUS +import Updater from '@/uni_modules/guyue-updater/index'; +// #endif const state = { location: Cache.get('LOCATION_DATA', true) || {}, @@ -99,9 +102,71 @@ const actions = { }).catch(() => { }); - } + }, + async INIT_CONFIG({ state, commit }, data = false) { + // let res = await getConfig(); + let res = { + data: { + version: '1.2.1', + version_info: { + "id": 4, + "title": "正式IOS测试", + "content": "IOS正式", + "type": 1, + "version": "1.2.1", + "dow_url": "https://worker-task.lihaink.cn/uploads/files/20230908/20230908174409996e65763.wgt", + "force": 0, + "quiet": 0, + "create_time": "2023-09-04 15:28:29", + "update_time": "2023-09-04 15:28:29", + "delete_time": null + } + } + } + const wgt_v = uni.getStorageSync('wgt_version')||'1.0.0'; + // #ifdef APP-PLUS + let os = uni.getSystemInfoSync(); + if(data) uni.showLoading({ + title: '检查更新中' + }) + // 版本更新 + if(compareVersions(res.data.version, os.appVersion||wgt_v)==1&&compareVersions(res.data.version, wgt_v)==1){ + try{ + let info = res.data.version_info||{}; + let version = { + title: info.title||'发现新版本', + content: info.content||'修复了部分BUG', + versionName: info.version||'1.0.1', + downUrl: info.dow_url||'', + force: info.force==1?true:false, // 是否强制更新 + quiet: info.quiet==1?true:false // 是否静默更新 + } + Updater.update(version); + }catch(e){ + console.log(e); + } + if(data) uni.hideLoading(); + } + // #endif + } }; +function compareVersions(version1, version2) { + const arr1 = version1.split('.').map(Number); + const arr2 = version2.split('.').map(Number); + for (let i = 0; i < Math.max(arr1.length, arr2.length); i++) { + const num1 = i < arr1.length ? arr1[i] : 0; + const num2 = i < arr2.length ? arr2[i] : 0; + + if (num1 > num2) { + return 1; + } else if (num1 < num2) { + return -1; + } + } + return 0; +} + export default { state, mutations, diff --git a/uni_modules/guyue-updater/assets/bg1.png b/uni_modules/guyue-updater/assets/bg1.png new file mode 100644 index 0000000..5eef28e Binary files /dev/null and b/uni_modules/guyue-updater/assets/bg1.png differ diff --git a/uni_modules/guyue-updater/assets/bg2.png b/uni_modules/guyue-updater/assets/bg2.png new file mode 100644 index 0000000..31ca5f1 Binary files /dev/null and b/uni_modules/guyue-updater/assets/bg2.png differ diff --git a/uni_modules/guyue-updater/assets/close.png b/uni_modules/guyue-updater/assets/close.png new file mode 100644 index 0000000..020cdf5 Binary files /dev/null and b/uni_modules/guyue-updater/assets/close.png differ diff --git a/uni_modules/guyue-updater/changelog.md b/uni_modules/guyue-updater/changelog.md new file mode 100644 index 0000000..c757be8 --- /dev/null +++ b/uni_modules/guyue-updater/changelog.md @@ -0,0 +1,6 @@ +## 1.0.2(2023-06-02) +修复部分bug,新增条件编译,仅限app平台调用 +## 1.0.1(2023-06-02) +去除ts,使用js进行开发,兼容性更好 +## 1.0.0(2023-05-04) +完成初始功能,支持apk安装以及wgt升级,支持显示弹窗升级、静默更新以及强制更新,支持进度显示,支持覆盖原生tabbar、原生导航栏 \ No newline at end of file diff --git a/uni_modules/guyue-updater/index.js b/uni_modules/guyue-updater/index.js new file mode 100644 index 0000000..b47abe5 --- /dev/null +++ b/uni_modules/guyue-updater/index.js @@ -0,0 +1,24 @@ + +import { download, install } from "./updater"; + +export default class Updater { + static async update(options ) { + // #ifdef APP-PLUS + // 静默更新 + if (options.quiet) { + download({ + url: options.downUrl, + onSuccess(filePath) { + install(filePath, false, options.versionName); + }, + }); + } else if (options.downUrl) { + uni.navigateTo({ + url: `/uni_modules/guyue-updater/pages/updater?data=${encodeURIComponent(JSON.stringify(options))}`, + animationType: "fade-in", + animationDuration: 200, + }); + } + // #endif + } +} \ No newline at end of file diff --git a/uni_modules/guyue-updater/interface.ts b/uni_modules/guyue-updater/interface.ts new file mode 100644 index 0000000..a2898b1 --- /dev/null +++ b/uni_modules/guyue-updater/interface.ts @@ -0,0 +1,12 @@ +export type UpdateParams = { + content: string; // 必填,更新内容,内容中使用 \n 进行换行 + downUrl: string; // 必填,wgt热更新请给出 .wgt 的文件地址,APK整包更新请设置下载apk地址,ios请设置苹果商店的连接地址; + title?: string; // 用于显示弹窗标题,默认 发现新版本 + versionName?: string; // 版本名,用于显示更新版本,如 1.0.0 + quiet?: boolean; // 是否是静默更新,开启后,不会有弹窗,会在后台下载更新文件,在下次启动APP时使用更新 + force?: boolean; // 是否是强制更新,开启后,弹窗无法被关闭,必须更新 + updateBtnText?: string; // 升级按钮文字,默认 立即升级 + downMsgTip?: string; // 仅android,默认 下载中,请稍后 + downSucTip?: string; // 仅android,默认 下载完成,安装中 + downErrorTip?: string; // 仅android,默认 下载失败,请重试 +} \ No newline at end of file diff --git a/uni_modules/guyue-updater/package.json b/uni_modules/guyue-updater/package.json new file mode 100644 index 0000000..e55784a --- /dev/null +++ b/uni_modules/guyue-updater/package.json @@ -0,0 +1,82 @@ +{ + "id": "guyue-updater", + "displayName": "App版本升级弹框和进度提示", + "version": "1.0.2", + "description": "app热更新模块,支持apk安装以及wgt升级,支持显示弹窗升级、静默更新以及强制更新,支持进度显示,支持覆盖原生tabar,原生导航栏", + "keywords": [ + "热更新", + "进度提示", + "版本升级", + "app自动升级", + "wgt自动升级" +], + "repository": "", + "engines": { + "HBuilderX": "^3.4.9" + }, + "dcloudext": { + "type": "sdk-js", + "sale": { + "regular": { + "price": "9.98" + }, + "sourcecode": { + "price": "16.80" + } + }, + "contact": { + "qq": "2292550932" + }, + "declaration": { + "ads": "无", + "data": "插件不采集任何数据", + "permissions": "无" + }, + "npmurl": "" + }, + "uni_modules": { + "dependencies": [], + "encrypt": [], + "platforms": { + "cloud": { + "tcb": "y", + "aliyun": "y" + }, + "client": { + "Vue": { + "vue2": "y", + "vue3": "y" + }, + "App": {}, + "H5-mobile": { + "Safari": "n", + "Android Browser": "n", + "微信浏览器(Android)": "n", + "QQ浏览器(Android)": "n" + }, + "H5-pc": { + "Chrome": "n", + "IE": "n", + "Edge": "n", + "Firefox": "n", + "Safari": "n" + }, + "小程序": { + "微信": "n", + "阿里": "n", + "百度": "n", + "字节跳动": "n", + "QQ": "n", + "钉钉": "n", + "快手": "n", + "飞书": "n", + "京东": "n" + }, + "快应用": { + "华为": "n", + "联盟": "n" + } + } + } + } +} \ No newline at end of file diff --git a/uni_modules/guyue-updater/pages/updater.vue b/uni_modules/guyue-updater/pages/updater.vue new file mode 100644 index 0000000..9ca259b --- /dev/null +++ b/uni_modules/guyue-updater/pages/updater.vue @@ -0,0 +1,315 @@ + + + + + diff --git a/uni_modules/guyue-updater/readme.md b/uni_modules/guyue-updater/readme.md new file mode 100644 index 0000000..ccb2b01 --- /dev/null +++ b/uni_modules/guyue-updater/readme.md @@ -0,0 +1,65 @@ +# App热更新 + +App热更新模块,支持apk安装以及wgt升级,支持显示弹窗升级、静默更新以及强制更新,支持进度显示,支持覆盖原生tabbar、原生导航栏。 +可用于自建热更新渠道,不依赖于云服务,无云服务费用支出,也可以适配官方更新中心。 + +## 使用说明 +### 1.将此项目导入自己的项目工程 +### 2.在page.json中注册页面,如下 + ```javascript + { + "path": "uni_modules/guyue-updater/pages/updater", + "style": { + "navigationStyle": "custom", + "backgroundColor": "transparent", + "disableScroll": true, + "app-plus": { + "backgroundColorTop": "transparent", + "background": "transparent", + "scrollIndicator": false, + "titleNView": false, + "popGesture": "none", + "bounce": "none", + "animationType": "fade-in", + "animationDuration": 200 + } + } + } + ``` +### 3.将版本检测函数导入需要使用的页面 + 一般在App.vue中的onLaunch导入或者首页导入,需要自行完成热更新检查,一般在APP启动时发起一个请求获取热更新数据,数据获取后,可以调用该组件完成更新。 + ```javascript + import Updater from '@/uni_modules/guyue-updater/index'; + + // 仅在app平台有效,其他平台调用无效 + Updater.update({ + title: '发现新版本', + content: '1. 我们更新了新的UI设计\n2. 我们更新了新的UI设计\n3. 我们更新了新的UI设计\n4. 我们更新了新的UI设计\n', + versionName: '1.3.6', + downUrl: 'https://cdn.xxx.cn/mp/__UNI__1F29D65.wgt', + force: false, + }) + ``` +## 参数说明 +```javascript +export type UpdateParams = { + content: string; // 必填,更新内容,内容中使用 \n 进行换行 + downUrl: string; // 必填,wgt热更新请给出 .wgt 的文件地址,APK整包更新请设置下载apk地址,ios请设置苹果商店的连接地址; + title?: string; // 用于显示弹窗标题,默认 发现新版本 + versionName?: string; // 版本名,用于显示更新版本,如 1.0.0 + quiet?: boolean; // 是否是静默更新,开启后,不会有弹窗,会在后台下载更新文件,在下次启动APP时使用更新 + force?: boolean; // 是否是强制更新,开启后,弹窗无法被关闭,必须更新 + updateBtnText?: string; // 升级按钮文字,默认 立即升级 + downMsgTip?: string; // 仅android,默认 下载中,请稍后 + downSucTip?: string; // 仅android,默认 下载完成,安装中 + downErrorTip?: string; // 仅android,默认 下载失败,请重试 +} +``` + +## Android如何跳转到应用市场更新 + +downUrl 设置为应用市场的地址即可,如: market://details?id={这里写你的应用包名} + +## iOS 如何跳转到AppStore + +downUrl 设置为AppStore的地址即可,如: itms-apps://itunes.apple.com/cn/app/hello-uni-app/id1417078253 \ No newline at end of file diff --git a/uni_modules/guyue-updater/updater.js b/uni_modules/guyue-updater/updater.js new file mode 100644 index 0000000..fbdf8b6 --- /dev/null +++ b/uni_modules/guyue-updater/updater.js @@ -0,0 +1,31 @@ + +export const download = ({ url, onProgress, onSuccess, onFail }) => { + const task = uni.downloadFile({ + url, + success(res) { + if (res.statusCode === 200) { + onSuccess && onSuccess(res.tempFilePath); + } + }, + fail() { + onFail && onFail(); + } + }); + task.onProgressUpdate(res => { + onProgress && onProgress(res.progress); + }); +}; + +export const install = (filePath, restart = false, version='') => { + plus.runtime.install(filePath, { + force: true + }, () => { + console.log('install success...'); + if (restart) { + uni.setStorageSync('wgt_version', version); + plus.runtime.restart(); + } + }, (e) => { + console.error('install fail...', e); + }); +}; \ No newline at end of file