BasicInfoForm.vue 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  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. const message = useMessage() // 消息弹窗
  120. const props = defineProps({
  121. propFormData: {
  122. type: Object as PropType<SpuType>,
  123. default: () => {}
  124. }
  125. })
  126. const AttributesAddFormRef = ref() // 添加商品属性表单
  127. const ProductManagementBasicInfoRef = ref() // 表单Ref
  128. const attributeList = ref([]) // 商品属性列表
  129. /** 添加商品属性 */
  130. const addAttribute = (property: any) => {
  131. attributeList.value.push(property)
  132. }
  133. const formData = reactive<SpuType>({
  134. name: '', // 商品名称
  135. categoryId: undefined, // 商品分类
  136. keyword: '', // 关键字
  137. unit: '', // 单位
  138. picUrl: '', // 商品封面图
  139. sliderPicUrls: [], // 商品轮播图
  140. introduction: '', // 商品简介
  141. deliveryTemplateId: 1, // 运费模版
  142. specType: false, // 商品规格
  143. subCommissionType: false, // 分销类型
  144. skus: []
  145. })
  146. const rules = reactive({
  147. name: [required],
  148. categoryId: [required],
  149. keyword: [required],
  150. unit: [required],
  151. introduction: [required],
  152. picUrl: [required],
  153. sliderPicUrls: [required],
  154. // deliveryTemplateId: [required],
  155. specType: [required],
  156. subCommissionType: [required]
  157. })
  158. /**
  159. * 将传进来的值赋值给formData
  160. */
  161. watch(
  162. () => props.propFormData,
  163. (data) => {
  164. if (!data) return
  165. copyValueToTarget(formData, data)
  166. },
  167. {
  168. deep: true,
  169. immediate: true
  170. }
  171. )
  172. const emit = defineEmits(['update:activeName'])
  173. /**
  174. * 表单校验
  175. */
  176. const validate = async () => {
  177. // 校验表单
  178. if (!ProductManagementBasicInfoRef) return
  179. return await unref(ProductManagementBasicInfoRef).validate((valid) => {
  180. if (!valid) {
  181. message.warning('商品信息未完善!!')
  182. emit('update:activeName', 'basicInfo')
  183. // 目的截断之后的校验
  184. throw new Error('商品信息未完善!!')
  185. } else {
  186. // 校验通过更新数据
  187. Object.assign(props.propFormData, formData)
  188. }
  189. })
  190. }
  191. defineExpose({ validate })
  192. // 分销类型
  193. const changeSubCommissionType = () => {
  194. // 默认为零,类型切换后也要重置为零
  195. for (const item of formData.skus) {
  196. item.subCommissionFirstPrice = 0
  197. item.subCommissionSecondPrice = 0
  198. }
  199. }
  200. // 选择规格
  201. const onChangeSpec = () => {
  202. console.log(111)
  203. // 重置商品属性列表
  204. attributeList.value = []
  205. // 重置sku列表
  206. formData.skus = [
  207. {
  208. price: 0,
  209. marketPrice: 0,
  210. costPrice: 0,
  211. barCode: '',
  212. picUrl: '',
  213. stock: 0,
  214. weight: 0,
  215. volume: 0,
  216. subCommissionFirstPrice: 0,
  217. subCommissionSecondPrice: 0
  218. }
  219. ]
  220. }
  221. const categoryList = ref() // 分类树
  222. onMounted(async () => {
  223. // 获得分类树
  224. const data = await ProductCategoryApi.getCategoryList({})
  225. categoryList.value = handleTree(data, 'id', 'parentId')
  226. })
  227. </script>