天波h5前端应用
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

667 rindas
26KB

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