From 23c8b8154fcf7c9251d07f028f3e14737ba4cecd Mon Sep 17 00:00:00 2001 From: dongyukun <1208714201@qq.com> Date: 星期二, 31 十二月 2024 19:04:35 +0800 Subject: [PATCH] 调度模型下发配置 --- src/components/DiyEditor/components/mobile/PromotionPoint/index.vue | 202 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 files changed, 202 insertions(+), 0 deletions(-) diff --git a/src/components/DiyEditor/components/mobile/PromotionPoint/index.vue b/src/components/DiyEditor/components/mobile/PromotionPoint/index.vue new file mode 100644 index 0000000..4acd93f --- /dev/null +++ b/src/components/DiyEditor/components/mobile/PromotionPoint/index.vue @@ -0,0 +1,202 @@ +<template> + <div ref="containerRef" :class="`box-content min-h-30px w-full flex flex-row flex-wrap`"> + <div + v-for="(spu, index) in spuList" + :key="index" + :style="{ + ...calculateSpace(index), + ...calculateWidth(), + borderTopLeftRadius: `${property.borderRadiusTop}px`, + borderTopRightRadius: `${property.borderRadiusTop}px`, + borderBottomLeftRadius: `${property.borderRadiusBottom}px`, + borderBottomRightRadius: `${property.borderRadiusBottom}px` + }" + class="relative box-content flex flex-row flex-wrap overflow-hidden bg-white" + > + <!-- 角标 --> + <div v-if="property.badge.show" class="absolute left-0 top-0 z-1 items-center justify-center"> + <el-image :src="property.badge.imgUrl" class="h-26px w-38px" fit="cover" /> + </div> + <!-- 商品封面图 --> + <div + :class="[ + 'h-140px', + { + 'w-full': property.layoutType !== 'oneColSmallImg', + 'w-140px': property.layoutType === 'oneColSmallImg' + } + ]" + > + <el-image :src="spu.picUrl" class="h-full w-full" fit="cover" /> + </div> + <div + :class="[ + ' flex flex-col gap-8px p-8px box-border', + { + 'w-full': property.layoutType !== 'oneColSmallImg', + 'w-[calc(100%-140px-16px)]': property.layoutType === 'oneColSmallImg' + } + ]" + > + <!-- 商品名称 --> + <div + v-if="property.fields.name.show" + :class="[ + 'text-14px ', + { + truncate: property.layoutType !== 'oneColSmallImg', + 'overflow-ellipsis line-clamp-2': property.layoutType === 'oneColSmallImg' + } + ]" + :style="{ color: property.fields.name.color }" + > + {{ spu.name }} + </div> + <!-- 商品简介 --> + <div + v-if="property.fields.introduction.show" + :style="{ color: property.fields.introduction.color }" + class="truncate text-12px" + > + {{ spu.introduction }} + </div> + <div> + <!-- 积分 --> + <span + v-if="property.fields.price.show" + :style="{ color: property.fields.price.color }" + class="text-16px" + > + {{ spu.point }}积分 + {{ !spu.pointPrice || spu.pointPrice === 0 ? '' : `+${fenToYuan(spu.pointPrice)}元` }} + </span> + <!-- 市场价 --> + <span + v-if="property.fields.marketPrice.show && spu.marketPrice" + :style="{ color: property.fields.marketPrice.color }" + class="ml-4px text-10px line-through" + > + ¥{{ fenToYuan(spu.marketPrice) }} + </span> + </div> + <div class="text-12px"> + <!-- 销量 --> + <span + v-if="property.fields.salesCount.show" + :style="{ color: property.fields.salesCount.color }" + > + 已兑{{ (spu.pointTotalStock || 0) - (spu.pointStock || 0) }}件 + </span> + <!-- 库存 --> + <span v-if="property.fields.stock.show" :style="{ color: property.fields.stock.color }"> + 库存{{ spu.pointTotalStock || 0 }} + </span> + </div> + </div> + <!-- 购买按钮 --> + <div class="absolute bottom-8px right-8px"> + <!-- 文字按钮 --> + <span + v-if="property.btnBuy.type === 'text'" + :style="{ + background: `linear-gradient(to right, ${property.btnBuy.bgBeginColor}, ${property.btnBuy.bgEndColor}` + }" + class="rounded-full p-x-12px p-y-4px text-12px text-white" + > + {{ property.btnBuy.text }} + </span> + <!-- 图片按钮 --> + <el-image + v-else + :src="property.btnBuy.imgUrl" + class="h-28px w-28px rounded-full" + fit="cover" + /> + </div> + </div> + </div> +</template> +<script lang="ts" setup> +import { PromotionPointProperty } from './config' +import * as ProductSpuApi from '@/api/mall/product/spu' +import { PointActivityApi, PointActivityVO, SpuExtension0 } from '@/api/mall/promotion/point' +import { fenToYuan } from '@/utils' + +/** 积分商城卡片 */ +defineOptions({ name: 'PromotionPoint' }) +// 定义属性 +const props = defineProps<{ property: PromotionPointProperty }>() +// 商品列表 +const spuList = ref<SpuExtension0[]>([]) +const spuIdList = ref<number[]>([]) +const pointActivityList = ref<PointActivityVO[]>([]) + +watch( + () => props.property.activityIds, + async () => { + try { + // 新添加的积分商城组件,是没有活动ID的 + const activityIds = props.property.activityIds + // 检查活动ID的有效性 + if (Array.isArray(activityIds) && activityIds.length > 0) { + // 获取积分商城活动详情列表 + pointActivityList.value = await PointActivityApi.getPointActivityListByIds(activityIds) + + // 获取积分商城活动的 SPU 详情列表 + spuList.value = [] + spuIdList.value = pointActivityList.value.map((activity) => activity.spuId) + if (spuIdList.value.length > 0) { + spuList.value = await ProductSpuApi.getSpuDetailList(spuIdList.value) + } + + // 更新 SPU 的最低兑换积分和所需兑换金额 + pointActivityList.value.forEach((activity) => { + // 匹配spuId + const spu = spuList.value.find((spu) => spu.id === activity.spuId) + if (spu) { + spu.pointStock = activity.stock + spu.pointTotalStock = activity.totalStock + spu.point = activity.point + spu.pointPrice = activity.price + } + }) + } + } catch (error) { + console.error('获取积分商城活动细节或 SPU 细节时出错:', error) + } + }, + { + immediate: true, + deep: true + } +) + +/** + * 计算商品的间距 + * @param index 商品索引 + */ +const calculateSpace = (index: number) => { + // 商品的列数 + const columns = props.property.layoutType === 'twoCol' ? 2 : 1 + // 第一列没有左边距 + const marginLeft = index % columns === 0 ? '0' : props.property.space + 'px' + // 第一行没有上边距 + const marginTop = index < columns ? '0' : props.property.space + 'px' + + return { marginLeft, marginTop } +} + +// 容器 +const containerRef = ref() +// 计算商品的宽度 +const calculateWidth = () => { + let width = '100%' + // 双列时每列的宽度为:(总宽度 - 间距)/ 2 + if (props.property.layoutType === 'twoCol') { + width = `${(containerRef.value.offsetWidth - props.property.space) / 2}px` + } + return { width } +} +</script> + +<style lang="scss" scoped></style> -- Gitblit v1.9.3