首次完整推送,

V:1.20240808.006
This commit is contained in:
fm453
2024-08-13 18:32:37 +08:00
parent 15be3e9373
commit c62d15b288
939 changed files with 111777 additions and 0 deletions

View File

@ -0,0 +1,55 @@
## 0.6.22024-04-11
- 更新 支持支付宝小程序云
## 0.6.12023-11-02
- 修复 输入更新内容时有长度限制的Bug
## 0.6.02023-02-24
- 修复 升级中心安卓应用市场不显示的Bug
## 0.5.12022-07-06
- 修复 上版带出云函数不存在的Bug
- 升级 uni-admin 大于等于 1.9.0 务必更新至此版本。uni-admin 版本小于 1.9.0 请不要更新,历史版本在 Gitee 有发行版。后续 uni-admin 会集成升级中心
## 0.5.02022-07-05
- 修复 版本列表默认显示全部版本的Bug
- 升级 uni-admin 1.9.0 务必更新至此版本。uni-admin 版本小于 1.9.0 请不要更新,后续 uni-admin 会集成升级中心
## 0.4.22021-12-07
- 更新 优化 list 页面显示,修复 list 页面报错
## 0.4.12021-12-01
- 修复 0.4.0版本带出来,发布新版时 appid、name 不会自动填充的Bug
## 0.4.02021-11-26
- 更新 升级中心移除应用管理现在由uni-admin接管。旧版本若没有应用管理请做升级处理
## 0.3.02021-11-18
- 兼容 uni-admin 新版内置 $request 函数改动
## 0.2.22021-09-06
- 解决 opendb-app-list表对应的schema名称冲突的问题
## 0.2.12021-09-03
- 修复 一个在添加菜单时报错createdate不与默认值匹配 的Bug
## 0.2.02021-08-25
- 兼容vue3.0
## 0.1.92021-08-13
- 更新 uni-forms使用validate校验字段
- 修复 报错dirty_data、create_date在数据库中并不存在
## 0.1.82021-08-09
- 修复 默认配置项配置错误
## 0.1.72021-08-09
- 移除测试时配置项
## 0.1.62021-08-09
- 修复 修改版本信息时,上传时间丢失问题
## 0.1.52021-07-21
- 更新 :value.sync 改为 :value 和 @update:value
## 0.1.42021-07-13
- 修复 uni-easyinput去除输入字符长度限制
- 更新文档 关于 uni-id缺少配置信息 错误。请查看安装指引第13条
## 0.1.32021-06-15
- 修复 wgt更新某些情况下获取数据错误
## 0.1.22021-06-04
- 修复 上传包时根据平台筛选文件
- 更新 文档
## 0.1.12021-05-18
- 更新uni-table中uni-tr组件的selectable属性为disabled
## 0.1.02021-04-07
- 更新版本对比函数 compare
## 0.0.62021-04-01
- 调整db_init.json
## 0.0.52021-03-25
- 调整为uni_modules目录
- 升级中心后台管理系统拆分为 Admin 后台管理 和 前台检查更新uni-upgrade-center-app

View File

@ -0,0 +1,44 @@
// 表单校验规则由 schema2code 生成,不建议直接修改校验规则,而建议通过 schema2code 生成, 详情: https://uniapp.dcloud.net.cn/uniCloud/schema
const validator = {
"appid": {
"rules": [
{
"required": true
},
{
"format": "string"
}
],
"label": "AppID"
},
"name": {
"rules": [
{
"required": true
},
{
"format": "string"
}
],
"label": "应用名称"
},
"description": {
"rules": [
{
"required": true
},
{
"format": "string"
}
],
"label": "应用描述"
}
}
const enumConverter = {}
export { validator, enumConverter }

View File

@ -0,0 +1,151 @@
// 表单校验规则由 schema2code 生成,不建议直接修改校验规则,而建议通过 schema2code 生成, 详情: https://uniapp.dcloud.net.cn/uniCloud/schema
const validator = {
"appid": {
"rules": [{
"required": true
},
{
"format": "string"
}
],
"label": "AppID"
},
"name": {
"rules": [{
"format": "string"
}],
"label": "应用名称"
},
"title": {
"rules": [{
"format": "string"
}],
"label": "更新标题"
},
"contents": {
"rules": [{
"required": true
},
{
"format": "string"
}
],
"label": "更新内容"
},
"platform": {
"rules": [{
"required": true
},
/* 此处不校验数据类型因为platform在发布app端是单选在发布wgt时可能是多选
{
"format": "array"
}, */
{
"range": [{
"value": "Android",
"text": "安卓"
},
{
"value": "iOS",
"text": "苹果"
}
]
}
],
"label": "平台"
},
"type": {
"rules": [{
"required": true
}, {
"format": "string"
},
{
"range": [{
"value": "native_app",
"text": "原生App安装包"
},
{
"value": "wgt",
"text": "wgt资源包"
}
]
}
],
"label": "安装包类型"
},
"version": {
"rules": [{
"required": true
},
{
"format": "string"
}
],
"label": "版本号"
},
"min_uni_version": {
"rules": [{
"format": "string"
}],
"label": "原生App最低版本"
},
"url": {
"rules": [{
"required": true
}, {
"format": "string"
}],
"label": "包地址"
},
"stable_publish": {
"rules": [{
"format": "bool"
}],
"label": "上线发行"
},
"create_date": {
"rules": [{
"format": "timestamp"
}],
"label": "上传时间"
},
"is_silently": {
"rules": [{
"format": "bool"
}],
"label": "静默更新",
"defaultValue": false
},
"is_mandatory": {
"rules": [{
"format": "bool"
}],
"label": "强制更新",
"defaultValue": false
}
}
const enumConverter = {
"platform_valuetotext": [{
"value": "Android",
"text": "安卓"
},
{
"value": "iOS",
"text": "苹果"
}
],
"type_valuetotext": {
"native_app": "原生App安装包",
"wgt": "wgt资源包"
}
}
export {
validator,
enumConverter
}

View File

@ -0,0 +1,10 @@
[{
"menu_id": "system_update",
"name": "升级中心",
"icon": "uni-icons-cloud-upload",
"url": "uni_modules/uni-upgrade-center/pages/version/list",
"sort": 1050,
"parent_id": "system_management",
"permission": [],
"enable": true
}]

View File

@ -0,0 +1,92 @@
{
"id": "uni-upgrade-center",
"displayName": "升级中心 uni-upgrade-center - Admin",
"version": "0.6.2",
"description": "uni升级中心 - 后台管理系统",
"keywords": [
"uniCloud",
"admin",
"update",
"升级",
"wgt"
],
"repository": "https://gitee.com/dcloud/uni-upgrade-center/tree/master/uni_modules/uni-upgrade-center",
"engines": {
"HBuilderX": "^3.3.10"
},
"dcloudext": {
"sale": {
"regular": {
"price": "0.00"
},
"sourcecode": {
"price": "0.00"
}
},
"contact": {
"qq": ""
},
"declaration": {
"ads": "无",
"data": "插件不采集任何数据",
"permissions": "无"
},
"npmurl": "",
"type": "unicloud-admin"
},
"uni_modules": {
"dependencies": [
"uni-data-checkbox",
"uni-data-picker",
"uni-dateformat",
"uni-easyinput",
"uni-file-picker",
"uni-forms",
"uni-icons",
"uni-pagination",
"uni-table"
],
"encrypt": [],
"platforms": {
"cloud": {
"tcb": "y",
"aliyun": "y",
"alipay": "y"
},
"client": {
"App": {
"app-vue": "y",
"app-nvue": "y"
},
"H5-mobile": {
"Safari": "y",
"Android Browser": "y",
"微信浏览器(Android)": "y",
"QQ浏览器(Android)": "y"
},
"H5-pc": {
"Chrome": "y",
"IE": "y",
"Edge": "y",
"Firefox": "y",
"Safari": "y"
},
"小程序": {
"微信": "y",
"阿里": "y",
"百度": "y",
"字节跳动": "y",
"QQ": "y"
},
"快应用": {
"华为": "y",
"联盟": "y"
},
"Vue": {
"vue2": "y",
"vue3": "y"
}
}
}
}
}

View File

@ -0,0 +1,52 @@
<template>
<view style="position: relative;">
<uni-icons @mouseenter.native="mouseenter" @mouseleave.native="showStableInfo = false"
style="padding:0 10px;color: #a8a8a8;cursor: pointer;" type="info" />
<view v-if="showStableInfo" class="show-stable" :style="{top:`${top}px`,left:`${left}px`}">
<text>{{content}}</text>
</view>
</view>
</template>
<script>
export default {
props: {
content: String,
top: {
type: [Number, String],
default: -60
},
left: {
type: [Number, String],
default: -100
}
},
data() {
return {
showStableInfo: false,
arrowStyle: {}
}
},
methods: {
mouseenter(e) {
this.showStableInfo = true
}
}
}
</script>
<style lang="scss" scoped>
$main_color: #fff;
$main_back_color: #303133;
.show-stable {
width: 200px;
position: absolute;
padding: 5px 10px;
background-color: $main_back_color;
color: $main_color;
border-radius: 4px;
border: 1px solid #e9e9eb;
z-index: 99999;
}
</style>

