551 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
			
		
		
	
	
			551 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
<template>
 | 
						|
  <view class="success-page">
 | 
						|
    <!-- 状态栏占位 -->
 | 
						|
    <view class="status-bar"></view>
 | 
						|
 | 
						|
    <!-- 成功状态区域 -->
 | 
						|
    <view v-show="payStatus === 1" class="success-section">
 | 
						|
      <view class="success-icon-wrapper">
 | 
						|
        <view class="success-icon">
 | 
						|
          <text class="success-checkmark">✓</text>
 | 
						|
        </view>
 | 
						|
        <view class="success-rings">
 | 
						|
          <view class="ring ring1"></view>
 | 
						|
          <view class="ring ring2"></view>
 | 
						|
          <view class="ring ring3"></view>
 | 
						|
        </view>
 | 
						|
      </view>
 | 
						|
 | 
						|
      <view class="success-title">{{ paySuccessText }}</view>
 | 
						|
      <view v-if="!isRecharge && payStatus === 1" class="success-subtitle"
 | 
						|
        >您的订单已支付完成</view
 | 
						|
      >
 | 
						|
    </view>
 | 
						|
    <!-- 查询中状态区域 -->
 | 
						|
    <view v-show="payStatus === 0" class="loading-section">
 | 
						|
      <view class="loading-icon-wrapper">
 | 
						|
        <view class="loading-icon">
 | 
						|
          <view class="loading-spinner">
 | 
						|
            <view class="spinner-dot dot1"></view>
 | 
						|
            <view class="spinner-dot dot2"></view>
 | 
						|
            <view class="spinner-dot dot3"></view>
 | 
						|
          </view>
 | 
						|
        </view>
 | 
						|
        <view class="loading-rings">
 | 
						|
          <view class="loading-ring ring1"></view>
 | 
						|
          <view class="loading-ring ring2"></view>
 | 
						|
        </view>
 | 
						|
      </view>
 | 
						|
 | 
						|
      <view class="loading-title">支付查询中...</view>
 | 
						|
      <view class="loading-subtitle">请稍等,正在确认您的支付状态</view>
 | 
						|
    </view>
 | 
						|
    <!-- 操作按钮区域 -->
 | 
						|
    <view class="action-section">
 | 
						|
      <view class="button-group">
 | 
						|
        <button
 | 
						|
          v-if="!isRecharge && !isSharePay"
 | 
						|
          class="btn btn-secondary"
 | 
						|
          @click="goToOrderList"
 | 
						|
        >
 | 
						|
          查看订单
 | 
						|
        </button>
 | 
						|
        <button v-if="!isSharePay" class="btn btn-primary" @click="goToHome">
 | 
						|
          返回首页
 | 
						|
        </button>
 | 
						|
        <button v-else class="btn btn-primary" @click="goToLogin">
 | 
						|
          去登录
 | 
						|
        </button>
 | 
						|
      </view>
 | 
						|
    </view>
 | 
						|
 | 
						|
    <successDialog @successClose="successClose" ref="successDialog" />
 | 
						|
  </view>
 | 
						|
</template>
 | 
						|
 | 
						|
<script>
 | 
						|
import { mapGetters } from 'vuex'
 | 
						|
import { payStatus, registerInfo, fansOrder } from '@/config/pay.js'
 | 
						|
import successDialog from '@/components/successDialog.vue'
 | 
						|
import { removeToken } from '@/config/auth.js'
 | 
						|
let paySetTimeoutFlag = null
 | 
						|
let getRegisterInfoTimeoutFlag = null
 | 
						|
