recharge.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. <template>
  2. <view class="recharge-page">
  3. <!-- 个人服务包选择 -->
  4. <view class="package-section">
  5. <view class="section-title">选择个人服务包</view>
  6. <!-- 服务包列表 -->
  7. <view v-if="servicePackages.length > 0" class="package-list">
  8. <view
  9. v-for="pkg in servicePackages"
  10. :key="pkg.grfwbid"
  11. class="package-item"
  12. :class="{ active: selectedPackage && String(selectedPackage.grfwbid) === String(pkg.grfwbid) }"
  13. @click="selectPackage(pkg)"
  14. >
  15. <view class="package-header">
  16. <view class="package-name">{{ pkg.mc }}</view>
  17. <view class="package-info">
  18. <text class="package-price">¥{{ pkg.num > 0 ? (pkg.jg * pkg.num) : pkg.jg }}</text>
  19. <text v-if="pkg.num > 0" class="package-weeks">{{ pkg.num }}周 × ¥{{ pkg.jg }}/周</text>
  20. </view>
  21. </view>
  22. <!-- 服务项目列表 -->
  23. <view v-if="pkg.grfwbmxList && pkg.grfwbmxList.length > 0" class="service-items">
  24. <view
  25. v-for="(item, index) in pkg.grfwbmxList"
  26. :key="index"
  27. class="service-item"
  28. >
  29. <text class="service-name">{{ item.mc }}</text>
  30. <view class="service-details">
  31. <text v-if="item.sfmf == 1" class="service-tag free">免费</text>
  32. <text v-if="item.sc > 0" class="service-tag">{{ item.sc }}分钟</text>
  33. <text v-if="item.cs > 0" class="service-tag">{{ item.cs }}次</text>
  34. <text v-if="item.ll > 0" class="service-tag">{{ item.ll }}MB</text>
  35. </view>
  36. </view>
  37. </view>
  38. </view>
  39. </view>
  40. <!-- 空状态 -->
  41. <view v-else class="empty-state">
  42. <text>暂无可用服务包</text>
  43. </view>
  44. </view>
  45. <!-- 支付金额显示 -->
  46. <view class="pay-info">
  47. <view class="pay-info-item">
  48. <text class="label">支付金额:</text>
  49. <text class="value">¥{{ finalAmount }}</text>
  50. </view>
  51. </view>
  52. <!-- 支付按钮 -->
  53. <view class="pay-btn-wrapper">
  54. <button
  55. class="pay-btn"
  56. :disabled="!finalAmount || finalAmount <= 0"
  57. @click="handlePay"
  58. >
  59. 立即购买
  60. </button>
  61. <!-- <button
  62. class="pay-btn mock-btn"
  63. type="default"
  64. @click="goToTestResult"
  65. >
  66. 测试支付结果页
  67. </button> -->
  68. </view>
  69. <!-- 充值说明 -->
  70. <!-- <view class="tips">
  71. <view class="tips-title">充值说明:</view>
  72. <view class="tips-item">1. 充值金额将实时到账</view>
  73. <view class="tips-item">2. 如有疑问请联系客服</view>
  74. <view class="tips-item">3. 充值成功后可在消费记录中查看</view>
  75. </view> -->
  76. </view>
  77. </template>
  78. <script setup>
  79. import { ref, computed } from 'vue'
  80. import { onLoad } from '@dcloudio/uni-app'
  81. import { grfwApi } from '@/api/grfw'
  82. // 兼容不同字段命名的支付参数
  83. const normalizePayParams = (res) => {
  84. const data = res?.data.ssData || res || {}
  85. const payload = data.prepay || data.payInfo || data
  86. if (!payload) {
  87. return {}
  88. }
  89. const rawPackage =
  90. payload.package ||
  91. payload.packageVal ||
  92. (payload.prepayId ? `prepay_id=${payload.prepayId}` : undefined)
  93. const normalizedPackage =
  94. typeof rawPackage === 'string' && rawPackage && !rawPackage.startsWith('prepay_id=')
  95. ? `prepay_id=${rawPackage}`
  96. : rawPackage
  97. return {
  98. appId: payload.appId,
  99. timeStamp: payload.timeStamp || payload.timestamp,
  100. nonceStr: payload.nonceStr || payload.noncestr,
  101. package: normalizedPackage,
  102. signType: payload.signType || payload.sign_type || 'RSA',
  103. paySign: payload.paySign || payload.sign,
  104. orderId: data.outTradeNo
  105. }
  106. }
  107. // 个人服务包数据
  108. const servicePackages = ref([]) // 服务包列表
  109. const selectedPackage = ref(null) // 选中的服务包
  110. const isPaymentInProgress = ref(false) // 支付进行中标志
  111. // 最终支付金额(使用服务包价格,周包需要乘以周数)
  112. const finalAmount = computed(() => {
  113. if (selectedPackage.value) {
  114. const pkg = selectedPackage.value
  115. // 如果是周包(num > 0),总价 = 单价 × 周数
  116. if (pkg.num > 0) {
  117. return parseFloat(pkg.jg * pkg.num) || 0
  118. }
  119. return parseFloat(pkg.jg) || 0
  120. }
  121. return 0
  122. })
  123. // 选择服务包
  124. const selectPackage = (pkg) => {
  125. selectedPackage.value = pkg
  126. console.log('选中服务包:', pkg)
  127. }
  128. // 发起支付
  129. const handlePay = async () => {
  130. if (isPaymentInProgress.value) {
  131. console.log('支付进行中,请勿重复点击')
  132. return
  133. }
  134. // 检查是否选择了服务包
  135. if (!selectedPackage.value) {
  136. uni.showToast({
  137. title: '请选择服务包',
  138. icon: 'none'
  139. })
  140. return
  141. }
  142. const amount = finalAmount.value
  143. if (!amount || amount <= 0) {
  144. uni.showToast({
  145. title: '支付金额异常',
  146. icon: 'none'
  147. })
  148. return
  149. }
  150. try {
  151. isPaymentInProgress.value = true
  152. // 1. 创建订单,获取支付参数
  153. console.log('创建个人服务包订单,包ID:', selectedPackage.value.grfwbid)
  154. console.log('支付金额:', amount)
  155. const payParamsRaw = await grfwApi.grfw_prepayGrfwb({
  156. grfwbid: selectedPackage.value.grfwbid
  157. })
  158. console.log('创建订单接口原始返回:', payParamsRaw.data.ssData)
  159. const payParams = normalizePayParams(payParamsRaw)
  160. if (!payParams.timeStamp || !payParams.nonceStr || !payParams.package || !payParams.paySign) {
  161. throw new Error('支付参数不完整')
  162. }
  163. console.log('支付参数:', payParams)
  164. // 2. 调用微信支付
  165. const paymentResult = await uni.requestPayment({
  166. provider: 'wxpay',
  167. timeStamp: String(payParams.timeStamp),
  168. nonceStr: payParams.nonceStr,
  169. package: payParams.package,
  170. signType: payParams.signType || 'RSA',
  171. paySign: payParams.paySign,
  172. })
  173. console.log('支付结果:', paymentResult)
  174. // 3. 支付成功 - 跳转到支付结果页面
  175. const resultOrderId = payParams.orderId
  176. if (resultOrderId) {
  177. // 跳转到支付结果页面,进行订单状态查询,传递 grfwbid 用于确认服务
  178. uni.redirectTo({
  179. url: `/pages/payment/result?orderId=${resultOrderId}&grfwbid=${selectedPackage.value.grfwbid}`
  180. })
  181. } else {
  182. // 如果没有订单ID,直接提示成功
  183. uni.showToast({
  184. title: '支付成功',
  185. icon: 'success',
  186. duration: 2000
  187. })
  188. setTimeout(() => {
  189. uni.navigateBack()
  190. }, 2000)
  191. }
  192. } catch (error) {
  193. console.error('支付失败:', error)
  194. // 处理不同的错误情况
  195. if (error.errMsg) {
  196. // 微信支付相关错误
  197. if (error.errMsg.includes('cancel')) {
  198. uni.showToast({
  199. title: '已取消支付',
  200. icon: 'none'
  201. })
  202. } else if (error.errMsg.includes('fail')) {
  203. uni.showToast({
  204. title: '支付失败,请重试',
  205. icon: 'none'
  206. })
  207. }
  208. } else {
  209. // 其他错误(比如创建订单失败)
  210. uni.showToast({
  211. title: error.message || '操作失败',
  212. icon: 'none'
  213. })
  214. }
  215. } finally {
  216. isPaymentInProgress.value = false
  217. }
  218. }
  219. // 加载个人服务包列表
  220. const loadServicePackages = async () => {
  221. try {
  222. const res = await grfwApi.grfw_initGrfwbBuy({})
  223. console.log('获取服务包列表返回:', res)
  224. console.log('res.data:', res.data)
  225. console.log('res.data.ssData:', res.data?.ssData)
  226. // 数据在 res.data.ssData 里
  227. if (res && res.data && res.data.ssData && Array.isArray(res.data.ssData)) {
  228. const ssData = res.data.ssData
  229. console.log('ssData 长度:', ssData.length)
  230. console.log('ssData 内容:', JSON.stringify(ssData))
  231. // 跳过第一个元素(minJzsj),从第二个元素开始获取服务包
  232. if (ssData.length > 1) {
  233. const packages = ssData.slice(1)
  234. console.log('提取的服务包(去除第一个元素):', packages)
  235. servicePackages.value = packages.filter(item => item.grfwbid)
  236. console.log('过滤后的服务包列表:', servicePackages.value)
  237. console.log('服务包数量:', servicePackages.value.length)
  238. } else {
  239. console.log('ssData 长度不足,没有服务包数据')
  240. }
  241. } else {
  242. console.log('res.data.ssData 不存在或不是数组')
  243. }
  244. } catch (error) {
  245. console.error('加载服务包失败:', error)
  246. uni.showToast({
  247. title: '加载服务包失败',
  248. icon: 'none'
  249. })
  250. }
  251. }
  252. // 页面加载
  253. onLoad((options) => {
  254. console.log('充值页面加载,参数:', options)
  255. // 加载服务包列表
  256. loadServicePackages()
  257. })
  258. // 测试跳转到支付结果页
  259. const goToTestResult = () => {
  260. // 使用固定的订单号测试
  261. const testOrderId = 'pmsgrfwb7m05clpYYgwHG8qlroZB1'
  262. const testGrfwbid = selectedPackage.value?.grfwbid || ''
  263. uni.redirectTo({
  264. url: `/pages/payment/result?orderId=${testOrderId}&grfwbid=${testGrfwbid}`
  265. })
  266. }
  267. </script>
  268. <style lang="scss" scoped>
  269. .recharge-page {
  270. min-height: 100vh;
  271. background: #f5f5f5;
  272. padding: 30rpx;
  273. }
  274. .package-section {
  275. background: #fff;
  276. border-radius: 20rpx;
  277. padding: 40rpx 30rpx;
  278. margin-bottom: 30rpx;
  279. }
  280. .section-title {
  281. font-size: 32rpx;
  282. font-weight: bold;
  283. color: #333;
  284. margin-bottom: 30rpx;
  285. }
  286. .package-list {
  287. display: flex;
  288. flex-direction: column;
  289. gap: 20rpx;
  290. }
  291. .package-item {
  292. padding: 30rpx;
  293. border: 2rpx solid #e5e5e5;
  294. border-radius: 12rpx;
  295. background: #fff;
  296. transition: all 0.3s;
  297. }
  298. .package-item.active {
  299. border-color: #07c160;
  300. background: #f0f9ff;
  301. }
  302. .package-header {
  303. display: flex;
  304. justify-content: space-between;
  305. align-items: center;
  306. margin-bottom: 20rpx;
  307. }
  308. .package-name {
  309. font-size: 32rpx;
  310. color: #333;
  311. font-weight: 500;
  312. }
  313. .package-item.active .package-name {
  314. color: #07c160;
  315. font-weight: bold;
  316. }
  317. .package-info {
  318. display: flex;
  319. flex-direction: column;
  320. align-items: flex-end;
  321. gap: 8rpx;
  322. }
  323. .package-price {
  324. font-size: 36rpx;
  325. color: #ff4d4f;
  326. font-weight: bold;
  327. }
  328. .package-item.active .package-price {
  329. color: #07c160;
  330. }
  331. .package-weeks {
  332. font-size: 24rpx;
  333. color: #999;
  334. background: #f5f5f5;
  335. padding: 4rpx 12rpx;
  336. border-radius: 4rpx;
  337. }
  338. .package-item.active .package-weeks {
  339. background: #e6f7ff;
  340. color: #1890ff;
  341. }
  342. .service-items {
  343. border-top: 1rpx solid #f0f0f0;
  344. padding-top: 20rpx;
  345. display: grid;
  346. grid-template-columns: repeat(2, 1fr);
  347. gap: 12rpx;
  348. }
  349. .service-item {
  350. display: flex;
  351. flex-direction: column;
  352. gap: 8rpx;
  353. padding: 12rpx 16rpx;
  354. background: #fafafa;
  355. border-radius: 8rpx;
  356. }
  357. .package-item.active .service-item {
  358. background: #f0f9ff;
  359. }
  360. .service-name {
  361. font-size: 28rpx;
  362. color: #666;
  363. font-weight: 500;
  364. }
  365. .service-details {
  366. display: flex;
  367. flex-wrap: wrap;
  368. gap: 6rpx;
  369. }
  370. .service-tag {
  371. font-size: 22rpx;
  372. color: #666;
  373. background: #fff;
  374. padding: 2rpx 8rpx;
  375. border-radius: 4rpx;
  376. border: 1rpx solid #e5e5e5;
  377. white-space: nowrap;
  378. }
  379. .service-tag.free {
  380. color: #52c41a;
  381. background: #f6ffed;
  382. border-color: #b7eb8f;
  383. }
  384. .empty-state {
  385. text-align: center;
  386. padding: 60rpx 0;
  387. color: #999;
  388. font-size: 28rpx;
  389. }
  390. .pay-info {
  391. background: #fff;
  392. border-radius: 20rpx;
  393. padding: 30rpx;
  394. margin-bottom: 30rpx;
  395. }
  396. .pay-info-item {
  397. display: flex;
  398. justify-content: space-between;
  399. align-items: center;
  400. }
  401. .pay-info-item .label {
  402. font-size: 32rpx;
  403. color: #333;
  404. }
  405. .pay-info-item .value {
  406. font-size: 40rpx;
  407. color: #ff4d4f;
  408. font-weight: bold;
  409. }
  410. .pay-btn-wrapper {
  411. padding: 40rpx 0;
  412. }
  413. .pay-btn {
  414. width: 100%;
  415. height: 88rpx;
  416. line-height: 88rpx;
  417. background: linear-gradient(135deg, #07c160 0%, #05a050 100%);
  418. color: #fff;
  419. border-radius: 44rpx;
  420. font-size: 32rpx;
  421. font-weight: bold;
  422. border: none;
  423. box-shadow: 0 8rpx 20rpx rgba(7, 193, 96, 0.3);
  424. }
  425. .pay-btn[disabled] {
  426. background: #e5e5e5;
  427. color: #999;
  428. box-shadow: none;
  429. }
  430. .pay-btn::after {
  431. border: none;
  432. }
  433. .mock-btn {
  434. margin-top: 20rpx;
  435. background: #f5f5f5;
  436. color: #333;
  437. border: 1px solid #d9d9d9;
  438. box-shadow: none;
  439. }
  440. .tips {
  441. background: #fff;
  442. border-radius: 20rpx;
  443. padding: 30rpx;
  444. }
  445. .tips-title {
  446. font-size: 28rpx;
  447. color: #333;
  448. font-weight: bold;
  449. margin-bottom: 20rpx;
  450. }
  451. .tips-item {
  452. font-size: 26rpx;
  453. color: #999;
  454. line-height: 40rpx;
  455. margin-bottom: 10rpx;
  456. }
  457. </style>