View File

@ -0,0 +1,209 @@
import {
validator,
enumConverter
} from '@/js_sdk/validator/opendb-app-versions.js';
const platform_iOS = 'iOS';
const platform_Android = 'Android';
const db = uniCloud.database();
function getValidator(fields) {
let reuslt = {}
for (let key in validator) {
if (fields.includes(key)) {
reuslt[key] = validator[key]
}
}
return reuslt
}
export const fields =
'appid,name,title,contents,platform,type,version,min_uni_version,url,stable_publish,is_silently,is_mandatory,create_date,store_list'
export default {
data() {
return {
labelWidth: '100px',
enableiOSWgt: true, // 是否开启iOS的wgt更新
silentlyContent: '静默更新App升级时会在后台下载wgt包并自行安装。新功能在下次启动App时生效',
mandatoryContent: '强制更新App升级弹出框不可取消',
stablePublishContent: '同时只可有一个线上发行版,线上发行不可更设为下线。\n未上线可以设为上线发行并自动替换当前线上发行版',
stablePublishContent2: '使用本包替换当前线上发行版',
uploadFileContent: '可下载安装包地址。上传文件到云存储自动填写,也可以手动填写',
minUniVersionContent: '上次使用新Api或打包新模块的App版本',
priorityContent: '检查更新时按照优先级从大到小依次尝试跳转商店。如果都跳转失败则会打开浏览器使用下载链接下载apk安装包',
latestStableData: [], // 库中最新已上线版
appFileList: null, // 上传包
type_valuetotext: enumConverter.type_valuetotext,
preUrl: '',
formData: {
"appid": "",
"name": "",
"title": "",
"contents": "",
"platform": [],
"store_list": [],
"type": "",
"version": "",
"min_uni_version": "",
"url": "",
"stable_publish": false,
"create_date": null
},
formOptions: {
"platform_localdata": [{
"value": "Android",
"text": "安卓"
},
{
"value": "iOS",
"text": "苹果"
}
],
"type_localdata": [{
"value": "native_app",
"text": "原生App安装包"
},
{
"value": "wgt",
"text": "App资源包"
}
]
},
rules: {
...getValidator([
"appid", "contents", "platform", "type",
"version", "min_uni_version", "url", "stable_publish",
"title", "name", "is_silently", "is_mandatory", "store_list"
])
}
}
},
onReady() {
this.$refs.form.setRules(this.rules)
},
computed: {
isWGT() {
return this.formData.type === 'wgt'
},
isiOS() {
return !this.isWGT ? this.formData.platform.includes(platform_iOS) : false;
},
hasPackage() {
return this.appFileList && !!Object.keys(this.appFileList).length
},
fileExtname() {
return this.isWGT ? ['wgt'] : ['apk']
},
platformLocaldata() {
return !this.isWGT ? this.formOptions.platform_localdata : this.enableiOSWgt ? this.formOptions
.platform_localdata : [this.formOptions.platform_localdata[0]]
},
uni_platform() {
return (this.isiOS ? platform_iOS : platform_Android).toLocaleLowerCase()
}
},
methods: {
getStoreList(appid) {
return db.collection('opendb-app-list')
.where({
appid
})
.get()
.then(res => {
const data = res.result.data[0]
return data ? data.store_list || [] : []
})
},
packageUploadSuccess(res) {
uni.showToast({
icon: 'success',
title: '上传成功',
duration: 800
})
this.preUrl = this.formData.url
this.formData.url = res.tempFilePaths[0]
},
deleteFile(fileList) {
return this.$request('deleteFile', {
fileList
}, {
functionName: 'uni-upgrade-center'
})
},
async packageDelete(res) {
if (!this.hasPackage) return;
await this.deleteFile([res.tempFilePath])
uni.showToast({
icon: 'success',
title: '删除成功',
duration: 800
})
this.formData.url = this.preUrl
this.$refs.form.clearValidate('url')
},
selectFile() {
if (this.hasPackage) {
uni.showToast({
icon: 'none',
title: '只可上传一个文件,请删除已上传后重试',
duration: 1000
});
}
},
createCenterRecord(value) {
return {
...value,
uni_platform: this.uni_platform,
create_env: 'upgrade-center'
}
},
createCenterQuery({
appid
}) {
return {
appid,
create_env: 'upgrade-center'
}
},
createStatQuery({
appid,
type,
version,
uni_platform
}) {
return {
appid,
type,
version,
uni_platform: uni_platform ? uni_platform : this.uni_platform,
create_env: 'uni-stat',
stable_publish: false
}
},
toUrl(url){
// #ifdef H5
window.open(url);
// #endif
// #ifndef H5
uni.showToast({
title: '请在浏览器中打开',
icon: 'none'
});
// #endif
},
getCloudStorageConfig(){
return uni.getStorageSync('uni-admin-cloud-storage-config') || {};
},
setCloudStorageConfig(data={}){
uni.setStorageSync('uni-admin-cloud-storage-config', data);
},
// 临时方法,后面会优化
setCloudStorage(data){
// uniCloud.setCloudStorage 不是标准的API临时挂载在uniCloud对象上的后面会优化
if (typeof uniCloud.setCloudStorage === "function") {
uniCloud.setCloudStorage(data);
}
}
}
}

View File

@ -0,0 +1,26 @@
// 判断arr是否为一个数组返回一个bool值
function isArray(arr) {
return Object.prototype.toString.call(arr) === '[object Array]';
}
// 深度克隆
export function deepClone(obj) {
// 对常见的“非”值,直接返回原来值
if ([null, undefined, NaN, false].includes(obj)) return obj;
if (typeof obj !== "object" && typeof obj !== 'function') {
//原始类型直接返回
return obj;
}
let o = isArray(obj) ? [] : {};
for (let i in obj) {
if (obj.hasOwnProperty(i)) {
o[i] = typeof obj[i] === "object" ? deepClone(obj[i]) : obj[i];
}
}
return o;
}
export const appListDbName = 'opendb-app-list'
export const appVersionListDbName = 'opendb-app-versions'
// 版本列表默认显示应用Appid
export const defaultDisplayApp = ''

View File