export default {
 | 
						|
  name: 'PaySuccess',
 | 
						|
  components: {
 | 
						|
    successDialog,
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      orderCode: '',
 | 
						|
      specialArea: null,
 | 
						|
      isRecharge: undefined,
 | 
						|
      payStatus: 0,
 | 
						|
    }
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    ...mapGetters(['userInfo']),
 | 
						|
    paySuccessText() {
 | 
						|
      if (this.isRecharge === undefined) {
 | 
						|
        return ''
 | 
						|
      }
 | 
						|
      if (this.isRecharge === true) {
 | 
						|
        return '充值成功'
 | 
						|
      }
 | 
						|
      return '支付成功'
 | 
						|
    },
 | 
						|
  },
 | 
						|
  onLoad(options) {
 | 
						|
    // 获取传递的参数
 | 
						|
    const extParam = JSON.parse(atob(options.extParam || '{}'))
 | 
						|
    console.log(extParam, '..extParam')
 | 
						|
    this.isSharePay = extParam.isSharePay
 | 
						|
    this.specialArea = extParam.specialArea
 | 
						|
    this.isRecharge = extParam.isRecharge
 | 
						|
    this.orderCode = extParam.orderCode || ''
 | 
						|
    if (this.isSharePay) {
 | 
						|
      uni.showLoading({
 | 
						|
        title: '正在获取注册信息,请不要关闭当前页面',
 | 
						|
      })
 | 
						|
      setTimeout(() => {
 | 
						|
        this.getShareRegistInfo()
 | 
						|
      }, 500)
 | 
						|
      return
 | 
						|
    }
 | 
						|
    if (this.orderCode) {
 | 
						|
      setTimeout(() => {
 | 
						|
        this.pollingPayStatus(this.orderCode)
 | 
						|
      }, 500)
 | 
						|
    }
 | 
						|
    // 设置导航栏
 | 
						|
    uni.setNavigationBarTitle({
 | 
						|
      title: '支付成功',
 | 
						|
    })
 | 
						|
  },
 | 
						|
  onUnload() {
 | 
						|
    clearTimeout(paySetTimeoutFlag)
 | 
						|
    clearTimeout(getRegisterInfoTimeoutFlag)
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    getRegisterInfo() {
 | 
						|
      registerInfo(this.orderCode).then(res => {
 | 
						|
        if (res.data) {
 | 
						|
          uni.hideLoading()
 | 
						|
          this.$refs.successDialog.showSuccess(res.data)
 | 
						|
        } else {
 | 
						|
          getRegisterInfoTimeoutFlag = setTimeout(() => {
 | 
						|
            this.getRegisterInfo()
 | 
						|
          }, 3000)
 | 
						|
        }
 | 
						|
      })
 | 
						|
    },
 | 
						|
    pollingPayStatus(orderCode) {
 | 
						|
      payStatus({ businessCode: orderCode }).then(res => {
 | 
						|
        if (res.data == 1) {
 | 
						|
          this.payStatus = 1
 | 
						|
          clearTimeout(paySetTimeoutFlag)
 | 
						|
          if ([1, 7, 24].includes(Number(this.specialArea))) {
 | 
						|
            uni.showLoading({
 | 
						|
              title: '注册信息加载中...',
 | 
						|
              mask: false,
 | 
						|
            })
 | 
						|
            setTimeout(() => {
 | 
						|
              this.getRegisterInfo()
 | 
						|
            }, 2000)
 | 
						|
          }
 | 
						|
        } else {
 | 
						|
          paySetTimeoutFlag = setTimeout(() => {
 | 
						|
            this.pollingPayStatus(orderCode)
 | 
						|
          }, 1000)
 | 
						|
        }
 | 
						|
      })
 | 
						|
    },
 | 
						|
 | 
						|
    // 跳转到订单列表
 | 
						|
    goToOrderList() {
 | 
						|
      uni.redirectTo({
 | 
						|
        url: '/pages/mine/order/index',
 | 
						|
      })
 | 
						|
    },
 | 
						|
    successClose() {
 | 
						|
      uni.redirectTo({
 | 
						|
        url: this.isSharePay ? '/pages/login/index' : '/pages/mine/order/index',
 | 
						|
      })
 | 
						|
    },
 | 
						|
 | 
						|
    // 返回首页
 | 
						|
    goToHome() {
 | 
						|
      uni.switchTab({
 | 
						|
        url: '/pages/index/index',
 | 
						|
      })
 | 
						|
    },
 | 
						|
    goToLogin() {
 | 
						|
      uni.redirectTo({
 | 
						|
        url: '/pages/login/index',
 | 
						|
      })
 | 
						|
    },
 | 
						|
    getShareRegistInfo() {
 | 
						|
      fansOrder(this.orderCode).then(res => {
 | 
						|
        if (res.data) {
 | 
						|
          removeToken()
 | 
						|
          uni.hideLoading()
 | 
						|
          this.$refs.successDialog.showSuccess(res.data)
 | 
						|
        } else {
 | 
						|
          getRegisterInfoTimeoutFlag = setTimeout(() => {
 | 
						|
            this.getShareRegistInfo()
 | 
						|
          }, 3000)
 | 
						|
        }
 | 
						|
      })
 | 
						|
    },
 | 
						|
  },
 | 
						|
}
 | 
						|
