detail.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. <template>
  2. <view class="uni-container">
  3. <view class="uni-header">
  4. <view class="uni-group">
  5. <view class="uni-title">包类型</view>
  6. <view class="uni-sub-title" style="display: flex;justify-content: center;align-items: center;">
  7. {{type_valuetotext[formData.type]}}
  8. </view>
  9. </view>
  10. <view v-if="!isStable" class="uni-group">
  11. <button class="uni-button" type="warn" size="mini" @click="deletePackage">删除</button>
  12. </view>
  13. </view>
  14. <uni-forms ref="form" :value="formData" validateTrigger="bind" :labelWidth="labelWidth">
  15. <uni-forms-item name="appid" label="AppID" required>
  16. <uni-easyinput :disabled="true" v-model="formData.appid" trim="both" />
  17. </uni-forms-item>
  18. <uni-forms-item name="name" label="应用名称">
  19. <uni-easyinput :disabled="true" v-model="formData.name" trim="both" />
  20. </uni-forms-item>
  21. <uni-forms-item name="title" label="更新标题">
  22. <uni-easyinput :disabled="detailsState" placeholder="更新标题" v-model="formData.title" />
  23. </uni-forms-item>
  24. <uni-forms-item name="contents" label="更新内容" required>
  25. <textarea auto-height style="box-sizing: content-box;" :disabled="detailsState"
  26. @input="binddata('contents', $event.detail.value)" class="uni-textarea-border"
  27. :value="formData.contents" @update:value="val => formData.contents = val"></textarea>
  28. </uni-forms-item>
  29. <uni-forms-item name="platform" label="平台" required>
  30. <uni-data-checkbox :disabled="true" :multiple="true" v-model="formData.platform"
  31. :localdata="platformLocaldata" />
  32. </uni-forms-item>
  33. <uni-forms-item name="version" label="版本号" required>
  34. <uni-easyinput :disabled="true" v-model="formData.version" placeholder="当前包版本号,必须大于当前已上线版本号" />
  35. </uni-forms-item>
  36. <uni-forms-item v-if="isWGT" key="min_uni_version" name="min_uni_version" label="原生App最低版本"
  37. :required="isWGT">
  38. <uni-easyinput :disabled="detailsState" placeholder="原生App最低版本" v-model="formData.min_uni_version" />
  39. <show-info :content="minUniVersionContent"></show-info>
  40. </uni-forms-item>
  41. <uni-forms-item label="存储选择" v-if="!detailsState">
  42. <view class="flex">
  43. <radio-group @change="e => uniFilePickerProvider = e.detail.value" style="width: 100%;">
  44. <view class="flex" style="flex-wrap: nowrap;">
  45. 上传至:
  46. <label>
  47. <radio value="unicloud" :checked="uniFilePickerProvider === 'unicloud'"/><text>内置存储</text>
  48. </label>
  49. <label style="margin-left: 20rpx;">
  50. <radio value="extStorage" :checked="uniFilePickerProvider === 'extStorage'"/><text>扩展存储</text>
  51. </label>
  52. </view>
  53. </radio-group>
  54. <text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;width: 100%;">内置存储是服务空间开通后自带的云存储,不支持自定义域名,不支持阶梯计费</text>
  55. <text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">扩展存储支持自定义域名、阶梯计费,越用越便宜、功能更强大</text>
  56. <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>
  57. </view>
  58. </uni-forms-item>
  59. <uni-forms-item label="自定义域名" v-if="uniFilePickerProvider === 'extStorage' && !detailsState">
  60. <view class="flex" style="flex-direction: column;align-items:flex-start;">
  61. <uni-easyinput placeholder="请输入扩展存储自定义域名" v-model="domain" :maxlength="-1" style="width: 550px;"/>
  62. <text class="uni-sub-title" style="margin-top: 10px;font-size: 12px;color: #666;">输入扩展存储绑定的域名,在服务空间-云存储-扩展存储页面可查看,如:cdn.example.com</text>
  63. </view>
  64. </uni-forms-item>
  65. <uni-forms-item v-if="!isiOS && !detailsState" :label="'上传'+fileExtname[0]+'包'">
  66. <uni-file-picker v-model="appFileList" :file-extname="fileExtname" :disabled="hasPackage"
  67. returnType="object" file-mediatype="all" limit="1" @success="packageUploadSuccess" :provider="uniFilePickerProvider"
  68. @delete="packageDelete">
  69. <view class="flex">
  70. <button type="primary" size="mini" @click="selectFile" style="margin: 0px;">选择文件</button>
  71. </view>
  72. </uni-file-picker>
  73. <text v-if="hasPackage"
  74. style="padding-left: 20px;color: #a8a8a8;">{{Number(appFileList.size / 1024 / 1024).toFixed(2)}}M</text>
  75. </uni-forms-item>
  76. <uni-forms-item key="url" name="url" :label="isiOS ? 'AppStore' : '下载链接'" required>
  77. <view class="flex" style="flex-direction: column;align-items:flex-start;flex: 1;">
  78. <view class="flex" style="width: 100%;">
  79. <uni-easyinput :disabled="detailsState" placeholder="下载链接" v-model="formData.url" :maxlength="-1" />
  80. <text style="margin-left: 10px;color: #2979ff;cursor: pointer;text-decoration: underline;" v-if="formData.url" @click="toUrl(formData.url)">测试下载</text>
  81. </view>
  82. <text style="margin-top: 10px;font-size: 12px;color: #666;" v-if="formData.url && !detailsState">建议点击【测试下载】能正常下载后,再进行发布</text>
  83. </view>
  84. </uni-forms-item>
  85. <uni-forms-item v-if="!isiOS && !isWGT && formData.store_list.length" label="Android应用市场" key="store_list"
  86. name="store_list" labelWidth="120">
  87. <view style="flex: 1;">
  88. <view v-for="(item,index) in formData.store_list" :key="item.id">
  89. <uni-card style="margin: 0px 0px 20px 0px;">
  90. <view style="display: flex;">
  91. <checkbox-group style="user-select: none;"
  92. @change="({detail:{value}}) => {item.enable = !!value.length}">
  93. <label class="title_padding">
  94. <checkbox :disabled="detailsState" value="scheme" :checked="item.enable" />
  95. <text>是否启用</text>
  96. </label>
  97. </checkbox-group>
  98. </view>
  99. <uni-forms-item label="商店名称">
  100. <uni-easyinput disabled v-model="item.name" trim="both"></uni-easyinput>
  101. </uni-forms-item>
  102. <uni-forms-item label="Scheme">
  103. <uni-easyinput disabled v-model="item.scheme" trim="both"></uni-easyinput>
  104. </uni-forms-item>
  105. <uni-forms-item label="优先级">
  106. <uni-easyinput :disabled="detailsState" v-model="item.priority" type="number">
  107. </uni-easyinput>
  108. <show-info :top="-100" :left="-180" :content="priorityContent"></show-info>
  109. </uni-forms-item>
  110. </uni-card>
  111. </view>
  112. </view>
  113. </uni-forms-item>
  114. <uni-forms-item v-if="isWGT" key="is_silently" name="is_silently" label="静默更新">
  115. <switch :disabled="detailsState"
  116. @change="binddata('is_silently', $event.detail.value),formData.is_silently=$event.detail.value"
  117. :checked="formData.is_silently" />
  118. <show-info :top="-80" :content="silentlyContent"></show-info>
  119. </uni-forms-item>
  120. <uni-forms-item v-if="!isiOS" key="is_mandatory" name="is_mandatory" label="强制更新">
  121. <switch :disabled="detailsState"
  122. @change="binddata('is_mandatory', $event.detail.value),formData.is_mandatory=$event.detail.value"
  123. :checked="formData.is_mandatory" />
  124. <show-info width="230" :top="-30" :content="mandatoryContent"></show-info>
  125. </uni-forms-item>
  126. <uni-forms-item name="stable_publish" label="上线发行">
  127. <switch :disabled="detailsState || isStable"
  128. @change="binddata('stable_publish', $event.detail.value),formData.stable_publish=$event.detail.value"
  129. :checked="formData.stable_publish" />
  130. <show-info v-if="isStable" :top="-50" width="350" :content="stablePublishContent"></show-info>
  131. <show-info v-else :top="-40" :content="stablePublishContent2"></show-info>
  132. </uni-forms-item>
  133. <uni-forms-item name="create_date" label="上传时间">
  134. <uni-dateformat format="yyyy-MM-dd hh:mm:ss" :date="formData.create_date" :threshold="[0, 0]" />
  135. </uni-forms-item>
  136. <uni-forms-item v-show="false" name="type" label="安装包类型">
  137. <uni-data-checkbox v-model="formData.type" :localdata="formOptions.type_localdata" />
  138. </uni-forms-item>
  139. <view class="uni-button-group">
  140. <button type="primary" class="uni-button" style="width: 100px;" @click="detailsState = false"
  141. v-if="detailsState">修改</button>
  142. <button type="primary" class="uni-button" style="width: 100px;" @click="submit"
  143. v-if="!detailsState">提交</button>
  144. <button type="warn" class="uni-button" style="width: 100px;" @click="cancelEdit"
  145. v-if="!detailsState">取消</button>
  146. <navigator open-type="navigateBack" style="margin-left: 15px;">
  147. <button class="uni-button" style="width: 100px;">返回</button>
  148. </navigator>
  149. </view>
  150. </uni-forms>
  151. </view>
  152. </template>
  153. <script>
  154. import {
  155. validator,
  156. enumConverter
  157. } from '@/js_sdk/validator/opendb-app-versions.js';
  158. import addAndDetail, {
  159. fields
  160. } from '../mixin/version_add_detail_mixin.js'
  161. import {
  162. deepClone,
  163. appVersionListDbName
  164. } from '../utils.js'
  165. const db = uniCloud.database();
  166. const dbCmd = db.command;
  167. const dbCollectionName = appVersionListDbName;
  168. const platform_iOS = 'iOS';
  169. const platform_Android = 'Android';
  170. function getValidator(fields) {
  171. let reuslt = {}
  172. for (let key in validator) {
  173. if (fields.includes(key)) {
  174. reuslt[key] = validator[key]
  175. }
  176. }
  177. return reuslt
  178. }
  179. export default {
  180. mixins: [addAndDetail],
  181. data() {
  182. return {
  183. showStableInfo: false,
  184. isStable: true, // 是否是线上发行版
  185. originalData: {}, // 原始数据,用于恢复状态
  186. detailsState: true, // 查看状态,
  187. uniFilePickerProvider: 'unicloud',
  188. domain: ""
  189. }
  190. },
  191. async onLoad(e) {
  192. let { domain, provider } = this.getCloudStorageConfig();
  193. if (domain) this.domain = domain;
  194. if (provider) this.uniFilePickerProvider = provider;
  195. const id = e.id
  196. this.formDataId = id
  197. await this.getDetail(id)
  198. this.isStable = this.formData.stable_publish;
  199. this.latestStableData = await this.getLatestVersion();
  200. if (this.isWGT) {
  201. this.rules.min_uni_version.rules.push({
  202. "required": true
  203. })
  204. }
  205. },
  206. onUnload() {
  207. // 临时处理,后面会再优化
  208. this.setCloudStorage({
  209. provider: null
  210. });
  211. },
  212. watch: {
  213. "domain"(val) {
  214. this.setCloudStorage({
  215. domain: val
  216. });
  217. if (this.formData.url) {
  218. // 替换 this.formData.url 内的域名
  219. if (!val) val = "请输入自定义域名"
  220. this.formData.url = this.formData.url.replace(/^(https?:\/\/)[^\/]+/, `$1${val}`);
  221. }
  222. },
  223. uniFilePickerProvider(val){
  224. this.setCloudStorage({
  225. provider: val
  226. });
  227. }
  228. },
  229. methods: {
  230. /**
  231. * 触发表单提交
  232. */
  233. submit() {
  234. uni.showLoading({
  235. mask: true
  236. })
  237. this.$refs.form.validate(['store_list']).then((res) => {
  238. if (res.store_list) {
  239. res.store_list.forEach(item => {
  240. item.priority = parseFloat(item.priority)
  241. })
  242. }
  243. this.submitForm(res)
  244. }).catch((errors) => {
  245. uni.hideLoading()
  246. })
  247. },
  248. async submitForm(value) {
  249. const collectionDB = db.collection(dbCollectionName)
  250. // 使用 clientDB 提交数据
  251. collectionDB.doc(this.formDataId).update(value).then(async (res) => {
  252. // 如果不是线上发行版,则在设置为上线发行时,需将之前的已上线版设为下线
  253. if (!this.isStable && value.stable_publish === true && this.latestStableData) {
  254. await collectionDB.doc(this.latestStableData._id).update({
  255. stable_publish: false
  256. })
  257. }
  258. uni.showToast({
  259. title: '修改成功'
  260. })
  261. this.getOpenerEventChannel().emit('refreshData')
  262. setTimeout(() => uni.navigateBack(), 500)
  263. }).catch((err) => {
  264. uni.showModal({
  265. content: err.message || '请求服务失败',
  266. showCancel: false
  267. })
  268. }).finally(() => {
  269. uni.hideLoading()
  270. })
  271. },
  272. /**
  273. * 获取表单数据
  274. * @param {Object} id
  275. */
  276. getDetail(id) {
  277. uni.showLoading({
  278. mask: true
  279. })
  280. return db.collection(dbCollectionName)
  281. .doc(id)
  282. .field(fields)
  283. .get()
  284. .then((res) => {
  285. const data = res.result.data[0]
  286. if (data) {
  287. if (!data.store_list) data.store_list = []
  288. this.formData = data
  289. this.originalData = deepClone(this.formData)
  290. }
  291. }).catch((err) => {
  292. uni.showModal({
  293. content: err.message || '请求服务失败',
  294. showCancel: false
  295. })
  296. }).finally(() => {
  297. uni.hideLoading()
  298. })
  299. },
  300. deletePackage() {
  301. uni.showModal({
  302. title: '提示',
  303. content: '是否删除该版本',
  304. success: res => {
  305. if (res.confirm) {
  306. uni.showLoading({
  307. mask: true
  308. })
  309. db.collection(dbCollectionName).doc(this.formDataId).remove()
  310. .then(() => {
  311. uni.showToast({
  312. title: '删除成功'
  313. })
  314. this.getOpenerEventChannel().emit('refreshData')
  315. setTimeout(() => uni.navigateBack(), 500)
  316. }).catch((err) => {
  317. uni.showModal({
  318. content: err.message || '请求服务失败',
  319. showCancel: false
  320. })
  321. }).finally(() => {
  322. uni.hideLoading()
  323. })
  324. }
  325. }
  326. });
  327. },
  328. async getLatestVersion() {
  329. const where = {
  330. appid: this.formData.appid,
  331. type: this.formData.type,
  332. stable_publish: true
  333. };
  334. if (!this.isWGT) {
  335. where.platform = this.formData.platform[0]
  336. }
  337. const latestStableData = await db.collection(dbCollectionName).where(where).get()
  338. return latestStableData.result.data.find(item => item.platform.toString() === this.formData.platform
  339. .toString());
  340. },
  341. cancelEdit() {
  342. let content = '';
  343. !this.isiOS && this.hasPackage ? content += '\n将会删除已上传的包' : '';
  344. uni.showModal({
  345. title: '取消修改',
  346. content,
  347. success: res => {
  348. if (res.confirm) {
  349. this.formData = deepClone(this.originalData)
  350. this.detailsState = true
  351. if (this.hasPackage) {
  352. this.deleteFile([this.appFileList.url])
  353. }
  354. }
  355. }
  356. });
  357. }
  358. }
  359. }
  360. </script>
  361. <style lang="scss">
  362. .show-stable-info {
  363. position: absolute;
  364. left: 165px;
  365. padding: 5px 10px;
  366. background-color: #f4f4f5;
  367. color: #909399;
  368. border-radius: 4px;
  369. border: 1px solid #e9e9eb;
  370. }
  371. ::v-deep .uni-forms-item__content {
  372. display: flex;
  373. align-items: center;
  374. }
  375. .uni-button-group {
  376. & button {
  377. margin-left: 15px;
  378. }
  379. & button:first-child {
  380. margin-left: 0px;
  381. }
  382. }
  383. </style>