HotGoodsAdd.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. <template>
  2. <el-dialog :close-on-click-modal="false" :title="dialogTitle" :visible.sync="hotGoodsAddVisible" width="30%" :before-close="handleBeforeClose">
  3. <el-form :model="form" :rules="rules" ref="formRef" label-width="120px">
  4. <el-form-item label="爆品分类" prop="categoryId">
  5. <el-select v-model="form.categoryId" placeholder="请选择爆品分类">
  6. <el-option v-for="item in categoryList" :key="item.id" :label="item.categoryName" :value="item.id"></el-option>
  7. </el-select>
  8. </el-form-item>
  9. <el-form-item label="产品" prop="productId">
  10. <el-select
  11. :disabled="!!this.form.id"
  12. @change="handleSelectGoods"
  13. style="width: 100%"
  14. v-model="form.productId"
  15. filterable
  16. remote
  17. reserve-keyword
  18. placeholder="请输入商品名称"
  19. :remote-method="handleSearchGoods"
  20. :loading="searchGoodsLoading"
  21. >
  22. <el-option v-for="item in goodsList" :key="item.productId" :label="item.productName" :value="item.productId">
  23. <div class="name">{{ item.productName }} - {{ item.sectionPrice }}</div>
  24. </el-option>
  25. </el-select>
  26. </el-form-item>
  27. <el-form-item label="爆品名称" prop="productName">
  28. <el-input v-model="form.productName" placeholder="请填写爆品名称"></el-input>
  29. </el-form-item>
  30. <el-form-item label="爆品价格" prop="productPrice">
  31. <el-input v-model="form.productPrice" prefix="¥" placeholder="请输入爆品价格"></el-input>
  32. </el-form-item>
  33. <el-form-item label="活动区域" prop="regionalCodes">
  34. <el-cascader
  35. style="width: 100%"
  36. v-model="regionArrDialog"
  37. :props="regionProps"
  38. size="large"
  39. placeholder="请选择区域"
  40. @change="form.regionalCodes = [...new Set(regionArrDialog.map((i) => i[i.length - 1]))].join(',')"
  41. ></el-cascader>
  42. <div v-if="String(form.regionalCodes)">已选ID:{{ form.regionalCodes }}</div>
  43. </el-form-item>
  44. <el-form-item label="是否上架" prop="isPut">
  45. <el-select style="width: 100%" v-model="form.isPut" placeholder="请选择">
  46. <el-option label="上架" :value="1"></el-option>
  47. <el-option label="下架" :value="0"></el-option>
  48. </el-select>
  49. </el-form-item>
  50. <el-form-item label="爆品主图链接" prop="productImage">
  51. <ImageUpload :limit="1" v-model="form.productImage"></ImageUpload>
  52. </el-form-item>
  53. </el-form>
  54. <span slot="footer" class="dialog-footer">
  55. <el-button @click="hotGoodsAddVisible = false">取 消</el-button>
  56. <el-button type="primary" :loading="isLoading" @click="handleConfirm">确 定</el-button>
  57. </span>
  58. </el-dialog>
  59. </template>
  60. <script>
  61. import ImageUpload from '@/components/ImageUpload'
  62. import { getClassifyGetAll } from '@/api/commodity'
  63. import { addHotGoods, editHotGoods } from '@/api/active/active_hot'
  64. import { getExplosiveProductCategory } from '@/api/explosiveProduct/category'
  65. import { getProvinceList, getChildAreaList } from '@/api/address'
  66. export default {
  67. components: { ImageUpload },
  68. data() {
  69. return {
  70. categoryList: [],
  71. goodsList: [],
  72. hotGoodsAddVisible: false,
  73. searchGoodsLoading: false,
  74. dialogTitle: '新增爆品商品',
  75. isLoading: false,
  76. form: {
  77. id: void 0,
  78. productId: '',
  79. productName: '',
  80. productImage: '',
  81. productPrice: '',
  82. isPut: 1,
  83. categoryId: '',
  84. regionalCodes: ''
  85. },
  86. regionProps: {
  87. checkStrictly: true,
  88. multiple: true,
  89. lazy: true,
  90. label: 'name',
  91. value: 'id',
  92. lazyLoad(node, resolve) {
  93. const { level, value } = node
  94. if (level === 0) {
  95. resolve([{ id: 0, parentId: 0, name: '全中国', shortName: '全国', longitude: '108.55', latitude: '34.32', level: 1, sort: 1, status: true }])
  96. } else if (level === 1) {
  97. getProvinceList().then((res) => {
  98. resolve(res.data)
  99. })
  100. } else if (level != 0 && level != 1) {
  101. getChildAreaList(value).then((res) => {
  102. resolve(
  103. res.data.map((item) => {
  104. item.leaf = level === 4
  105. return item
  106. })
  107. )
  108. })
  109. }
  110. }
  111. },
  112. regionArrDialog: [],
  113. rules: {
  114. categoryId: [{ required: true, message: '请选择爆品分类', trigger: 'blur' }],
  115. productId: [{ required: true, message: '请选择商品', trigger: 'blur' }],
  116. productName: [{ required: true, message: '请输入爆品名称', trigger: 'blur' }],
  117. productImage: [{ required: true, message: '请输入爆品主图链接', trigger: 'blur' }],
  118. productPrice: [
  119. { required: true, message: '请输入爆品价格', trigger: 'blur' },
  120. {
  121. validator: (rule, value, callback) => {
  122. const priceRegex = /^\d+(\.\d{1,2})?$/
  123. if (!value) {
  124. callback(new Error('请输入爆品价格'))
  125. } else if (!priceRegex.test(value) || parseFloat(value) <= 0) {
  126. callback(new Error('请输入有效的价格(正数,最多两位小数)'))
  127. } else {
  128. callback()
  129. }
  130. },
  131. trigger: 'blur'
  132. }
  133. ],
  134. regionalCodes: [{ required: true, message: '请选择地址', trigger: 'change' }],
  135. isPut: [{ required: true, message: '请选择是否上架', trigger: 'change' }]
  136. }
  137. }
  138. },
  139. watch: {
  140. hotGoodsAddVisible(visible) {
  141. if (!visible) {
  142. this.reset()
  143. }
  144. }
  145. },
  146. methods: {
  147. open(row) {
  148. if (row && typeof row === 'object') {
  149. this.form.id = row.id
  150. this.form.productId = row.productId
  151. this.form.productName = row.productName
  152. this.form.productImage = row.productImage
  153. this.form.productPrice = row.productPrice
  154. this.form.isPut = row.isPut
  155. this.form.regionalCodes = row.regionalCodesSource || ''
  156. this.form.categoryId = row.categoryId || ''
  157. this.dialogTitle = '编辑爆款产品'
  158. } else {
  159. this.dialogTitle = '新增爆品商品'
  160. }
  161. this.getCategory()
  162. this.hotGoodsAddVisible = true
  163. },
  164. async getCategory() {
  165. const res = await getExplosiveProductCategory({ page: 1, pageSize: 10000 })
  166. if (res.code === '200') {
  167. this.categoryList = res.data.list
  168. }
  169. },
  170. handleBeforeClose(done) {
  171. this.reset()
  172. done()
  173. },
  174. async handleSearchGoods(query) {
  175. if (query !== '') {
  176. this.searchGoodsLoading = true
  177. try {
  178. const queryData = {
  179. productName: query,
  180. page: 1,
  181. pageSize: 20
  182. }
  183. const res = await getClassifyGetAll(queryData)
  184. this.goodsList = res.data.list
  185. } catch (error) {
  186. this.goodsList = []
  187. } finally {
  188. this.searchGoodsLoading = false
  189. }
  190. } else {
  191. this.goodsList = []
  192. }
  193. },
  194. // 添加商品
  195. handleSelectGoods(productId) {
  196. const currentGoods = this.goodsList.find((item) => item.productId === productId)
  197. if (currentGoods) {
  198. this.form.productId = productId
  199. this.form.productName = currentGoods.productName
  200. this.form.productImage = currentGoods.image
  201. this.form.productPrice = Number(currentGoods.price) || 0
  202. }
  203. },
  204. // 重置数据
  205. reset() {
  206. this.form = {
  207. productId: '',
  208. productName: '',
  209. productImage: '',
  210. productPrice: '',
  211. isPut: 1,
  212. categoryId: '',
  213. regionalCodes: ''
  214. }
  215. this.goodsList = []
  216. },
  217. close() {
  218. this.hotGoodsAddVisible = false
  219. },
  220. async handleConfirm() {
  221. try {
  222. this.isLoading = true
  223. await this.$refs.formRef.validate()
  224. const api = this.form.id ? editHotGoods : addHotGoods
  225. const res = await api(this.form)
  226. if (res.code === '200') {
  227. this.$message({
  228. message: this.form.id ? '编辑成功' : '添加成功',
  229. type: 'success'
  230. })
  231. this.reset()
  232. this.$emit('refresh')
  233. this.close()
  234. }
  235. } finally {
  236. this.isLoading = false
  237. }
  238. }
  239. }
  240. }
  241. </script>
  242. <style lang="scss" scoped></style>