BasicInfoForm.vue 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. <template>
  2. <el-form ref="ProductManagementBasicInfoRef" :model="formData" :rules="rules" label-width="120px">
  3. <el-row>
  4. <el-col :span="12">
  5. <el-form-item label="商品名称" prop="name">
  6. <el-input v-model="formData.name" placeholder="请输入商品名称" />
  7. </el-form-item>
  8. </el-col>
  9. <el-col :span="12">
  10. <el-form-item label="商品分类" prop="categoryId">
  11. <el-tree-select
  12. v-model="formData.categoryId"
  13. :data="categoryList"
  14. :props="defaultProps"
  15. check-strictly
  16. node-key="id"
  17. placeholder="请选择商品分类"
  18. />
  19. </el-form-item>
  20. </el-col>
  21. <el-col :span="12">
  22. <el-form-item label="商品关键字" prop="keyword">
  23. <el-input v-model="formData.keyword" placeholder="请输入商品关键字" />
  24. </el-form-item>
  25. </el-col>
  26. <el-col :span="12">
  27. <el-form-item label="单位" prop="unit">
  28. <el-select v-model="formData.unit" placeholder="请选择单位">
  29. <el-option
  30. v-for="dict in getIntDictOptions(DICT_TYPE.PRODUCT_UNIT)"
  31. :key="dict.value"
  32. :label="dict.label"
  33. :value="dict.value"
  34. />
  35. </el-select>
  36. </el-form-item>
  37. </el-col>
  38. <el-col :span="12">
  39. <el-form-item label="商品简介" prop="introduction">
  40. <el-input
  41. v-model="formData.introduction"
  42. :rows="3"
  43. placeholder="请输入商品简介"
  44. type="textarea"
  45. />
  46. </el-form-item>
  47. </el-col>
  48. <el-col :span="12">
  49. <el-form-item label="商品封面图" prop="picUrl">
  50. <UploadImg v-model="formData.picUrl" height="80px" />
  51. </el-form-item>
  52. </el-col>
  53. <el-col :span="24">
  54. <el-form-item label="商品轮播图" prop="sliderPicUrls">
  55. <UploadImgs v-model="formData.sliderPicUrls" />
  56. </el-form-item>
  57. </el-col>
  58. <el-col :span="12">
  59. <el-form-item label="运费模板" prop="deliveryTemplateId">
  60. <el-select v-model="formData.deliveryTemplateId" placeholder="请选择" style="width: 100%">
  61. <el-option v-for="item in []" :key="item.id" :label="item.name" :value="item.id" />
  62. </el-select>
  63. </el-form-item>
  64. </el-col>
  65. <el-col :span="12">
  66. <el-button class="ml-20px">运费模板</el-button>
  67. </el-col>
  68. <el-col :span="12">
  69. <el-form-item label="商品规格" props="specType">
  70. <el-radio-group v-model="formData.specType" @change="onChangeSpec">
  71. <el-radio :label="false" class="radio">单规格</el-radio>
  72. <el-radio :label="true">多规格</el-radio>
  73. </el-radio-group>
  74. </el-form-item>
  75. </el-col>
  76. <el-col :span="12">
  77. <el-form-item label="分销类型" props="subCommissionType">
  78. <el-radio-group v-model="formData.subCommissionType" @change="changeSubCommissionType">
  79. <el-radio :label="false">默认设置</el-radio>
  80. <el-radio :label="true" class="radio">自行设置</el-radio>
  81. </el-radio-group>
  82. </el-form-item>
  83. </el-col>
  84. <!-- 多规格添加-->
  85. <el-col :span="24">
  86. <el-form-item v-if="formData.specType" label="商品属性">
  87. <el-button class="mr-15px mb-10px" @click="AttributesAddFormRef.open()"
  88. >添加规格
  89. </el-button>
  90. <ProductAttributes :attribute-data="attributeList" />
  91. </el-form-item>
  92. <template v-if="formData.specType && attributeList.length > 0">
  93. <el-form-item label="批量设置">
  94. <SkuList :attributeList="attributeList" :is-batch="true" :prop-form-data="formData" />
  95. </el-form-item>
  96. <el-form-item label="属性列表">
  97. <SkuList :attributeList="attributeList" :prop-form-data="formData" />
  98. </el-form-item>
  99. </template>
  100. <el-form-item v-if="!formData.specType">
  101. <SkuList :attributeList="attributeList" :prop-form-data="formData" />
  102. </el-form-item>
  103. </el-col>
  104. </el-row>
  105. </el-form>
  106. <ProductAttributesAddForm ref="AttributesAddFormRef" @success="addAttribute" />
  107. </template>
  108. <script lang="ts" name="ProductManagementBasicInfoForm" setup>
  109. import { PropType } from 'vue'
  110. import { defaultProps, handleTree } from '@/utils/tree'
  111. import { ElInput } from 'element-plus'
  112. import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
  113. import type { SpuType } from '@/api/mall/product/management/type/spuType'
  114. import { UploadImg, UploadImgs } from '@/components/UploadFile'
  115. import { copyValueToTarget } from '@/utils/object'
  116. import { ProductAttributes, ProductAttributesAddForm, SkuList } from './index'
  117. // 业务Api
  118. import * as ProductCategoryApi from '@/api/mall/product/category'
  119. import { propTypes } from '@/utils/propTypes'
  120. const message = useMessage() // 消息弹窗
  121. const props = defineProps({
  122. propFormData: {
  123. type: Object as PropType<SpuType>,
  124. default: () => {}
  125. },
  126. activeName: propTypes.string.def('')
  127. })
  128. const AttributesAddFormRef = ref() // 添加商品属性表单
  129. const ProductManagementBasicInfoRef = ref() // 表单Ref
  130. const attributeList = ref([]) // 商品属性列表
  131. /** 添加商品属性 */
  132. const addAttribute = (property: any) => {
  133. if (Array.isArray(property)) {
  134. attributeList.value = property
  135. return
  136. }
  137. attributeList.value.push(property)
  138. }
  139. const formData = reactive<SpuType>({
  140. name: '', // 商品名称
  141. categoryId: undefined, // 商品分类
  142. keyword: '', // 关键字
  143. unit: '', // 单位
  144. picUrl: '', // 商品封面图
  145. sliderPicUrls: [], // 商品轮播图
  146. introduction: '', // 商品简介
  147. deliveryTemplateId: 1, // 运费模版
  148. specType: false, // 商品规格
  149. subCommissionType: false, // 分销类型
  150. skus: []
  151. })
  152. const rules = reactive({
  153. name: [required],
  154. categoryId: [required],
  155. keyword: [required],
  156. unit: [required],
  157. introduction: [required],
  158. picUrl: [required],
  159. sliderPicUrls: [required],
  160. // deliveryTemplateId: [required],
  161. specType: [required],
  162. subCommissionType: [required]
  163. })
  164. /**
  165. * 将传进来的值赋值给formData
  166. */
  167. watch(
  168. () => props.propFormData,
  169. (data) => {
  170. if (!data) return
  171. copyValueToTarget(formData, data)
  172. },
  173. {
  174. deep: true,
  175. immediate: true
  176. }
  177. )
  178. const emit = defineEmits(['update:activeName'])
  179. /**
  180. * 表单校验
  181. */
  182. const validate = async () => {
  183. // 校验表单
  184. if (!ProductManagementBasicInfoRef) return
  185. return await unref(ProductManagementBasicInfoRef).validate((valid) => {
  186. if (!valid) {
  187. message.warning('商品信息未完善!!')
  188. emit('update:activeName', 'basicInfo')
  189. // 目的截断之后的校验
  190. throw new Error('商品信息未完善!!')
  191. } else {
  192. // 校验通过更新数据
  193. Object.assign(props.propFormData, formData)
  194. }
  195. })
  196. }
  197. defineExpose({ validate, addAttribute })
  198. // 分销类型
  199. const changeSubCommissionType = () => {
  200. // 默认为零,类型切换后也要重置为零
  201. for (const item of formData.skus) {
  202. item.subCommissionFirstPrice = 0
  203. item.subCommissionSecondPrice = 0
  204. }
  205. }
  206. // 选择规格
  207. const onChangeSpec = () => {
  208. // 重置商品属性列表
  209. attributeList.value = []
  210. // 重置sku列表
  211. formData.skus = [
  212. {
  213. price: 0,
  214. marketPrice: 0,
  215. costPrice: 0,
  216. barCode: '',
  217. picUrl: '',
  218. stock: 0,
  219. weight: 0,
  220. volume: 0,
  221. subCommissionFirstPrice: 0,
  222. subCommissionSecondPrice: 0
  223. }
  224. ]
  225. }
  226. const categoryList = ref() // 分类树
  227. onMounted(async () => {
  228. // 获得分类树
  229. const data = await ProductCategoryApi.getCategoryList({})
  230. categoryList.value = handleTree(data, 'id', 'parentId')
  231. })
  232. </script>