潘志宝
2024-09-29 9907560470c1a6821f4998453bb885cd9fee3445
提交 | 用户 | 时间
820397 1 <template>
H 2   <div :class="`box-content min-h-30px w-full flex flex-row flex-wrap`" ref="containerRef">
3     <div
4       class="relative box-content flex flex-row flex-wrap overflow-hidden bg-white"
5       :style="{
6         ...calculateSpace(index),
7         ...calculateWidth(),
8         borderTopLeftRadius: `${property.borderRadiusTop}px`,
9         borderTopRightRadius: `${property.borderRadiusTop}px`,
10         borderBottomLeftRadius: `${property.borderRadiusBottom}px`,
11         borderBottomRightRadius: `${property.borderRadiusBottom}px`
12       }"
13       v-for="(spu, index) in spuList"
14       :key="index"
15     >
16       <!-- 角标 -->
17       <div v-if="property.badge.show" class="absolute left-0 top-0 z-1 items-center justify-center">
18         <el-image fit="cover" :src="property.badge.imgUrl" class="h-26px w-38px" />
19       </div>
20       <!-- 商品封面图 -->
21       <div
22         :class="[
23           'h-140px',
24           {
25             'w-full': property.layoutType !== 'oneColSmallImg',
26             'w-140px': property.layoutType === 'oneColSmallImg'
27           }
28         ]"
29       >
30         <el-image fit="cover" class="h-full w-full" :src="spu.picUrl" />
31       </div>
32       <div
33         :class="[
34           ' flex flex-col gap-8px p-8px box-border',
35           {
36             'w-full': property.layoutType !== 'oneColSmallImg',
37             'w-[calc(100%-140px-16px)]': property.layoutType === 'oneColSmallImg'
38           }
39         ]"
40       >
41         <!-- 商品名称 -->
42         <div
43           v-if="property.fields.name.show"
44           :class="[
45             'text-14px ',
46             {
47               truncate: property.layoutType !== 'oneColSmallImg',
48               'overflow-ellipsis line-clamp-2': property.layoutType === 'oneColSmallImg'
49             }
50           ]"
51           :style="{ color: property.fields.name.color }"
52         >
53           {{ spu.name }}
54         </div>
55         <!-- 商品简介 -->
56         <div
57           v-if="property.fields.introduction.show"
58           class="truncate text-12px"
59           :style="{ color: property.fields.introduction.color }"
60         >
61           {{ spu.introduction }}
62         </div>
63         <div>
64           <!-- 价格 -->
65           <span
66             v-if="property.fields.price.show"
67             class="text-16px"
68             :style="{ color: property.fields.price.color }"
69           >
70             ¥{{ spu.price }}
71           </span>
72           <!-- 市场价 -->
73           <span
74             v-if="property.fields.marketPrice.show && spu.marketPrice"
75             class="ml-4px text-10px line-through"
76             :style="{ color: property.fields.marketPrice.color }"
77             >¥{{ spu.marketPrice }}</span
78           >
79         </div>
80         <div class="text-12px">
81           <!-- 销量 -->
82           <span
83             v-if="property.fields.salesCount.show"
84             :style="{ color: property.fields.salesCount.color }"
85           >
86             已售{{ (spu.salesCount || 0) + (spu.virtualSalesCount || 0) }}件
87           </span>
88           <!-- 库存 -->
89           <span v-if="property.fields.stock.show" :style="{ color: property.fields.stock.color }">
90             库存{{ spu.stock || 0 }}
91           </span>
92         </div>
93       </div>
94       <!-- 购买按钮 -->
95       <div class="absolute bottom-8px right-8px">
96         <!-- 文字按钮 -->
97         <span
98           v-if="property.btnBuy.type === 'text'"
99           class="rounded-full p-x-12px p-y-4px text-12px text-white"
100           :style="{
101             background: `linear-gradient(to right, ${property.btnBuy.bgBeginColor}, ${property.btnBuy.bgEndColor}`
102           }"
103         >
104           {{ property.btnBuy.text }}
105         </span>
106         <!-- 图片按钮 -->
107         <el-image
108           v-else
109           class="h-28px w-28px rounded-full"
110           fit="cover"
111           :src="property.btnBuy.imgUrl"
112         />
113       </div>
114     </div>
115   </div>
116 </template>
117 <script setup lang="ts">
118 import { ProductCardProperty } from './config'
119 import * as ProductSpuApi from '@/api/mall/product/spu'
120
121 /** 商品卡片 */
122 defineOptions({ name: 'ProductCard' })
123 // 定义属性
124 const props = defineProps<{ property: ProductCardProperty }>()
125 // 商品列表
126 const spuList = ref<ProductSpuApi.Spu[]>([])
127 watch(
128   () => props.property.spuIds,
129   async () => {
130     spuList.value = await ProductSpuApi.getSpuDetailList(props.property.spuIds)
131   },
132   {
133     immediate: true,
134     deep: true
135   }
136 )
137
138 /**
139  * 计算商品的间距
140  * @param index 商品索引
141  */
142 const calculateSpace = (index: number) => {
143   // 商品的列数
144   const columns = props.property.layoutType === 'twoCol' ? 2 : 1
145   // 第一列没有左边距
146   const marginLeft = index % columns === 0 ? '0' : props.property.space + 'px'
147   // 第一行没有上边距
148   const marginTop = index < columns ? '0' : props.property.space + 'px'
149
150   return { marginLeft, marginTop }
151 }
152
153 // 容器
154 const containerRef = ref()
155 // 计算商品的宽度
156 const calculateWidth = () => {
157   let width = '100%'
158   // 双列时每列的宽度为:(总宽度 - 间距)/ 2
159   if (props.property.layoutType === 'twoCol') {
160     width = `${(containerRef.value.offsetWidth - props.property.space) / 2}px`
161   }
162   return { width }
163 }
164 </script>
165
166 <style scoped lang="scss"></style>