@ -0,0 +1,478 @@
<template>
<view class="uni-container">
<view class="uni-header">
<view class="uni-group">
<view class="uni-title">包类型</view>
<view class="uni-sub-title">{{type_valuetotext[formData.type]}}</view>
</view>
</view>
<uni-forms ref="form" :value="formData" validateTrigger="bind" :labelWidth="labelWidth">
<uni-forms-item name="appid" label="AppID" required>
<uni-easyinput :disabled="true" v-model="formData.appid" trim="both" />
</uni-forms-item>
<uni-forms-item name="name" label="应用名称">
<uni-easyinput :disabled="true" v-model="formData.name" trim="both" />
</uni-forms-item>
<uni-forms-item name="title" label="更新标题">
<uni-easyinput placeholder="更新标题" v-model="formData.title" />
</uni-forms-item>
<uni-forms-item name="contents" label="更新内容" required>
<textarea auto-height style="box-sizing: content-box;" :maxlength="-1"
@input="binddata('contents', $event.detail.value)" class="uni-textarea-border"
:value="formData.contents" @update:value="val => formData.contents = val"></textarea>
</uni-forms-item>
<uni-forms-item name="platform" label="平台" required>
<uni-data-checkbox :multiple="isWGT" v-model="formData.platform" :localdata="platformLocaldata" />
</uni-forms-item>
<uni-forms-item name="version" label="版本号" required>
<uni-easyinput v-model="formData.version" placeholder="当前包版本号,必须大于当前线上发行版本号" />
</uni-forms-item>
<uni-forms-item v-if="isWGT" key="min_uni_version" name="min_uni_version" label="原生App最低版本"
:required="isWGT">
<uni-easyinput placeholder="原生App最低版本" v-model="formData.min_uni_version" />
<show-info :content="minUniVersionContent"></show-info>
</uni-forms-item>
<uni-forms-item label="存储选择">
<view class="flex">
<radio-group @change="e => uniFilePickerProvider = e.detail.value" style="width: 100%;">
<view class="flex" style="flex-wrap: nowrap;">
上传至
<label>
<radio value="unicloud" :checked="uniFilePickerProvider === 'unicloud'"/><text>内置存储</text>
</label>
<label style="margin-left: 20rpx;">
<radio value="extStorage" :checked="uniFilePickerProvider === 'extStorage'"/><text>扩展存储</text>
</label>
</view>
</radio-group>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;width: 100%;">内置存储是服务空间开通后自带的云存储不支持自定义域名不支持阶梯计费</text>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">扩展存储支持自定义域名阶梯计费越用越便宜功能更强大</text>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #2979ff;cursor: pointer;text-decoration: underline; margin-left: 10px;" @click="toUrl('https://doc.dcloud.net.cn/uniCloud/ext-storage/service.html')">扩展存储开通文档</text>
</view>
</uni-forms-item>
<uni-forms-item label="自定义域名" v-if="uniFilePickerProvider === 'extStorage'">
<view class="flex" style="flex-direction: column;align-items:flex-start;">
<uni-easyinput placeholder="请输入扩展存储自定义域名" v-model="domain" :maxlength="-1" style="width: 550px;" />
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">输入扩展存储绑定的域名在服务空间-云存储-扩展存储页面可查看cdn.example.com</text>
</view>
</uni-forms-item>
<uni-forms-item v-if="!isiOS" :label="'上传'+fileExtname[0]+'包'">
<uni-file-picker v-model="appFileList" :file-extname="fileExtname" :disabled="hasPackage"
returnType="object" file-mediatype="all" limit="1" @success="packageUploadSuccess" :provider="uniFilePickerProvider"
@delete="packageDelete">
<view class="flex">
<button type="primary" size="mini" @click="selectFile" style="margin: 0px;">选择文件</button>
</view>
<view class="flex">
<text style="margin-top: 10px;font-size: 12px;color: #666;">上传{{fileExtname[0]}}到当前服务空间的云存储中上传成功后会自动使用云存储地址填充下载链接</text>
<text style="margin-top: 10px;font-size: 12px;color: #666;">上传文件后同步到各地cdn缓存节点有延迟请适当等候再提交新版信息入库触发客户端更新提示</text>
</view>
</uni-file-picker>
<text v-if="hasPackage" style="padding-left: 20px;color: #a8a8a8;">{{Number(appFileList.size / 1024 / 1024).toFixed(2)}}M</text>
</uni-forms-item>
<uni-forms-item key="url" name="url" :label="isiOS ? 'AppStore' : '下载链接'" required>
<view class="flex" style="flex-direction: column;align-items:flex-start;flex: 1;">
<view class="flex" style="width: 100%;">
<uni-easyinput placeholder="链接" v-model="formData.url" :maxlength="-1" />
<text style="margin-left: 10px;color: #2979ff;cursor: pointer;text-decoration: underline;" v-if="formData.url" @click="toUrl(formData.url)">测试下载</text>
</view>
<text style="margin-top: 10px;font-size: 12px;color: #666;" v-if="formData.url">建议点击测试下载能正常下载后再进行发布</text>
</view>
</uni-forms-item>
<uni-forms-item v-if="!isiOS && !isWGT && formData.store_list.length" label="Android应用市场" labelWidth="125px"
key="store_list" name="store_list">
<view style="flex: 1;">
<view v-for="(item) in formData.store_list" :key="item.id">
<uni-card style="margin: 0px 0px 20px 0px;">
<view style="display: flex;">
<checkbox-group style="user-select: none;"
@change="({detail:{value}}) => {item.enable = !!value.length}">
<label class="title_padding">
<checkbox value="scheme" :checked="item.enable" />
<text>是否启用</text>
</label>
</checkbox-group>
</view>
<uni-forms-item label="商店名称">
<uni-easyinput disabled v-model="item.name" trim="both"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="Scheme">
<uni-easyinput disabled v-model="item.scheme" trim="both"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="优先级">
<uni-easyinput v-model="item.priority" type="number"></uni-easyinput>
<show-info :top="-100" :left="-180" :content="priorityContent"></show-info>
</uni-forms-item>
</uni-card>
</view>
</view>
</uni-forms-item>
<uni-forms-item v-if="isWGT" key="is_silently" name="is_silently" label="静默更新">
<switch @change="binddata('is_silently', $event.detail.value)" :checked="formData.is_silently" />
<show-info :top="-80" :content="silentlyContent"></show-info>
</uni-forms-item>
<uni-forms-item v-if="!isiOS" key="is_mandatory" name="is_mandatory" label="强制更新">
<switch @change="binddata('is_mandatory', $event.detail.value)" :checked="formData.is_mandatory" />
<show-info :content="mandatoryContent"></show-info>
</uni-forms-item>
<uni-forms-item name="stable_publish" label="上线发行">
<switch @change="binddata('stable_publish', $event.detail.value)" :checked="formData.stable_publish" />
<show-info :top="-40" :content="stablePublishContent2"></show-info>
</uni-forms-item>
<uni-forms-item v-show="false" name="type" label="安装包类型">
<uni-data-checkbox v-model="formData.type" :localdata="formOptions.type_localdata" />
</uni-forms-item>
<view class="uni-button-group">
<button type="primary" class="uni-button" style="width: 100px;" @click="submit">发布</button>
<button type="warn" class="uni-button" style="width: 100px;margin-left: 15px;" @click="back">取消</button>
</view>
</uni-forms>
</view>
</template>
<script>
import {
validator,
enumConverter
} from '@/js_sdk/validator/opendb-app-versions.js';
import addAndDetail, {
fields
} from '../mixin/version_add_detail_mixin.js';
import {
appVersionListDbName
} from '../utils.js';
const db = uniCloud.database();
const dbCmd = db.command;
const dbCollectionName = appVersionListDbName;
const platform_iOS = 'iOS';
const platform_Android = 'Android';
/**
* 对比版本号,如需要,请自行修改判断规则
* 支持比对 ("3.0.0.0.0.1.0.1", "3.0.0.0.0.1") ("3.0.0.1", "3.0") ("3.1.1", "3.1.1.1") 之类的
* @param {Object} v1
* @param {Object} v2
* v1 > v2 return 1
* v1 < v2 return -1
* v1 == v2 return 0
*/
function compare(v1 = '0', v2 = '0') {
v1 = String(v1).split('.')
v2 = String(v2).split('.')
const minVersionLens = Math.min(v1.length, v2.length);
let result = 0;
for (let i = 0; i < minVersionLens; i++) {
const curV1 = Number(v1[i])
const curV2 = Number(v2[i])
if (curV1 > curV2) {
result = 1
break;
} else if (curV1 < curV2) {
result = -1
break;
}
}
if (result === 0 && (v1.length !== v2.length)) {
const v1BiggerThenv2 = v1.length > v2.length;
const maxLensVersion = v1BiggerThenv2 ? v1 : v2;
for (let i = minVersionLens; i < maxLensVersion.length; i++) {
const curVersion = Number(maxLensVersion[i])
if (curVersion > 0) {
v1BiggerThenv2 ? result = 1 : result = -1
break;
}
}
}
return result;
}
export default {
mixins: [addAndDetail],
data() {
return {
latestVersion: '0.0.0',
lastVersionId: '',
uniFilePickerProvider: 'unicloud',
domain: ""
}
},
async onLoad({
appid,
name,
type
}) {
let { domain, provider } = this.getCloudStorageConfig();
if (domain) this.domain = domain;
if (provider) this.uniFilePickerProvider = provider;
if (appid && type && name) {
const store_list = await this.getStoreList(appid)
this.formData = {
...this.formData,
...{
appid,
name,
type,
store_list,
}
}
this.latestStableData = await this.getDetail(appid, type)
// 如果有数据否则为发布第一版默认为Android
if (!this.isWGT && this.latestStableData.length) {
this.setFormData(platform_Android)
}
// 如果是wgt ,则需要将 min_uni_version 设为必填
if (this.isWGT) {
this.rules.min_uni_version.rules.push({
"required": true
})
}
}
},
onUnload() {
// 临时处理,后面会再优化
this.setCloudStorage({
provider: null
});
},
watch: {
isiOS(val) {
if (!val && this.hasPackage) {
this.formData.url = this.appFileList.url
return;
}
this.formData.url = ''
},
"formData.platform"(val) {
this.setFormData(val)
},
"domain"(val) {
this.setCloudStorage({
domain: val
});
if (this.formData.url) {
// 替换 this.formData.url 内的域名
if (!val) val = "请输入自定义域名"
this.formData.url = this.formData.url.replace(/^(https?:\/\/)[^\/]+/, `$1${val}`);
}
},
uniFilePickerProvider:{
immediate: true,
handler(val){
this.setCloudStorage({
provider: val
});
}
}
},
methods: {
setFormData(os) {
uni.showLoading({
mask: true
})
// 每次需初始化 版本 与 id ,因为可能是新增第一版
this.latestVersion = '0.0.0';
this.lastVersionId = ''
const data = this.getData(this.latestStableData, os)[0]
if (data) {
const {
_id,
version,
name,
platform,
min_uni_version,
url
} = data
this.lastVersionId = _id
this.latestVersion = version;
this.formData.name = name
// 如果不是wgt则需要删除 min_uni_version 字段
if (!this.isWGT) {
delete this.formData.min_uni_version;
this.formData.platform = platform[0]
// iOS需要带出上一版本的AppStore链接
if (this.isiOS) {
this.formData.url = url;
}
} else {
this.formData.min_uni_version = min_uni_version
// this.formData.platform = [os]
}
} else if (this.isWGT) {
this.formData.min_uni_version = ''
}
uni.hideLoading()
},
/**
* 触发表单提交
*/
submit() {
uni.showLoading({
mask: true
})
this.$refs.form.validate(['store_list']).then((res) => {
if (compare(this.latestVersion, res.version) >= 0) {
uni.showModal({
content: `版本号必须大于当前已上线版本(${this.latestVersion}`,
showCancel: false
})
throw new Error('版本号必须大于已上线版本(${this.latestVersion}');
}
// 如果不是 wgt 更新,则需将 platform 字段还原为 array
if (!this.isWGT) {
res.platform = [res.platform]
}
if (this.isiOS || this.isWGT) delete res.store_list;
if (res.store_list) {
res.store_list.forEach(item => {
item.priority = parseFloat(item.priority)
})
}
this.submitForm(res)
}).catch((errors) => {
uni.hideLoading()
})
},
async submitForm(value) {
value = this.createCenterRecord(value)
const collectionDB = db.collection(dbCollectionName)
// uni-stat 会创建这些字段 appid
let recordCreateByUniStat = []
if (!this.isWGT) {
recordCreateByUniStat = await this.getDetail(value.appid, value.type, this.createStatQuery(value))
}
let dbOperate
if (!recordCreateByUniStat.length) {
dbOperate = collectionDB.add(value)
} else {
value.create_date = Date.now()
dbOperate = collectionDB.doc(recordCreateByUniStat[0]._id).update(value)
}
// 使用 clientDB 提交数据
dbOperate.then(async (res) => {
// 如果新增版本为上线发行,且之前有该平台的上线发行,则自动将上一版设为下线
if (value.stable_publish && this.lastVersionId) {
await collectionDB.doc(this.lastVersionId).update({
stable_publish: false
})
}
uni.showToast({
title: '新增成功'
})
this.getOpenerEventChannel().emit('refreshData')
setTimeout(() => uni.navigateBack(), 500)
}).catch((err) => {
uni.showModal({
content: err.message || '请求服务失败',
showCancel: false
})
}).finally(() => {
uni.hideLoading()
})
this.setCloudStorageConfig({
provider: this.uniFilePickerProvider,
domain: this.domain,
});
},
/**
* 获取表单数据
* @param {Object} id
*/
getDetail(appid, type, args = {}) {
uni.showLoading({
mask: true
})
return db.collection(dbCollectionName)
.where(
Object.assign({
appid,
type,
stable_publish: true
}, args)
)
.field(fields)
.get()
.then((res) => res.result.data)
.catch((err) => {
uni.showModal({
content: err.message || '请求服务失败',
showCancel: false
})
}).finally(() => {
uni.hideLoading()
})
},
getData(data = [], platform) {
if (typeof platform === 'string') {
return data.filter(item => item.platform.includes(platform))
} else {
return data.filter(item => item.platform.toString() === platform.toString())
}
},
back() {
uni.showModal({
title: '取消发布',
content: this.hasPackage ? '将会删除已上传的包' : undefined,
success: res => {
if (res.confirm) {
// 若已上传包但取消发布,则自动将包删除
if (this.hasPackage) {
this.deleteFile([this.appFileList.url])
}
uni.navigateBack()
}
}
});
}
}
}
</script>
<style lang="scss">
::v-deep .uni-forms-item__content {
display: flex;
align-items: center;
}
.uni-button-group {
& button {
margin-left: 15px;
}
& button:first-child {
margin-left: 0px;
}
}
.title_padding {
padding-bottom: 15px;
display: block;
}
::v-deep .uni-file-picker__files {
max-width: 100%;
}
</style>

