天波h5前端应用
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

732 lines
30KB

  1. <!--
  2. * @Date: 2022-03-29 16:57:58
  3. * @LastEditors: JinxChen
  4. * @LastEditTime: 2023-03-13 08:55:33
  5. * @FilePath: \TelpoH5FrontendWeb\src\views\package-list\index.vue
  6. * @description: TODO 小台风充值h5
  7. -->
  8. <template>
  9. <div class="package-list-container">
  10. <van-nav-bar :left-arrow="false" :border="true">
  11. <template #title>
  12. <h5 style="font-size: 16px">{{topupTitle}}</h5>
  13. </template>
  14. </van-nav-bar>
  15. <!-- 灰色线条 -->
  16. <div class="gray-line"></div>
  17. <!-- 套餐列表 -->
  18. <div class="topup-container">
  19. <div class="main">
  20. <!-- 无套餐时显示 -->
  21. <div class="noData_container" v-show="packageOrderList.length === 0 && isShowNoData">
  22. <p>暂无相关套餐数据,请您联系管理员~</p>
  23. </div>
  24. <!-- 套餐订购 -->
  25. <div
  26. class="package-order-container"
  27. v-for="(item, index) in packageOrderList"
  28. :key="index"
  29. >
  30. <!-- 推荐 -->
  31. <div class="recom" v-show="false">
  32. <div class="shape"></div>
  33. <div class="square">
  34. <p>推荐</p>
  35. </div>
  36. </div>
  37. <!-- 套餐内容 -->
  38. <div class="order-content">
  39. <div class="title">
  40. <p>{{item.packageName}}<!-- :{{(item.packagePrice/item.packageIssue).toFixed(0)}}元/月 --></p>
  41. </div>
  42. <div class="remark">
  43. <p>每月200分钟通话时长,1G流量</p>
  44. </div>
  45. <div class="details">
  46. <p>
  47. <span class="orange large">¥{{(item.packagePrice/(item.packageIssue === 0 ? 1: item.packageIssue)).toFixed(0)}}</span>元/月,
  48. </p>
  49. <p class="total">
  50. <span>合计</span><span class="orange price">{{item.packagePrice}}元</span>
  51. </p>
  52. </div>
  53. <!-- <div class="package-buy">
  54. <div class="buy-btn" @click="onBuy(item)">
  55. <p>第一步:充值话费</p>
  56. </div>
  57. </div> -->
  58. <div class="radios-con">
  59. <div class="pay-type">
  60. <p>支付方式:</p>
  61. </div>
  62. <van-radio-group v-model="radio" direction="horizontal" @change="onRaidoChange">
  63. <van-radio name="1">
  64. <template #default>
  65. <div class="radio-con">
  66. <span>微信</span>
  67. <img src="../../assets/wx-pay.png"/>
  68. </div>
  69. </template>
  70. </van-radio>
  71. <van-radio name="2">
  72. <template #default>
  73. <div class="radio-con">
  74. <span>支付宝</span>
  75. <img src="../../assets/alipay.png"/>
  76. </div>
  77. </template>
  78. </van-radio>
  79. <van-radio name="3">
  80. <template #default>
  81. <div class="radio-con">
  82. <span>12期花呗</span>
  83. <img src="../../assets/antpay.png"/>
  84. </div>
  85. </template>
  86. </van-radio>
  87. </van-radio-group>
  88. </div>
  89. <div class="package-buy">
  90. <div class="buy-btn" @click="onBuy(item)">
  91. <p>第一步:充值话费</p>
  92. </div>
  93. </div>
  94. </div>
  95. </div>
  96. </div>
  97. </div>
  98. </div>
  99. </template>
  100. <script>
  101. import APIWx from "@/api/wx";
  102. import AppId from "@/config/appId";
  103. import { APIPay } from "@/api/pay";
  104. import APICore from "@/api/core";
  105. let wx = require("weixin-js-sdk"); // TODO 再封装,可拦截错误提示等操作
  106. import { isNotNull } from "@/utils/index";
  107. export default {
  108. name: "packageList",
  109. data() {
  110. return {
  111. topupTitle: "请选择套餐充值电话卡", //充值页面标题
  112. // 套餐列表, todo 需要从接口获取
  113. packageOrderList: [
  114. {
  115. packageName: '语音卡套餐(1年)',
  116. packagePayType: 1,
  117. packagePrice: 240,
  118. packageIssue: 12,
  119. payWxPayProductId: process.env.NODE_ENV === "production" ? '1629407413618294784' : '1629405716684029952', //微信支付
  120. payAliPayProductId: process.env.NODE_ENV === "production" ? '1629407413618294784' : '1629405716684029952', //支付宝全额支付
  121. payAntPayProductId: process.env.NODE_ENV === "production" ? '1629407705558630400': '1629405558344859648', //支付宝花呗支付
  122. }
  123. ],
  124. outTradeNo: "", //订单号
  125. price: "", //价格,
  126. isShowNoData: false, //是否显示无套餐内容, 默认false
  127. // 获取从路由的参数,import 调取激活接口需要用到
  128. params: {
  129. imei: '',
  130. iccid: '',
  131. manufactorId: '', //厂商id,一般用来获取core接口token
  132. appId: '', //公众号appId,需要对接多个不同公众号时动态从页面路由获取,
  133. },
  134. radio: '1', //支付方式单选按钮默认值
  135. payProductId: null, //套餐id
  136. packageIssue: null, //套餐分期
  137. payType: '1', //支付方式 1 微信, 2 支付宝,支付宝又分为花呗和全额支付,全额支付分期数传0 或者1 ,花呗则传 3 6 12
  138. };
  139. },
  140. created() {
  141. this.getAuth();
  142. this.getParams();
  143. this.getWxAutograph();
  144. // todo 接口列表
  145. /* this.getLiveBasePackage(); */
  146. },
  147. /* mounted() {
  148. this.getParams();
  149. this.getWxAutograph();
  150. this.getLiveBasePackage();
  151. }, */
  152. methods: {
  153. // 根据code获取openId
  154. getOpenId() {
  155. let code = this.$store.getters.wxAuthCode;
  156. let openId = this.$store.getters.openId;
  157. if(isNotNull(openId)) {
  158. console.log("已经存在openId");
  159. } else {
  160. APIPay.getOpenId(code).then(res => {
  161. let data= res.data;
  162. if(data.code === 20000) {
  163. this.$store.commit("openId", data.data.openId);
  164. }
  165. })
  166. }
  167. },
  168. // 获取b端接口的token
  169. getAuth() {
  170. let manufactorId = "5bf13062-a41e-4d00-ba14-1101aad12650";
  171. APICore.getAuth({ manufactorId: manufactorId }).then(res => {
  172. this.$store.commit("gatewayToken", res.data.data);
  173. });
  174. },
  175. // 获取url传过来的参数
  176. getParams() {
  177. let params = this.$route.query;
  178. if (params) {
  179. let url = window.location.href.split("?code=")[1];
  180. if ( isNotNull(url) || window.location.href.indexOf("code") > -1) {
  181. let timeStamp = new Date().getTime();
  182. let code = url.split("&")[0];
  183. if (isNotNull(code)) {
  184. this.$store.commit("wxAuthCode", `${code}`);
  185. this.getOpenId();
  186. }
  187. }
  188. this.params = {...params};
  189. }
  190. },
  191. // 获取基本套餐信息
  192. getLiveBasePackage() {
  193. this.$toast.loading({
  194. message: "获取套餐中",
  195. duration: 1500
  196. });
  197. let reqBody = {
  198. manufactorId:"",
  199. productModelId:0,
  200. iccid:"",
  201. pageNumber:1,
  202. begNumber:20
  203. }
  204. APICore.cardPackageList(reqBody)
  205. .then(res => {
  206. console.log("data", res.data);
  207. if (res.data.code === 106 || res.data.code === 104) {
  208. // token过期
  209. this.getAuth();
  210. setTimeout(() => {
  211. this.getLiveBasePackage();
  212. }, 1500);
  213. } else if (res.data.code === 0 && res.data.data === null) {
  214. this.isShowNoData = true;
  215. }else {
  216. let data = res.data.data.list;
  217. if(data === null) {
  218. this.isShowNoData = true;
  219. } else {
  220. this.packageOrderList = data.filter(item => {
  221. return item.pechargeUrl === '1629405716684029952' || item.pechargeUrl === '1629405558344859648';
  222. });
  223. console.log("套餐数据::", data);
  224. }
  225. }
  226. this.$toast.success({
  227. message: "成功获取套餐",
  228. duration: 1500
  229. });
  230. })
  231. .catch(error => {
  232. this.$dialog.confirm({
  233. title: "获取套餐数据失败",
  234. message: error,
  235. showCancelButton: false,
  236. });
  237. })
  238. .finally(() => {
  239. setTimeout(() => {
  240. this.$toast.clear();
  241. }, 1500);
  242. });
  243. },
  244. // 返回
  245. onNavBack() {
  246. },
  247. // 获取微信jssdk
  248. getWxAutograph() {
  249. let that = this;
  250. return new Promise((resolve, reject) => {
  251. APIWx.createJSSDK({
  252. sUrl: window.location.href.split("#")[0],
  253. userId: '',
  254. appId: this.params.appId || AppId
  255. })
  256. .then(res => {
  257. let item = res.data.data;
  258. wx.config({
  259. debug: false, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
  260. appId: item.appId, // 必填,公众号的唯一标识
  261. timestamp: item.timeStamp, // 必填,生成签名的时间戳
  262. nonceStr: item.nonceStr, // 必填,生成签名的随机串
  263. signature: item.signature, // 必填,签名
  264. jsApiList: ["chooseWXPay"] // 必填,需要使用的JS接口列表
  265. });
  266. wx.ready(() => {
  267. resolve(true);
  268. });
  269. })
  270. .catch(err => {
  271. reject(false);
  272. console.log(err);
  273. });
  274. });
  275. },
  276. // 话费充值
  277. onBuy(data) {
  278. this.price = data.packagePrice;
  279. // 需要区分是要用微信支付还是支付宝花呗支付
  280. if (this.payType === '2') {
  281. this.payProductId = process.env.NODE_ENV === "production" ? '1629407413618294784' : '1629405716684029952', //支付宝全额支付
  282. this.packageIssue = 0;
  283. this.aliPay(data);
  284. } else if (this.payType === '3') {
  285. this.payProductId = process.env.NODE_ENV === "production" ? '1629407705558630400': '1629405558344859648', //支付宝花呗支付
  286. this.packageIssue = 12;
  287. this.payType = '2'
  288. this.aliPay(data);
  289. } else {
  290. let openId = this.$store.getters.openId;
  291. this.payProductId = process.env.NODE_ENV === "production" ? '1629407413618294784': '1629405716684029952'; //支付宝花呗支付
  292. if(openId === null || openId === 'null') {
  293. this.$dialog.confirm({
  294. message: '获取OpenId失败,请您重新进入',
  295. showCancelButton: false,
  296. })
  297. } else {
  298. this.wxPay(data);
  299. }
  300. }
  301. },
  302. // 微信支付
  303. wxPay(data) {
  304. this.$toast.loading({
  305. message: "加载中"
  306. });
  307. console.log("微信支付", data);
  308. let orderData = data;
  309. let reqBody = {
  310. openId: this.$store.getters.openId, //openId
  311. imei: this.params.imei, //imei
  312. productId: this.payProductId, //套餐id
  313. packageName: /* data.productModel */ data.packageName, //套餐名字
  314. packagePayType: Number(this.payType), //支付类型
  315. packageIssue: 12, //分期
  316. packagePrice: process.env.NODE_ENV === "production" ? data.packagePrice * 100 : 1 //总金额单位为分,测试环境写死
  317. };
  318. this.$toast.clear();
  319. APICore.payLiveBaseDevice(reqBody)
  320. .then(res => {
  321. if (res.data.code === 104 || res.data.code === 106) {
  322. this.getAuth();
  323. setTimeout(() => {
  324. this.wxPay(orderData);
  325. }, 1000);
  326. } else if (res.data.code === 104) {
  327. this.$dialog.confirm({
  328. message: `${res.data.message}`
  329. })
  330. }
  331. let that = this;
  332. let wxData = res.data.data;
  333. that.outTradeNo = wxData.out_trade_no;
  334. console.log("wxData", wxData);
  335. // 本地测试
  336. /* that.$router.push({
  337. name: "payResult",
  338. query: {
  339. outTradeNo: that.outTradeNo,
  340. price: that.price,
  341. rechargeUrl: data.rechargeUrl || '',
  342. iccid: this.params.iccid,
  343. isAdmin: this.$route.query.isAdmin,
  344. serialNo: this.params.imei,
  345. routerName: this.params.routerName,
  346. issue: data.packageIssue
  347. }
  348. }); */
  349. wx.chooseWXPay({
  350. timestamp: wxData.timeStamp, // 支付签名时间戳,注意微信 jssdk 中的所有使用 timestamp 字段均为小写。但最新版的支付后台生成签名使用的 timeStamp 字段名需大写其中的 S 字符
  351. nonceStr: wxData.nonceStr, // 支付签名随机串,不长于 32 位
  352. package: wxData.package, // 统一支付接口返回的prepay_id参数值,提交格式如:prepay_id=\*\*\*)
  353. signType: wxData.signType, // 微信支付V3的传入 RSA ,微信支付V2的传入格式与V2统一下单的签名格式保持一致
  354. paySign: wxData.paySign, // 支付签名
  355. success: function(res) {
  356. // 支付成功后的回调函数
  357. that.$router.replace({
  358. name: "payResult",
  359. query: {
  360. outTradeNo: that.outTradeNo,
  361. price: that.price,
  362. rechargeUrl: data.rechargeUrl,
  363. iccid: that.params.iccid,
  364. isAdmin: that.$route.query.isAdmin || false,
  365. serialNo: that.params.imei,
  366. issue: that.packageIssue
  367. }
  368. });
  369. console.log("微信支付成功::", res);
  370. },
  371. fail: err => {
  372. console.log("支付出错了::", err);
  373. that.$dialog.confirm({
  374. title: "温馨提示",
  375. message: "出错了,请您重新进入",
  376. showCancelButton: false
  377. });
  378. },
  379. cancel: function(err) {
  380. // 用户取消支付
  381. that.$dialog.confirm({
  382. title: "温馨提示",
  383. message: "您取消了支付",
  384. showCancelButton: false
  385. });
  386. console.log("用户取消了支付::", err);
  387. }
  388. });
  389. })
  390. .catch(error => {
  391. console.log("error", error);
  392. })
  393. .finally(() => {
  394. this.$toast.clear();
  395. });
  396. },
  397. // 跳转到支付宝花呗外部链接
  398. aliPay(data) {
  399. console.log("选择了支付宝::", data);
  400. this.$toast.loading({
  401. message: "加载中"
  402. });
  403. let orderData = data;
  404. let reqBody = {
  405. openId: this.$store.getters.openId, //openId
  406. imei: this.params.imei, //imei
  407. productId: this.payProductId, //套餐id
  408. packageName: /* data.productModel + ',' + */data.packageName, //套餐名字
  409. packagePayType: Number(this.payType), //支付类型
  410. packageIssue: this.packageIssue, //分期
  411. packagePrice: process.env.NODE_ENV === "production" ? data.packagePrice * 100 : 1 //总金额单位为分,测试环境写死
  412. };
  413. this.$toast.clear();
  414. APICore.payLiveBaseDevice(reqBody)
  415. .then(res => {
  416. if (res.data.code === 104 || res.data.code === 106) {
  417. this.getAuth();
  418. setTimeout(() => {
  419. this.aliPay(orderData);
  420. }, 1000);
  421. }
  422. let that = this;
  423. let alipayData = res.data.data.xmlStrMap;
  424. that.outTradeNo = alipayData.outTradeNo;
  425. let alipayForm = decodeURI(alipayData.payXmlStr);
  426. that.$store.commit("isFromWx", true);
  427. let alipayUserId = process.env.NODE_ENV === "production" ? 42 : 18;
  428. this.$router.replace({
  429. name: "payResult",
  430. query: {
  431. rechargeUrl:
  432. data.rechargeUrl ||
  433. `https://id.ssjlai.com/frontend/#/alipay`,
  434. outTradeNo: that.outTradeNo,
  435. price: that.price,
  436. alipayForm: alipayForm,
  437. iccid: that.params.iccid,
  438. isAdmin: that.$route.query.isAdmin || false,
  439. serialNo: that.params.imei,
  440. alipayUserId: alipayUserId,
  441. productId: this.payProductId
  442. }
  443. });
  444. })
  445. .catch(error => {
  446. console.log("error", error);
  447. this.$dialog.confirm({
  448. message: `${error.message}`,
  449. showCancelButton: false
  450. })
  451. })
  452. .finally(() => {
  453. this.$toast.clear();
  454. });
  455. },
  456. // 支付类型切换
  457. onRaidoChange(value) {
  458. console.log("选择的支付类型是", value);
  459. this.payType = value;
  460. }
  461. }
  462. };
  463. </script>
  464. <style lang="scss">
  465. .van-nav-bar__title {
  466. max-width: 80% !important;
  467. font-size: 16px;
  468. }
  469. </style>
  470. <style lang="scss" scoped>
  471. .package-list-container {
  472. background-color: white;
  473. height: 100vh;
  474. .topup-container {
  475. position: relative;
  476. /* margin: 20px; */
  477. display: flex;
  478. justify-content: space-between;
  479. align-items: center;
  480. flex-direction: column;
  481. background-color: white;
  482. overflow: scroll;
  483. height: calc(100vh - 88px);
  484. .main {
  485. display: flex;
  486. justify-content: center;
  487. align-items: center;
  488. flex-direction: column;
  489. .tips {
  490. padding: 10px;
  491. }
  492. .noData_container {
  493. margin: 100px auto 0;
  494. height: 120px;
  495. /* background: url(../../../assets/img/news-noData.png) center no-repeat; */
  496. background-size: 165px 120px;
  497. display: flex;
  498. justify-content: center;
  499. align-items: flex-end;
  500. font-size: 16px;
  501. color: #999;
  502. p {
  503. font-size: 16px;
  504. }
  505. /* @include colorAndFont(#999, 28); */
  506. }
  507. p {
  508. padding: 10px;
  509. font-size: 16px;
  510. }
  511. .cancel-button {
  512. width: 30vw;
  513. border-radius: 5px;
  514. background-color: #2599ff;
  515. color: #fff;
  516. min-height: 40px;
  517. display: flex;
  518. justify-content: center;
  519. align-items: center;
  520. font-size: 16px;
  521. }
  522. .package-order-container {
  523. position: relative;
  524. margin: 10px 0;
  525. /* padding: 0 0 0 20px; */
  526. z-index: 999;
  527. box-shadow: rgba(14, 30, 37, 0.12) 0 3px 5px 0,
  528. rgba(14, 30, 37, 0.32) 0 2px 16px 0;
  529. .recom {
  530. position: absolute;
  531. top: -7px;
  532. left: -26px;
  533. border-style: solid;
  534. border-width: 0 40px 40px;
  535. border-color: transparent transparent red;
  536. transform: rotate(-45deg);
  537. text-align: center;
  538. z-index: 9999;
  539. p {
  540. padding: 0;
  541. color: white;
  542. font-size: 14px;
  543. }
  544. .shape {
  545. position: absolute;
  546. top: -1px;
  547. left: -21px;
  548. border-style: solid;
  549. border-width: 0 21px 21px;
  550. border-color: transparent transparent white;
  551. }
  552. .square {
  553. height: 15px;
  554. width: 35px;
  555. position: absolute;
  556. top: 21px;
  557. left: -20px;
  558. background: red;
  559. font-size: 14px;
  560. }
  561. }
  562. .order-content {
  563. padding: 15px 10px;
  564. .title {
  565. display: flex;
  566. justify-content: flex-start;
  567. p {
  568. font-size: 16px;
  569. font-weight: bold;
  570. }
  571. }
  572. .details {
  573. display: flex;
  574. justify-content: flex-start;
  575. align-items: center;
  576. p {
  577. height: 40px;
  578. @include center();
  579. font-size: 14px;
  580. padding: 5px 0 5px 10px;
  581. }
  582. .total {
  583. font-size: 16px;
  584. padding: 0;
  585. }
  586. .orange {
  587. color: orange;
  588. }
  589. .large {
  590. font-size: 26px;
  591. margin: 0 2px;
  592. }
  593. .price {
  594. font-size: 16px;
  595. }
  596. /* .buy-btn {
  597. height: 30px;
  598. width: 100px;
  599. display: flex;
  600. justify-content: center;
  601. align-items: center;
  602. background: orange;
  603. border-radius: 45px;
  604. p {
  605. font-size: 16px;
  606. padding: 0;
  607. color: red;
  608. }
  609. } */
  610. }
  611. .remark {
  612. display: flex;
  613. justify-content: flex-start;
  614. p {
  615. font-size: 14px;
  616. padding: 5px 10px;
  617. /* font-weight: bold; */
  618. }
  619. }
  620. .package-buy {
  621. padding: 5px 8px;
  622. @include center();
  623. .buy-btn {
  624. height: 40px;
  625. width: 200px;
  626. padding: 0 5px;
  627. display: flex;
  628. justify-content: center;
  629. align-items: center;
  630. background: orange;
  631. border-radius: 20px;
  632. p {
  633. font-size: 16px;
  634. padding: 0;
  635. color: white;
  636. }
  637. }
  638. }
  639. .radios-con {
  640. /* padding: 20px 10px; */
  641. /* @include center(); */
  642. /* align-items: center; */
  643. padding: 10px 0 10px 10px;
  644. font-size: 14px;
  645. /* box-shadow: rgba(14, 30, 37, 0.12) 0 3px 5px 0,
  646. rgba(14, 30, 37, 0.32) 0 2px 16px 0; */
  647. img {
  648. height: 20px;
  649. width: 20px;
  650. margin: 2px;
  651. }
  652. .pay-type {
  653. text-align: left;
  654. p {
  655. font-size: 14px;
  656. padding: 0 0 10px 0;
  657. }
  658. }
  659. .radio-con {
  660. @include center();
  661. }
  662. .van-cell-text {
  663. display: flex;
  664. justify-content: flex-start;
  665. align-items: center;
  666. span {
  667. font-size: 14px;
  668. padding: 0 5px;
  669. }
  670. }
  671. }
  672. }
  673. .content {
  674. font-size: 14px;
  675. }
  676. }
  677. }
  678. }
  679. .gray-line {
  680. height: 10px;
  681. width: 100%;
  682. background: #f2f4f5;
  683. }
  684. .order-description {
  685. height: 60px;
  686. display: flex;
  687. justify-content: flex-start;
  688. align-items: flex-start;
  689. flex-direction: column;
  690. padding: 5px 0 10px 10px;
  691. h5 {
  692. font-size: 14px;
  693. padding: 10px 0 0 0;
  694. }
  695. }
  696. .pay-radios {
  697. position: relative;
  698. top: 0;
  699. width: 100%;
  700. /* background-color: red; */
  701. /* padding: 0 15px; */
  702. .radios-con {
  703. padding: 20px 10px;
  704. /* @include center(); */
  705. align-items: center;
  706. font-size: 18px;
  707. box-shadow: rgba(14, 30, 37, 0.12) 0 3px 5px 0,
  708. rgba(14, 30, 37, 0.32) 0 2px 16px 0;
  709. img {
  710. height: 35px;
  711. width: auto;
  712. }
  713. .van-cell-text {
  714. display: flex;
  715. justify-content: flex-start;
  716. align-items: center;
  717. span {
  718. padding: 0 5px;
  719. }
  720. }
  721. }
  722. }
  723. }
  724. </style>