<template>
|
<ContentWrap v-loading="formLoading">
|
<el-tabs v-model="activeName">
|
<el-tab-pane label="基础设置" name="info">
|
<InfoForm
|
ref="infoRef"
|
v-model:activeName="activeName"
|
:is-detail="isDetail"
|
:propFormData="formData"
|
/>
|
</el-tab-pane>
|
<el-tab-pane label="价格库存" name="sku">
|
<SkuForm
|
ref="skuRef"
|
v-model:activeName="activeName"
|
:is-detail="isDetail"
|
:propFormData="formData"
|
/>
|
</el-tab-pane>
|
<el-tab-pane label="物流设置" name="delivery">
|
<DeliveryForm
|
ref="deliveryRef"
|
v-model:activeName="activeName"
|
:is-detail="isDetail"
|
:propFormData="formData"
|
/>
|
</el-tab-pane>
|
<el-tab-pane label="商品详情" name="description">
|
<DescriptionForm
|
ref="descriptionRef"
|
v-model:activeName="activeName"
|
:is-detail="isDetail"
|
:propFormData="formData"
|
/>
|
</el-tab-pane>
|
<el-tab-pane label="其它设置" name="other">
|
<OtherForm
|
ref="otherRef"
|
v-model:activeName="activeName"
|
:is-detail="isDetail"
|
:propFormData="formData"
|
/>
|
</el-tab-pane>
|
</el-tabs>
|
<el-form>
|
<el-form-item style="float: right">
|
<el-button v-if="!isDetail" :loading="formLoading" type="primary" @click="submitForm">
|
保存
|
</el-button>
|
<el-button @click="close">返回</el-button>
|
</el-form-item>
|
</el-form>
|
</ContentWrap>
|
</template>
|
<script lang="ts" setup>
|
import { cloneDeep } from 'lodash-es'
|
import { useTagsViewStore } from '@/store/modules/tagsView'
|
import * as ProductSpuApi from '@/api/mall/product/spu'
|
import InfoForm from './InfoForm.vue'
|
import DescriptionForm from './DescriptionForm.vue'
|
import OtherForm from './OtherForm.vue'
|
import SkuForm from './SkuForm.vue'
|
import DeliveryForm from './DeliveryForm.vue'
|
import { convertToInteger, floatToFixed2, formatToFraction } from '@/utils'
|
|
defineOptions({ name: 'ProductSpuForm' })
|
|
const { t } = useI18n() // 国际化
|
const message = useMessage() // 消息弹窗
|
const { push, currentRoute } = useRouter() // 路由
|
const { params, name } = useRoute() // 查询参数
|
const { delView } = useTagsViewStore() // 视图操作
|
|
const formLoading = ref(false) // 表单的加载中:1)修改时的数据加载;2)提交的按钮禁用
|
const activeName = ref('info') // Tag 激活的窗口
|
const isDetail = ref(false) // 是否查看详情
|
const infoRef = ref() // 商品信息 Ref
|
const skuRef = ref() // 商品规格 Ref
|
const deliveryRef = ref() // 物流设置 Ref
|
const descriptionRef = ref() // 商品详情 Ref
|
const otherRef = ref() // 其他设置 Ref
|
// SPU 表单数据
|
const formData = ref<ProductSpuApi.Spu>({
|
name: '', // 商品名称
|
categoryId: undefined, // 商品分类
|
keyword: '', // 关键字
|
picUrl: '', // 商品封面图
|
sliderPicUrls: [], // 商品轮播图
|
introduction: '', // 商品简介
|
deliveryTypes: [], // 配送方式数组
|
deliveryTemplateId: undefined, // 运费模版
|
brandId: undefined, // 商品品牌
|
specType: false, // 商品规格
|
subCommissionType: false, // 分销类型
|
skus: [
|
{
|
price: 0, // 商品价格
|
marketPrice: 0, // 市场价
|
costPrice: 0, // 成本价
|
barCode: '', // 商品条码
|
picUrl: '', // 图片地址
|
stock: 0, // 库存
|
weight: 0, // 商品重量
|
volume: 0, // 商品体积
|
firstBrokeragePrice: 0, // 一级分销的佣金
|
secondBrokeragePrice: 0 // 二级分销的佣金
|
}
|
],
|
description: '', // 商品详情
|
sort: 0, // 商品排序
|
giveIntegral: 0, // 赠送积分
|
virtualSalesCount: 0 // 虚拟销量
|
})
|
|
/** 获得详情 */
|
const getDetail = async () => {
|
if ('ProductSpuDetail' === name) {
|
isDetail.value = true
|
}
|
const id = params.id as unknown as number
|
if (id) {
|
formLoading.value = true
|
try {
|
const res = (await ProductSpuApi.getSpu(id)) as ProductSpuApi.Spu
|
res.skus?.forEach((item) => {
|
if (isDetail.value) {
|
item.price = floatToFixed2(item.price)
|
item.marketPrice = floatToFixed2(item.marketPrice)
|
item.costPrice = floatToFixed2(item.costPrice)
|
item.firstBrokeragePrice = floatToFixed2(item.firstBrokeragePrice)
|
item.secondBrokeragePrice = floatToFixed2(item.secondBrokeragePrice)
|
} else {
|
// 回显价格分转元
|
item.price = formatToFraction(item.price)
|
item.marketPrice = formatToFraction(item.marketPrice)
|
item.costPrice = formatToFraction(item.costPrice)
|
item.firstBrokeragePrice = formatToFraction(item.firstBrokeragePrice)
|
item.secondBrokeragePrice = formatToFraction(item.secondBrokeragePrice)
|
}
|
})
|
formData.value = res
|
} finally {
|
formLoading.value = false
|
}
|
}
|
}
|
|
/** 提交按钮 */
|
const submitForm = async () => {
|
// 提交请求
|
formLoading.value = true
|
try {
|
// 校验各表单
|
await unref(infoRef)?.validate()
|
await unref(skuRef)?.validate()
|
await unref(deliveryRef)?.validate()
|
await unref(descriptionRef)?.validate()
|
await unref(otherRef)?.validate()
|
// 深拷贝一份, 这样最终 server 端不满足,不需要影响原始数据
|
const deepCopyFormData = cloneDeep(unref(formData.value)) as ProductSpuApi.Spu
|
deepCopyFormData.skus!.forEach((item) => {
|
// 给sku name赋值
|
item.name = deepCopyFormData.name
|
// sku相关价格元转分
|
item.price = convertToInteger(item.price)
|
item.marketPrice = convertToInteger(item.marketPrice)
|
item.costPrice = convertToInteger(item.costPrice)
|
item.firstBrokeragePrice = convertToInteger(item.firstBrokeragePrice)
|
item.secondBrokeragePrice = convertToInteger(item.secondBrokeragePrice)
|
})
|
// 处理轮播图列表
|
const newSliderPicUrls: any[] = []
|
deepCopyFormData.sliderPicUrls!.forEach((item: any) => {
|
// 如果是前端选的图
|
typeof item === 'object' ? newSliderPicUrls.push(item.url) : newSliderPicUrls.push(item)
|
})
|
deepCopyFormData.sliderPicUrls = newSliderPicUrls
|
// 校验都通过后提交表单
|
const data = deepCopyFormData as ProductSpuApi.Spu
|
const id = params.id as unknown as number
|
if (!id) {
|
await ProductSpuApi.createSpu(data)
|
message.success(t('common.createSuccess'))
|
} else {
|
await ProductSpuApi.updateSpu(data)
|
message.success(t('common.updateSuccess'))
|
}
|
close()
|
} finally {
|
formLoading.value = false
|
}
|
}
|
|
/** 关闭按钮 */
|
const close = () => {
|
delView(unref(currentRoute))
|
push({ name: 'ProductSpu' })
|
}
|
|
/** 初始化 */
|
onMounted(async () => {
|
await getDetail()
|
})
|
</script>
|