</script>
 | 
						|
 | 
						|
<style lang="scss" scoped>
 | 
						|
.success-page {
 | 
						|
  min-height: 100vh;
 | 
						|
  background: linear-gradient(135deg, #f8fafc 0%, #e3f2fd 100%);
 | 
						|
  position: relative;
 | 
						|
  display: flex;
 | 
						|
  flex-direction: column;
 | 
						|
}
 | 
						|
 | 
						|
.status-bar {
 | 
						|
  height: var(--status-bar-height);
 | 
						|
  background: transparent;
 | 
						|
}
 | 
						|
 | 
						|
.success-section {
 | 
						|
  flex: 1;
 | 
						|
  display: flex;
 | 
						|
  flex-direction: column;
 | 
						|
  align-items: center;
 | 
						|
  justify-content: center;
 | 
						|
  padding: 60rpx 40rpx 40rpx;
 | 
						|
  position: relative;
 | 
						|
}
 | 
						|
 | 
						|
.success-icon-wrapper {
 | 
						|
  position: relative;
 | 
						|
  margin-bottom: 40rpx;
 | 
						|
}
 | 
						|
 | 
						|
.success-icon {
 | 
						|
  width: 120rpx;
 | 
						|
  height: 120rpx;
 | 
						|
  background: #005bac;
 | 
						|
  border-radius: 50%;
 | 
						|
  display: flex;
 | 
						|
  align-items: center;
 | 
						|
  justify-content: center;
 | 
						|
  position: relative;
 | 
						|
  z-index: 2;
 | 
						|
  box-shadow: 0 8rpx 24rpx rgba(0, 91, 172, 0.3);
 | 
						|
  animation: successBounce 0.6s ease-out;
 | 
						|
}
 | 
						|
 | 
						|
.success-checkmark {
 | 
						|
  color: #ffffff;
 | 
						|
  font-size: 60rpx;
 | 
						|
  font-weight: bold;
 | 
						|
  line-height: 1;
 | 
						|
}
 | 
						|
 | 
						|
.success-rings {
 | 
						|
  position: absolute;
 | 
						|
  top: 50%;
 | 
						|
  left: 50%;
 | 
						|
  transform: translate(-50%, -50%);
 | 
						|
  z-index: 1;
 | 
						|
}
 | 
						|
 | 
						|
.ring {
 | 
						|
  position: absolute;
 | 
						|
  border: 2rpx solid #005bac;
 | 
						|
  border-radius: 50%;
 | 
						|
  opacity: 0;
 | 
						|
  top: 50%;
 | 
						|
  left: 50%;
 | 
						|
  transform: translate(-50%, -50%);
 | 
						|
}
 | 
						|
 | 
						|
.ring1 {
 | 
						|
  width: 140rpx;
 | 
						|
  height: 140rpx;
 | 
						|
  animation: ripple 1.5s ease-out 0.2s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.ring2 {
 | 
						|
  width: 180rpx;
 | 
						|
  height: 180rpx;
 | 
						|
  animation: ripple 1.5s ease-out 0.6s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.ring3 {
 | 
						|
  width: 220rpx;
 | 
						|
  height: 220rpx;
 | 
						|
  animation: ripple 1.5s ease-out 1s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.success-title {
 | 
						|
  font-size: 48rpx;
 | 
						|
  font-weight: 600;
 | 
						|
  color: #1a1a1a;
 | 
						|
  margin-bottom: 16rpx;
 | 
						|
  text-align: center;
 | 
						|
}
 | 
						|
 | 
						|
.success-subtitle {
 | 
						|
  font-size: 28rpx;
 | 
						|
  color: #666666;
 | 
						|
  text-align: center;
 | 
						|
  line-height: 1.4;
 | 
						|
}
 | 
						|
 | 
						|
.action-section {
 | 
						|
  padding: 32rpx;
 | 
						|
}
 | 
						|
 | 
						|
.button-group {
 | 
						|
  display: flex;
 | 
						|
  gap: 24rpx;
 | 
						|
}
 | 
						|
 | 
						|
.btn {
 | 
						|
  flex: 1;
 | 
						|
  height: 88rpx;
 | 
						|
  border-radius: 44rpx;
 | 
						|
  font-size: 32rpx;
 | 
						|
  font-weight: 500;
 | 
						|
  border: none;
 | 
						|
  transition: all 0.3s ease;
 | 
						|
  position: relative;
 | 
						|
  overflow: hidden;
 | 
						|
 | 
						|
  &::after {
 | 
						|
    border: none;
 | 
						|
  }
 | 
						|
 | 
						|
  &.btn-secondary {
 | 
						|
    background: #ffffff;
 | 
						|
    color: #005bac;
 | 
						|
    border: 2rpx solid #005bac;
 | 
						|
 | 
						|
    &:active {
 | 
						|
      background: #f8fafc;
 | 
						|
      transform: scale(0.98);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  &.btn-primary {
 | 
						|
    background: linear-gradient(135deg, #005bac 0%, #0066cc 100%);
 | 
						|
    color: #ffffff;
 | 
						|
    box-shadow: 0 4rpx 16rpx rgba(0, 91, 172, 0.3);
 | 
						|
 | 
						|
    &:active {
 | 
						|
      background: linear-gradient(135deg, #004691 0%, #0052a3 100%);
 | 
						|
      transform: scale(0.98);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// 动画效果
 | 
						|
@keyframes successBounce {
 | 
						|
  0% {
 | 
						|
    transform: scale(0);
 | 
						|
    opacity: 0;
 | 
						|
  }
 | 
						|
  50% {
 | 
						|
    transform: scale(1.2);
 | 
						|
    opacity: 1;
 | 
						|
  }
 | 
						|
  100% {
 | 
						|
    transform: scale(1);
 | 
						|
    opacity: 1;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
@keyframes ripple {
 | 
						|
  0% {
 | 
						|
    opacity: 0.6;
 | 
						|
    transform: translate(-50%, -50%) scale(0.1);
 | 
						|
  }
 | 
						|
  100% {
 | 
						|
    opacity: 0;
 | 
						|
    transform: translate(-50%, -50%) scale(1);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
.spinner-dot {
 | 
						|
  position: relative;
 | 
						|
  width: 12rpx;
 | 
						|
  height: 12rpx;
 | 
						|
  background: #ffffff;
 | 
						|
  border-radius: 50%;
 | 
						|
  box-shadow: 0 2rpx 4rpx rgba(255, 255, 255, 0.3);
 | 
						|
}
 | 
						|
 | 
						|
.spinner-dot.dot1 {
 | 
						|
  animation: dotBlink 1.4s ease-in-out infinite;
 | 
						|
}
 | 
						|
 | 
						|
.spinner-dot.dot2 {
 | 
						|
  animation: dotBlink 1.4s ease-in-out 0.2s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.spinner-dot.dot3 {
 | 
						|
  animation: dotBlink 1.4s ease-in-out 0.4s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.loading-section {
 | 
						|
  flex: 1;
 | 
						|
  display: flex;
 | 
						|
  flex-direction: column;
 | 
						|
  align-items: center;
 | 
						|
  justify-content: center;
 | 
						|
  padding: 60rpx 40rpx 40rpx;
 | 
						|
  position: relative;
 | 
						|
}
 | 
						|
 | 
						|
.loading-icon-wrapper {
 | 
						|
  position: relative;
 | 
						|
  margin-bottom: 40rpx;
 | 
						|
}
 | 
						|
 | 
						|
.loading-icon {
 | 
						|
  width: 120rpx;
 | 
						|
  height: 120rpx;
 | 
						|
  background: linear-gradient(135deg, #005bac 0%, #0066cc 100%);
 | 
						|
  border-radius: 50%;
 | 
						|
  display: flex;
 | 
						|
  align-items: center;
 | 
						|
  justify-content: center;
 | 
						|
  position: relative;
 | 
						|
  z-index: 2;
 | 
						|
  box-shadow: 0 8rpx 32rpx rgba(0, 91, 172, 0.4);
 | 
						|
}
 | 
						|
 | 
						|
.loading-spinner {
 | 
						|
  position: relative;
 | 
						|
  width: 80rpx;
 | 
						|
  height: 20rpx;
 | 
						|
  display: flex;
 | 
						|
  align-items: center;
 | 
						|
  justify-content: space-between;
 | 
						|
}
 | 
						|
 | 
						|
.loading-rings {
 | 
						|
  position: absolute;
 | 
						|
  top: 50%;
 | 
						|
  left: 50%;
 | 
						|
  transform: translate(-50%, -50%);
 | 
						|
  z-index: 1;
 | 
						|
}
 | 
						|
 | 
						|
.loading-ring {
 | 
						|
  position: absolute;
 | 
						|
  border: 2rpx solid #005bac;
 | 
						|
  border-radius: 50%;
 | 
						|
  opacity: 0;
 | 
						|
  top: 50%;
 | 
						|
  left: 50%;
 | 
						|
  transform: translate(-50%, -50%);
 | 
						|
}
 | 
						|
 | 
						|
.loading-ring.ring1 {
 | 
						|
  width: 140rpx;
 | 
						|
  height: 140rpx;
 | 
						|
  animation: loadingRipple 2s ease-out infinite;
 | 
						|
}
 | 
						|
 | 
						|
.loading-ring.ring2 {
 | 
						|
  width: 180rpx;
 | 
						|
  height: 180rpx;
 | 
						|
  animation: loadingRipple 2s ease-out 0.8s infinite;
 | 
						|
}
 | 
						|
 | 
						|
.loading-title {
 | 
						|
  font-size: 48rpx;
 | 
						|
  font-weight: 600;
 | 
						|
  color: #2c3e50;
 | 
						|
  margin-bottom: 16rpx;
 | 
						|
  text-align: center;
 | 
						|
  text-shadow: 0 2rpx 4rpx rgba(0, 0, 0, 0.1);
 | 
						|
}
 | 
						|
 | 
						|
.loading-subtitle {
 | 
						|
  font-size: 28rpx;
 | 
						|
  color: #7f8c8d;
 | 
						|
  text-align: center;
 | 
						|
  line-height: 1.4;
 | 
						|
  opacity: 0.9;
 | 
						|
}
 | 
						|
 | 
						|
@keyframes dotBlink {
 | 
						|
  0% {
 | 
						|
    opacity: 0.4;
 | 
						|
  }
 | 
						|
  50% {
 | 
						|
    opacity: 1;
 | 
						|
  }
 | 
						|
  100% {
 | 
						|
    opacity: 0.4;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
@keyframes loadingRipple {
 | 
						|
  0% {
 | 
						|
    opacity: 0.8;
 | 
						|
    transform: translate(-50%, -50%) scale(0.1);
 | 
						|
  }
 | 
						|
  100% {
 | 
						|
    opacity: 0;
 | 
						|
    transform: translate(-50%, -50%) scale(1);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// 暗黑模式适配
 | 
						|
@media (prefers-color-scheme: dark) {
 | 
						|
  .success-page {
 | 
						|
    background: linear-gradient(135deg, #1a1a1a 0%, #2d3748 100%);
 | 
						|
  }
 | 
						|
 | 
						|
  .success-title {
 | 
						|
    color: #ffffff;
 | 
						|
  }
 | 
						|
 | 
						|
  .success-subtitle {
 | 
						|
    color: #a0aec0;
 | 
						|
  }
 | 
						|
 | 
						|
  .loading-title {
 | 
						|
    color: #ffffff;
 | 
						|
  }
 | 
						|
 | 
						|
  .loading-subtitle {
 | 
						|
    color: #a0aec0;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// 适配不同屏幕尺寸
 | 
						|
@media screen and (max-width: 375px) {
 | 
						|
  .success-icon {
 | 
						|
    width: 100rpx;
 | 
						|
    height: 100rpx;
 | 
						|
  }
 | 
						|
 | 
						|
  .success-checkmark {
 | 
						|
    font-size: 50rpx;
 | 
						|
  }
 | 
						|
 | 
						|
  .success-title {
 | 
						|
    font-size: 42rpx;
 | 
						|
  }
 | 
						|
 | 
						|
  .button-group {
 | 
						|
    flex-direction: column;
 | 
						|
    gap: 16rpx;
 | 
						|
  }
 | 
						|
}
 | 
						|
</style>
 |