View File

@ -0,0 +1,401 @@
<template>
<view class="uni-container">
<view class="uni-header">
<view class="uni-group">
<view class="uni-title">包类型</view>
<view class="uni-sub-title" style="display: flex;justify-content: center;align-items: center;">
{{type_valuetotext[formData.type]}}
</view>
</view>
<view v-if="!isStable" class="uni-group">
<button class="uni-button" type="warn" size="mini" @click="deletePackage">删除</button>
</view>
</view>
<uni-forms ref="form" :value="formData" validateTrigger="bind" :labelWidth="labelWidth">
<uni-forms-item name="appid" label="AppID" required>
<uni-easyinput :disabled="true" v-model="formData.appid" trim="both" />
</uni-forms-item>
<uni-forms-item name="name" label="应用名称">
<uni-easyinput :disabled="true" v-model="formData.name" trim="both" />
</uni-forms-item>
<uni-forms-item name="title" label="更新标题">
<uni-easyinput :disabled="detailsState" placeholder="更新标题" v-model="formData.title" />
</uni-forms-item>
<uni-forms-item name="contents" label="更新内容" required>
<textarea auto-height style="box-sizing: content-box;" :disabled="detailsState"
@input="binddata('contents', $event.detail.value)" class="uni-textarea-border"
:value="formData.contents" @update:value="val => formData.contents = val"></textarea>
</uni-forms-item>
<uni-forms-item name="platform" label="平台" required>
<uni-data-checkbox :disabled="true" :multiple="true" v-model="formData.platform"
:localdata="platformLocaldata" />
</uni-forms-item>
<uni-forms-item name="version" label="版本号" required>
<uni-easyinput :disabled="true" v-model="formData.version" placeholder="当前包版本号,必须大于当前已上线版本号" />
</uni-forms-item>
<uni-forms-item v-if="isWGT" key="min_uni_version" name="min_uni_version" label="原生App最低版本"
:required="isWGT">
<uni-easyinput :disabled="detailsState" placeholder="原生App最低版本" v-model="formData.min_uni_version" />
<show-info :content="minUniVersionContent"></show-info>
</uni-forms-item>
<uni-forms-item label="存储选择" v-if="!detailsState">
<view class="flex">
<radio-group @change="e => uniFilePickerProvider = e.detail.value" style="width: 100%;">
<view class="flex" style="flex-wrap: nowrap;">
上传至
<label>
<radio value="unicloud" :checked="uniFilePickerProvider === 'unicloud'"/><text>内置存储</text>
</label>
<label style="margin-left: 20rpx;">
<radio value="extStorage" :checked="uniFilePickerProvider === 'extStorage'"/><text>扩展存储</text>
</label>
</view>
</radio-group>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;width: 100%;">内置存储是服务空间开通后自带的云存储不支持自定义域名不支持阶梯计费</text>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">扩展存储支持自定义域名阶梯计费越用越便宜功能更强大</text>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #2979ff;cursor: pointer;text-decoration: underline; margin-left: 10px;" @click="toUrl('https://doc.dcloud.net.cn/uniCloud/ext-storage/service.html')">扩展存储开通文档</text>
</view>
</uni-forms-item>
<uni-forms-item label="自定义域名" v-if="uniFilePickerProvider === 'extStorage' && !detailsState">
<view class="flex" style="flex-direction: column;align-items:flex-start;">
<uni-easyinput placeholder="请输入扩展存储自定义域名" v-model="domain" :maxlength="-1" style="width: 550px;"/>
<text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">输入扩展存储绑定的域名在服务空间-云存储-扩展存储页面可查看cdn.example.com</text>
</view>
</uni-forms-item>
<uni-forms-item v-if="!isiOS && !detailsState" :label="'上传'+fileExtname[0]+'包'">
<uni-file-picker v-model="appFileList" :file-extname="fileExtname" :disabled="hasPackage"
returnType="object" file-mediatype="all" limit="1" @success="packageUploadSuccess" :provider="uniFilePickerProvider"
@delete="packageDelete">
<view class="flex">
<button type="primary" size="mini" @click="selectFile" style="margin: 0px;">选择文件</button>
</view>
</uni-file-picker>
<text v-if="hasPackage"
style="padding-left: 20px;color: #a8a8a8;">{{Number(appFileList.size / 1024 / 1024).toFixed(2)}}M</text>
</uni-forms-item>
<uni-forms-item key="url" name="url" :label="isiOS ? 'AppStore' : '下载链接'" required>
<view class="flex" style="flex-direction: column;align-items:flex-start;flex: 1;">
<view class="flex" style="width: 100%;">
<uni-easyinput :disabled="detailsState" placeholder="下载链接" v-model="formData.url" :maxlength="-1" />
<text style="margin-left: 10px;color: #2979ff;cursor: pointer;text-decoration: underline;" v-if="formData.url" @click="toUrl(formData.url)">测试下载</text>
</view>
<text style="margin-top: 10px;font-size: 12px;color: #666;" v-if="formData.url && !detailsState">建议点击测试下载能正常下载后再进行发布</text>
</view>
</uni-forms-item>
<uni-forms-item v-if="!isiOS && !isWGT && formData.store_list.length" label="Android应用市场" key="store_list"
name="store_list" labelWidth="120">
<view style="flex: 1;">
<view v-for="(item,index) in formData.store_list" :key="item.id">
<uni-card style="margin: 0px 0px 20px 0px;">
<view style="display: flex;">
<checkbox-group style="user-select: none;"
@change="({detail:{value}}) => {item.enable = !!value.length}">
<label class="title_padding">
<checkbox :disabled="detailsState" value="scheme" :checked="item.enable" />
<text>是否启用</text>
</label>
</checkbox-group>
</view>
<uni-forms-item label="商店名称">
<uni-easyinput disabled v-model="item.name" trim="both"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="Scheme">
<uni-easyinput disabled v-model="item.scheme" trim="both"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="优先级">
<uni-easyinput :disabled="detailsState" v-model="item.priority" type="number">
</uni-easyinput>
<show-info :top="-100" :left="-180" :content="priorityContent"></show-info>
</uni-forms-item>
</uni-card>
</view>
</view>
</uni-forms-item>
<uni-forms-item v-if="isWGT" key="is_silently" name="is_silently" label="静默更新">
<switch :disabled="detailsState"
@change="binddata('is_silently', $event.detail.value),formData.is_silently=$event.detail.value"
:checked="formData.is_silently" />
<show-info :top="-80" :content="silentlyContent"></show-info>
</uni-forms-item>
<uni-forms-item v-if="!isiOS" key="is_mandatory" name="is_mandatory" label="强制更新">
<switch :disabled="detailsState"
@change="binddata('is_mandatory', $event.detail.value),formData.is_mandatory=$event.detail.value"
:checked="formData.is_mandatory" />
<show-info width="230" :top="-30" :content="mandatoryContent"></show-info>
</uni-forms-item>
<uni-forms-item name="stable_publish" label="上线发行">
<switch :disabled="detailsState || isStable"
@change="binddata('stable_publish', $event.detail.value),formData.stable_publish=$event.detail.value"
:checked="formData.stable_publish" />
<show-info v-if="isStable" :top="-50" width="350" :content="stablePublishContent"></show-info>
<show-info v-else :top="-40" :content="stablePublishContent2"></show-info>
</uni-forms-item>
<uni-forms-item name="create_date" label="上传时间">
<uni-dateformat format="yyyy-MM-dd hh:mm:ss" :date="formData.create_date" :threshold="[0, 0]" />
</uni-forms-item>
<uni-forms-item v-show="false" name="type" label="安装包类型">
<uni-data-checkbox v-model="formData.type" :localdata="formOptions.type_localdata" />
</uni-forms-item>
<view class="uni-button-group">
<button type="primary" class="uni-button" style="width: 100px;" @click="detailsState = false"
v-if="detailsState">修改</button>
<button type="primary" class="uni-button" style="width: 100px;" @click="submit"
v-if="!detailsState">提交</button>
<button type="warn" class="uni-button" style="width: 100px;" @click="cancelEdit"
v-if="!detailsState">取消</button>
<navigator open-type="navigateBack" style="margin-left: 15px;">
<button class="uni-button" style="width: 100px;">返回</button>
</navigator>
</view>
</uni-forms>
</view>
</template>
<script>
import {
validator,
enumConverter
} from '@/js_sdk/validator/opendb-app-versions.js';
import addAndDetail, {
fields
} from '../mixin/version_add_detail_mixin.js'
import {
deepClone,
appVersionListDbName
} from '../utils.js'
const db = uniCloud.database();
const dbCmd = db.command;
const dbCollectionName = appVersionListDbName;
const platform_iOS = 'iOS';
const platform_Android = 'Android';
function getValidator(fields) {
let reuslt = {}
for (let key in validator) {
if (fields.includes(key)) {
reuslt[key] = validator[key]
}
}
return reuslt
}
export default {
mixins: [addAndDetail],
data() {
return {
showStableInfo: false,
isStable: true, // 是否是线上发行版
originalData: {}, // 原始数据,用于恢复状态
detailsState: true, // 查看状态,
uniFilePickerProvider: 'unicloud',
domain: ""
}
},
async onLoad(e) {
let { domain, provider } = this.getCloudStorageConfig();
if (domain) this.domain = domain;
if (provider) this.uniFilePickerProvider = provider;
const id = e.id
this.formDataId = id
await this.getDetail(id)
this.isStable = this.formData.stable_publish;
this.latestStableData = await this.getLatestVersion();
if (this.isWGT) {
this.rules.min_uni_version.rules.push({
"required": true
})
}
},
onUnload() {
// 临时处理,后面会再优化
this.setCloudStorage({
provider: null
});
},
watch: {
"domain"(val) {
this.setCloudStorage({
domain: val
});
if (this.formData.url) {
// 替换 this.formData.url 内的域名
if (!val) val = "请输入自定义域名"
this.formData.url = this.formData.url.replace(/^(https?:\/\/)[^\/]+/, `$1${val}`);
}
},
uniFilePickerProvider(val){
this.setCloudStorage({
provider: val
});
}
},
methods: {
/**
* 触发表单提交
*/
submit() {
uni.showLoading({
mask: true
})
this.$refs.form.validate(['store_list']).then((res) => {
if (res.store_list) {
res.store_list.forEach(item => {
item.priority = parseFloat(item.priority)
})
}
this.submitForm(res)
}).catch((errors) => {
uni.hideLoading()
})
},
async submitForm(value) {
const collectionDB = db.collection(dbCollectionName)
// 使用 clientDB 提交数据
collectionDB.doc(this.formDataId).update(value).then(async (res) => {
// 如果不是线上发行版,则在设置为上线发行时,需将之前的已上线版设为下线
if (!this.isStable && value.stable_publish === true && this.latestStableData) {
await collectionDB.doc(this.latestStableData._id).update({
stable_publish: false
})
}
uni.showToast({
title: '修改成功'
})
this.getOpenerEventChannel().emit('refreshData')
setTimeout(() => uni.navigateBack(), 500)
}).catch((err) => {
uni.showModal({
content: err.message || '请求服务失败',
showCancel: false
})
}).finally(() => {
uni.hideLoading()
})
},
/**
* 获取表单数据
* @param {Object} id
*/
getDetail(id) {
uni.showLoading({
mask: true
})
return db.collection(dbCollectionName)
.doc(id)
.field(fields)
.get()
.then((res) => {
const data = res.result.data[0]
if (data) {
if (!data.store_list) data.store_list = []
this.formData = data
this.originalData = deepClone(this.formData)
}
}).catch((err) => {
uni.showModal({
content: err.message || '请求服务失败',
showCancel: false
})
}).finally(() => {
uni.hideLoading()
})
},
deletePackage() {
uni.showModal({
title: '提示',
content: '是否删除该版本',
success: res => {
if (res.confirm) {
uni.showLoading({
mask: true
})
db.collection(dbCollectionName).doc(this.formDataId).remove()
.then(() => {
uni.showToast({
title: '删除成功'
})
this.getOpenerEventChannel().emit('refreshData')
setTimeout(() => uni.navigateBack(), 500)
}).catch((err) => {
uni.showModal({
content: err.message || '请求服务失败',
showCancel: false
})
}).finally(() => {
uni.hideLoading()
})
}
}
});
},
async getLatestVersion() {
const where = {
appid: this.formData.appid,
type: this.formData.type,
stable_publish: true
};
if (!this.isWGT) {
where.platform = this.formData.platform[0]
}
const latestStableData = await db.collection(dbCollectionName).where(where).get()
return latestStableData.result.data.find(item => item.platform.toString() === this.formData.platform
.toString());
},
cancelEdit() {
let content = '';
!this.isiOS && this.hasPackage ? content += '\n将会删除已上传的包' : '';
uni.showModal({
title: '取消修改',
content,
success: res => {
if (res.confirm) {
this.formData = deepClone(this.originalData)
this.detailsState = true
if (this.hasPackage) {
this.deleteFile([this.appFileList.url])
}
}
}
});
}
}
}
</script>
<style lang="scss">
.show-stable-info {
position: absolute;
left: 165px;
padding: 5px 10px;
background-color: #f4f4f5;
color: #909399;
border-radius: 4px;
border: 1px solid #e9e9eb;
}
::v-deep .uni-forms-item__content {
display: flex;
align-items: center;
}
.uni-button-group {
& button {
margin-left: 15px;
}
& button:first-child {
margin-left: 0px;
}
}
</style>

