feat:【ele/antd】discountActivity todo 优化
This commit is contained in:
@@ -73,7 +73,6 @@ export function useFormSchema(): VbenFormSchema[] {
|
|||||||
fieldName: 'spuIds',
|
fieldName: 'spuIds',
|
||||||
label: '活动商品',
|
label: '活动商品',
|
||||||
component: 'Input',
|
component: 'Input',
|
||||||
rules: 'required',
|
|
||||||
formItemClass: 'col-span-2',
|
formItemClass: 'col-span-2',
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -1,30 +1,52 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
|
import type { MallSpuApi } from '#/api/mall/product/spu';
|
||||||
import type { MallDiscountActivityApi } from '#/api/mall/promotion/discount/discountActivity';
|
import type { MallDiscountActivityApi } from '#/api/mall/promotion/discount/discountActivity';
|
||||||
|
import type {
|
||||||
|
PropertyAndValues,
|
||||||
|
RuleConfig,
|
||||||
|
SpuProperty,
|
||||||
|
} from '#/views/mall/product/spu/components';
|
||||||
|
|
||||||
import { computed, ref } from 'vue';
|
import { computed, nextTick, ref } from 'vue';
|
||||||
|
|
||||||
import { useVbenForm, useVbenModal } from '@vben/common-ui';
|
import { useVbenForm, useVbenModal } from '@vben/common-ui';
|
||||||
|
import {
|
||||||
|
convertToInteger,
|
||||||
|
erpCalculatePercentage,
|
||||||
|
formatToFraction,
|
||||||
|
yuanToFen,
|
||||||
|
} from '@vben/utils';
|
||||||
|
|
||||||
import { message } from 'ant-design-vue';
|
import { Button, InputNumber, message } from 'ant-design-vue';
|
||||||
|
|
||||||
|
import { VxeColumn } from '#/adapter/vxe-table';
|
||||||
|
import { getSpuDetailList } from '#/api/mall/product/spu';
|
||||||
import {
|
import {
|
||||||
createDiscountActivity,
|
createDiscountActivity,
|
||||||
getDiscountActivity,
|
getDiscountActivity,
|
||||||
updateDiscountActivity,
|
updateDiscountActivity,
|
||||||
} from '#/api/mall/promotion/discount/discountActivity';
|
} from '#/api/mall/promotion/discount/discountActivity';
|
||||||
import { $t } from '#/locales';
|
import { $t } from '#/locales';
|
||||||
import { SpuShowcase } from '#/views/mall/product/spu/components';
|
import {
|
||||||
|
getPropertyList,
|
||||||
|
SpuAndSkuList,
|
||||||
|
SpuSkuSelect,
|
||||||
|
} from '#/views/mall/product/spu/components';
|
||||||
|
|
||||||
import { useFormSchema } from '../data';
|
import { useFormSchema } from '../data';
|
||||||
|
|
||||||
defineOptions({ name: 'DiscountActivityForm' });
|
defineOptions({ name: 'DiscountActivityForm' });
|
||||||
|
|
||||||
const emit = defineEmits(['success']);
|
const emit = defineEmits(['success']);
|
||||||
const formData = ref<
|
|
||||||
Partial<MallDiscountActivityApi.DiscountActivity> & {
|
/** 折扣类型枚举 */
|
||||||
spuIds?: number[];
|
const PromotionDiscountTypeEnum = {
|
||||||
}
|
PRICE: { type: 1 }, // 满减
|
||||||
>({});
|
PERCENT: { type: 2 }, // 折扣
|
||||||
|
};
|
||||||
|
|
||||||
|
// ================= 表单相关 =================
|
||||||
|
const formData = ref<Partial<MallDiscountActivityApi.DiscountActivity>>({});
|
||||||
const getTitle = computed(() => {
|
const getTitle = computed(() => {
|
||||||
return formData.value?.id
|
return formData.value?.id
|
||||||
? $t('ui.actionTitle.edit', ['限时折扣活动'])
|
? $t('ui.actionTitle.edit', ['限时折扣活动'])
|
||||||
@@ -44,27 +66,203 @@ const [Form, formApi] = useVbenForm({
|
|||||||
showDefaultActions: false,
|
showDefaultActions: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// ================= 商品选择相关 =================
|
||||||
|
/** SKU 扩展类型 */
|
||||||
|
interface SkuExtension extends MallSpuApi.Sku {
|
||||||
|
productConfig: MallDiscountActivityApi.DiscountProduct;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** SPU 扩展类型 */
|
||||||
|
interface SpuExtension extends MallSpuApi.Spu {
|
||||||
|
skus?: SkuExtension[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const spuSelectRef = ref(); // 商品选择组件 Ref
|
||||||
|
const spuAndSkuListRef = ref(); // SKU 列表组件 Ref
|
||||||
|
const spuList = ref<SpuExtension[]>([]); // 选择的 SPU 列表
|
||||||
|
const spuPropertyList = ref<SpuProperty<SpuExtension>[]>([]); // SPU 属性列表
|
||||||
|
const spuIdList = ref<number[]>([]); // 已选择的 SPU ID 列表
|
||||||
|
|
||||||
|
/** SKU 校验规则配置 */
|
||||||
|
const ruleConfig: RuleConfig[] = [
|
||||||
|
{
|
||||||
|
name: 'productConfig.discountPrice',
|
||||||
|
rule: (arg) => arg > 0,
|
||||||
|
message: '商品优惠金额不能为 0 !!!',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/** 打开商品选择弹窗 */
|
||||||
|
function openSpuSelect() {
|
||||||
|
spuSelectRef.value?.open();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 选择商品后的回调 */
|
||||||
|
function handleSpuSelected(spuId: number, skuIds?: number[]) {
|
||||||
|
getSpuDetails(spuId, skuIds);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取 SPU 详情 */
|
||||||
|
async function getSpuDetails(
|
||||||
|
spuId: number,
|
||||||
|
skuIdArr?: number[],
|
||||||
|
products?: MallDiscountActivityApi.DiscountProduct[],
|
||||||
|
type?: string,
|
||||||
|
) {
|
||||||
|
// 如果已经包含该 SPU 则跳过
|
||||||
|
if (spuIdList.value.includes(spuId)) {
|
||||||
|
if (type !== 'load') {
|
||||||
|
message.error('数据重复选择!');
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
spuIdList.value.push(spuId);
|
||||||
|
|
||||||
|
const res = (await getSpuDetailList([spuId])) as SpuExtension[];
|
||||||
|
if (res.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const spu = res[0]!;
|
||||||
|
// 筛选 SKU
|
||||||
|
const selectSkus =
|
||||||
|
skuIdArr === undefined
|
||||||
|
? spu.skus
|
||||||
|
: spu.skus?.filter((sku) => skuIdArr.includes(sku.id!));
|
||||||
|
|
||||||
|
// 为每个 SKU 添加折扣配置
|
||||||
|
selectSkus?.forEach((sku) => {
|
||||||
|
let config: MallDiscountActivityApi.DiscountProduct = {
|
||||||
|
skuId: sku.id!,
|
||||||
|
spuId: spu.id!,
|
||||||
|
discountType: 1,
|
||||||
|
discountPercent: 0,
|
||||||
|
discountPrice: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 编辑时,使用已有的配置
|
||||||
|
if (products !== undefined) {
|
||||||
|
const product = products.find((item) => item.skuId === sku.id);
|
||||||
|
if (product) {
|
||||||
|
// 转换为元显示
|
||||||
|
config = {
|
||||||
|
...product,
|
||||||
|
discountPercent: Number(formatToFraction(product.discountPercent)),
|
||||||
|
discountPrice: Number(formatToFraction(product.discountPrice)),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
(sku as SkuExtension).productConfig = config;
|
||||||
|
});
|
||||||
|
|
||||||
|
spu.skus = selectSkus as SkuExtension[];
|
||||||
|
spuPropertyList.value.push({
|
||||||
|
spuId: spu.id!,
|
||||||
|
spuDetail: spu,
|
||||||
|
propertyList: getPropertyList(spu) as PropertyAndValues[],
|
||||||
|
});
|
||||||
|
spuList.value.push(spu);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除 SPU */
|
||||||
|
function handleDeleteSpu(spuId: number) {
|
||||||
|
const spuIndex = spuIdList.value.indexOf(spuId);
|
||||||
|
if (spuIndex !== -1) {
|
||||||
|
spuIdList.value.splice(spuIndex, 1);
|
||||||
|
}
|
||||||
|
const propertyIndex = spuPropertyList.value.findIndex(
|
||||||
|
(item) => item.spuId === spuId,
|
||||||
|
);
|
||||||
|
if (propertyIndex !== -1) {
|
||||||
|
spuPropertyList.value.splice(propertyIndex, 1);
|
||||||
|
}
|
||||||
|
const listIndex = spuList.value.findIndex((item) => item.id === spuId);
|
||||||
|
if (listIndex !== -1) {
|
||||||
|
spuList.value.splice(listIndex, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 处理 SKU 优惠金额变动 */
|
||||||
|
function handleSkuDiscountPriceChange(row: SkuExtension) {
|
||||||
|
if (row.productConfig.discountPrice <= 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 设置优惠类型:满减
|
||||||
|
row.productConfig.discountType = PromotionDiscountTypeEnum.PRICE.type;
|
||||||
|
// 计算折扣百分比
|
||||||
|
const price = typeof row.price === 'number' ? row.price : Number(row.price);
|
||||||
|
const percent = erpCalculatePercentage(
|
||||||
|
price - yuanToFen(row.productConfig.discountPrice),
|
||||||
|
price,
|
||||||
|
);
|
||||||
|
row.productConfig.discountPercent =
|
||||||
|
typeof percent === 'number' ? percent : Number(percent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 处理 SKU 折扣百分比变动 */
|
||||||
|
function handleSkuDiscountPercentChange(row: SkuExtension) {
|
||||||
|
if (
|
||||||
|
row.productConfig.discountPercent <= 0 ||
|
||||||
|
row.productConfig.discountPercent >= 100
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 设置优惠类型:折扣
|
||||||
|
row.productConfig.discountType = PromotionDiscountTypeEnum.PERCENT.type;
|
||||||
|
// 计算优惠金额
|
||||||
|
const price = typeof row.price === 'number' ? row.price : Number(row.price);
|
||||||
|
row.productConfig.discountPrice = Number(
|
||||||
|
formatToFraction(price - price * (row.productConfig.discountPercent / 100)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 重置表单 */
|
||||||
|
async function resetForm() {
|
||||||
|
spuList.value = [];
|
||||||
|
spuPropertyList.value = [];
|
||||||
|
spuIdList.value = [];
|
||||||
|
formData.value = {};
|
||||||
|
await nextTick();
|
||||||
|
await formApi.resetForm();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ================= 弹窗相关 =================
|
||||||
const [Modal, modalApi] = useVbenModal({
|
const [Modal, modalApi] = useVbenModal({
|
||||||
async onConfirm() {
|
async onConfirm() {
|
||||||
const { valid } = await formApi.validate();
|
const { valid } = await formApi.validate();
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
modalApi.lock();
|
|
||||||
// 提交表单
|
|
||||||
const data =
|
|
||||||
(await formApi.getValues()) as MallDiscountActivityApi.DiscountActivity;
|
|
||||||
|
|
||||||
// 确保必要的默认值
|
// 校验是否选择了商品
|
||||||
if (!data.products) {
|
if (spuList.value.length === 0) {
|
||||||
data.products = [];
|
message.warning('请选择活动商品');
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
modalApi.lock();
|
||||||
try {
|
try {
|
||||||
|
// 获取折扣商品配置
|
||||||
|
const products = structuredClone(
|
||||||
|
spuAndSkuListRef.value?.getSkuConfigs('productConfig') || [],
|
||||||
|
) as MallDiscountActivityApi.DiscountProduct[];
|
||||||
|
|
||||||
|
// 转换金额为分
|
||||||
|
products.forEach((item) => {
|
||||||
|
item.discountPercent = convertToInteger(item.discountPercent);
|
||||||
|
item.discountPrice = convertToInteger(item.discountPrice);
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = structuredClone(
|
||||||
|
await formApi.getValues(),
|
||||||
|
) as MallDiscountActivityApi.DiscountActivity;
|
||||||
|
data.products = products;
|
||||||
|
|
||||||
|
// 提交请求
|
||||||
await (formData.value?.id
|
await (formData.value?.id
|
||||||
? updateDiscountActivity(data)
|
? updateDiscountActivity(data)
|
||||||
: createDiscountActivity(data));
|
: createDiscountActivity(data));
|
||||||
// 关闭并提示
|
|
||||||
await modalApi.close();
|
await modalApi.close();
|
||||||
emit('success');
|
emit('success');
|
||||||
message.success($t('ui.actionMessage.operationSuccess'));
|
message.success($t('ui.actionMessage.operationSuccess'));
|
||||||
@@ -74,19 +272,45 @@ const [Modal, modalApi] = useVbenModal({
|
|||||||
},
|
},
|
||||||
async onOpenChange(isOpen: boolean) {
|
async onOpenChange(isOpen: boolean) {
|
||||||
if (!isOpen) {
|
if (!isOpen) {
|
||||||
formData.value = {};
|
await resetForm();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 加载数据
|
// 加载数据
|
||||||
const data = modalApi.getData<MallDiscountActivityApi.DiscountActivity>();
|
const data = modalApi.getData<MallDiscountActivityApi.DiscountActivity>();
|
||||||
if (!data || !data.id) {
|
if (!data || !data.id) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
modalApi.lock();
|
modalApi.lock();
|
||||||
try {
|
try {
|
||||||
formData.value = await getDiscountActivity(data.id);
|
const activityData = await getDiscountActivity(data.id);
|
||||||
// 设置到 values
|
formData.value = activityData;
|
||||||
await formApi.setValues(formData.value);
|
|
||||||
|
// 加载商品详情
|
||||||
|
if (activityData.products && activityData.products.length > 0) {
|
||||||
|
// 按 spuId 分组
|
||||||
|
const spuProductsMap = new Map<
|
||||||
|
number,
|
||||||
|
MallDiscountActivityApi.DiscountProduct[]
|
||||||
|
>();
|
||||||
|
for (const product of activityData.products) {
|
||||||
|
const spuId = product.spuId;
|
||||||
|
if (!spuProductsMap.has(spuId)) {
|
||||||
|
spuProductsMap.set(spuId, []);
|
||||||
|
}
|
||||||
|
spuProductsMap.get(spuId)!.push(product);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载每个 SPU 的详情
|
||||||
|
for (const [spuId, products] of spuProductsMap) {
|
||||||
|
const skuIdArr = products.map((p) => p.skuId);
|
||||||
|
await getSpuDetails(spuId, skuIdArr, products, 'load');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置表单值
|
||||||
|
await formApi.setValues(activityData);
|
||||||
} finally {
|
} finally {
|
||||||
modalApi.unlock();
|
modalApi.unlock();
|
||||||
}
|
}
|
||||||
@@ -95,12 +319,59 @@ const [Modal, modalApi] = useVbenModal({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<Modal class="w-3/5" :title="getTitle">
|
<Modal class="w-[70%]" :title="getTitle">
|
||||||
<Form>
|
<Form>
|
||||||
<!-- 自定义插槽:商品选择 -->
|
<!-- 自定义插槽:商品选择 -->
|
||||||
<template #spuIds>
|
<template #spuIds>
|
||||||
<SpuShowcase v-model="formData.spuIds" />
|
<div class="w-full">
|
||||||
|
<Button class="mb-4" @click="openSpuSelect">选择商品</Button>
|
||||||
|
<SpuAndSkuList
|
||||||
|
ref="spuAndSkuListRef"
|
||||||
|
:deletable="true"
|
||||||
|
:rule-config="ruleConfig"
|
||||||
|
:spu-list="spuList"
|
||||||
|
:spu-property-list-p="spuPropertyList"
|
||||||
|
@delete="handleDeleteSpu"
|
||||||
|
>
|
||||||
|
<!-- 扩展列:限时折扣活动特有配置 -->
|
||||||
|
<template #default>
|
||||||
|
<VxeColumn align="center" min-width="168" title="优惠金额(元)">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<InputNumber
|
||||||
|
v-model:value="row.productConfig.discountPrice"
|
||||||
|
:max="Number(formatToFraction(row.price))"
|
||||||
|
:min="0"
|
||||||
|
:precision="2"
|
||||||
|
:step="0.1"
|
||||||
|
class="w-full"
|
||||||
|
@change="handleSkuDiscountPriceChange(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</VxeColumn>
|
||||||
|
<VxeColumn align="center" min-width="168" title="折扣百分比(%)">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<InputNumber
|
||||||
|
v-model:value="row.productConfig.discountPercent"
|
||||||
|
:max="100"
|
||||||
|
:min="0"
|
||||||
|
:precision="2"
|
||||||
|
:step="0.1"
|
||||||
|
class="w-full"
|
||||||
|
@change="handleSkuDiscountPercentChange(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</VxeColumn>
|
||||||
|
</template>
|
||||||
|
</SpuAndSkuList>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Form>
|
</Form>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|
||||||
|
<!-- 商品选择弹窗 -->
|
||||||
|
<SpuSkuSelect
|
||||||
|
ref="spuSelectRef"
|
||||||
|
:is-select-sku="true"
|
||||||
|
@select="handleSpuSelected"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -75,7 +75,6 @@ export function useFormSchema(): VbenFormSchema[] {
|
|||||||
fieldName: 'spuIds',
|
fieldName: 'spuIds',
|
||||||
label: '活动商品',
|
label: '活动商品',
|
||||||
component: 'Input',
|
component: 'Input',
|
||||||
rules: 'required',
|
|
||||||
formItemClass: 'col-span-2',
|
formItemClass: 'col-span-2',
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -1,30 +1,52 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
|
import type { MallSpuApi } from '#/api/mall/product/spu';
|
||||||
import type { MallDiscountActivityApi } from '#/api/mall/promotion/discount/discountActivity';
|
import type { MallDiscountActivityApi } from '#/api/mall/promotion/discount/discountActivity';
|
||||||
|
import type {
|
||||||
|
PropertyAndValues,
|
||||||
|
RuleConfig,
|
||||||
|
SpuProperty,
|
||||||
|
} from '#/views/mall/product/spu/components';
|
||||||
|
|
||||||
import { computed, ref } from 'vue';
|
import { computed, nextTick, ref } from 'vue';
|
||||||
|
|
||||||
import { useVbenForm, useVbenModal } from '@vben/common-ui';
|
import { useVbenForm, useVbenModal } from '@vben/common-ui';
|
||||||
|
import {
|
||||||
|
convertToInteger,
|
||||||
|
erpCalculatePercentage,
|
||||||
|
formatToFraction,
|
||||||
|
yuanToFen,
|
||||||
|
} from '@vben/utils';
|
||||||
|
|
||||||
import { ElMessage } from 'element-plus';
|
import { ElButton, ElInputNumber, ElMessage } from 'element-plus';
|
||||||
|
|
||||||
|
import { VxeColumn } from '#/adapter/vxe-table';
|
||||||
|
import { getSpuDetailList } from '#/api/mall/product/spu';
|
||||||
import {
|
import {
|
||||||
createDiscountActivity,
|
createDiscountActivity,
|
||||||
getDiscountActivity,
|
getDiscountActivity,
|
||||||
updateDiscountActivity,
|
updateDiscountActivity,
|
||||||
} from '#/api/mall/promotion/discount/discountActivity';
|
} from '#/api/mall/promotion/discount/discountActivity';
|
||||||
import { $t } from '#/locales';
|
import { $t } from '#/locales';
|
||||||
import { SpuShowcase } from '#/views/mall/product/spu/components';
|
import {
|
||||||
|
getPropertyList,
|
||||||
|
SpuAndSkuList,
|
||||||
|
SpuSkuSelect,
|
||||||
|
} from '#/views/mall/product/spu/components';
|
||||||
|
|
||||||
import { useFormSchema } from '../data';
|
import { useFormSchema } from '../data';
|
||||||
|
|
||||||
defineOptions({ name: 'DiscountActivityForm' });
|
defineOptions({ name: 'DiscountActivityForm' });
|
||||||
|
|
||||||
|
/** 折扣类型枚举 */
|
||||||
|
const PromotionDiscountTypeEnum = {
|
||||||
|
PRICE: { type: 1 }, // 满减
|
||||||
|
PERCENT: { type: 2 }, // 折扣
|
||||||
|
};
|
||||||
|
|
||||||
const emit = defineEmits(['success']);
|
const emit = defineEmits(['success']);
|
||||||
const formData = ref<
|
|
||||||
Partial<MallDiscountActivityApi.DiscountActivity> & {
|
// ================= 表单相关 =================
|
||||||
spuIds?: number[];
|
const formData = ref<Partial<MallDiscountActivityApi.DiscountActivity>>({});
|
||||||
}
|
|
||||||
>({});
|
|
||||||
const getTitle = computed(() => {
|
const getTitle = computed(() => {
|
||||||
return formData.value?.id
|
return formData.value?.id
|
||||||
? $t('ui.actionTitle.edit', ['限时折扣活动'])
|
? $t('ui.actionTitle.edit', ['限时折扣活动'])
|
||||||
@@ -44,28 +66,203 @@ const [Form, formApi] = useVbenForm({
|
|||||||
showDefaultActions: false,
|
showDefaultActions: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
// TODO @puhui999:antd 和 ele 里,修改时,商品都没展示。
|
// ================= 商品选择相关 =================
|
||||||
|
/** SKU 扩展类型 */
|
||||||
|
interface SkuExtension extends MallSpuApi.Sku {
|
||||||
|
productConfig: MallDiscountActivityApi.DiscountProduct;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** SPU 扩展类型 */
|
||||||
|
interface SpuExtension extends MallSpuApi.Spu {
|
||||||
|
skus?: SkuExtension[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const spuSelectRef = ref(); // 商品选择组件 Ref
|
||||||
|
const spuAndSkuListRef = ref(); // SKU 列表组件 Ref
|
||||||
|
const spuList = ref<SpuExtension[]>([]); // 选择的 SPU 列表
|
||||||
|
const spuPropertyList = ref<SpuProperty<SpuExtension>[]>([]); // SPU 属性列表
|
||||||
|
const spuIdList = ref<number[]>([]); // 已选择的 SPU ID 列表
|
||||||
|
|
||||||
|
/** SKU 校验规则配置 */
|
||||||
|
const ruleConfig: RuleConfig[] = [
|
||||||
|
{
|
||||||
|
name: 'productConfig.discountPrice',
|
||||||
|
rule: (arg) => arg > 0,
|
||||||
|
message: '商品优惠金额不能为 0 !!!',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/** 打开商品选择弹窗 */
|
||||||
|
function openSpuSelect() {
|
||||||
|
spuSelectRef.value?.open();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 选择商品后的回调 */
|
||||||
|
function handleSpuSelected(spuId: number, skuIds?: number[]) {
|
||||||
|
getSpuDetails(spuId, skuIds);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取 SPU 详情 */
|
||||||
|
async function getSpuDetails(
|
||||||
|
spuId: number,
|
||||||
|
skuIdArr?: number[],
|
||||||
|
products?: MallDiscountActivityApi.DiscountProduct[],
|
||||||
|
type?: string,
|
||||||
|
) {
|
||||||
|
// 如果已经包含该 SPU 则跳过
|
||||||
|
if (spuIdList.value.includes(spuId)) {
|
||||||
|
if (type !== 'load') {
|
||||||
|
ElMessage.error('数据重复选择!');
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
spuIdList.value.push(spuId);
|
||||||
|
|
||||||
|
const res = (await getSpuDetailList([spuId])) as SpuExtension[];
|
||||||
|
if (res.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const spu = res[0]!;
|
||||||
|
// 筛选 SKU
|
||||||
|
const selectSkus =
|
||||||
|
skuIdArr === undefined
|
||||||
|
? spu.skus
|
||||||
|
: spu.skus?.filter((sku) => skuIdArr.includes(sku.id!));
|
||||||
|
|
||||||
|
// 为每个 SKU 添加折扣配置
|
||||||
|
selectSkus?.forEach((sku) => {
|
||||||
|
let config: MallDiscountActivityApi.DiscountProduct = {
|
||||||
|
skuId: sku.id!,
|
||||||
|
spuId: spu.id!,
|
||||||
|
discountType: 1,
|
||||||
|
discountPercent: 0,
|
||||||
|
discountPrice: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 编辑时,使用已有的配置
|
||||||
|
if (products !== undefined) {
|
||||||
|
const product = products.find((item) => item.skuId === sku.id);
|
||||||
|
if (product) {
|
||||||
|
// 转换为元显示
|
||||||
|
config = {
|
||||||
|
...product,
|
||||||
|
discountPercent: Number(formatToFraction(product.discountPercent)),
|
||||||
|
discountPrice: Number(formatToFraction(product.discountPrice)),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
(sku as SkuExtension).productConfig = config;
|
||||||
|
});
|
||||||
|
|
||||||
|
spu.skus = selectSkus as SkuExtension[];
|
||||||
|
spuPropertyList.value.push({
|
||||||
|
spuId: spu.id!,
|
||||||
|
spuDetail: spu,
|
||||||
|
propertyList: getPropertyList(spu) as PropertyAndValues[],
|
||||||
|
});
|
||||||
|
spuList.value.push(spu);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除 SPU */
|
||||||
|
function handleDeleteSpu(spuId: number) {
|
||||||
|
const spuIndex = spuIdList.value.indexOf(spuId);
|
||||||
|
if (spuIndex !== -1) {
|
||||||
|
spuIdList.value.splice(spuIndex, 1);
|
||||||
|
}
|
||||||
|
const propertyIndex = spuPropertyList.value.findIndex(
|
||||||
|
(item) => item.spuId === spuId,
|
||||||
|
);
|
||||||
|
if (propertyIndex !== -1) {
|
||||||
|
spuPropertyList.value.splice(propertyIndex, 1);
|
||||||
|
}
|
||||||
|
const listIndex = spuList.value.findIndex((item) => item.id === spuId);
|
||||||
|
if (listIndex !== -1) {
|
||||||
|
spuList.value.splice(listIndex, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 处理 SKU 优惠金额变动 */
|
||||||
|
function handleSkuDiscountPriceChange(row: SkuExtension) {
|
||||||
|
if (row.productConfig.discountPrice <= 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 设置优惠类型:满减
|
||||||
|
row.productConfig.discountType = PromotionDiscountTypeEnum.PRICE.type;
|
||||||
|
// 计算折扣百分比
|
||||||
|
const price = typeof row.price === 'number' ? row.price : Number(row.price);
|
||||||
|
const percent = erpCalculatePercentage(
|
||||||
|
price - yuanToFen(row.productConfig.discountPrice),
|
||||||
|
price,
|
||||||
|
);
|
||||||
|
row.productConfig.discountPercent =
|
||||||
|
typeof percent === 'number' ? percent : Number(percent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 处理 SKU 折扣百分比变动 */
|
||||||
|
function handleSkuDiscountPercentChange(row: SkuExtension) {
|
||||||
|
if (
|
||||||
|
row.productConfig.discountPercent <= 0 ||
|
||||||
|
row.productConfig.discountPercent >= 100
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 设置优惠类型:折扣
|
||||||
|
row.productConfig.discountType = PromotionDiscountTypeEnum.PERCENT.type;
|
||||||
|
// 计算优惠金额
|
||||||
|
const price = typeof row.price === 'number' ? row.price : Number(row.price);
|
||||||
|
row.productConfig.discountPrice = Number(
|
||||||
|
formatToFraction(price - price * (row.productConfig.discountPercent / 100)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 重置表单 */
|
||||||
|
async function resetForm() {
|
||||||
|
spuList.value = [];
|
||||||
|
spuPropertyList.value = [];
|
||||||
|
spuIdList.value = [];
|
||||||
|
formData.value = {};
|
||||||
|
await nextTick();
|
||||||
|
await formApi.resetForm();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ================= 弹窗相关 =================
|
||||||
const [Modal, modalApi] = useVbenModal({
|
const [Modal, modalApi] = useVbenModal({
|
||||||
async onConfirm() {
|
async onConfirm() {
|
||||||
const { valid } = await formApi.validate();
|
const { valid } = await formApi.validate();
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
modalApi.lock();
|
|
||||||
// 提交表单
|
|
||||||
const data =
|
|
||||||
(await formApi.getValues()) as MallDiscountActivityApi.DiscountActivity;
|
|
||||||
|
|
||||||
// 确保必要的默认值
|
// 校验是否选择了商品
|
||||||
if (!data.products) {
|
if (spuList.value.length === 0) {
|
||||||
data.products = [];
|
ElMessage.warning('请选择活动商品');
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
modalApi.lock();
|
||||||
try {
|
try {
|
||||||
|
// 获取折扣商品配置
|
||||||
|
const products = structuredClone(
|
||||||
|
spuAndSkuListRef.value?.getSkuConfigs('productConfig') || [],
|
||||||
|
) as MallDiscountActivityApi.DiscountProduct[];
|
||||||
|
|
||||||
|
// 转换金额为分
|
||||||
|
products.forEach((item) => {
|
||||||
|
item.discountPercent = convertToInteger(item.discountPercent);
|
||||||
|
item.discountPrice = convertToInteger(item.discountPrice);
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = structuredClone(
|
||||||
|
await formApi.getValues(),
|
||||||
|
) as MallDiscountActivityApi.DiscountActivity;
|
||||||
|
data.products = products;
|
||||||
|
|
||||||
|
// 提交请求
|
||||||
await (formData.value?.id
|
await (formData.value?.id
|
||||||
? updateDiscountActivity(data)
|
? updateDiscountActivity(data)
|
||||||
: createDiscountActivity(data));
|
: createDiscountActivity(data));
|
||||||
// 关闭并提示
|
|
||||||
await modalApi.close();
|
await modalApi.close();
|
||||||
emit('success');
|
emit('success');
|
||||||
ElMessage.success($t('ui.actionMessage.operationSuccess'));
|
ElMessage.success($t('ui.actionMessage.operationSuccess'));
|
||||||
@@ -75,19 +272,45 @@ const [Modal, modalApi] = useVbenModal({
|
|||||||
},
|
},
|
||||||
async onOpenChange(isOpen: boolean) {
|
async onOpenChange(isOpen: boolean) {
|
||||||
if (!isOpen) {
|
if (!isOpen) {
|
||||||
formData.value = {};
|
await resetForm();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 加载数据
|
// 加载数据
|
||||||
const data = modalApi.getData<MallDiscountActivityApi.DiscountActivity>();
|
const data = modalApi.getData<MallDiscountActivityApi.DiscountActivity>();
|
||||||
if (!data || !data.id) {
|
if (!data || !data.id) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
modalApi.lock();
|
modalApi.lock();
|
||||||
try {
|
try {
|
||||||
formData.value = await getDiscountActivity(data.id);
|
const activityData = await getDiscountActivity(data.id);
|
||||||
// 设置到 values
|
formData.value = activityData;
|
||||||
await formApi.setValues(formData.value);
|
|
||||||
|
// 加载商品详情
|
||||||
|
if (activityData.products && activityData.products.length > 0) {
|
||||||
|
// 按 spuId 分组
|
||||||
|
const spuProductsMap = new Map<
|
||||||
|
number,
|
||||||
|
MallDiscountActivityApi.DiscountProduct[]
|
||||||
|
>();
|
||||||
|
for (const product of activityData.products) {
|
||||||
|
const spuId = product.spuId;
|
||||||
|
if (!spuProductsMap.has(spuId)) {
|
||||||
|
spuProductsMap.set(spuId, []);
|
||||||
|
}
|
||||||
|
spuProductsMap.get(spuId)!.push(product);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载每个 SPU 的详情
|
||||||
|
for (const [spuId, products] of spuProductsMap) {
|
||||||
|
const skuIdArr = products.map((p) => p.skuId);
|
||||||
|
await getSpuDetails(spuId, skuIdArr, products, 'load');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置表单值
|
||||||
|
await formApi.setValues(activityData);
|
||||||
} finally {
|
} finally {
|
||||||
modalApi.unlock();
|
modalApi.unlock();
|
||||||
}
|
}
|
||||||
@@ -96,12 +319,59 @@ const [Modal, modalApi] = useVbenModal({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<Modal class="w-3/5" :title="getTitle">
|
<Modal class="w-[70%]" :title="getTitle">
|
||||||
<Form>
|
<Form>
|
||||||
<!-- 自定义插槽:商品选择 -->
|
<!-- 自定义插槽:商品选择 -->
|
||||||
<template #spuIds>
|
<template #spuIds>
|
||||||
<SpuShowcase v-model="formData.spuIds" />
|
<div class="w-full">
|
||||||
|
<ElButton class="mb-4" @click="openSpuSelect">选择商品</ElButton>
|
||||||
|
<SpuAndSkuList
|
||||||
|
ref="spuAndSkuListRef"
|
||||||
|
:deletable="true"
|
||||||
|
:rule-config="ruleConfig"
|
||||||
|
:spu-list="spuList"
|
||||||
|
:spu-property-list-p="spuPropertyList"
|
||||||
|
@delete="handleDeleteSpu"
|
||||||
|
>
|
||||||
|
<!-- 扩展列:限时折扣活动特有配置 -->
|
||||||
|
<template #default>
|
||||||
|
<VxeColumn align="center" min-width="168" title="优惠金额(元)">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<ElInputNumber
|
||||||
|
v-model="row.productConfig.discountPrice"
|
||||||
|
:max="Number(formatToFraction(row.price))"
|
||||||
|
:min="0"
|
||||||
|
:precision="2"
|
||||||
|
:step="0.1"
|
||||||
|
class="w-full"
|
||||||
|
@change="handleSkuDiscountPriceChange(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</VxeColumn>
|
||||||
|
<VxeColumn align="center" min-width="168" title="折扣百分比(%)">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<ElInputNumber
|
||||||
|
v-model="row.productConfig.discountPercent"
|
||||||
|
:max="100"
|
||||||
|
:min="0"
|
||||||
|
:precision="2"
|
||||||
|
:step="0.1"
|
||||||
|
class="w-full"
|
||||||
|
@change="handleSkuDiscountPercentChange(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</VxeColumn>
|
||||||
|
</template>
|
||||||
|
</SpuAndSkuList>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</Form>
|
</Form>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|
||||||
|
<!-- 商品选择弹窗 -->
|
||||||
|
<SpuSkuSelect
|
||||||
|
ref="spuSelectRef"
|
||||||
|
:is-select-sku="true"
|
||||||
|
@select="handleSpuSelected"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
Reference in New Issue
Block a user