index.vue 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583
  1. <template>
  2. <div class="p-3">
  3. <div class="bg-fff flex1 ov-hd d-flex flex-cln">
  4. <div class="d-flex a-c pd-16 border-bottom">
  5. <div class="f-s-20 c-333 f-w-7 mr-10">{{ form.id ? '编辑' : '新增' }}会议</div>
  6. <el-button @click="router.go(-1)" type="primary" text>
  7. <el-icon>
  8. <Back />
  9. </el-icon>
  10. 返回上一级
  11. </el-button>
  12. </div>
  13. <div class="flex1 over-auto">
  14. <el-form ref="formRef" label-width="auto" label-position="top" :model="form" :rules="rules" :scroll-into-view-options="scrollOptions" scroll-to-error>
  15. <div class="pd-16 border-bottom ov-hd">
  16. <div class="info-title mb-10">会议信息</div>
  17. <el-row :gutter="20">
  18. <el-col :span="6">
  19. <el-form-item label="会议名称" prop="trainingName">
  20. <el-input v-model="form.trainingName" placeholder="请输入会议名称" clearable />
  21. </el-form-item>
  22. </el-col>
  23. <el-col :span="6">
  24. <el-form-item label="会议时间" prop="trainingTime">
  25. <el-date-picker v-model="form.trainingTime" type="datetimerange" value-format="YYYY-MM-DD HH:mm:ss" date-format="YYYY-MM-DD" time-format="HH:mm" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间" />
  26. </el-form-item>
  27. </el-col>
  28. <el-col :span="6">
  29. <el-form-item label="报名时间" prop="signupTime">
  30. <el-date-picker v-model="form.signupsTime" type="datetimerange" value-format="YYYY-MM-DD HH:mm:ss" date-format="YYYY-MM-DD" time-format="HH:mm" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间" />
  31. </el-form-item>
  32. </el-col>
  33. <el-col :span="6">
  34. <el-form-item label="会议方式" prop="joinType">
  35. <!-- 单选框 -->
  36. <el-radio-group v-model="form.joinType">
  37. <el-radio v-for="item in lm_training_join_type" :key="item.value" :label="item.value">{{ item.label }}</el-radio>
  38. </el-radio-group>
  39. </el-form-item>
  40. </el-col>
  41. <el-col v-if="form.joinType === '0'" :span="6">
  42. <el-form-item label="会议地点" prop="trainingLocation">
  43. <el-input v-model="form.trainingLocation" placeholder="请输入会议地点" clearable />
  44. </el-form-item>
  45. </el-col>
  46. <el-col :span="6">
  47. <el-form-item label="会议联系人" prop="contactName">
  48. <el-input v-model="form.contactName" maxlength="40" placeholder="请输入会议联系人" clearable />
  49. </el-form-item>
  50. </el-col>
  51. <el-col :span="6">
  52. <el-form-item label="联系电话" prop="tel">
  53. <el-input v-model="form.tel" maxlength="20" placeholder="请输入联系电话" clearable />
  54. </el-form-item>
  55. </el-col>
  56. <el-col :span="12">
  57. <el-form-item label="可报名人员类型" prop="conditions.typeCheck">
  58. <el-checkbox-group v-model="checkedVipLevels" @change="handleCheckedChange">
  59. <el-checkbox v-for="city in form.conditions.typeCheck" :key="city" :label="city" :value="city">
  60. {{ selectDictLabels(lm_check_join_type, city.vipLevel, ',') }}
  61. </el-checkbox>
  62. </el-checkbox-group>
  63. <div @click="handleCheckAllChange(true)" v-if="!checkAll" class="pl-10 c-s-p">
  64. <u>全选</u>
  65. </div>
  66. <div @click="handleCheckAllChange(false)" v-else class="pl-10 c-s-p"><u>取消全选</u></div>
  67. </el-form-item>
  68. </el-col>
  69. <el-col :span="6">
  70. <el-form-item label="报名人数" prop="conditions.totalCheck">
  71. <div class="d-flex a-c">
  72. <el-radio-group v-model="form.conditions.totalCheck" style="flex-wrap: nowrap">
  73. <el-radio label="0">不限制</el-radio>
  74. <el-radio label="1">限制</el-radio>
  75. </el-radio-group>
  76. </div>
  77. </el-form-item>
  78. </el-col>
  79. <el-col :span="12" v-if="form.conditions.totalCheck == '1'">
  80. <el-form-item prop="restrictiveConditions">
  81. <template #label>
  82. <span>限制条件</span>
  83. <span class="c-999 f-s-12 f-w-4">(不限制的条件可不填)</span>
  84. </template>
  85. <div class="d-flex flex-cln pl-10">
  86. <div class="d-flex">
  87. <div class="c-#606266 f-w-6" style="">报名总人数限制:</div>
  88. <el-input class="flex1 pl-5" v-model="form.conditions.total" maxlength="20" placeholder="请输入报名人数" clearable :disabled="form.conditions.totalCheck !== '1'" style="max-width: 200px;" />
  89. </div>
  90. <div class="c-#606266 f-w-6" style="">条件限制:</div>
  91. <div class="d-flex">
  92. <div class="d-flex flex-cln">
  93. <el-checkbox v-model="form.conditions.levelTotalCheck" label="按单位类型限制" size="large" true-value="1" false-value="0" />
  94. <div v-if="form.conditions.levelTotalCheck == '1' && form.conditions.cpyTotalCheck == '0'" class="pl-10 pr-10 pt-5 pb-5 border">
  95. <template v-for="(item, index) in form.conditions.levelCheck" :key="index">
  96. <div class="d-flex a-c" v-if="checkedVipLevels.some(items => items.vipLevel === item.vipLevel)">
  97. <el-checkbox v-model="item.check" true-value="1" false-value="0" :label="'所有' + selectDictLabels(lm_check_join_type, item.vipLevel, ',') + '参会人数≤'" size="large" />
  98. <el-input class="pl-10" v-model="item.total" maxlength="20" placeholder="请输入报名人数" :disabled="!+item.check" />
  99. </div>
  100. </template>
  101. <el-empty :image-size="20" description="请先选择报名人员类型" v-if="checkedVipLevels.length == 0" />
  102. </div>
  103. </div>
  104. <div class="d-flex flex-cln ml-20">
  105. <el-checkbox v-model="form.conditions.cpyTotalCheck" label="按每家单位人数限制" size="large" true-value="1" false-value="0" />
  106. <div v-if="form.conditions.cpyTotalCheck == '1' && form.conditions.levelTotalCheck == '0'" class="pl-10 pr-10 pt-5 pb-5 border">
  107. <template v-for="(item, index) in form.conditions.cpyCheck" :key="index">
  108. <div class="d-flex a-c" v-if="checkedVipLevels.some(items => items.vipLevel === item.vipLevel)">
  109. <el-checkbox v-model="item.check" :label="selectDictLabels(lm_check_join_type, item.vipLevel, ',') + '限制每家单位人数≤'" size="large" true-value="1" false-value="0" />
  110. <el-input class="pl-10" v-model="item.total" maxlength="20" placeholder="请输入报名人数" :disabled="!+item.check" />
  111. </div>
  112. </template>
  113. <el-empty :image-size="20" description="请先选择报名人员类型" v-if="checkedVipLevels.length == 0" />
  114. </div>
  115. </div>
  116. </div>
  117. <div class="d-flex border" v-if="form.conditions.levelTotalCheck == '1' && form.conditions.cpyTotalCheck == '1'">
  118. <div class="pl-10 pr-10 pt-5 pb-5 ">
  119. <template v-for="(item, index) in form.conditions.levelCheck" :key="index">
  120. <div class="d-flex a-c" v-if="checkedVipLevels.some(items => items.vipLevel === item.vipLevel)">
  121. <el-checkbox v-model="item.check" :label="'所有' + selectDictLabels(lm_check_join_type, item.vipLevel, ',') + '参会人数≤'" size="large" true-value="1" false-value="0" />
  122. <el-input class="pl-10" v-model="item.total" maxlength="20" placeholder="请输入报名人数" :disabled="!+item.check" />
  123. </div>
  124. </template>
  125. </div>
  126. <div class="pl-10 pr-10 pt-5 pb-5">
  127. <template v-for="(item, index) in form.conditions.cpyCheck" :key="index">
  128. <div class="d-flex a-c" v-if="checkedVipLevels.some(items => items.vipLevel === item.vipLevel)">
  129. <el-checkbox v-model="item.check" :label="selectDictLabels(lm_check_join_type, item.vipLevel, ',') + '限制每家单位人数≤'" size="large" true-value="1" false-value="0" />
  130. <el-input class="pl-10" v-model="item.total" maxlength="20" placeholder="请输入报名人数" :disabled="!+item.check" />
  131. </div>
  132. </template>
  133. </div>
  134. <div class="flex1 ml--10" v-if="checkedVipLevels.length == 0"><el-empty :image-size="20" description="请先选择报名人员类型" /></div>
  135. </div>
  136. </div>
  137. </el-form-item>
  138. </el-col>
  139. <el-col :span="6">
  140. <el-form-item label="是否电子手签" prop="eleSignature">
  141. <div class="d-flex a-c">
  142. <el-radio-group v-model="form.eleSignature" style="flex-wrap: nowrap">
  143. <el-radio label="1">是</el-radio>
  144. <el-radio label="0">否</el-radio>
  145. </el-radio-group>
  146. </div>
  147. </el-form-item>
  148. </el-col>
  149. <el-col :span="12">
  150. <div class="d-flex" style="align-items: flex-end;">
  151. <el-form-item label="发放积分" prop="pointsFlag" class="">
  152. <el-radio-group v-model="form.pointsFlag" style="flex-wrap: nowrap">
  153. <el-radio label="1">是</el-radio>
  154. <el-radio label="0">否</el-radio>
  155. </el-radio-group>
  156. </el-form-item>
  157. <el-form-item label="" prop="points" v-if="form.pointsFlag == '1'" class="flex1 pl-10">
  158. <div class="d-flex f-s-14">
  159. <div>每成功参会(签到成功)1人发放</div>
  160. <el-input v-model="form.points" style="width: 60px" />
  161. <div>个单位积分。</div>
  162. </div>
  163. </el-form-item>
  164. </div>
  165. </el-col>
  166. </el-row>
  167. <el-row :gutter="20">
  168. <el-col :span="12">
  169. <el-form-item label="会议详情" prop="description">
  170. <el-input v-model="form.description" :rows="4" type="textarea" placeholder="请输入会议详情" />
  171. </el-form-item>
  172. </el-col>
  173. <el-col :span="12">
  174. <el-form-item label="与会须知" prop="description">
  175. <el-input v-model="form.notice" :rows="4" type="textarea" placeholder="请输入与会须知" />
  176. </el-form-item>
  177. </el-col>
  178. <el-col :span="12">
  179. <el-form-item label="会议微信群聊二维码" prop="wechatQrCode">
  180. <ImageUpload v-model="form.wechatQrCode" :fileSize="20"></ImageUpload>
  181. </el-form-item>
  182. </el-col>
  183. <el-col :span="12">
  184. <el-form-item label="封面图" prop="coverImg">
  185. <ImageUpload v-model="form.coverImg" :limit="1" :fileSize="20"></ImageUpload>
  186. </el-form-item>
  187. </el-col>
  188. <el-col :span="12">
  189. <el-form-item label="会议图" prop="trainingImg">
  190. <ImageUpload v-model="form.trainingImg" :fileSize="20"></ImageUpload>
  191. </el-form-item>
  192. </el-col>
  193. <el-col :span="12">
  194. <el-form-item label="会议备注" prop="remark">
  195. <el-input v-model="form.remark" :rows="4" type="textarea" placeholder="请输入会议备注" />
  196. </el-form-item>
  197. </el-col>
  198. <el-col :span="12">
  199. <el-form-item label="相关文件" prop="attachments">
  200. <template #label>
  201. <span>相关文件</span>
  202. <span class="c-999 f-s-12 f-w-4">(此模块报名审核通过后才可查看)</span>
  203. </template>
  204. <div class="d-flex flex-cln mt-10 f-s-12 c-#606266">
  205. <FileUpload v-model="form.attachments" format="array" :limit="20" :fileSize="100"></FileUpload>
  206. </div>
  207. </el-form-item>
  208. </el-col>
  209. </el-row>
  210. </div>
  211. <div class="pd-16 border-bottom d-flex">
  212. <div class="w-50% ov-at h-1000">
  213. <div class="info-title mb-10">证书信息</div>
  214. <div class="d-flex j-start a-c">
  215. <el-form-item label="是否颁发证书" prop="certFlag">
  216. <el-radio-group v-model="form.certFlag">
  217. <el-radio v-for="item in yes_no" :key="item.value" :label="item.value">
  218. {{ item.label }}
  219. </el-radio>
  220. </el-radio-group>
  221. </el-form-item>
  222. <div v-if="+form.certFlag" class="ml-20">
  223. <el-button @click="addCertInfo" type="primary">新增证书</el-button>
  224. </div>
  225. </div>
  226. <template v-if="form.certFlag">
  227. <template v-for="(item, index) in form.certificateInfo" :key="index">
  228. <el-row :gutter="20" class="bg-#f4f4f4 pd-16 mb-10">
  229. <el-col :span="10">
  230. <el-form-item label="证书名称" :prop="`certificateInfo.${index}.certType`" :rules="[{ required: true, message: '请选择证书名称', trigger: 'change' }]">
  231. <el-select v-model="item.certType" placeholder="证书名称" clearable>
  232. <el-option v-for="item in lm_training_cert" :key="item.value" :label="item.label" :value="item.value" />
  233. </el-select>
  234. </el-form-item>
  235. </el-col>
  236. <el-col :span="10">
  237. <el-form-item :prop="`certificateInfo.${index}.certImg`" :rules="[{ required: true, message: '请上传证书图片', trigger: 'change' }]">
  238. <template #label>
  239. <span>证书模板图片</span>
  240. <el-button @click="goEditor()" type="primary" text>去编辑模板图片</el-button>
  241. </template>
  242. <ImageUpload v-model="item.certImg" :fileSize="40" :limit="1"></ImageUpload>
  243. </el-form-item>
  244. </el-col>
  245. <el-col :span="4">
  246. <el-form-item>
  247. <el-button type="danger" @click="deleteItem(index)">删除</el-button>
  248. </el-form-item>
  249. </el-col>
  250. </el-row>
  251. </template>
  252. </template>
  253. </div>
  254. <div class="w-50% d-flex flex-cln j-c a-c pl-20">
  255. <el-button type="primary" class="w-100%" plain style="height: 70px; margin-bottom: 20px;" @click="showSignIn = true">点击去编辑报名信息></el-button>
  256. <div class="w-400 h-700 border over-auto">
  257. <div class="pd-10 border bg-#fafafa">
  258. <div class="pt-10 f-s-20 f-w-6 d-flex j-c a-c flex-cln ">报名信息</div>
  259. <div class="f-s-12 f-w-4 d-flex j-start c-red">此页面为意向人报名时所见页面:</div>
  260. </div>
  261. <template v-for="(item, index) in fixedField" :key="index">
  262. <div class="pd-15 border1 c-#D7D7D7 d-flex j-sb">
  263. <span class="c-#606266 f-s-16 f-w-6">{{ item.label }}</span>
  264. <span class="f-s-14">请输入</span>
  265. </div>
  266. </template>
  267. <template v-for="(field, index) in fields" :key="index">
  268. <div class="meeting-custom-wrapper" @click.stop="showSignIn = true">
  269. <meetingCustomPreview :field="field" style="pointer-events: none" v-if="field" />
  270. </div>
  271. </template>
  272. <div class="pd-15 border1 c-#D7D7D7 d-flex j-sb">
  273. <span class="c-#606266 f-s-16 f-w-6">备注</span>
  274. <span class="f-s-14">请输入</span>
  275. </div>
  276. </div>
  277. </div>
  278. </div>
  279. </el-form>
  280. <MeetingEditors v-if="showSignIn" v-model:show="showSignIn" v-model:info="fields" />
  281. </div>
  282. <div class="d-flex a-c j-c pd-16">
  283. <el-button @click="Cancel">取消</el-button>
  284. <el-button @click="save" type="primary">提交</el-button>
  285. </div>
  286. </div>
  287. </div>
  288. </template>
  289. <script setup name="meeting-add" lang="ts">
  290. import { trainingAdd, trainingDetail, trainingUpdate } from '@/api/training';
  291. import { debounce } from 'lodash';
  292. import { onMounted, reactive, ref } from 'vue';
  293. import { useRouter } from 'vue-router';
  294. import { FieldDefinition } from '../models/type';
  295. // 需要添加以下导入
  296. import meetingCustomPreview from '../models/meeting-custom-preview.vue';
  297. import MeetingEditors from '../models/meeting-editors.vue';
  298. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  299. const { lm_training_join_type, yes_no, lm_training_cert, lm_check_join_type } = toRefs<any>(proxy?.useDict('lm_training_join_type', 'yes_no', 'lm_training_cert', 'lm_check_join_type'));
  300. const fields = ref<FieldDefinition[]>([])
  301. const showSignIn = ref(false);
  302. const fixedField = ref<FieldDefinition[]>([{
  303. name: `ent-${generateSecureRandomString()}`,
  304. label: '企业名称', type: '1',
  305. required: '1', readonly: '0',
  306. }, {
  307. name: `name-${generateSecureRandomString()}`,
  308. label: '姓名', type: '1', readonly: '0',
  309. required: '1'
  310. }, {
  311. name: `pos-${generateSecureRandomString()}`,
  312. label: '职务', type: '1', readonly: '0',
  313. required: '1'
  314. }, {
  315. name: `Con-${generateSecureRandomString()}`,
  316. label: '联系方式', type: '1', readonly: '0',
  317. required: '1'
  318. }])
  319. const scrollOptions = {
  320. block: 'center',
  321. behavior: 'smooth'
  322. };
  323. function generateSecureRandomString(length = 8) {
  324. const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
  325. const randomValues = new Uint32Array(length);
  326. window.crypto.getRandomValues(randomValues);
  327. let result = '';
  328. randomValues.forEach((value) => {
  329. result += chars[value % chars.length];
  330. });
  331. return result;
  332. }
  333. const Cancel = () => {
  334. router.go(-1)
  335. }
  336. const router = useRouter();
  337. const route = useRoute();
  338. const levelTypeCheck = computed(() =>
  339. String(Number(checkedVipLevels.value.length > 0))
  340. );
  341. const form = ref<any>({
  342. id: undefined,
  343. conditions: {
  344. levelTotalCheck: '0',
  345. cpyTotalCheck: '0',
  346. levelTypeCheck: levelTypeCheck,
  347. typeCheck: [
  348. {
  349. vipLevel: "0",
  350. check: "0"
  351. },
  352. {
  353. vipLevel: "1",
  354. check: "0"
  355. },
  356. {
  357. vipLevel: "3",
  358. check: "0"
  359. },
  360. {
  361. vipLevel: "5",
  362. check: "0"
  363. },
  364. {
  365. vipLevel: "6",
  366. check: "0"
  367. },
  368. {
  369. vipLevel: "7",
  370. check: "0"
  371. },
  372. {
  373. vipLevel: "9",
  374. check: "0"
  375. },
  376. {
  377. vipLevel: "10",
  378. check: "0"
  379. }
  380. ],
  381. levelCheck: [{
  382. vipLevel: '0',
  383. check: "0",
  384. total: ''
  385. }, {
  386. vipLevel: '1',
  387. check: "0",
  388. total: ''
  389. }, {
  390. vipLevel: '3',
  391. check: "0",
  392. total: ''
  393. }, {
  394. vipLevel: '5',
  395. check: "0",
  396. total: ''
  397. },
  398. {
  399. vipLevel: "6",
  400. check: "0",
  401. total: ''
  402. },
  403. {
  404. vipLevel: "7",
  405. check: "0",
  406. total: ''
  407. },
  408. {
  409. vipLevel: "9",
  410. check: "0",
  411. },
  412. {
  413. vipLevel: "10",
  414. total: ''
  415. }],
  416. cpyCheck: [{
  417. vipLevel: '0',
  418. check: "0",
  419. total: ''
  420. }, {
  421. vipLevel: '1',
  422. check: "0",
  423. total: ''
  424. }, {
  425. vipLevel: '3',
  426. check: "0",
  427. total: ''
  428. }, {
  429. vipLevel: '5',
  430. check: "0",
  431. total: ''
  432. }, {
  433. vipLevel: '6',
  434. check: "0",
  435. total: ''
  436. }, {
  437. vipLevel: '7',
  438. check: "0",
  439. total: ''
  440. }, {
  441. vipLevel: '9',
  442. check: "0",
  443. total: ''
  444. }, {
  445. vipLevel: '10',
  446. check: "0",
  447. total: ''
  448. }]
  449. }
  450. });
  451. const checkAll = ref(false)
  452. const checkedVipLevels = ref([])
  453. // 选项变化时的处理
  454. const handleCheckedChange = (selectedValues) => {
  455. const selectedLevelMap = new Map(selectedValues.map(item => [item.vipLevel, true]));
  456. form.value.conditions.typeCheck.forEach(item => {
  457. item.check = selectedLevelMap.has(item.vipLevel) ? '1' : '0';
  458. });
  459. }
  460. // 全选/取消全选
  461. const handleCheckAllChange = (val: boolean) => {
  462. checkedVipLevels.value = val
  463. ? form.value.conditions.typeCheck.map(item => item)
  464. : []
  465. handleCheckedChange(checkedVipLevels.value)
  466. checkAll.value = val
  467. }
  468. const rules = reactive({
  469. // 自动生成全部
  470. trainingName: [{ required: true, message: '请输入会议名称', trigger: 'blur' }],
  471. trainingTime: [{ required: true, message: '请选择会议时间', trigger: 'blur' }],
  472. joinType: [{ required: true, message: '请选择会议方式', trigger: 'change' }],
  473. trainingLocation: [{ required: true, message: '请输入会议地点', trigger: 'blur' }],
  474. 'conditions.totalCheck': [{ required: true, message: '请选择是否限制报名人数', trigger: 'change' }],
  475. 'conditions.typeCheck': [
  476. {
  477. validator: (rule, value, callback) => {
  478. setTimeout(() => {
  479. const isChecked = value.some(item => item.check == "1");
  480. if (!isChecked) {
  481. callback(new Error('请至少选择一种可报名单位类型'));
  482. } else {
  483. callback();
  484. }
  485. }, 300);
  486. },
  487. trigger: 'change' // 触发校验的时机
  488. }
  489. ],
  490. certFlag: [{ required: true, message: '请选择是否颁发证书', trigger: 'change' }],
  491. certificateInfo: [{ required: true, message: '请选择证书名称', trigger: 'change' }],
  492. description: [{ required: true, message: '请输入培训详情', trigger: 'blur' }],
  493. eleSignature: [{ required: true, message: '请选择是否电子手签', trigger: 'blur' }],
  494. pointsFlag: [{ required: true, message: '请选择是否发放积分', trigger: 'blur' }],
  495. // coverImg: [{ required: true, message: '请上传封面图', trigger: 'change' }],
  496. // trainingImg: [{ required: true, message: '请上传会议图', trigger: 'change' }],
  497. contactName: [{ required: true, message: '请输入会议联系人', trigger: 'blur' }],
  498. tel: [{ required: true, message: '请输入联系电话', trigger: 'blur' }],
  499. });
  500. const formRef = ref();
  501. const save = debounce(async () => {
  502. await formRef.value.validate();
  503. form.value.conditions.typeCheck.forEach(typeItem => {
  504. if (typeItem.check === "0") {
  505. // Update cpyCheck
  506. const cpyItem = form.value.conditions.cpyCheck.find(item => item.vipLevel === typeItem.vipLevel);
  507. if (cpyItem) {
  508. cpyItem.check = "0";
  509. }
  510. // Update levelCheck
  511. const levelItem = form.value.conditions.levelCheck.find(item => item.vipLevel === typeItem.vipLevel);
  512. if (levelItem) {
  513. levelItem.check = "0";
  514. }
  515. }
  516. });
  517. const params = {
  518. ...form.value,
  519. trainingStart: form.value.trainingTime ? form.value.trainingTime[0] : undefined,
  520. trainingEnd: form.value.trainingTime ? form.value.trainingTime[1] : undefined,
  521. signupStart: form.value.signupsTime ? form.value.signupsTime[0] : undefined,
  522. signupEnd: form.value.signupsTime ? form.value.signupsTime[1] : undefined,
  523. certificateInfo: +form.value.certFlag ? form.value.certificateInfo : undefined,
  524. questions: fields.value.map((item, index) => ({
  525. ...item,
  526. sort: index + 1 // 从 1 开始
  527. }))
  528. };
  529. const res = form.value.id ? await trainingUpdate(params) : await trainingAdd(params);
  530. if (res && res.code === 200) {
  531. router.go(-1);
  532. }
  533. }, 500);
  534. const goEditor = () => {
  535. window.open('https://lm.yujin.shuziyunyao.com/poster#/editor', '_blank');
  536. }
  537. const addCertInfo = () => {
  538. if (!form.value.certificateInfo) {
  539. form.value.certificateInfo = [];
  540. }
  541. form.value.certificateInfo.push({
  542. certName: undefined,
  543. certImg: undefined
  544. });
  545. };
  546. const deleteItem = (index: number) => {
  547. form.value.certificateInfo.splice(index, 1);
  548. };
  549. // 获取专家详情
  550. const getMeetingDetail = async () => {
  551. if (route.query?.id) {
  552. const res = await trainingDetail(route.query?.id as string);
  553. if (!res || res.code !== 200) return;
  554. form.value = {
  555. ...res.data,
  556. trainingTime: res.data.trainingStart && res.data.trainingEnd ? [res.data.trainingStart, res.data.trainingEnd] : undefined,
  557. signupsTime: res.data.signupStart && res.data.signupEnd ? [res.data.signupStart, res.data.signupEnd] : undefined
  558. };
  559. fields.value = res.data.questions
  560. form.value.conditions.typeCheck.forEach((i) => {
  561. if (i.check == '1') {
  562. checkedVipLevels.value.push(i)
  563. }
  564. })
  565. }
  566. };
  567. onMounted(() => {
  568. getMeetingDetail();
  569. });
  570. </script>
  571. <style scoped>
  572. .border {
  573. border: 1px solid #dcdfe6;
  574. }
  575. </style>