View File

@ -0,0 +1,364 @@
<template>
<view class="main">
<view v-if="loaded">
<view class="uni-header">
<view class="uni-group">
<view class="uni-sub-title">当前应用</view>
<view class="uni-title app-list">
<picker @change="(e) => showAppIndex = e.detail.value" :value="showAppIndex"
:range="appNameList">
<view class="uni-input" style="font-size: 14px;">
{{appNameList[showAppIndex]}}
<uni-icons type="bottom"></uni-icons>
</view>
</picker>
</view>
</view>
<view class="uni-group">
<input class="uni-search" type="text" v-model="query" @confirm="search" placeholder="请输入搜索内容" />
<button class="uni-button" type="default" size="mini" @click="search">搜索</button>
<button class="uni-button publish" type="primary" size="mini" @click="publish">发布新版</button>
<button class="uni-button" type="warn" size="mini" :disabled="!selectedIndexs.length"
@click="delTable">批量删除</button>
</view>
</view>
<view class="uni-container">
<unicloud-db ref="udb" :collection="appVersionListDbName"
field="store_list,appid,contents,platform,type,version,min_uni_version,url,stable_publish,create_date,title,name"
:where="where" page-data="replace" :orderby="orderby" :getcount="true" :page-size="options.pageSize"
:page-current="options.pageCurrent" v-slot:default="{data,pagination,loading,error,options}"
:options="options">
<uni-table style="overflow-y: hidden;" :loading="loading" :emptyText="error.message || '没有更多数据'"
border stripe type="selection" @selection-change="selectionChange">
<uni-tr>
<uni-th align="center">AppID</uni-th>
<uni-th align="center">更新标题</uni-th>
<uni-th align="center">安装包类型</uni-th>
<uni-th align="center">平台</uni-th>
<uni-th align="center">已上架应用市场</uni-th>
<uni-th align="center">版本号</uni-th>
<uni-th align="center">安装包状态</uni-th>
<uni-th align="center">上传时间</uni-th>
<uni-th align="center">操作</uni-th>
</uni-tr>
<uni-tr v-for="(item,index) in data" :key="index" :disabled="item.stable_publish">
<uni-td align="center"> {{item.appid}} </uni-td>
<uni-td align="center"> {{item.title || '-'}} </uni-td>
<uni-td align="center">
<text :style="{
padding: '5px 8px',
backgroundColor: item.type === 'wgt' ? '#f0f9eb' : '#ecf5ff',
color: item.type === 'wgt' ? '#67c23a' : '#409eff',
border: `1px solid ${item.type === 'wgt' ? '#e1f3d8' : '#d9ecff'}`,
borderRadius: '4px'
}">{{options.type_valuetotext[item.type]}}</text>
</uni-td>
<uni-td align="center">
<uni-data-picker :localdata="options.platform_valuetotext" :value="item.platform"
:border="false" :readonly="true" split="," />
</uni-td>
<uni-td align="center">
<text>{{store_list_key(item.store_list)}}</text>
</uni-td>
<uni-td align="center"> {{item.version}} </uni-td>
<uni-td align="center"> {{item.stable_publish == true ? '已上线' : '已下线'}} </uni-td>
<uni-td align="center">
<uni-dateformat format="yyyy-MM-dd hh:mm:ss" :date="item.create_date"
:threshold="[0, 0]" />
</uni-td>
<uni-td align="center">
<button @click="navigateTo('./detail?id='+item._id, false)" class="uni-button" size="mini" type="primary">详情</button>
</uni-td>
</uni-tr>
</uni-table>
<view class="uni-pagination-box">
<uni-pagination show-icon :page-size="pagination.size" v-model="pagination.current"
:total="pagination.count" @change="onPageChanged" />
</view>
</unicloud-db>
</view>
</view>
<view v-else class="page-loading" :style="containerTop">
<i class="uni-icon_toast uni-loading"></i>
</view>
</view>
</template>
<script>
import {
enumConverter
} from '@/js_sdk/validator/opendb-app-versions.js';
import {
appListDbName,
appVersionListDbName,
defaultDisplayApp
} from '../utils.js'
import {
mapState
} from 'vuex'
const db = uniCloud.database()
const dbCmd = db.command
// 表查询配置
const dbOrderBy = 'stable_publish desc,create_date desc' // 排序字段
const dbSearchFields = ['name', 'title', 'stable_publish', 'type'] // 模糊搜索字段,支持模糊搜索的字段列表
// 分页配置
const pageSize = 20
const pageCurrent = 1
const appidKey = '__app_version_appid'
const nameKey = '__app_version_name'
function getScreenHeight() {
return document.documentElement ? document.documentElement.clientHeight : window.innerHeight;
}
function createListQuery(condition = {}) {
return {
create_env: dbCmd.neq("uni-stat"),
...condition
}
}
export default {
data() {
return {
backButtonHover: false,
appVersionListDbName,
currentAppid: '',
currentAppName: '',
query: '',
where: '',
orderby: dbOrderBy,
selectedIndexs: [],
options: {
pageSize,
pageCurrent,
...enumConverter
},
imageStyles: {
width: 64,
height: 64
},
loaded: false,
containerTop: {},
appList: [],
showAppIndex: 0
}
},
async onLoad({
appid
}) {
await this.getAppList()
if (!this.appList.length) {
this.showModalToAppManager()
return
}
this.loaded = true
this.appList.forEach((item, index) => {
if (item.appid === appid || defaultDisplayApp) {
this.showAppIndex = index
}
})
this.setAppInfo(this.showAppIndex)
this.where = createListQuery({
appid: this.currentAppid
})
},
computed: {
...mapState('app', ['appid']),
appNameList() {
return this.appList.map(item => item.name)
}
},
watch: {
showAppIndex(val) {
this.setAppInfo(val)
this.where = createListQuery({
appid: this.currentAppid
})
}
},
onReady() {
this.containerTop.height = `${getScreenHeight()}px`
},
methods: {
setAppInfo(index) {
this.currentAppid = this.appList[index].appid
this.currentAppName = this.appList[index].name
},
navigateBack() {
uni.navigateBack()
},
getWhere() {
const query = this.query.trim()
if (!query) {
return ''
}
const queryRe = new RegExp(query, 'i')
return dbSearchFields.map(name => queryRe + '.test(' + name + ')').join(' || ')
},
search() {
const newWhere = this.getWhere()
const isSameWhere = newWhere === this.where
this.where = newWhere
if (this.where) {
this.where = `(${this.where}) && `
}
this.where += `${new RegExp(this.currentAppid, 'i')}.test(appid)`
if (isSameWhere) { // 相同条件时,手动强制刷新
this.loadData()
}
},
loadData(clear = true) {
this.$refs.udb.loadData({
clear
})
},
onPageChanged(e) {
this.$refs.udb.loadData({
current: e.current
})
},
navigateTo(url, clear) {
// clear 表示刷新列表时是否清除页码true 表示刷新并回到列表第 1 页,默认为 true
uni.navigateTo({
url,
events: {
refreshData: () => {
this.loadData(clear)
}
}
})
},
// 多选处理
selectedItems() {
let dataList = this.$refs.udb.dataList
return this.selectedIndexs.map(i => dataList[i]._id)
},
// 批量删除
delTable() {
this.$refs.udb.remove(this.selectedItems())
},
// 多选
selectionChange(e) {
this.selectedIndexs = e.detail.index
},
confirmDelete(id) {
this.$refs.udb.remove(id)
},
publish(e) {
// #ifdef H5
const {
top,
left,
width,
height
} = document.querySelector('.uni-button.publish').getBoundingClientRect()
// #endif
const platforms = Object.keys(this.options.type_valuetotext)
uni.showActionSheet({
itemList: Object.values(this.options.type_valuetotext),
// #ifdef H5
popover: {
top: top + height,
left,
width
},
// #endif
success: async (res) => {
this.navigateTo(
`./add?appid=${this.currentAppid}&name=${this.currentAppName}&type=${platforms[res.tapIndex]}`
)
}
});
},
async getAppList() {
try {
const {
result
} = await db.collection(appListDbName).get()
if (result && result.data && result.data.length > 0) {
this.appList = result.data.filter(item => item.appid !== this.appid)
} else {
this.showModalToAppManager()
}
} catch (e) {
const arr = ['TOKEN_INVALID_TOKEN_EXPIRED', 'TOKEN_INVALID_ANONYMOUS_USER']
if (arr.indexOf(e.code) === -1)
this.showModalToAppManager()
}
},
showModalToAppManager() {
let timer = null
let second = 3
function jump() {
uni.navigateTo({
url: '/pages/system/app/list'
})
clearInterval(timer)
}
timer = setInterval(() => {
if (--second <= 0) {
jump()
}
}, 1000)
uni.showModal({
title: '请先添加应用',
content: '即将跳转至应用管理……',
showCancel: false,
confirmText: '立即跳转',
success: (res) => jump()
})
},
store_list_key(store_list) {
const arr = store_list ? store_list.filter(item => item.enable) : []
return arr.length ?
arr.sort((a, b) => b.priority - a.priority)
.map(item => item.name).join(',') :
'-'
}
}
}
</script>
<style lang="scss">
.page-loading {
display: flex;
justify-content: center;
align-items: center;
flex-direction: column;
flex: 1;
i {
$icon-size: 80rpx;
width: $icon-size;
height: $icon-size;
}
}
page,
page .main,
.page-loading {
height: 100%;
}
.app-list {
display: flex;
justify-content: space-between;
align-items: center;
padding: 5px 10px;
border-radius: 4px;
border: 1px solid #2e76ba;
color: #3A8EE6;
uni-text {
margin-left: 10px;
}
}
</style>

