天波h5前端应用
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

726 行
29KB

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