index.vue 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. <template>
  2. <form class="form" @submit="checkForm">
  3. <view class="input-section">
  4. <view class="section-hd">{{$t(`支付金额`)}}</view>
  5. <view class="section-bd">
  6. <view class="input-group">
  7. {{$t(`¥`)}}
  8. <input v-model.number="money" class="input" name="money" type="digit" @input="inputChange" placeholder="0.00" />
  9. </view>
  10. <view v-if="payPrice" class="discount">{{$t(`会员优惠价`)}}:{{$t(`¥`)}}{{ payPrice }}</view>
  11. </view>
  12. </view>
  13. <view class="radio-section">
  14. <view class="section-hd">{{$t(`支付方式`)}}</view>
  15. <radio-group class="section-bd" name="method">
  16. <label class="item" v-if="yuePay">
  17. <text class="iconfont icon-yue"></text>
  18. <text class="name">
  19. {{$t(`余额支付`)}}
  20. <text class="money">{{$t(`可用余额`)}}:{{$t(`¥`)}}{{ now_money || 0 }}</text>
  21. </text>
  22. <radio value="yue" :checked="payType === 'yue'" />
  23. </label>
  24. <label v-if="wxpay" class="item">
  25. <text class="iconfont icon-weixinzhifu"></text>
  26. <text class="name">{{$t(`微信支付`)}}</text>
  27. <radio value="weixin" :checked="payType === 'weixin'" />
  28. </label>
  29. </radio-group>
  30. </view>
  31. <button class="button" form-type="submit">{{$t(`确认`)}}</button>
  32. <view class="alipay" v-html="alipayHtml"></view>
  33. </form>
  34. </template>
  35. <script>
  36. import {
  37. offlineCheckPrice,
  38. offlineCreate,
  39. orderOfflinePayType
  40. } from '@/api/order.js';
  41. import {
  42. toLogin
  43. } from '@/libs/login.js';
  44. import {
  45. mapGetters
  46. } from "vuex";
  47. const app = getApp();
  48. export default {
  49. data() {
  50. return {
  51. money: '',
  52. payPrice: '',
  53. payType: 'weixin',
  54. alipayHtml: '',
  55. alipay: false,
  56. wxpay: false,
  57. yuePay: false,
  58. paying: false,
  59. now_money: 0,
  60. isWeixin: false,
  61. site_name: '',
  62. isCommitted: false
  63. };
  64. },
  65. watch: {
  66. money(newValue, oldValue) {
  67. if (newValue && typeof newValue === 'number') {
  68. this.checkPrice();
  69. } else {
  70. this.payPrice = '';
  71. }
  72. }
  73. },
  74. computed: mapGetters(['isLogin']),
  75. onLoad(options) {
  76. if (!this.isLogin) {
  77. toLogin()
  78. }
  79. // #ifdef H5
  80. if (options.code) {
  81. let spread = app.globalData.spid ? app.globalData.spid : '';
  82. wechatAuthV2(options.code, spread).then(res => {
  83. location.href = decodeURIComponent(
  84. decodeURIComponent(options.back_url)
  85. )
  86. })
  87. }
  88. // #endif
  89. },
  90. onShow() {
  91. if (this.isLogin) {
  92. this.getPayType();
  93. }
  94. //#ifdef H5
  95. this.isWeixin = this.$wechat.isWeixin();
  96. //#endif
  97. },
  98. methods: {
  99. inputChange(e){
  100. var that = this
  101. e.target.value = (e.target.value.match(/^\d*(.?\d{0,2})/g)[0]) || ""
  102. this.$nextTick(() => {
  103. this.money = e.target.value
  104. this.checkPrice()
  105. })
  106. },
  107. getPayType() {
  108. orderOfflinePayType()
  109. .then(res => {
  110. const {
  111. ali_pay_status,
  112. pay_weixin_open,
  113. yue_pay_status,
  114. offline_pay_status,
  115. site_name,
  116. now_money
  117. } = res.data;
  118. this.alipay = ali_pay_status === '1' ? true : false;
  119. this.wxpay = pay_weixin_open === 1 ? true : false;
  120. this.yuePay = yue_pay_status === 1 ? true : false;
  121. this.now_money = now_money;
  122. this.site_name = site_name;
  123. if (!offline_pay_status) {
  124. uni.showModal({
  125. title: this.$t(`支付提醒`),
  126. content: this.$t(`线下支付已关闭,请点击确认按钮返回主页`),
  127. showCancel: false,
  128. success() {
  129. uni.switchTab({
  130. url: '/pages/index/index'
  131. })
  132. }
  133. });
  134. }
  135. if (site_name) {
  136. uni.setNavigationBarTitle({
  137. title: site_name
  138. });
  139. }
  140. })
  141. .catch(err => {
  142. uni.showToast({
  143. title: err,
  144. icon: 'none'
  145. });
  146. });
  147. },
  148. checkForm(e) {
  149. const {
  150. money,
  151. method
  152. } = e.detail.value;
  153. if (money) {
  154. this.combData(method);
  155. } else {
  156. uni.showToast({
  157. title: this.$t(`请输入支付金额`),
  158. icon: 'none'
  159. });
  160. }
  161. },
  162. // 优惠价
  163. checkPrice() {
  164. offlineCheckPrice({
  165. pay_price: this.money
  166. })
  167. .then(res => {
  168. this.payPrice = res.data.pay_price;
  169. })
  170. .catch(err => {
  171. uni.showToast({
  172. title: err,
  173. icon: 'none'
  174. });
  175. });
  176. },
  177. // 组合数据
  178. combData(payType) {
  179. let data = {
  180. type: 3,
  181. pay_type: payType,
  182. from: 'weixinh5',
  183. price: this.payPrice || this.money,
  184. money: this.money
  185. };
  186. // #ifdef H5
  187. if (this.isWeixin) {
  188. data.from = 'weixin';
  189. }
  190. // #endif
  191. // #ifdef MP
  192. data.from = 'routine';
  193. // #endif
  194. if (this.paying) {
  195. return;
  196. }
  197. this.paying = true;
  198. uni.showLoading({
  199. title: this.$t(`正在确认`)
  200. });
  201. offlineCreate(data)
  202. .then(res => {
  203. uni.hideLoading();
  204. this.callPay(res);
  205. })
  206. .catch(err => {
  207. this.paying = false;
  208. uni.showToast({
  209. title: err,
  210. icon: 'none'
  211. });
  212. });
  213. },
  214. // 调用支付
  215. callPay(res) {
  216. const {
  217. status,
  218. result
  219. } = res.data, {
  220. orderId,
  221. jsConfig
  222. } = result,
  223. goPages = '/pages/annex/offline_result/index?site_name=' + this.site_name;
  224. switch (status) {
  225. case 'ORDER_EXIST':
  226. case 'EXTEND_ORDER':
  227. case 'PAY_ERROR':
  228. this.paying = false;
  229. this.$util.Tips({
  230. title: res.msg
  231. }, {
  232. tab: 5,
  233. url: goPages
  234. });
  235. break;
  236. case 'SUCCESS':
  237. this.paying = false;
  238. this.money = '';
  239. this.$util.Tips({
  240. title: res.msg,
  241. icon: 'success'
  242. }, {
  243. tab: 5,
  244. url: goPages
  245. });
  246. break;
  247. case 'WECHAT_PAY':
  248. // #ifdef MP
  249. let that = this;
  250. let mp_pay_name=''
  251. if(uni.requestOrderPayment){
  252. mp_pay_name='requestOrderPayment'
  253. }else{
  254. mp_pay_name='requestPayment'
  255. }
  256. uni[mp_pay_name]({
  257. timeStamp: jsConfig.timestamp,
  258. nonceStr: jsConfig.nonceStr,
  259. package: jsConfig.package,
  260. signType: jsConfig.signType,
  261. paySign: jsConfig.paySign,
  262. success: function(res) {
  263. that.$util.Tips({
  264. title: that.$t(`支付成功`),
  265. icon: 'success'
  266. }, {
  267. tab: 5,
  268. url: '/pages/annex/offline_result/index'
  269. });
  270. },
  271. fail: function() {
  272. uni.showToast({
  273. title: that.$t(`取消支付`),
  274. icon: 'none',
  275. success: function() {
  276. that.paying = false;
  277. }
  278. });
  279. },
  280. complete: function() {
  281. that.paying = false;
  282. uni.hideLoading();
  283. }
  284. });
  285. // #endif
  286. // #ifndef MP
  287. this.$wechat
  288. .pay(result.jsConfig)
  289. .then(res => {
  290. this.paying = false;
  291. this.$util.Tips({
  292. title: this.$t(`支付成功`),
  293. icon: 'success'
  294. }, {
  295. tab: 5,
  296. url: '/pages/annex/offline_result/index'
  297. });
  298. })
  299. .catch(err => {
  300. this.paying = false;
  301. if (err.errMsg == 'chooseWXPay:cancel') {
  302. uni.showToast({
  303. title: this.$t(`取消支付`),
  304. icon: 'none'
  305. });
  306. }
  307. });
  308. // #endif
  309. break;
  310. case 'PAY_DEFICIENCY':
  311. this.paying = false;
  312. this.$util.Tips({
  313. title: res.msg
  314. });
  315. break;
  316. case 'WECHAT_H5_PAY':
  317. this.paying = false;
  318. uni.showToast({
  319. title: res.msg,
  320. success() {
  321. location.href = jsConfig.mweb_url;
  322. }
  323. });
  324. break;
  325. case 'ALIPAY_PAY':
  326. this.paying = false;
  327. // #ifdef H5
  328. if (this.$wechat.isWeixin()) {
  329. uni.navigateTo({
  330. url: `/pages/users/alipay_invoke/index?id=${orderId}&link=${jsConfig.qrCode}`
  331. });
  332. } else {
  333. this.alipayHtml = jsConfig;
  334. this.$nextTick(() => {
  335. document.getElementById('alipaysubmit').submit();
  336. });
  337. }
  338. // #endif
  339. // #ifdef MP
  340. uni.navigateTo({
  341. url: `/pages/users/alipay_invoke/index?id=${orderId}&link=${jsConfig.qrCode}`
  342. });
  343. // #endif
  344. break;
  345. }
  346. }
  347. }
  348. };
  349. </script>
  350. <style>
  351. page {
  352. background-color: #ffffff;
  353. }
  354. </style>
  355. <style lang="scss" scoped>
  356. /deep/uni-radio .uni-radio-input.uni-radio-input-checked {
  357. border: 1px solid #FDC383 !important;
  358. background-color: #FDC383 !important;
  359. }
  360. .input-section {
  361. .section-hd {
  362. padding: 30rpx;
  363. font-size: 28rpx;
  364. color: #666666;
  365. }
  366. .section-bd {
  367. padding-right: 30rpx;
  368. padding-left: 30rpx;
  369. }
  370. .input-group {
  371. display: flex;
  372. align-items: flex-end;
  373. padding: 45rpx 20rpx 47rpx;
  374. font-size: 80rpx;
  375. color: #999999;
  376. }
  377. .input {
  378. flex: 1;
  379. height: 110rpx;
  380. margin-left: 15rpx;
  381. font-size: 100rpx;
  382. color: #282828;
  383. }
  384. .discount {
  385. padding: 27rpx 20rpx;
  386. border-top: 1rpx solid #eeeeee;
  387. font-size: 28rpx;
  388. color: #e93323;
  389. }
  390. }
  391. .radio-section {
  392. border-top: 20rpx solid #f5f5f5;
  393. .section-hd {
  394. padding: 30rpx;
  395. font-size: 28rpx;
  396. color: #666666;
  397. }
  398. .section-bd {
  399. padding-left: 50rpx;
  400. }
  401. .item {
  402. display: flex;
  403. align-items: center;
  404. padding-top: 30rpx;
  405. padding-right: 30rpx;
  406. padding-bottom: 30rpx;
  407. border-bottom: 1rpx solid #f5f5f5;
  408. }
  409. .iconfont {
  410. font-size: 44rpx;
  411. }
  412. .icon-yue {
  413. color: #fe960f;
  414. }
  415. .icon-weixinzhifu {
  416. color: #41b035;
  417. }
  418. .icon-zhifubao {
  419. color: #099bdf;
  420. }
  421. .name {
  422. flex: 1;
  423. margin-left: 30rpx;
  424. font-size: 30rpx;
  425. color: #333333;
  426. }
  427. .money {
  428. float: right;
  429. padding-right: 20rpx;
  430. font-size: 20rpx;
  431. }
  432. }
  433. .button {
  434. height: 86rpx;
  435. border-radius: 43rpx;
  436. margin: 114rpx 30rpx 30rpx;
  437. background: linear-gradient(90deg, #FEE2B7 0%, #FDC383 100%);
  438. font-size: 30rpx;
  439. line-height: 86rpx;
  440. color: #5D3324;
  441. }
  442. .alipay {
  443. display: none;
  444. }
  445. </style>