View File

@ -0,0 +1,233 @@
## uni-admin 1.9.3+ 已内置,此插件不再维护 [点击查看文档](https://uniapp.dcloud.net.cn/uniCloud/upgrade-center.html)
- `uni-admin < 1.9.0`:请前往 [Gitee](https://gitee.com/dcloud/uni-upgrade-center/releases) 下载 `tag v0.4.2` 版本使用
-`1.9.0 <= uni-admin < 1.9.2` :请前往 [Gitee](https://gitee.com/dcloud/uni-upgrade-center/releases) 下载 `tag v0.5.1` 版本使用
- `uni-admin >= 1.9.3` uni-admin 已内置 升级中心,直接使用即可 [详情](https://uniapp.dcloud.io/uniCloud/admin.html#app-manager)。并且云函数 `upgrade-center` 废弃,使用 `uni-upgrade-center` 云函数。
# uni-upgrade-center - Admin
### 概述
> 统一管理App及App在`Android`、`iOS`平台上`App安装包`和`wgt资源包`的发布升级
> 本插件为uni升级中心后台管理系统客户端检查更新插件请点击查看 [uni-upgrade-center-app](https://ext.dcloud.net.cn/plugin?id=4542)
### 基于uniCloud的App升级中心本插件具有如下特征
- 云端基于uniCloud云函数实现
- 数据库遵循opendb规范
- 遵循uni-Admin框架规范可直接导入uni-admin项目中
- 支持App整包升级及wgt资源包升级
## 升级中心解决了什么问题?
升级中心是一款uni-admin插件负责App版本更新业务。包含后台管理界面、更新检查逻辑App内只要调用弹出提示即可。
升级中心有以下功能点:
- 应用管理对App的信息记录和应用版本管理
- 版本管理可以发布新版也可方便直观的对当前App历史版本以及线上发行版本进行查看、编辑和删除操作
- 版本发布信息管理,包括 更新标题,更新内容,版本号,静默更新,强制更新,灵活上线发行 的设置和修改
- 原生App安装包发布Apk更新用于App的整包更新可设置是否强制更新
- wgt资源包发布wgt更新用于App的热更新可设置是否强制更新静默更新
- App管理列表及App版本记录列表搜索
只需导入插件,初始化数据库即可拥有上述功能。
您也可以自己修改逻辑自定义数据库字段,和随意定制 UI 样式。
## 安装指引
1. 使用`HBuilderX 3.1.0+`,因为要使用到`uni_modules`
2. 使用已有`uniCloud-admin`项目或新建项目:`打开HBuilderX` -> `文件` -> `新建` -> `项目` -> `uni-app` 选择 `uniCloud admin`模板,键入一个名字,确定
3. 鼠标右键选择`关联云服务空间``运行云服务空间初始化向导`
3. 在插件市场打开本插件页面,在右侧点击`使用 HBuilderX 导入插件`,选择 `uniCloud admin` 项目点击确定
4. 等待下载安装完毕。由于本插件依赖一些uni-ui插件下载完成后会显示合并插件页面自行选择即可
5. 找到`/uni_modules/uni-upgrade-center/uniCloud/cloudfunctions/upgrade-center`,右键上传部署
7.`pages.json`中添加页面路径
```json
//此结构与uniCloud admin中的pages.json结构一致
{
"pages": [
// ……其他页面配置
{
"path": "uni_modules/uni-upgrade-center/pages/version/list",
"style": {
"navigationBarTitleText": "版本列表"
}
}, {
"path": "uni_modules/uni-upgrade-center/pages/version/add",
"style": {
"navigationBarTitleText": "新版发布"
}
}, {
"path": "uni_modules/uni-upgrade-center/pages/version/detail",
"style": {
"navigationBarTitleText": "版本信息查看"
}
}
]
}
```
8.`manifest.json -> 源码视图`中添加以下配置:
```js
"networkTimeout":{
"uploadFile":1200000 //ms 如果不配置,上传大文件可能会超时
}
```
9. 运行项目到`Chrome`
10. 添加菜单
- `vue2`
运行起来uniCloud admin菜单管理模块会自动读取`/uni_modules/uni-upgrade-center/menu.json`文件中的菜单配置,生成【待添加菜单】,选中升级中心,点击`添加选中的菜单`即可
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/16dc338e-7d5b-4290-98a9-adb7f0c23754.png" width="800"></img>
</div>
- `vue3`
可将 `/uni_modules/uni-upgrade-center/menu.json` 拷贝至 `uniCloud/database/db_init.json` 中的 `opendb-admin-menus` 节点下,并右键初始化数据库即可。
11. 添加成功后,就可以在左侧的菜单栏中找到`升级中心`菜单
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/fcf04804-0c4c-4342-9a8b-dfc273dfc83c.png" width="300"></img>
</div>
12. 在进入`升级中心`之前:
1. 需要到`uni-admin`的`应用管理`中添加一个应用,才可以在`升级中心`中发布对应应用的版本。
2. 当你有多个应用时,可以在`/uni_modules/uni-upgrade-center/pages/utils.js`中修改`defaultDisplayApp`字段来设置默认显示应用的`appid`。
3. 如果不设置或设置应用不存在则默认从数据库中查出来的第一个应用。
13. 由于插件依赖的uni-ui的一些组件建议右键`/uni_modules/uni-upgrade-center`安装一下第三方依赖,否则可能会出现一些问题
14. 运行在`uniCloud`,由于本插件使用了`clientDB`,因此可能需要配置一下`uni-config-center插件`关于`uni-id`的配置信息。如提示`公用模块uni-id缺少配置信息`请这样做:
1. 点击[uni-config-center](https://ext.dcloud.net.cn/plugin?id=4425)导入插件
2. 在`/uniCloud/cloudfunctions/common/uni-config-center/`下创建`uni-id`文件夹,文件夹内创建`config.json`文件。
3. 点击[config.json默认配置](https://uniapp.dcloud.net.cn/uniCloud/uni-id?id=start)。将内容拷贝至`config.json`中。**注:一定要把注释去除!**
## 使用指南
### 升级中心
#### 应用列表
1. 点击菜单 `应用管理`,这里展示你所添加的 App点击右上角 `新增` 可以新增一个 App
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/7f85aa6a-eff3-4cc6-bb32-9feaaeaf97d0.png" width="400"></img>
</div>
2. 将App的信息都填写完善后你可以在列表的操作列进行`修改`应用信息或者`删除`该应用。
**Tips**
- 删除应用会把该应用的所有版本记录同时删除
#### 版本管理
1. 在版本管理list的右上角点击`发布新版`,可以发布`原生App安装包`和`wgt资源包`。在左上角点击`下拉列表`,可以切换展示应用。
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/442e84e7-e7f3-4d27-9c98-45568e5db835.png" width="800"></img>
</div>
- #### 发布原生App安装包
1. 在上传安装包界面填写此次发版信息
<div align="center" >
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/67932ae3-1a7a-4f21-9849-ba3bcc500c36.png" width="400"></img>
</div>
2. `包地址`
- 可以选择手动上传一个文件到 `云存储`,会自动将地址填入该项
- 也可以手动填写一个地址,就可以不用再上传文件
- 如果是发布`苹果`版本,包地址则为 应用在`AppStore的链接`
3. `强制更新`
- 如果使用强制更新App端接收到该字段后App升级弹出框不可取消
4. `上线发行`
- 可设置当前包是否上线发行,只有已上线才会进行更新检测
- 同时只可有一个线上发行版,线上发行不可更设为下线。未上线可以设为上线发行并自动替换当前线上发行版
- 修改当前包为上线发行,自动替换当前线上发行版
**注:版本号请填写以`.`分隔字符串,例如:`0.0.1`**
- #### 发布wgt资源包
1. 大部分配置与发布 `原生App安装包` 一致
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/ec916cde-0d0e-4bf3-a735-643ea2a45b74.png" width="400"></img>
</div>
2. `原生App最低版本`
- 上次使用新Api或打包新模块的App版本
- 如果此次打包wgt使用了`新的api`或者打包了`新的模块`,则在发布 `wgt资源包` 的时候,将此版本更新为本次版本
- 如果已有正式版`wgt资源包`,则本次新增会自动带出
2. `静默更新`
- App升级时会在后台下载wgt包并自行安装。新功能在下次启动App时生效
- **静默更新后不重启应用,可能会导致正在访问的应用的页面数据错乱,请谨慎使用!**
**注:版本号请填写以`.`分隔字符串,例如:`0.0.1`**
- #### 发布完成页面
<div align="center">
<img src="https://vkceyugu.cdn.bspapp.com/VKCEYUGU-a90b5f95-90ba-4d30-a6a7-cd4d057327db/c5470d8c-cc37-4b41-8d56-6d50f8daac62.png" width="800"></img>
</div>
**Tips**
1. `pages/system/upgradecenter/version/add.vue`中有版本对比函数compare
- 使用多段式版本格式(如:"3.0.0.0.0.1.0.1", "3.0.0.0.0.1")。如果不满足对比规则,请自行修改。
## 项目代码说明
### uniCloud 数据表
数据表基于 [openDB](https://gitee.com/dcloud/opendb/tree/master) 规范,它约定了一个标准用户表的表名和字段定义,并且基于 nosql 的特性,可以由开发者自行扩展字段。
本项目用到了 2 个表:
- opendb-app-listapp管理列表。记录应用的 appid、name、description 用于展示。[详见](https://gitee.com/dcloud/opendb/tree/master/collection/opendb-app-list)
- opendb-app-versions应用版本管理表。记录管理应用的版本信息。[详见](https://gitee.com/dcloud/opendb/tree/master/collection/opendb-app-versions)
### 前端页面
点击`升级中心`,会进入应用管理列表,在这里你可以新增应用,或者在`应用详情`中查看、修改或删除一个已经录入的应用。
在应用管理列表中点击某个应用的`版本管理`,进入该应用的所有版本记录。列表排序为:先排序已上线版本,剩下已下线版本根据创建时间排列。
在应用版本列表中点击`详情`,即可进入该版本的信息详情中查看、修改或删除该记录。
**Tips**
- 升级中心设计之初就支持iOS的wgt更新
- iOS的wgt更新肯定是违反apple政策的注意事项
- 审核期间请不要弹窗升级
- 升级完后尽量不要自行重启
- 尽量使用静默更新
- 可以通过以下修改支持iOS的wgt更新
> \uni_modules\uni-upgrade-center\pages\mixin\version_add_detail_mixin.js
>
> 将 `data` 中的 `enableiOSWgt: false` 中 改为 `enableiOSWgt: true`
**常见问题**
- 以下问题可以通过升级插件版本解决:
- createdate不与默认值匹配
- ["create_date"]在数据库中并不存在
- 提交的字段["dirty_data"]在数据库中并不存在
- 集合[opendb-app-list]对应的schema内存在错误详细信息opendb-app-list表对应的schema名称冲突这是什么意思呢
- 没有/找不到 [opendb-app-list] 集合/表。**解决方案:**升级 admin 至 1.6.0+ 即可
- 测试时发布了高版本的包,测试完了发布包提示需要大于版本号 (x.x.x)。**解决方案:**直接在控制台修改数据库

View File

@ -0,0 +1,329 @@
{
"bsonType": "object",
"required": [
"appid",
"name"
],
"permission": {
"read": false,
"create": false,
"update": false,
"delete": false
},
"properties": {
"_id": {
"description": "ID系统自动生成"
},
"appid": {
"bsonType": "string",
"description": "应用的AppID",
"label": "AppID",
"componentForEdit": {
"name": "uni-easyinput",
"props": {
":disabled": true
}
}
},
"name": {
"bsonType": "string",
"description": "应用名称",
"label": "应用名称",
"componentForEdit": {
"name": "uni-easyinput",
"props": {
":disabled": true
}
}
},
"description": {
"bsonType": "string",
"description": "应用描述",
"label": "应用描述",
"componentForEdit": {
"name": "textarea"
},
"componentForShow": {
"name": "textarea",
"props": {
":disabled": true
}
}
},
"creator_uid": {
"description": "创建者的user_id创建者必然是用户不随应用转让而改变",
"bsonType": "string"
},
"owner_type": {
"bsonType": "int",
"description": "应用当前归属者类型1个人2企业"
},
"owner_id": {
"bsonType": "string",
"description": "应用当前归属者的iduser_id or enterprise_id"
},
"managers": {
"bsonType": "array",
"description": "应用管理员ID列表"
},
"members": {
"bsonType": "array",
"description": "团队成员ID列表"
},
"icon_url": {
"bsonType": "string",
"trim": "both",
"description": "应用图标链接",
"label": "应用图标"
},
"introduction": {
"bsonType": "string",
"trim": "both",
"description": "应用简介",
"label": "应用简介",
"componentForEdit": {
"name": "uni-easyinput",
"props": {
"disabled": true
}
}
},
"screenshot": {
"bsonType": "array",
"description": "应用截图",
"label": "应用截图"
},
"app_android": {
"bsonType": "object",
"description": "安卓 App 相关信息",
"properties": {
"name": {
"bsonType": "string",
"description": "快应用名称",
"label": "快应用名称"
},
"url": {
"bsonType": "string",
"description": "安卓可下载安装包地址",
"label": "安卓下载地址"
}
}
},
"app_ios": {
"bsonType": "object",
"description": "苹果 App 相关信息",
"properties": {
"name": {
"bsonType": "string",
"description": "快应用名称",
"label": "快应用名称"
},
"url": {
"bsonType": "string",
"description": "AppStore 上架地址",
"label": "AppStore 地址"
}
}
},
"mp_weixin": {
"bsonType": "object",
"description": "微信小程序相关信息",
"label": "微信小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_alipay": {
"bsonType": "object",
"description": "支付宝小程序相关信息",
"label": "支付宝小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_baidu": {
"bsonType": "object",
"description": "百度小程序相关信息",
"label": "百度小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_toutiao": {
"bsonType": "object",
"description": "头条小程序相关信息",
"label": "头条小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_qq": {
"bsonType": "object",
"description": "QQ小程序相关信息",
"label": "QQ小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_kuaishou": {
"bsonType": "object",
"description": "快手小程序相关信息",
"label": "快手小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_lark": {
"bsonType": "object",
"description": "飞书小程序相关信息",
"label": "飞书小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_jd": {
"bsonType": "object",
"description": "京东小程序相关信息",
"label": "京东小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"mp_dingtalk": {
"bsonType": "object",
"description": "钉钉小程序相关信息",
"label": "钉钉小程序",
"properties": {
"name": {
"bsonType": "string",
"description": "小程序名字"
},
"qrcode_url": {
"bsonType": "string",
"description": "二维码url"
}
}
},
"h5": {
"bsonType": "object",
"properties": {
"url": {
"bsonType": "string",
"description": "H5 可访问链接"
}
}
},
"quickapp": {
"bsonType": "object",
"properties": {
"name": {
"bsonType": "string",
"description": "快应用名称",
"label": "快应用名称"
},
"qrcode_url": {
"bsonType": "string",
"description": "快应用二维码url"
}
}
},
"store_list": {
"bsonType": "array",
"description": "发布的应用市场",
"label": "应用市场",
"properties": {
"id": {
"bsonType": "string",
"description": "应用id自动生成",
"label": "id"
},
"name": {
"bsonType": "string",
"description": "应用名称",
"label": "应用名称"
},
"scheme": {
"bsonType": "string",
"description": "应用 scheme",
"label": "应用 scheme"
},
"enable": {
"bsonType": "bool",
"description": "是否启用"
},
"priority": {
"bsonType": "int",
"description": "按照从大到小排序",
"label": "优先级"
}
}
},
"create_date": {
"bsonType": "timestamp",
"label": "创建时间",
"forceDefaultValue": {
"$env": "now"
},
"componentForEdit": {
"name": "uni-dateformat"
}
}
},
"version": "0.0.1"
}