| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496 | <template>	<view>		<s-layout :onShareAppMessage="shareInfo" navbar="goods">			<!-- 标题栏 -->			<detailNavbar v-model="state.goodsInfo" />			<!-- 骨架屏 -->			<detailSkeleton v-if="state.skeletonLoading" />			<!-- 下架/售罄提醒 -->			<s-empty v-else-if="state.goodsInfo === null" text="商品不存在或已下架" icon="/static/soldout-empty.png" showAction				actionText="再逛逛" actionUrl="/pages/goods/list" />			<block v-else>				<view class="detail-swiper-selector">					<!-- 商品轮播图  -->					<su-swiper class="ss-m-b-14" isPreview :list="formatGoodsSwiper(state.goodsInfo.sliderPicUrls)"						otStyle="tag" imageMode="widthFix" dotCur="bg-mask-40" :seizeHeight="750" />					<!-- 价格+标题 -->					<view class="title-card detail-card ss-p-y-40 ss-p-x-20">						<view class="ss-flex ss-row-between ss-col-center ss-m-b-26">							<view class="price-box ss-flex ss-col-center">								<view class="price-text ss-m-r-16 ss-flex ss-col-center">									<image src="@/static/icon/points.png"  v-if="state.goodsInfo.spuPayType == 2" class="ss-m-r-10" style="width:40rpx;height:40rpx"   ></image>									<text v-else>¥</text>									<text v-if="state.goodsInfo.highPrecision">										{{ (state.selectedSku.highPrecisionPrice || state.goodsInfo.highPrecisionPrice)/100 }}									</text>									<text v-else>										{{ fen2yuan(state.selectedSku.price || state.goodsInfo.price) }}									</text>								</view>								<view class="origin-price-text" v-if="state.goodsInfo.promotionFee >= 0">									积分:{{ fen2yuan(state.selectedSku.promotionFee || state.goodsInfo.promotionFee)  }}								</view>							</view>							<view class="sales-text">								<!-- {{ formatSales('exact', state.goodsInfo.salesCount) }} -->							</view>						</view>						<view class="discounts-box ss-flex ss-row-between ss-m-b-28">							<!-- 满减送/限时折扣活动的提示 -->							<div class="tag-content">								<view class="tag-box ss-flex">									<view class="tag ss-m-r-10" v-for="promos in state.activityInfo" :key="promos.id"										@tap="onActivity">										{{ promos.name }}									</view>								</view>							</div>							<!-- 优惠劵 -->							<view class="get-coupon-box ss-flex ss-col-center ss-m-l-20" @tap="state.showModel = true"								v-if="state.couponInfo.length">								<view class="discounts-title ss-m-r-8">领券</view>								<text class="cicon-forward"></text>							</view>						</view>						<view class="title-text ss-line-2 ss-m-b-6">{{ state.goodsInfo.name }}</view>						<view class="subtitle-text ss-line-1">{{ state.goodsInfo.introduction }}</view>					</view>					<!-- 功能卡片 -->					<view class="detail-cell-card detail-card ss-flex-col">						<detail-cell-sku v-model="state.selectedSku.goods_sku_text" :sku="state.selectedSku"							@tap="state.showSelectSku = true" />					</view>					<!-- 规格与数量弹框 -->					<s-select-sku :goodsInfo="state.goodsInfo" :show="state.showSelectSku" @addCart="onAddCart"						@buy="onBuy" @change="onSkuChange" @close="state.showSelectSku = false" />				</view>				<!-- 评价 -->				<detail-comment-card class="detail-comment-selector" :goodsId="state.goodsId" />				<!-- 详情 -->				<detail-content-card class="detail-content-selector" :content="state.goodsInfo.description" />				<!-- 活动跳转:拼团/秒杀/砍价活动 -->				<!-- <detail-activity-tip v-if="state.activityList.length > 0" :activity-list="state.activityList" /> -->				<!-- 详情 tabbar -->												<detail-tabbar v-model="state.goodsInfo">					<!-- {{state.goodsInfo.spuType }}  {{state.goodsInfo.spuPayType }} -->					<!-- 如果有库存并且不是积分商品和虚拟商品 可以加入购物车和购买 -->					<view class="buy-box ss-flex ss-col-center ss-p-r-20" v-if="state.goodsInfo.stock > 0 && state.goodsInfo.spuType && state.goodsInfo.spuPayType == 1">						<button class="ss-reset-button add-btn ui-Shadow-Main"  @tap="state.showSelectSku = true">							加入购物车						</button>						<button class="ss-reset-button buy-btn ui-Shadow-Main" @tap="state.showSelectSku = true">							立即购买						</button>					</view>					<view class="buy-box ss-flex ss-col-center ss-p-r-20"  v-else-if="state.goodsInfo.stock > 0 &&  state.goodsInfo.spuPayType != 1 || !state.goodsInfo.spuType">						<button class="ss-reset-button disabled-btn buy-btn ui-Shadow-Main " @tap="state.showSelectSku = true">							<template v-if="state.goodsInfo.spuPayType != 1 && !state.goodsInfo.spuType" >								立即兑换							</template>								 <template v-else-if="!state.goodsInfo.spuType">								 立即购买							 </template>								 <template v-else>								 立即兑换							 </template>						</button>					</view>					<view class="buy-box ss-flex ss-col-center ss-p-r-20" v-else>						<button class="ss-reset-button disabled-btn" disabled style="background: #999999;"> 已售罄 </button>					</view>				</detail-tabbar>				<!-- 优惠劵弹窗 -->				<!-- <s-coupon-get v-model="state.couponInfo" :show="state.showModel" @close="state.showModel = false"					@get="onGet" /> -->				<!-- 满减送/限时折扣活动弹窗 -->				<!-- <s-activity-pop v-model="state.activityInfo" :show="state.showActivityModel"					@close="state.showActivityModel = false" /> -->			</block>		</s-layout>	</view></template><script setup>	import {		reactive,		computed,		provide,		watchEffect,		watch	} from 'vue';	import {		onLoad,		onPageScroll	} from '@dcloudio/uni-app';		import sheep from '@/sheep';	import CouponApi from '@/sheep/api/promotion/coupon';	import ActivityApi from '@/sheep/api/promotion/activity';	import FavoriteApi from '@/sheep/api/product/favorite';	import {		formatSales,		formatGoodsSwiper,		fen2yuan	} from '@/sheep/hooks/useGoods';	import detailNavbar from './components/detail/detail-navbar.vue';	import detailCellSku from './components/detail/detail-cell-sku.vue';	import detailTabbar from './components/detail/detail-tabbar.vue';	import detailSkeleton from './components/detail/detail-skeleton.vue';	import detailCommentCard from './components/detail/detail-comment-card.vue';	import detailContentCard from './components/detail/detail-content-card.vue';	import detailActivityTip from './components/detail/detail-activity-tip.vue';	import {		isEmpty	} from 'lodash';	import SpuApi from '@/sheep/api/product/spu';	import ShareApi from '@/sheep/api/distri/share';	import { showAuthModal } from '@/sheep/hooks/useModal';	onPageScroll(() => {});	const state = reactive({		goodsId: 0,		skeletonLoading: true, // SPU 加载中		goodsInfo: {}, // SPU 信息		showSelectSku: false, // 是否展示 SKU 选择弹窗		selectedSku: {}, // 选中的 SKU		showModel: false, // 是否展示 Coupon 优惠劵的弹窗		couponInfo: [], // 可领取的 Coupon 优惠劵的列表		showActivityModel: false, // 【满减送/限时折扣】是否展示 Activity 营销活动的弹窗		activityInfo: [], // 【满减送/限时折扣】可参与的 Activity 营销活动的列表		activityList: [], // 【秒杀/拼团/砍价】可参与的 Activity 营销活动的列表		linkId:0, // 分享时的id	});		// 规格变更	function onSkuChange(e) {		state.selectedSku = e;	}	// 添加购物车	function onAddCart(e) {		if (!e.id) {			sheep.$helper.toast('请选择商品规格');			return;		}		if(!isLogin.value){			showAuthModal();			return;		}		sheep.$store('cart').add(e);	}	// 立即购买	function onBuy(e) {		if (!state.selectedSku.id) {			sheep.$helper.toast('请选择商品规格');			return;		}		sheep.$router.go('/pages/order/confirm', {			data: JSON.stringify({				items: [{					skuId: e.id,					count: e.goods_num				}],				spuType:state.goodsInfo.spuType,				spuPayType:state.goodsInfo.spuPayType,				// TODO 芋艿:后续清理掉这 2 参数				deliveryType: 1,				pointStatus: false,			}),		});	}	// 营销活动	function onActivity() {		state.showActivityModel = true;	}	// 立即领取	async function onGet(id) {		const {			code		} = await CouponApi.takeCoupon(id);		if (code !== 0) {			return;		}		uni.showToast({			title: '领取成功',		});		setTimeout(() => {			getCoupon();		}, 1000);	}		//  TODO 芋艿:待测试	const shareInfo = computed(() => {		if (isEmpty(state.goodsInfo)) return {};		return sheep.$platform.share.getShareInfo({			title: state.goodsInfo.name,			image: sheep.$url.cdn(state.goodsInfo.image),			desc: state.goodsInfo.subtitle,			params: {				page: '2',				// query: state.linkId,			},		}, {			type: 'goods', // 商品海报			title: state.goodsInfo.name, // 商品标题			introduction:state.goodsInfo.introduction,			// image: sheep.$url.cdn(state.goodsInfo.image), // 商品主图			image: sheep.$url.cdn(state.goodsInfo.picUrl), // 商品主图			price: fen2yuan(state.goodsInfo.price), // 商品价格			original_price: fen2yuan(state.goodsInfo.maretPrice), // 商品原价		}, );	});	async function getCoupon() {		const {			code,			data		} = await CouponApi.getCouponTemplateList(state.goodsId, 2, 10);		if (code === 0) {			state.couponInfo = data;		}	}	async function getSpuDetail(id){		SpuApi.getSpuDetail(id).then((res) => {			// 未找到商品			if (res.code !== 0 || !res.data) {				state.goodsInfo = null;				return;			}			// 加载到商品			state.skeletonLoading = false;			state.goodsInfo = res.data;			console.log(state.goodsInfo )			if(!isLogin.value){				return;			}			// 加载是否收藏			FavoriteApi.isFavoriteExists(state.goodsId, 'goods').then((res) => {				if (res.code !== 0) {					return;				}				state.goodsInfo.favorite = res.data;			});		});	}	async function getLink(){		//加载分享时用到的linkId		ShareApi.getLinkId(3,state.goodsId).then((res) => {			if (res.code !== 0) {				return;			}			state.linkId = res.data.linkId			// console.log(state.linkId)		});	}	const isLogin = computed(() => sheep.$store('user').isLogin)	watch(() => isLogin.value, (newVal) => {		if (newVal) {			getSpuDetail(state.goodsId)		}	})	onLoad((options) => {		// console.log("options.id",options.id,"options.linkId",options.linkId)		// 没有spuId和linkId的话 就有问题		if (!options.id && !options.linkId) {			state.goodsInfo = null;			return;		}		state.goodsId = options.id || 0;		// 如果只是没有spuId有linkId的话 要去linkId对应的spuId				if(!options.id && options.linkId){			console.log("如果只是没有spuId有linkId的话 要去linkId对应的spuId")			const linkId =  options.linkId.toString()			ShareApi.getObjectIdByLinkId(linkId).then((res) => {				if (res.code !== 0) {					return;				}				state.goodsId = res.data				getSpuDetail(state.goodsId)				getLink()			});			uni.setStorageSync("linkId",options.linkId)		}else{			// 1. 加载商品信息			getSpuDetail(state.goodsId)		}		// 2. 加载优惠劵信息		// getCoupon();		// 3. 加载营销活动信息		ActivityApi.getActivityListBySpuId(state.goodsId).then((res) => {			if (res.code !== 0) {				return;			}			res.data.forEach(activity => {				if ([1, 2, 3].includes(activity.type)) { // 情况一:拼团/秒杀/砍价					state.activityList.push(activity);				} else if (activity.type === 5) { // 情况二:满减送					state.activityInfo.push(activity);				} else { // 情况三:限时折扣 TODO 芋艿					console.log('待实现!优先级不高');				}			})		});	});</script><style lang="scss" scoped>	.detail-card {		background-color: #ffff;		margin: 14rpx 20rpx;		border-radius: 10rpx;		overflow: hidden;	}	// 价格标题卡片	.title-card {		.price-box {			.price-text {				font-size: 42rpx;				font-weight: 500;				color: #ff3000;				line-height: 30rpx;				font-family: OPPOSANS;				// &::before {				// 	content: '¥';				// 	font-size: 30rpx;				// }			}			.origin-price-text {				font-size: 26rpx;				font-weight: 400;				color: $gray-c;				font-family: OPPOSANS;				background: #ffca3e;				padding: 2px 8px;				border-radius: 4px;				color: #597533;			}		}		.sales-text {			font-size: 26rpx;			font-weight: 500;			color: $gray-c;		}		.discounts-box {			.tag-content {				flex: 1;				min-width: 0;				white-space: nowrap;			}			.tag-box {				overflow: hidden;				text-overflow: ellipsis;			}			.tag {				flex-shrink: 0;				padding: 4rpx 10rpx;				font-size: 24rpx;				font-weight: 500;				border-radius: 4rpx;				color: var(--ui-BG-Main);				background: var(--ui-BG-Main-tag);			}			.discounts-title {				font-size: 24rpx;				font-weight: 500;				color: var(--ui-BG-Main);				line-height: normal;			}			.cicon-forward {				color: var(--ui-BG-Main);				font-size: 24rpx;				line-height: normal;				margin-top: 4rpx;			}		}		.title-text {			font-size: 30rpx;			font-weight: bold;			line-height: 42rpx;		}		.subtitle-text {			font-size: 26rpx;			font-weight: 400;			color: $dark-9;			line-height: 42rpx;		}	}	// 购买	.buy-box {		.add-btn {			width: 180rpx;			height: 72rpx;			font-weight: 500;			font-size: 28rpx;			border-radius: 40rpx 0 0 40rpx;			background-color: var(--ui-BG-Main-light);			color: var(--ui-BG-Main);		}		.buy-btn {			width: 180rpx;			height: 72rpx;			font-weight: 500;			font-size: 28rpx;			border-radius: 0 40rpx 40rpx 0;			background: linear-gradient(90deg, var(--ui-BG-Main), var(--ui-BG-Main-gradient));			color: $white;		}		.disabled-btn {			width: 428rpx;			height: 72rpx;			border-radius: 40rpx;						color: $white;		}	}	.model-box {		height: 60vh;		.model-content {			height: 56vh;		}		.title {			font-size: 36rpx;			font-weight: bold;			color: #333333;		}		.subtitle {			font-size: 26rpx;			font-weight: 500;			color: #333333;		}	}</style>
 |