BasicInfoForm.vue 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. <template>
  2. <el-form ref="productSpuBasicInfoRef" :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. <!-- TODO @puhui999:只能选根节点 fix: 已完善-->
  11. <el-form-item label="商品分类" prop="categoryId">
  12. <el-tree-select
  13. v-model="formData.categoryId"
  14. :data="categoryList"
  15. :props="defaultProps"
  16. check-strictly
  17. class="w-1/1"
  18. node-key="id"
  19. placeholder="请选择商品分类"
  20. @change="nodeClick"
  21. />
  22. </el-form-item>
  23. </el-col>
  24. <el-col :span="12">
  25. <el-form-item label="商品关键字" prop="keyword">
  26. <el-input v-model="formData.keyword" placeholder="请输入商品关键字" />
  27. </el-form-item>
  28. </el-col>
  29. <el-col :span="12">
  30. <el-form-item label="单位" prop="unit">
  31. <el-select v-model="formData.unit" class="w-1/1" placeholder="请选择单位">
  32. <el-option
  33. v-for="dict in getIntDictOptions(DICT_TYPE.PRODUCT_UNIT)"
  34. :key="dict.value"
  35. :label="dict.label"
  36. :value="dict.value"
  37. />
  38. </el-select>
  39. </el-form-item>
  40. </el-col>
  41. <el-col :span="12">
  42. <el-form-item label="商品简介" prop="introduction">
  43. <el-input
  44. v-model="formData.introduction"
  45. :rows="3"
  46. placeholder="请输入商品简介"
  47. type="textarea"
  48. />
  49. </el-form-item>
  50. </el-col>
  51. <el-col :span="12">
  52. <el-form-item label="商品封面图" prop="picUrl">
  53. <UploadImg v-model="formData.picUrl" height="80px" />
  54. </el-form-item>
  55. </el-col>
  56. <el-col :span="24">
  57. <el-form-item label="商品轮播图" prop="sliderPicUrls">
  58. <UploadImgs v-model:modelValue="formData.sliderPicUrls" />
  59. </el-form-item>
  60. </el-col>
  61. <el-col :span="12">
  62. <el-form-item label="运费模板" prop="deliveryTemplateId">
  63. <el-select v-model="formData.deliveryTemplateId" placeholder="请选择">
  64. <el-option v-for="item in []" :key="item.id" :label="item.name" :value="item.id" />
  65. </el-select>
  66. <el-button class="ml-20px">运费模板</el-button>
  67. </el-form-item>
  68. </el-col>
  69. <el-col :span="12">
  70. <el-form-item label="品牌" prop="brandId">
  71. <el-select v-model="formData.brandId" placeholder="请选择">
  72. <el-option
  73. v-for="item in brandList"
  74. :key="item.id"
  75. :label="item.name"
  76. :value="item.id"
  77. />
  78. </el-select>
  79. </el-form-item>
  80. </el-col>
  81. <el-col :span="12">
  82. <el-form-item label="商品规格" props="specType">
  83. <el-radio-group v-model="formData.specType" @change="onChangeSpec">
  84. <el-radio :label="false" class="radio">单规格</el-radio>
  85. <el-radio :label="true">多规格</el-radio>
  86. </el-radio-group>
  87. </el-form-item>
  88. </el-col>
  89. <el-col :span="12">
  90. <el-form-item label="分销类型" props="subCommissionType">
  91. <el-radio-group v-model="formData.subCommissionType" @change="changeSubCommissionType">
  92. <el-radio :label="false">默认设置</el-radio>
  93. <el-radio :label="true" class="radio">自行设置</el-radio>
  94. </el-radio-group>
  95. </el-form-item>
  96. </el-col>
  97. <!-- 多规格添加-->
  98. <el-col :span="24">
  99. <el-form-item v-if="formData.specType" label="商品属性">
  100. <el-button class="mr-15px mb-10px" @click="attributesAddFormRef.open">添加规格</el-button>
  101. <ProductAttributes :propertyList="propertyList" @success="generateSkus" />
  102. </el-form-item>
  103. <template v-if="formData.specType && propertyList.length > 0">
  104. <el-form-item label="批量设置">
  105. <SkuList :is-batch="true" :prop-form-data="formData" :propertyList="propertyList" />
  106. </el-form-item>
  107. <el-form-item label="属性列表">
  108. <SkuList ref="skuListRef" :prop-form-data="formData" :propertyList="propertyList" />
  109. </el-form-item>
  110. </template>
  111. <el-form-item v-if="!formData.specType">
  112. <SkuList :prop-form-data="formData" :propertyList="propertyList" />
  113. </el-form-item>
  114. </el-col>
  115. </el-row>
  116. </el-form>
  117. <ProductAttributesAddForm ref="attributesAddFormRef" :propertyList="propertyList" />
  118. </template>
  119. <script lang="ts" name="ProductSpuBasicInfoForm" setup>
  120. import { PropType } from 'vue'
  121. import { copyValueToTarget } from '@/utils'
  122. import { propTypes } from '@/utils/propTypes'
  123. import { checkSelectedNode, defaultProps, handleTree } from '@/utils/tree'
  124. import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
  125. import type { Spu } from '@/api/mall/product/spu'
  126. import { UploadImg, UploadImgs } from '@/components/UploadFile'
  127. import { ProductAttributes, ProductAttributesAddForm, SkuList } from './index'
  128. import * as ProductCategoryApi from '@/api/mall/product/category'
  129. import { getSimpleBrandList } from '@/api/mall/product/brand'
  130. const message = useMessage() // 消息弹窗
  131. const props = defineProps({
  132. propFormData: {
  133. type: Object as PropType<Spu>,
  134. default: () => {}
  135. },
  136. activeName: propTypes.string.def('')
  137. })
  138. const attributesAddFormRef = ref() // 添加商品属性表单
  139. const productSpuBasicInfoRef = ref() // 表单 Ref
  140. const propertyList = ref([]) // 商品属性列表
  141. const skuListRef = ref() // 商品属性列表Ref
  142. /** 调用 SkuList generateTableData 方法*/
  143. const generateSkus = (propertyList) => {
  144. skuListRef.value.generateTableData(propertyList)
  145. }
  146. const formData = reactive<Spu>({
  147. name: '', // 商品名称
  148. categoryId: null, // 商品分类
  149. keyword: '', // 关键字
  150. unit: '', // 单位
  151. picUrl: '', // 商品封面图
  152. sliderPicUrls: [], // 商品轮播图
  153. introduction: '', // 商品简介
  154. deliveryTemplateId: 1, // 运费模版
  155. brandId: null, // 商品品牌
  156. specType: false, // 商品规格
  157. subCommissionType: false, // 分销类型
  158. skus: []
  159. })
  160. const rules = reactive({
  161. name: [required],
  162. categoryId: [required],
  163. keyword: [required],
  164. unit: [required],
  165. introduction: [required],
  166. picUrl: [required],
  167. sliderPicUrls: [required],
  168. // deliveryTemplateId: [required],
  169. brandId: [required],
  170. specType: [required],
  171. subCommissionType: [required]
  172. })
  173. /**
  174. * 将传进来的值赋值给 formData
  175. */
  176. watch(
  177. () => props.propFormData,
  178. (data) => {
  179. if (!data) {
  180. return
  181. }
  182. copyValueToTarget(formData, data)
  183. formData.sliderPicUrls = data['sliderPicUrls'].map((item) => ({
  184. url: item
  185. }))
  186. // 只有是多规格才处理
  187. if (!formData.specType) return
  188. // 直接拿返回的 skus 属性逆向生成出 propertyList
  189. const properties = []
  190. formData.skus.forEach((sku) => {
  191. sku.properties.forEach(({ propertyId, propertyName, valueId, valueName }) => {
  192. // 添加属性
  193. if (!properties.some((item) => item.id === propertyId)) {
  194. properties.push({ id: propertyId, name: propertyName, values: [] })
  195. }
  196. // 添加属性值
  197. const index = properties.findIndex((item) => item.id === propertyId)
  198. if (!properties[index].values.some((value) => value.id === valueId)) {
  199. properties[index].values.push({ id: valueId, name: valueName })
  200. }
  201. })
  202. })
  203. propertyList.value = properties
  204. },
  205. {
  206. immediate: true
  207. }
  208. )
  209. /**
  210. * 表单校验
  211. */
  212. const emit = defineEmits(['update:activeName'])
  213. const validate = async () => {
  214. // 校验 sku
  215. if (!skuListRef.value.validateSku()) {
  216. message.warning('商品相关价格不能低于0.01元!!')
  217. throw new Error('商品相关价格不能低于0.01元!!')
  218. }
  219. // 校验表单
  220. if (!productSpuBasicInfoRef) return
  221. return await unref(productSpuBasicInfoRef).validate((valid) => {
  222. if (!valid) {
  223. message.warning('商品信息未完善!!')
  224. emit('update:activeName', 'basicInfo')
  225. // 目的截断之后的校验
  226. throw new Error('商品信息未完善!!')
  227. } else {
  228. // 校验通过更新数据
  229. Object.assign(props.propFormData, formData)
  230. }
  231. })
  232. }
  233. defineExpose({ validate })
  234. /** 分销类型 */
  235. const changeSubCommissionType = () => {
  236. // 默认为零,类型切换后也要重置为零
  237. for (const item of formData.skus) {
  238. item.subCommissionFirstPrice = 0
  239. item.subCommissionSecondPrice = 0
  240. }
  241. }
  242. /** 选择规格 */
  243. const onChangeSpec = () => {
  244. // 重置商品属性列表
  245. propertyList.value = []
  246. // 重置sku列表
  247. formData.skus = [
  248. {
  249. price: 0,
  250. marketPrice: 0,
  251. costPrice: 0,
  252. barCode: '',
  253. picUrl: '',
  254. stock: 0,
  255. weight: 0,
  256. volume: 0,
  257. subCommissionFirstPrice: 0,
  258. subCommissionSecondPrice: 0
  259. }
  260. ]
  261. }
  262. const categoryList = ref([]) // 分类树
  263. /**
  264. * 选择分类时触发校验
  265. */
  266. const nodeClick = () => {
  267. if (!checkSelectedNode(categoryList.value, formData.categoryId)) {
  268. formData.categoryId = null
  269. message.warning('必须选择二级节点!!')
  270. }
  271. }
  272. const brandList = ref([]) // 精简商品品牌列表
  273. onMounted(async () => {
  274. // 获得分类树
  275. const data = await ProductCategoryApi.getCategoryList({})
  276. categoryList.value = handleTree(data, 'id', 'parentId')
  277. // 获取商品品牌列表
  278. brandList.value = await getSimpleBrandList()
  279. })
  280. </script>