天波h5前端应用
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

719 lines
28KB

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