1422 lines
37 KiB
Vue
1422 lines
37 KiB
Vue
<template>
|
||
<div class="pay-page">
|
||
<!-- 导航栏 -->
|
||
<nav class="navbar">
|
||
<div class="nav-left">
|
||
<el-button
|
||
type="text"
|
||
@click="handleGoBack"
|
||
class="back-btn"
|
||
>
|
||
<el-icon><ArrowLeft /></el-icon>
|
||
</el-button>
|
||
</div>
|
||
<div class="nav-center">
|
||
<h1 class="nav-title">确认订单</h1>
|
||
</div>
|
||
<div class="nav-right">
|
||
<!-- 占位元素,保持标题居中 -->
|
||
</div>
|
||
</nav>
|
||
|
||
<div v-loading="loading" class="page-content">
|
||
<!-- 收货地址 -->
|
||
<div class="address-section">
|
||
<div class="address-header">
|
||
<el-icon><Location /></el-icon>
|
||
<span class="address-label">收货地址</span>
|
||
<el-button
|
||
type="text"
|
||
@click="goToAddressManage"
|
||
class="manage-address-btn"
|
||
>
|
||
管理地址
|
||
</el-button>
|
||
</div>
|
||
<div class="address-content">
|
||
<el-select
|
||
v-model="selectedAddressId"
|
||
placeholder="请选择收货地址"
|
||
class="address-select"
|
||
@change="handleAddressChange"
|
||
>
|
||
<el-option
|
||
v-for="address in addresses"
|
||
:key="address.id"
|
||
:label="`${address.recipientName} ${address.recipientPhone} ${address.province}${address.city}${address.district}${address.detailAddress}`"
|
||
:value="address.id"
|
||
>
|
||
<div class="address-option">
|
||
<div class="address-info">
|
||
<span class="recipient-info">{{ address.recipientName }} {{ address.recipientPhone }}</span>
|
||
<el-tag v-if="address.isDefault" type="danger" size="small" class="default-tag">默认</el-tag>
|
||
</div>
|
||
<div class="address-detail">{{ address.province }}{{ address.city }}{{ address.district }}{{ address.detailAddress }}</div>
|
||
</div>
|
||
</el-option>
|
||
</el-select>
|
||
<div v-if="addresses.length === 0" class="no-address">
|
||
<span class="no-address-text">暂无收货地址</span>
|
||
<el-button type="text" @click="goToAddressManage" class="add-address-btn">
|
||
立即添加
|
||
</el-button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 支付金额 -->
|
||
<div class="amount-section">
|
||
<h3 class="section-title">支付</h3>
|
||
<div class="amount-display">
|
||
<div class="total-amount-large">
|
||
<!-- 根据选择的支付方式动态显示支付格式 -->
|
||
<template v-if="selectedPaymentMethod === 'mixed' && paymentData.pointsAmount > 0 && paymentData.beansAmount > 0">
|
||
<!-- 积分+融豆混合支付 -->
|
||
<div class="payment-price-container">
|
||
<div class="payment-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-rongdou-icon" />
|
||
<span class="payment-rongdou-price">{{ totalRongdouPrice }}</span>
|
||
</div>
|
||
<div class="payment-sub-price">
|
||
<el-icon class="payment-points-icon"><Coin /></el-icon>
|
||
<span class="payment-points-price">{{ totalPointsPrice }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else-if="selectedPaymentMethod === 'beans' && paymentData.beansAmount > 0">
|
||
<!-- 仅融豆支付 -->
|
||
<div class="payment-price-container">
|
||
<div class="payment-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-rongdou-icon" />
|
||
<span class="payment-rongdou-price">{{ totalRongdouPrice }}</span>
|
||
</div>
|
||
<div class="payment-sub-price">
|
||
<el-icon class="payment-points-icon"><Coin /></el-icon>
|
||
<span class="payment-points-price">{{ totalPointsPrice }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else-if="selectedPaymentMethod === 'points' && paymentData.pointsAmount > 0">
|
||
<!-- 仅积分支付 -->
|
||
<div class="payment-price-container">
|
||
<div class="payment-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-rongdou-icon" />
|
||
<span class="payment-rongdou-price">{{ totalRongdouPrice }}</span>
|
||
</div>
|
||
<div class="payment-sub-price">
|
||
<el-icon class="payment-points-icon"><Coin /></el-icon>
|
||
<span class="payment-points-price">{{ totalPointsPrice }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else>
|
||
<!-- 未选择支付方式时的默认显示 -->
|
||
<div class="payment-price-container">
|
||
<div class="payment-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-rongdou-icon" />
|
||
<span class="payment-rongdou-price">{{ totalRongdouPrice }}</span>
|
||
</div>
|
||
<div class="payment-sub-price">
|
||
<el-icon class="payment-points-icon"><Coin /></el-icon>
|
||
<span class="payment-points-price">{{ totalPointsPrice }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
</div>
|
||
<div class="amount-breakdown" v-if="selectedPaymentMethod && (paymentData.pointsAmount > 0 || paymentData.beansAmount > 0)">
|
||
<div v-if="paymentData.pointsAmount > 0" class="breakdown-item">
|
||
<el-icon><Coin /></el-icon>
|
||
<span>积分:{{ paymentData.pointsAmount }}</span>
|
||
</div>
|
||
<div v-if="paymentData.beansAmount > 0" class="breakdown-item">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="breakdown-icon" />
|
||
<span>融豆:{{ paymentData.beansAmount }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 商品列表 -->
|
||
<div class="items-section" v-if="paymentData.items && paymentData.items.length > 0">
|
||
<h3 class="section-title">商品清单 ({{ paymentData.items.length }})</h3>
|
||
<div class="items-list">
|
||
<div
|
||
v-for="item in paymentData.items"
|
||
:key="item.id || item.productId"
|
||
class="item-card"
|
||
>
|
||
<div class="item-image">
|
||
<img :src="item.image || '/imgs/productdetail/商品主图.png'" :alt="item.name" />
|
||
</div>
|
||
<div class="item-info">
|
||
<div class="item-name">{{ item.name }}</div>
|
||
<div class="item-details">
|
||
<span v-if="item.category" class="item-category">{{ item.category }}</span>
|
||
<span v-if="item.size" class="item-size">{{ item.size }}</span>
|
||
</div>
|
||
<div class="item-price">
|
||
<div class="item-price-container">
|
||
<div class="item-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="item-rongdou-icon" />
|
||
<span class="item-rongdou-price">{{ item.rongdouPrice }}</span>
|
||
</div>
|
||
<div class="item-sub-price">
|
||
<el-icon class="item-points-icon"><Coin /></el-icon>
|
||
<span class="item-points-price">{{ item.points }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="item-quantity">
|
||
<span class="quantity-label">x{{ item.quantity }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 支付方式选择 -->
|
||
<div class="payment-method-section">
|
||
<h3 class="section-title">支付方式</h3>
|
||
<div class="payment-options">
|
||
<!-- 融豆支付选项 -->
|
||
<div
|
||
v-if="shouldShowPaymentMethod('beans')"
|
||
class="payment-option"
|
||
:class="{
|
||
active: selectedPaymentMethod === 'beans',
|
||
disabled: !isPaymentMethodAvailable('beans')
|
||
}"
|
||
@click="selectPaymentMethod('beans')"
|
||
>
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-icon-img" />
|
||
<div class="payment-info">
|
||
<div class="payment-name">融豆支付</div>
|
||
<div class="payment-desc">
|
||
<span v-if="isPaymentMethodAvailable('beans')">使用账户融豆进行支付</span>
|
||
<span v-else class="insufficient-balance">余额不足(当前:{{ userBalance.beans }})</span>
|
||
</div>
|
||
</div>
|
||
<el-icon class="check-icon" v-if="selectedPaymentMethod === 'beans'"><Check /></el-icon>
|
||
</div>
|
||
|
||
<!-- 积分支付选项 -->
|
||
<div
|
||
v-if="shouldShowPaymentMethod('points')"
|
||
class="payment-option"
|
||
:class="{
|
||
active: selectedPaymentMethod === 'points',
|
||
disabled: !isPaymentMethodAvailable('points')
|
||
}"
|
||
@click="selectPaymentMethod('points')"
|
||
>
|
||
<el-icon class="payment-icon"><Coin /></el-icon>
|
||
<div class="payment-info">
|
||
<div class="payment-name">积分支付</div>
|
||
<div class="payment-desc">
|
||
<span v-if="isPaymentMethodAvailable('points')">使用账户积分进行支付</span>
|
||
<span v-else class="insufficient-balance">余额不足(当前:{{ userBalance.points }})</span>
|
||
</div>
|
||
</div>
|
||
<el-icon class="check-icon" v-if="selectedPaymentMethod === 'points'"><Check /></el-icon>
|
||
</div>
|
||
|
||
<!-- 积分+融豆混合支付选项 -->
|
||
<div
|
||
v-if="shouldShowPaymentMethod('mixed')"
|
||
class="payment-option"
|
||
:class="{
|
||
active: selectedPaymentMethod === 'mixed',
|
||
disabled: !isPaymentMethodAvailable('mixed')
|
||
}"
|
||
@click="selectPaymentMethod('mixed')"
|
||
>
|
||
<div class="payment-icon-group">
|
||
<el-icon class="payment-icon"><Coin /></el-icon>
|
||
<span class="plus-sign">+</span>
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="payment-icon-img" />
|
||
</div>
|
||
<div class="payment-info">
|
||
<div class="payment-name">积分+融豆</div>
|
||
<div class="payment-desc">
|
||
<span v-if="isPaymentMethodAvailable('mixed')">使用积分和融豆组合支付</span>
|
||
<span v-else class="insufficient-balance">余额不足(当前:融豆{{ userBalance.beans }} + 积分{{ userBalance.points }})</span>
|
||
</div>
|
||
</div>
|
||
<el-icon class="check-icon" v-if="selectedPaymentMethod === 'mixed'"><Check /></el-icon>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 底部支付按钮 -->
|
||
<div class="bottom-actions">
|
||
<div class="payment-summary">
|
||
<div class="total-amount">
|
||
<span>实付:</span>
|
||
<!-- 根据选择的支付方式动态显示实付格式 -->
|
||
<template v-if="selectedPaymentMethod === 'mixed' && paymentData.pointsAmount > 0 && paymentData.beansAmount > 0">
|
||
<!-- 积分+融豆混合支付 -->
|
||
<div class="bottom-price-container">
|
||
<div class="bottom-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="bottom-rongdou-icon" />
|
||
<span class="bottom-rongdou-price">{{ paymentData.beansAmount }}</span>
|
||
</div>
|
||
<span class="bottom-plus-sign">+</span>
|
||
<div class="bottom-sub-price">
|
||
<el-icon class="bottom-points-icon"><Coin /></el-icon>
|
||
<span class="bottom-points-price">{{ paymentData.pointsAmount }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else-if="selectedPaymentMethod === 'beans' && paymentData.beansAmount > 0">
|
||
<!-- 仅融豆支付 -->
|
||
<div class="bottom-price-container">
|
||
<div class="bottom-main-price">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="bottom-rongdou-icon" />
|
||
<span class="bottom-rongdou-price">{{ paymentData.beansAmount }}</span>
|
||
</div>
|
||
<span class="bottom-plus-sign" v-if="paymentData.pointsAmount > 0">+</span>
|
||
<div class="bottom-sub-price" v-if="paymentData.pointsAmount > 0">
|
||
<el-icon class="bottom-points-icon"><Coin /></el-icon>
|
||
<span class="bottom-points-price">{{ paymentData.pointsAmount || 0 }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else-if="selectedPaymentMethod === 'points' && paymentData.pointsAmount > 0">
|
||
<!-- 仅积分支付 -->
|
||
<div class="bottom-price-container">
|
||
<div class="bottom-main-price" v-if="paymentData.beansAmount > 0">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="bottom-rongdou-icon" />
|
||
<span class="bottom-rongdou-price">{{ paymentData.beansAmount || 0 }}</span>
|
||
</div>
|
||
<span class="bottom-plus-sign" v-if="paymentData.beansAmount > 0 && paymentData.pointsAmount > 0">+</span>
|
||
<div class="bottom-sub-price">
|
||
<el-icon class="bottom-points-icon"><Coin /></el-icon>
|
||
<span class="bottom-points-price">{{ paymentData.pointsAmount }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<template v-else>
|
||
<!-- 未选择支付方式时的默认显示 -->
|
||
<div class="bottom-price-container">
|
||
<div class="bottom-main-price" v-if="paymentData.beansAmount > 0">
|
||
<img src='/imgs/profile/rongdou.png' alt="融豆" class="bottom-rongdou-icon" />
|
||
<span class="bottom-rongdou-price">{{ paymentData.beansAmount || 0 }}</span>
|
||
</div>
|
||
<span class="bottom-plus-sign" v-if="paymentData.beansAmount > 0 && paymentData.pointsAmount > 0">+</span>
|
||
<div class="bottom-sub-price" v-if="paymentData.pointsAmount > 0">
|
||
<el-icon class="bottom-points-icon"><Coin /></el-icon>
|
||
<span class="bottom-points-price">{{ paymentData.pointsAmount || 0 }}</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
</div>
|
||
</div>
|
||
<el-button
|
||
type="primary"
|
||
size="large"
|
||
class="pay-button"
|
||
@click="confirmPayment"
|
||
:disabled="paying || !selectedPaymentMethod"
|
||
:loading="paying"
|
||
>
|
||
{{ paying ? '支付中...' : '确认支付' }}
|
||
</el-button>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed, onMounted, onUnmounted } from 'vue'
|
||
import { useRoute, useRouter } from 'vue-router'
|
||
import { ElMessage, ElMessageBox } from 'element-plus'
|
||
import {
|
||
ArrowLeft,
|
||
Location,
|
||
Coin,
|
||
Orange,
|
||
Check
|
||
} from '@element-plus/icons-vue'
|
||
import api from '@/utils/api'
|
||
import { getImageUrl } from '@/config'
|
||
|
||
const route = useRoute()
|
||
const router = useRouter()
|
||
|
||
// 响应式数据
|
||
const loading = ref(false)
|
||
const paying = ref(false)
|
||
|
||
const selectedPaymentMethod = ref('') // 当前选择的支付方式
|
||
const paymentData = ref({
|
||
totalAmount: 0,
|
||
pointsAmount: 0,
|
||
beansAmount: 0,
|
||
cartId: null,
|
||
items: [] // 添加商品列表
|
||
})
|
||
|
||
// 计算商品总价
|
||
const totalRongdouPrice = computed(() => {
|
||
return paymentData.value.items.reduce((sum, item) => {
|
||
const rongdouPrice = item.rongdouPrice || (item.rongdou_price || 0)
|
||
return sum + (rongdouPrice * item.quantity)
|
||
}, 0)
|
||
})
|
||
|
||
const totalPointsPrice = computed(() => {
|
||
return paymentData.value.items.reduce((sum, item) => {
|
||
const pointsPrice = item.points || (item.points_price || 0)
|
||
return sum + (pointsPrice * item.quantity)
|
||
}, 0)
|
||
})
|
||
// 用户余额数据
|
||
const userBalance = ref({
|
||
points: 0, // 用户积分
|
||
beans: 0 // 用户融豆
|
||
})
|
||
// 地址相关数据
|
||
const addresses = ref([])
|
||
const selectedAddressId = ref('')
|
||
const selectedAddress = ref(null)
|
||
|
||
// 方法
|
||
|
||
const selectPaymentMethod = async (method) => {
|
||
// 仅基于商品的points_price价值总和计算支付总额
|
||
const totalPointsPrice = paymentData.value.items.reduce((sum, item) => sum + (item.points_price * item.quantity), 0)
|
||
const EXCHANGE_RATE = 10000 // 1融豆 = 10000积分
|
||
|
||
// 检查支付方式是否可用
|
||
if (!isPaymentMethodAvailable(method)) {
|
||
let message = ''
|
||
if (method === 'beans') {
|
||
const requiredBeans = Math.ceil(totalPointsPrice / EXCHANGE_RATE)
|
||
message = `融豆余额不足,当前余额:${userBalance.value.beans},需要:${requiredBeans}`
|
||
} else if (method === 'points') {
|
||
message = `积分余额不足,当前余额:${userBalance.value.points},需要:${totalPointsPrice}`
|
||
} else if (method === 'mixed') {
|
||
message = `余额不足,无法完成支付`
|
||
}
|
||
ElMessage.warning(message)
|
||
return
|
||
}
|
||
|
||
selectedPaymentMethod.value = method
|
||
|
||
// 根据选择的支付方式更新显示的金额
|
||
if (method === 'beans') {
|
||
// 融豆支付:按积分/10000计算所需融豆
|
||
const requiredBeans = Math.ceil(totalPointsPrice / EXCHANGE_RATE)
|
||
paymentData.value.totalAmount = totalPointsPrice
|
||
paymentData.value.pointsAmount = 0
|
||
paymentData.value.beansAmount = requiredBeans
|
||
} else if (method === 'points') {
|
||
// 积分支付:直接使用积分
|
||
paymentData.value.totalAmount = totalPointsPrice
|
||
paymentData.value.pointsAmount = totalPointsPrice
|
||
paymentData.value.beansAmount = 0
|
||
} else if (method === 'mixed') {
|
||
// 积分+融豆支付:积分必须是10000的倍数,按(总积分-当前积分)/10000计算所需融豆
|
||
let availablePoints = Math.min(userBalance.value.points, totalPointsPrice)
|
||
// 确保积分是10000的倍数
|
||
availablePoints = Math.floor(availablePoints / 10000) * 10000
|
||
const remainingPoints = totalPointsPrice - availablePoints
|
||
const requiredBeans = Math.ceil(remainingPoints / EXCHANGE_RATE)
|
||
|
||
paymentData.value.totalAmount = totalPointsPrice
|
||
paymentData.value.pointsAmount = availablePoints
|
||
paymentData.value.beansAmount = requiredBeans
|
||
}
|
||
}
|
||
|
||
// 获取用户地址列表
|
||
const getAddressList = async () => {
|
||
try {
|
||
const response = await api.get('/addresses')
|
||
console.log('获取地址列表响应:', response)
|
||
if (response.data.success) {
|
||
// 根据接口文档转换数据格式,与Address.vue保持一致
|
||
const addressList = response.data.data || []
|
||
addresses.value = addressList.map(addr => ({
|
||
id: addr.id,
|
||
recipientName: addr.receiver_name,
|
||
recipientPhone: addr.receiver_phone,
|
||
province: addr.province_name,
|
||
city: addr.city_name,
|
||
district: addr.district_name,
|
||
detailAddress: addr.detailed_address,
|
||
isDefault: addr.is_default,
|
||
labelName: addr.label_name,
|
||
labelColor: addr.label_color
|
||
}))
|
||
|
||
// 如果有默认地址,自动选中
|
||
const defaultAddress = addresses.value.find(addr => addr.isDefault)
|
||
if (defaultAddress) {
|
||
selectedAddressId.value = defaultAddress.id
|
||
selectedAddress.value = defaultAddress
|
||
}
|
||
} else {
|
||
throw new Error(response.data.message || '获取地址列表失败')
|
||
}
|
||
} catch (error) {
|
||
console.error('获取地址列表失败:', error)
|
||
ElMessage.error(error.message || '获取地址列表失败')
|
||
}
|
||
}
|
||
|
||
// 处理地址选择变化
|
||
const handleAddressChange = (addressId) => {
|
||
selectedAddress.value = addresses.value.find(addr => addr.id === addressId)
|
||
}
|
||
|
||
// 跳转到地址管理页面
|
||
const goToAddressManage = () => {
|
||
router.push('/address')
|
||
}
|
||
|
||
// 获取用户余额信息
|
||
const fetchUserBalance = async () => {
|
||
try {
|
||
// 获取用户积分
|
||
const pointsResponse = await api.get('/user/points')
|
||
userBalance.value.points = pointsResponse.data?.currentPoints ?? pointsResponse.data?.points ?? 0
|
||
|
||
// 获取用户融豆(从用户资料接口获取,需要取反)
|
||
const profileResponse = await api.get('/user/profile')
|
||
if (profileResponse.data.success && profileResponse.data.user) {
|
||
userBalance.value.beans = -(parseFloat(profileResponse.data.user.balance) || 0)
|
||
}
|
||
} catch (error) {
|
||
console.error('获取用户余额失败:', error)
|
||
ElMessage.error('获取用户余额失败')
|
||
userBalance.value = { points: 0, beans: 0 }
|
||
}
|
||
}
|
||
|
||
// 检查支付方式是否可用
|
||
const isPaymentMethodAvailable = (method) => {
|
||
// 仅基于商品的points_price价值总和计算
|
||
const totalPointsPrice = paymentData.value.items.reduce((sum, item) => sum + (item.points_price * item.quantity), 0)
|
||
const EXCHANGE_RATE = 10000 // 1融豆 = 10000积分
|
||
|
||
switch (method) {
|
||
case 'beans':
|
||
// 融豆支付:检查总积分是否大于等于10000且融豆是否足够支付总积分价值
|
||
if (totalPointsPrice < 10000) {
|
||
return false
|
||
}
|
||
const requiredBeans = Math.ceil(totalPointsPrice / EXCHANGE_RATE)
|
||
return userBalance.value.beans >= requiredBeans
|
||
case 'points':
|
||
// 积分支付:检查积分是否足够
|
||
return userBalance.value.points >= totalPointsPrice
|
||
case 'mixed':
|
||
// 混合支付:检查积分+融豆换算后是否足够
|
||
const totalUserBalanceInPoints = userBalance.value.points + (userBalance.value.beans * EXCHANGE_RATE)
|
||
return totalUserBalanceInPoints >= totalPointsPrice
|
||
default:
|
||
return true
|
||
}
|
||
}
|
||
|
||
// 检查支付方式是否应该显示
|
||
const shouldShowPaymentMethod = (method) => {
|
||
// 仅基于商品的points_price价值总和计算
|
||
const totalPointsPrice = paymentData.value.items.reduce((sum, item) => sum + (item.points_price * item.quantity), 0)
|
||
const EXCHANGE_RATE = 10000 // 1融豆 = 10000积分
|
||
|
||
switch (method) {
|
||
case 'beans':
|
||
// 当总积分大于等于10000且用户融豆足够支付总积分价值时显示融豆支付选项
|
||
if (totalPointsPrice < 10000) {
|
||
return false
|
||
}
|
||
const requiredBeans = Math.ceil(totalPointsPrice / EXCHANGE_RATE)
|
||
return totalPointsPrice > 0 && userBalance.value.beans >= requiredBeans
|
||
case 'points':
|
||
// 当用户积分足够支付时显示积分支付选项
|
||
return totalPointsPrice > 0 && userBalance.value.points >= totalPointsPrice
|
||
case 'mixed':
|
||
// 当用户积分不够但积分+融豆换算足够时显示混合支付选项,且用户积分必须大于等于10000
|
||
const pointsNotEnough = totalPointsPrice > 0 && userBalance.value.points < totalPointsPrice
|
||
const hasEnoughPoints = userBalance.value.points >= 10000
|
||
return pointsNotEnough && hasEnoughPoints && isPaymentMethodAvailable('mixed')
|
||
default:
|
||
return false
|
||
}
|
||
}
|
||
|
||
const fetchPaymentData = async () => {
|
||
try {
|
||
loading.value = true
|
||
|
||
// 从路由参数获取订单ID
|
||
const orderId = route.params.orderId
|
||
if (!orderId) {
|
||
throw new Error('订单ID不存在')
|
||
}
|
||
|
||
// 先获取用户余额信息
|
||
await fetchUserBalance()
|
||
|
||
// 获取地址列表
|
||
await getAddressList()
|
||
|
||
// 从后端获取当前用户的待支付订单信息
|
||
const response = await api.get(`/orders/pending-payment/${orderId}`)
|
||
|
||
if (response.data.success) {
|
||
const data = response.data.data
|
||
const items = (data.items || []).map(item => ({
|
||
id: item.id,
|
||
productId: item.product_id,
|
||
name: item.product_name,
|
||
image: item.image_url,
|
||
points: item.points_price,
|
||
points_price: item.points_price, // 添加points_price字段
|
||
price: item.price,
|
||
rongdouPrice: item.rongdou_price,
|
||
quantity: item.quantity,
|
||
description: item.description,
|
||
specInfo: item.spec_info
|
||
}))
|
||
|
||
// 计算总积分价格和总融豆价格
|
||
const totalPointsPrice = items.reduce((sum, item) => sum + (item.points * item.quantity), 0)
|
||
const totalBeansPrice = items.reduce((sum, item) => sum + (item.rongdouPrice * item.quantity), 0)
|
||
|
||
paymentData.value = {
|
||
totalAmount: 0, // 初始不设置总金额,等用户选择支付方式后再设置
|
||
pointsAmount: totalPointsPrice, // 积分金额
|
||
beansAmount: totalBeansPrice, // 融豆金额
|
||
orderId: data.id || null, // 订单ID
|
||
orderNo: data.order_no || '', // 订单号
|
||
items: items
|
||
}
|
||
|
||
// 根据用户余额自动选择支付方式
|
||
if (shouldShowPaymentMethod('beans')) {
|
||
await selectPaymentMethod('beans')
|
||
} else if (shouldShowPaymentMethod('points')) {
|
||
await selectPaymentMethod('points')
|
||
} else if (shouldShowPaymentMethod('mixed')) {
|
||
await selectPaymentMethod('mixed')
|
||
}
|
||
// 如果没有可用的支付方式,不自动选择,让用户看到所有选项都不可用
|
||
} else {
|
||
throw new Error(response.data.message || '获取订单信息失败')
|
||
}
|
||
} catch (error) {
|
||
ElMessage.error(error.message || '获取订单信息失败')
|
||
} finally {
|
||
loading.value = false
|
||
}
|
||
}
|
||
|
||
const handleGoBack = async () => {
|
||
try {
|
||
await ElMessageBox.confirm(
|
||
'确认要取消订单吗?取消后可以在个人中心-我的订单里查看。',
|
||
'取消订单',
|
||
{
|
||
confirmButtonText: '确认取消',
|
||
cancelButtonText: '继续支付',
|
||
type: 'warning'
|
||
}
|
||
)
|
||
|
||
// 用户确认取消订单
|
||
router.go(-1)
|
||
} catch {
|
||
// 用户取消,什么都不做,留在当前页面
|
||
}
|
||
}
|
||
|
||
const confirmPayment = async () => {
|
||
if (!selectedPaymentMethod.value) {
|
||
ElMessage.error('请选择支付方式')
|
||
return
|
||
}
|
||
|
||
if (!selectedAddress.value) {
|
||
ElMessage.error('请选择收货地址')
|
||
return
|
||
}
|
||
|
||
try {
|
||
// 根据商品价格显示确认信息
|
||
let confirmMessage = ''
|
||
const totalPoints = paymentData.value.pointsAmount
|
||
const totalRongdou = paymentData.value.beansAmount
|
||
|
||
if (totalPoints > 0 && totalRongdou > 0) {
|
||
confirmMessage = `确认支付 ${totalPoints} 积分 + ${totalRongdou} 融豆?`
|
||
} else if (totalPoints === 0 && totalRongdou > 0) {
|
||
confirmMessage = `确认支付 ${totalRongdou} 融豆?`
|
||
} else if (totalRongdou === 0 && totalPoints > 0) {
|
||
confirmMessage = `确认支付 ${totalPoints} 积分?`
|
||
}
|
||
|
||
await ElMessageBox.confirm(
|
||
confirmMessage,
|
||
'确认支付',
|
||
{
|
||
confirmButtonText: '确定支付',
|
||
cancelButtonText: '取消',
|
||
type: 'warning'
|
||
}
|
||
)
|
||
|
||
paying.value = true
|
||
|
||
// 创建订单数据
|
||
const orderData = {
|
||
orderId: paymentData.value.orderId,
|
||
addressId: selectedAddress.value.id,
|
||
paymentMethod: selectedPaymentMethod.value,
|
||
pointsAmount: paymentData.value.pointsAmount,
|
||
beansAmount: paymentData.value.beansAmount
|
||
}
|
||
|
||
// 向后端发送订单支付请求
|
||
const response = await api.post('/orders/confirm-payment', orderData)
|
||
console.log(orderData)
|
||
|
||
if (response.data.success) {
|
||
console.log(orderData)
|
||
ElMessage.success('订单创建成功!')
|
||
|
||
// 跳转到PayLoading页面,传递订单ID
|
||
router.push({
|
||
path: '/payloading',
|
||
query: {
|
||
orderId: paymentData.value.orderId
|
||
}
|
||
})
|
||
} else {
|
||
console.log(orderData)
|
||
throw new Error(response.data.message || '创建订单失败')
|
||
}
|
||
} catch (error) {
|
||
console.log(orderData)
|
||
if (error !== 'cancel') {
|
||
ElMessage.error(error.message || '创建订单失败,请重试')
|
||
}
|
||
} finally {
|
||
paying.value = false
|
||
}
|
||
}
|
||
|
||
// 页面初始化
|
||
onMounted(() => {
|
||
fetchPaymentData()
|
||
})
|
||
|
||
|
||
</script>
|
||
|
||
<style scoped>
|
||
.pay-page {
|
||
min-height: 100vh;
|
||
background: #f5f5f5;
|
||
display: flex;
|
||
flex-direction: column;
|
||
}
|
||
|
||
.navbar {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
padding: 12px 16px;
|
||
background: white;
|
||
border-bottom: 1px solid #eee;
|
||
position: relative;
|
||
}
|
||
|
||
.nav-left,
|
||
.nav-right {
|
||
width: 48px;
|
||
display: flex;
|
||
justify-content: center;
|
||
}
|
||
|
||
.nav-center {
|
||
position: absolute;
|
||
left: 50%;
|
||
transform: translateX(-50%);
|
||
}
|
||
|
||
.nav-title {
|
||
font-size: 18px;
|
||
font-weight: 500;
|
||
margin: 0;
|
||
}
|
||
|
||
.back-btn {
|
||
color: #333;
|
||
padding: 0;
|
||
}
|
||
|
||
.page-content {
|
||
flex: 1;
|
||
padding: 0;
|
||
}
|
||
|
||
.countdown-section {
|
||
background: white;
|
||
padding: 24px 16px;
|
||
margin-bottom: 8px;
|
||
text-align: center;
|
||
}
|
||
|
||
.countdown-header {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
gap: 8px;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.clock-icon {
|
||
color: #ff4757;
|
||
font-size: 18px;
|
||
}
|
||
|
||
.countdown-label {
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
}
|
||
|
||
.countdown-display {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
gap: 8px;
|
||
margin-bottom: 12px;
|
||
}
|
||
|
||
.time-block {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
background: #ff4757;
|
||
color: white;
|
||
padding: 12px 16px;
|
||
border-radius: 8px;
|
||
min-width: 60px;
|
||
}
|
||
|
||
.time-number {
|
||
font-size: 24px;
|
||
font-weight: bold;
|
||
line-height: 1;
|
||
}
|
||
|
||
.time-label {
|
||
font-size: 12px;
|
||
margin-top: 4px;
|
||
}
|
||
|
||
.time-separator {
|
||
font-size: 24px;
|
||
font-weight: bold;
|
||
color: #ff4757;
|
||
}
|
||
|
||
.countdown-tip {
|
||
font-size: 14px;
|
||
color: #666;
|
||
}
|
||
|
||
.timeout-text {
|
||
color: #ff4757;
|
||
font-weight: 500;
|
||
}
|
||
|
||
.amount-section,
|
||
.items-section,
|
||
.payment-method-section {
|
||
background: white;
|
||
padding: 16px;
|
||
margin-bottom: 8px;
|
||
}
|
||
|
||
.section-title {
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
margin: 0 0 12px 0;
|
||
color: #333;
|
||
}
|
||
|
||
.items-list {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 12px;
|
||
}
|
||
|
||
.item-card {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 12px;
|
||
padding: 12px;
|
||
border: 1px solid #eee;
|
||
border-radius: 8px;
|
||
background: #fafafa;
|
||
}
|
||
|
||
.item-image {
|
||
width: 60px;
|
||
height: 60px;
|
||
border-radius: 6px;
|
||
overflow: hidden;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.item-image img {
|
||
width: 100%;
|
||
height: 100%;
|
||
object-fit: cover;
|
||
}
|
||
|
||
.item-info {
|
||
flex: 1;
|
||
min-width: 0;
|
||
}
|
||
|
||
.item-name {
|
||
font-size: 14px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
margin-bottom: 4px;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.item-details {
|
||
display: flex;
|
||
gap: 8px;
|
||
margin-bottom: 6px;
|
||
}
|
||
|
||
.item-category,
|
||
.item-size {
|
||
font-size: 12px;
|
||
color: #666;
|
||
background: #f0f0f0;
|
||
padding: 2px 6px;
|
||
border-radius: 4px;
|
||
}
|
||
|
||
.item-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
font-size: 14px;
|
||
color: #ffae00;
|
||
font-weight: 500;
|
||
}
|
||
|
||
.item-quantity {
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.quantity-label {
|
||
font-size: 14px;
|
||
color: #666;
|
||
font-weight: 500;
|
||
}
|
||
|
||
.amount-display {
|
||
text-align: center;
|
||
padding: 20px 0;
|
||
}
|
||
|
||
.total-amount-large {
|
||
display: flex;
|
||
align-items: baseline;
|
||
justify-content: center;
|
||
gap: 4px;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.currency-icon {
|
||
width: 24px;
|
||
height: 24px;
|
||
margin-right: 4px;
|
||
vertical-align: middle;
|
||
}
|
||
|
||
.currency-icon-el {
|
||
font-size: 24px;
|
||
color: #ffae00;
|
||
margin-right: 4px;
|
||
}
|
||
|
||
.currency-icon-group {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
margin-right: 4px;
|
||
}
|
||
|
||
.plus-sign-small {
|
||
font-size: 14px;
|
||
color: #666;
|
||
font-weight: bold;
|
||
margin: 0 2px;
|
||
}
|
||
|
||
.mixed-payment-item {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.mixed-amount {
|
||
font-size: 20px;
|
||
font-weight: bold;
|
||
color: #ff4757;
|
||
}
|
||
|
||
.mixed-payment-item-small {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.mixed-amount-small {
|
||
font-size: 14px;
|
||
font-weight: bold;
|
||
color: #ff4757;
|
||
}
|
||
|
||
.amount-number {
|
||
font-size: 36px;
|
||
font-weight: bold;
|
||
color: #ff4757;
|
||
}
|
||
|
||
.amount-breakdown {
|
||
display: flex;
|
||
justify-content: center;
|
||
gap: 16px;
|
||
flex-wrap: wrap;
|
||
}
|
||
|
||
.breakdown-item {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
font-size: 14px;
|
||
color: #666;
|
||
background: #f8f9fa;
|
||
padding: 6px 12px;
|
||
border-radius: 12px;
|
||
}
|
||
|
||
.breakdown-icon {
|
||
width: 16px;
|
||
height: 16px;
|
||
}
|
||
|
||
.payment-options {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 12px;
|
||
}
|
||
|
||
.payment-option {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 12px;
|
||
padding: 16px;
|
||
border: 1px solid #eee;
|
||
border-radius: 8px;
|
||
cursor: pointer;
|
||
transition: all 0.2s;
|
||
}
|
||
|
||
.payment-option:hover:not(.disabled) {
|
||
border-color: #ffae00;
|
||
background: #fff7e6;
|
||
}
|
||
|
||
.payment-option.active {
|
||
border-color: #ffae00;
|
||
background: #fff7e6;
|
||
}
|
||
|
||
.payment-option.disabled {
|
||
opacity: 0.5;
|
||
cursor: not-allowed;
|
||
background: #f5f5f5;
|
||
border-color: #ddd;
|
||
}
|
||
|
||
.payment-option.disabled:hover {
|
||
border-color: #ddd;
|
||
background: #f5f5f5;
|
||
}
|
||
|
||
.insufficient-balance {
|
||
color: #ff4757;
|
||
font-size: 12px;
|
||
}
|
||
|
||
/* 地址选择样式 */
|
||
.address-section {
|
||
background: white;
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
margin-bottom: 20px;
|
||
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
|
||
}
|
||
|
||
.address-header {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 8px;
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.address-label {
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
flex: 1;
|
||
}
|
||
|
||
.manage-address-btn {
|
||
color: #409eff;
|
||
font-size: 14px;
|
||
padding: 0;
|
||
}
|
||
|
||
.address-select {
|
||
width: 100%;
|
||
}
|
||
|
||
.address-option {
|
||
padding: 8px 0;
|
||
}
|
||
|
||
.address-info {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 12px;
|
||
margin-bottom: 4px;
|
||
}
|
||
|
||
.recipient-info {
|
||
font-weight: 500;
|
||
color: #333;
|
||
}
|
||
|
||
.default-tag {
|
||
background: #ff4757;
|
||
color: white;
|
||
font-size: 12px;
|
||
padding: 2px 6px;
|
||
border-radius: 4px;
|
||
}
|
||
|
||
.address-detail {
|
||
color: #666;
|
||
font-size: 14px;
|
||
line-height: 1.4;
|
||
}
|
||
|
||
.no-address {
|
||
text-align: center;
|
||
padding: 40px 20px;
|
||
color: #999;
|
||
}
|
||
|
||
.no-address-text {
|
||
display: block;
|
||
margin-bottom: 12px;
|
||
}
|
||
|
||
.add-address-btn {
|
||
color: #409eff;
|
||
font-size: 14px;
|
||
padding: 0;
|
||
}
|
||
|
||
/* 订单商品信息样式 */
|
||
.order-items-section {
|
||
background: white;
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
margin-bottom: 20px;
|
||
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
|
||
}
|
||
|
||
.section-header {
|
||
margin-bottom: 16px;
|
||
}
|
||
|
||
.section-header .title {
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
}
|
||
|
||
.order-items-section .items-list {
|
||
margin-top: 16px;
|
||
}
|
||
|
||
|
||
|
||
.payment-icon {
|
||
color: #ffae00;
|
||
font-size: 20px;
|
||
}
|
||
|
||
.payment-icon-img {
|
||
width: 24px;
|
||
height: 24px;
|
||
margin-right: 12px;
|
||
}
|
||
|
||
.payment-icon-group {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
}
|
||
|
||
.plus-sign {
|
||
font-size: 14px;
|
||
color: #666;
|
||
font-weight: bold;
|
||
}
|
||
|
||
.payment-info {
|
||
flex: 1;
|
||
}
|
||
|
||
.payment-name {
|
||
font-size: 14px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
margin-bottom: 4px;
|
||
}
|
||
|
||
.payment-desc {
|
||
font-size: 12px;
|
||
color: #666;
|
||
}
|
||
|
||
.check-icon {
|
||
color: #ffae00;
|
||
font-size: 18px;
|
||
}
|
||
|
||
.bottom-actions {
|
||
padding: 16px;
|
||
background: white;
|
||
border-top: 1px solid #eee;
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 16px;
|
||
}
|
||
|
||
.payment-info {
|
||
flex: 1;
|
||
}
|
||
|
||
.payment-summary {
|
||
flex: 1;
|
||
}
|
||
|
||
.total-amount {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
font-size: 16px;
|
||
}
|
||
|
||
.amount {
|
||
color: #ff4757;
|
||
font-size: 18px;
|
||
font-weight: bold;
|
||
}
|
||
|
||
.amount-icon {
|
||
width: 18px;
|
||
height: 18px;
|
||
margin-right: 4px;
|
||
vertical-align: middle;
|
||
}
|
||
|
||
.amount-icon-el {
|
||
font-size: 18px;
|
||
color: #ffae00;
|
||
margin-right: 4px;
|
||
}
|
||
|
||
.amount-icon-group {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
margin-right: 4px;
|
||
}
|
||
|
||
.pay-button {
|
||
min-width: 120px;
|
||
height: 48px;
|
||
background: #ffae00;
|
||
border: none;
|
||
border-radius: 24px;
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
}
|
||
|
||
.pay-button:hover {
|
||
background: #e69900;
|
||
}
|
||
|
||
.pay-button:disabled {
|
||
background: #ccc;
|
||
cursor: not-allowed;
|
||
}
|
||
|
||
/* 商品单价双价格显示样式 */
|
||
.item-price-container {
|
||
flex-direction: column;
|
||
align-items: flex-end;
|
||
gap: 2px;
|
||
}
|
||
|
||
.item-main-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.item-rongdou-icon {
|
||
width: 10px;
|
||
height: 10px;
|
||
}
|
||
|
||
.item-rongdou-price {
|
||
font-size: 16px;
|
||
font-weight: 500;
|
||
color: #333;
|
||
}
|
||
|
||
.item-sub-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.item-points-icon {
|
||
font-size: 12px;
|
||
color: #ffae00;
|
||
}
|
||
|
||
.item-points-price {
|
||
font-size: 12px;
|
||
color: #666;
|
||
}
|
||
|
||
/* 支付金额双价格显示样式 */
|
||
.payment-price-container {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
gap: 4px;
|
||
}
|
||
|
||
.payment-main-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
}
|
||
|
||
.payment-rongdou-icon {
|
||
width: 24px;
|
||
height: 24px;
|
||
}
|
||
|
||
.payment-rongdou-price {
|
||
font-size: 24px;
|
||
font-weight: bold;
|
||
color: #333;
|
||
}
|
||
|
||
.payment-sub-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.payment-points-icon {
|
||
font-size: 16px;
|
||
color: #ffae00;
|
||
}
|
||
|
||
.payment-points-price {
|
||
font-size: 16px;
|
||
color: #666;
|
||
}
|
||
|
||
/* 底部实付双价格显示样式 */
|
||
.bottom-price-container {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 8px;
|
||
}
|
||
|
||
.bottom-main-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
}
|
||
|
||
.bottom-rongdou-icon {
|
||
width: 18px;
|
||
height: 18px;
|
||
}
|
||
|
||
.bottom-rongdou-price {
|
||
font-size: 18px;
|
||
font-weight: bold;
|
||
color: #ff4757;
|
||
}
|
||
|
||
.bottom-plus-sign {
|
||
font-size: 16px;
|
||
font-weight: bold;
|
||
color: #666;
|
||
margin: 0 2px;
|
||
}
|
||
|
||
.bottom-sub-price {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
margin-left: 0px;
|
||
padding-left: 0px;
|
||
}
|
||
|
||
.bottom-points-icon {
|
||
font-size: 14px;
|
||
color: #ffae00;
|
||
}
|
||
|
||
.bottom-points-price {
|
||
font-size: 14px;
|
||
color: #666;
|
||
}
|
||
|
||
.bottom-points-price {
|
||
font-size: 14px;
|
||
color: #666;
|
||
}
|
||
</style> |