form-club-device.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491
  1. <template>
  2. <div class="club-device">
  3. <template v-for="formItem in formList">
  4. <div :key="formItem.uid">
  5. <el-form :model="formItem" :rules="rules" ref="form">
  6. <el-form-item prop="productName" :label="`设备名称${formItem.uuid}:`">
  7. <el-select
  8. v-model="formItem.productName"
  9. filterable
  10. allow-create
  11. placeholder="请输入新设备名称或选择已有设备"
  12. @change="onProductNameChange(formItem, $event)"
  13. >
  14. <el-option
  15. v-for="item in deviceList"
  16. :key="item.productTypeId"
  17. :label="item.name"
  18. :value="item.productTypeId"
  19. />
  20. </el-select>
  21. </el-form-item>
  22. <el-form-item prop="productImage" label="设备图片:">
  23. <br />
  24. <el-input v-show="false" v-model="formItem.productImage"></el-input>
  25. <SimpleUploadImage
  26. :disabled="Boolean(formItem.productTypeId)"
  27. :limit="1"
  28. :image-list="formItem.productImageList"
  29. :before-upload="beforeProductImageUpload"
  30. @success="uploadProductImageSuccess(formItem, $event)"
  31. @remove="handleProductImageRemove(formItem, $event)"
  32. />
  33. </el-form-item>
  34. <el-form-item label="所属品牌:" prop="brandId">
  35. <el-select v-model="formItem.brandId" placeholder="请选择品牌">
  36. <el-option
  37. v-for="item in brandList"
  38. :key="item.id"
  39. :label="item.name"
  40. :value="item.id"
  41. />
  42. </el-select>
  43. </el-form-item>
  44. <el-form-item prop="purchaseWay" label="购买渠道:">
  45. <el-input
  46. placeholder="请输入购买渠道"
  47. v-model="formItem.purchaseWay"
  48. ></el-input>
  49. </el-form-item>
  50. <el-form-item prop="invoiceImage" label="发票:">
  51. <br />
  52. <el-input v-show="false" v-model="formItem.invoiceImage"></el-input>
  53. <SimpleUploadImage
  54. :limit="1"
  55. :image-list="formItem.invoiceImageList"
  56. :before-upload="beforeInvoiceImageUpload"
  57. @success="uploadInvoiceImageSuccess(formItem, $event)"
  58. @remove="handleInvoiceImageRemove(formItem, $event)"
  59. />
  60. </el-form-item>
  61. <el-form-item prop="snCode" label="设备SN码:">
  62. <el-input
  63. placeholder="请输入设备SN码"
  64. v-model="formItem.snCode"
  65. ></el-input>
  66. </el-form-item>
  67. <el-form-item prop="paramList" label="设备参数:">
  68. <br />
  69. <div class="device-param-list">
  70. <span class="add-param" @click="insertParam(formItem)"
  71. >添加参数</span
  72. >
  73. <template v-for="(param, index) in formItem.paramList">
  74. <div :key="index">
  75. <div class="param flex justify-between mb-4">
  76. <el-input
  77. style="width: 40%"
  78. placeholder="例如:品牌"
  79. class="mr-2"
  80. v-model="param.paramName"
  81. ></el-input>
  82. <el-input
  83. placeholder="请输入参数信息"
  84. v-model="param.paramContent"
  85. ></el-input>
  86. <span
  87. class="remove el-icon-close"
  88. @click="removeParam(formItem, index)"
  89. v-if="formItem.paramList.length > 4"
  90. ></span>
  91. </div>
  92. </div>
  93. </template>
  94. </div>
  95. </el-form-item>
  96. </el-form>
  97. <el-divider></el-divider>
  98. </div>
  99. </template>
  100. <div class="add-device" @click="insertOne">
  101. <div class="add-icon"></div>
  102. 添加设备
  103. </div>
  104. <SimpleDialog
  105. v-model="active"
  106. @confirm="active = false"
  107. :cancel="false"
  108. description="请慎重填写机构信息,认证通过后将无法更改!"
  109. :center="true"
  110. />
  111. </div>
  112. </template>
  113. <script>
  114. import SimpleUploadImage from '@/components/SimpleUploadImage'
  115. import { mapGetters } from 'vuex'
  116. export default {
  117. components: {
  118. SimpleUploadImage,
  119. },
  120. data() {
  121. const productNameValidate = (rule, value, callback) => {
  122. if (value.toString().length > 50) {
  123. callback(new Error('设备名称长度需要在50个字符内'))
  124. } else {
  125. callback()
  126. }
  127. }
  128. return {
  129. active: true,
  130. uuid: 0, // 表单id
  131. productImageList: [],
  132. rules: {
  133. productName: [
  134. { required: true, message: '设备名称不能为空', trigger: ['change'] },
  135. { validator: productNameValidate, trigger: ['change'] },
  136. ],
  137. productImage: [
  138. { required: true, message: '设备图片不能为空', trigger: ['change'] },
  139. ],
  140. brandId: [
  141. { required: true, message: '所属品牌不能为空', trigger: ['change'] },
  142. ],
  143. snCode: [
  144. { required: true, message: '设备SN码不能为空', trigger: ['blur'] },
  145. ],
  146. paramList: [{ required: true, message: '参数不能为空' }],
  147. },
  148. formList: [],
  149. brandList: [],
  150. deviceList: [],
  151. }
  152. },
  153. computed: {
  154. ...mapGetters(['authUserId']),
  155. },
  156. created() {
  157. this.fetchBrandList()
  158. this.fetchDeviceList()
  159. this.initFormList()
  160. },
  161. methods: {
  162. // 表单验证
  163. validate() {
  164. this.$emit('step', this.formatFormList())
  165. return Promise.all(this.$refs.form.map((item) => item.validate()))
  166. },
  167. formatFormList() {
  168. const list = []
  169. this.formList.forEach((formItem) => {
  170. const obj = {}
  171. obj.productImage = formItem.productImage
  172. obj.productName = formItem.productName
  173. obj.snCode = formItem.snCode
  174. obj.brandId = formItem.brandId
  175. obj.productTypeId = formItem.productTypeId
  176. obj.purchaseWay = formItem.purchaseWay
  177. obj.invoiceImage = formItem.invoiceImage
  178. obj.paramList = formItem.paramList
  179. list.push(obj)
  180. })
  181. return list
  182. },
  183. generateFormData() {
  184. return {
  185. uuid: ++this.uuid,
  186. authUserId: '',
  187. authId: '', // 授权id
  188. createBy: '', // 创建人id
  189. // 设备参数列表
  190. paramList: this.initParams(),
  191. productId: '', // 授权设备id
  192. productImage: '', // 设备图片
  193. productName: '', // 设备名称
  194. snCode: '', // 设备SN码
  195. brandId: '',
  196. productTypeId: '',
  197. purchaseWay: '', // 购买渠道
  198. invoiceImage: '', // 发票
  199. productImageList: [],
  200. invoiceImageList: [],
  201. }
  202. },
  203. generageProductParam() {
  204. return {
  205. paramContent: '',
  206. paramName: '',
  207. }
  208. },
  209. initParams() {
  210. const list = []
  211. for (let i = 0; i < 4; i++) {
  212. list.push(this.generageProductParam())
  213. }
  214. return list
  215. },
  216. insertParam(formItem) {
  217. formItem.paramList.push(this.generateFormData())
  218. },
  219. removeParam(formItem, index) {
  220. formItem.paramList.splice(index, 1)
  221. },
  222. initFormList() {
  223. this.formList.push(this.generateFormData())
  224. console.log(this.formList)
  225. },
  226. insertOne() {
  227. this.formList.push(this.generateFormData())
  228. },
  229. removeOne() {},
  230. onProductNameChange(formItem, value) {
  231. if (typeof value === 'number') {
  232. formItem.productTypeId = value
  233. const deviceInfo = this.deviceList.find(
  234. (item) => item.productTypeId === value
  235. )
  236. formItem.productImage = deviceInfo.image
  237. formItem.productImageList = [{ name: '', url: deviceInfo.image }]
  238. } else {
  239. formItem.productTypeId = ''
  240. formItem.productImage = ''
  241. formItem.productImageList = []
  242. }
  243. },
  244. // 获取品牌列表
  245. async fetchBrandList() {
  246. try {
  247. const res = await this.$http.api.fetchBrandList({
  248. type: 3,
  249. authUserId: this.authUserId,
  250. })
  251. this.brandList = res.data
  252. } catch (error) {
  253. console.log(error)
  254. }
  255. },
  256. // 获取设备列表
  257. async fetchDeviceList() {
  258. try {
  259. const res = await this.$http.api.fetchProductSelectList({
  260. authUserId: this.authUserId,
  261. })
  262. this.deviceList = res.data
  263. } catch (error) {
  264. console.log(error)
  265. }
  266. },
  267. // 产品图片上传
  268. beforeProductImageUpload(file) {
  269. const flag = file.size / 1024 / 1024 < 1
  270. if (!flag) {
  271. this.$message.error('上传产品图片大小不能超过 1MB!')
  272. }
  273. return flag
  274. },
  275. uploadProductImageSuccess(formItem, { response, file, fileList }) {
  276. formItem.productImageList = fileList
  277. formItem.productImage = response.data
  278. },
  279. handleProductImageRemove(formItem, { file, fileList }) {
  280. formItem.productImageList = fileList
  281. formItem.productImage = ''
  282. },
  283. // 发票上传
  284. beforeInvoiceImageUpload(file) {
  285. const flag = file.size / 1024 / 1024 < 1
  286. if (!flag) {
  287. this.$message.error('发票图片大小不能超过 1MB!')
  288. }
  289. return flag
  290. },
  291. uploadInvoiceImageSuccess(formItem, { response, file, fileList }) {
  292. formItem.invoiceImageList = fileList
  293. formItem.invoiceImage = response.data
  294. },
  295. handleInvoiceImageRemove(formItem, { file, fileList }) {
  296. formItem.invoiceImageList = fileList
  297. formItem.invoiceImage = ''
  298. },
  299. },
  300. }
  301. </script>
  302. <style lang="scss" scoped>
  303. // pc端
  304. @media screen and (min-width: 768px) {
  305. .el-select {
  306. width: 100%;
  307. }
  308. .device-param-list {
  309. position: relative;
  310. .add-param {
  311. position: absolute;
  312. cursor: pointer;
  313. top: -40px;
  314. right: 0;
  315. text-decoration: underline;
  316. font-size: 14px;
  317. @include themify($themes) {
  318. color: themed('color');
  319. }
  320. }
  321. .param {
  322. position: relative;
  323. .remove {
  324. position: absolute;
  325. right: 0;
  326. top: 0;
  327. width: 20px;
  328. height: 20px;
  329. background: #f94b4b;
  330. border-radius: 2px;
  331. cursor: pointer;
  332. color: #fff;
  333. font-size: 14px;
  334. text-align: center;
  335. line-height: 20px;
  336. }
  337. }
  338. }
  339. .add-device {
  340. display: flex;
  341. justify-content: center;
  342. align-items: center;
  343. width: 162px;
  344. height: 46px;
  345. border-radius: 4px;
  346. box-sizing: border-box;
  347. font-size: 18px;
  348. margin: 0 auto;
  349. cursor: pointer;
  350. @include themify($themes) {
  351. border: 1px solid themed('color');
  352. color: themed('color');
  353. }
  354. .add-icon {
  355. width: 20px;
  356. height: 20px;
  357. position: relative;
  358. margin-right: 16px;
  359. &::before,
  360. &::after {
  361. position: absolute;
  362. width: 3px;
  363. height: 20px;
  364. left: 50%;
  365. top: 50%;
  366. transform: translate(-50%, -50%);
  367. border-radius: 1px;
  368. content: '';
  369. display: block;
  370. @include themify($themes) {
  371. background: themed('color');
  372. }
  373. }
  374. &::after {
  375. transform: translate(-50%, -50%) rotateZ(90deg);
  376. }
  377. }
  378. }
  379. }
  380. // 移动端
  381. @media screen and (max-width: 768px) {
  382. ::v-deep {
  383. .el-form-item__label {
  384. font-size: 3.4vw;
  385. }
  386. }
  387. .el-select {
  388. width: 100%;
  389. }
  390. .device-param-list {
  391. position: relative;
  392. .add-param {
  393. position: absolute;
  394. cursor: pointer;
  395. top: -40px;
  396. right: 0;
  397. font-size: 3.4vw;
  398. @include themify($themes) {
  399. color: themed('color');
  400. }
  401. }
  402. .param {
  403. position: relative;
  404. .remove {
  405. position: absolute;
  406. right: 0;
  407. top: 0;
  408. width: 4.4vw;
  409. height: 4.4vw;
  410. background: #f94b4b;
  411. border-radius: 0.2vw;
  412. cursor: pointer;
  413. color: #fff;
  414. font-size: 3.4vw;
  415. text-align: center;
  416. line-height: 4.4vw;
  417. }
  418. }
  419. }
  420. .add-device {
  421. display: flex;
  422. justify-content: center;
  423. align-items: center;
  424. width: 31vw;
  425. height: 8.8vw;
  426. border-radius: 0.4vw;
  427. box-sizing: border-box;
  428. font-size: 3.4vw;
  429. margin: 0 auto;
  430. cursor: pointer;
  431. @include themify($themes) {
  432. border: 1px solid themed('color');
  433. color: themed('color');
  434. }
  435. .add-icon {
  436. width: 20px;
  437. height: 20px;
  438. position: relative;
  439. margin-right: 16px;
  440. &::before,
  441. &::after {
  442. position: absolute;
  443. width: 0.6vw;
  444. height: 4.1vw;
  445. left: 50%;
  446. top: 50%;
  447. transform: translate(-50%, -50%);
  448. border-radius: 1px;
  449. content: '';
  450. display: block;
  451. @include themify($themes) {
  452. background: themed('color');
  453. }
  454. }
  455. &::after {
  456. transform: translate(-50%, -50%) rotateZ(90deg);
  457. }
  458. }
  459. }
  460. }
  461. </style>