前端H5微信支付寶支付實現(xiàn)方法(uniapp為例)
以uniapp項目為例
支付寶的
首先是一個支付類型選擇頁面,在選擇支付寶支付后,跳轉(zhuǎn)到一個空白頁,用于支付寶支付的中轉(zhuǎn)頁面。
在點擊立即支付之后,直接跳轉(zhuǎn)至自行設置好的空白頁就好,并把你生成訂單所需要的數(shù)據(jù)一并帶過去。
uni.navigateTo({ url: `/pages/alipaying/alipay?orderNum=${this.orderNum}&price=${this.orderinfo.price}`, });
接下來就是重點操作了(其實非常簡單),在跳轉(zhuǎn)這個頁面之后直接調(diào)用后端接口,將數(shù)據(jù)傳給后端生成訂單,并且還需要兩個地址,一個是取消支付的回調(diào)地址(quitUrl),一個是支付完之后的回調(diào)地址(returnUrl),后端會根據(jù)這些信息生成一個form表單返回給你,我們只需要將這個表單渲染到這個空白頁面,并進行提交表單的操作,就會跳轉(zhuǎn)至支付寶的頁面去進行支付操作,在取消或支付完成會跳轉(zhuǎn)至之前傳給后端規(guī)定好的地址,在那里去調(diào)后端接口查詢支付結(jié)果或者重新支付等其他操作就行了。
空白頁面內(nèi)容如下:
<template> <!-- #ifdef H5 --> <div v-html="html"> </div> <!-- #endif --> </template> <script> import api from "../../utils/api"; //這個utils是對一些東西做了處理,其實沒必要,utils.getStorage直接用uni.getStorage一樣的 import utils from "../../utils"; export default { onLoad(options) { this.orderNum=options.orderNum this.price=options.price this.couponId = options.couponId }, mounted(){ const result = utils.getStorage("weixinCallback"); if(result!=3){//這是弄了個變量來判斷是否已經(jīng)進入了支付,防止用戶直接左滑的返回一直在觸發(fā)支付 this.alipay() } }, data() { return { orderNum:"", price:"", html:"", couponId:"" }; }, methods: { async alipay(){ //這里是獲取支付寶返回的表單用的接口,那兩個地址我在api里面寫了默認值,所以這個地方就不傳了,接口詳細內(nèi)容就和普通調(diào)接口一樣,該傳傳,該接接 const res = await api.alipayOrder(this.orderNum,this.price,this.couponId) //將表單渲染進頁面 document.querySelector('body').innerHTML = res; //在渲染完立即提交表單,就會進入支付寶支付的界面 this.$nextTick(()=>{ utils.setStorage("weixinCallback", 3); window.document.forms[0].submit() }) } }, }; </script>
不過這里的實現(xiàn)支持正常瀏覽器h5,但是微信內(nèi)置瀏覽器的話會提示在瀏覽器打開該鏈接,非常的繁瑣,查了一下午解決方案,就是沒有解決方案(可能也是我沒找到),索性直接微信內(nèi)置瀏覽器的話就不顯示支付寶支付,反正需求很奇怪,都打開微信了還要支付寶干啥,對吧
微信的
微信支付比支付寶支付稍微麻煩一點點,但也不難,對于前端來說調(diào)接口就完事了。
微信的話有兩種,一種外部瀏覽器,一種微信內(nèi)置瀏覽器
所以首先需要一個方法來判斷瀏覽器的類型
isWeiXin(){ let ua = window.navigator.userAgent.toLowerCase(); if (ua.match(/MicroMessenger/i) == 'micromessenger') { this.payway.forEach(item=>{ if(item.value==1){ item.flag = false } }) return true; } else { return false; } },
是微信內(nèi)置就返回true,不是返回false
先說外部瀏覽器的情況,在立即支付之后去判斷一下瀏覽器是什么,返回false就走下面代碼,調(diào)接口就完了
//把生成訂單需要的數(shù)據(jù)傳給后端,然后后端會返回一個地址 const res2 = await api.weixinPayOrder(this.orderNum,this.couponId); const name = "_self"; //因為微信支付沒有傳什么回調(diào)地址,在支付完之后還會回到這個頁面,所以防止重復觸發(fā)這里設置一個值來控制 utils.setStorage("weixinCallback", 1); //打開后端返回的地址,就可以去微信支付了 window.open(res2.mweb_url, name);
在頁面展示的時候onShow鉤子,判斷weixinCallback這個值來執(zhí)行不同的操作,如果是1說明已經(jīng)發(fā)起過支付,那么需要一個執(zhí)行另一個方法來調(diào)用查詢支付結(jié)果的接口來跳轉(zhuǎn)頁面,下面是查詢結(jié)果的方法
getOrderDetail() { clearTimeout(this.timer); this.timer = setTimeout(() => { let initTime = +new Date(); let loop = () => { //這里調(diào)用后端給的查詢支付結(jié)果的接口, api.weixinPayOrderBack(this.orderNum).then((res) => { if (res.status == 1) { //支付成功跳轉(zhuǎn) uni.navigateTo({ url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=yes`, }); } else { let now = +new Date(); if (now - initTime < 45000) { loop(); } else { //支付失敗跳轉(zhuǎn) uni.navigateTo({ url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=no`, }); } } }); }; loop(); }, 2000); },
這樣其實就完成了外部瀏覽器的微信支付前端部分,可能還有不足或bug,至少現(xiàn)在我沒發(fā)現(xiàn)。
接下來是微信內(nèi)置瀏覽器支付,首先需要一個(jweixin-module),下載這個依賴npm、yarn都行
在調(diào)用isWeiXin()為true時,表示是微信內(nèi)置瀏覽器
首先,調(diào)用微信官方提供的鏈接獲取用戶授權(quán)信息code
//應用的appid const appId = "wxxxxxxxxxxxx"; //重定向的地址,重定向回當前頁面 const local = window.location.href //打開微信官方提供的鏈接,傳入appid和回調(diào)地址,在用戶確認授權(quán)后會在回調(diào)地址后面拼接上code window.location. + encodeURIComponent(local) + '&response_type=code&scope=snsapi_base&state=1#wechat_redirect';
在授權(quán)后回到這個頁面看看地址上有沒有code,有則進行下一步
寫一個方法取出地址里的code
getUrlParam(name) { var reg = new RegExp('(^|&)' + name + '=([^&]*)(&|$)') let url = window.location.href let search = url.split('?')[1] if (search) { var r = search.substr(0).match(reg) if (r !== null)return unescape(r[2]) return null } else return null },
判斷當前地址有沒有code,有則準備進行支付操作,沒有則去到讓用戶授權(quán)
const code = this.getUrlParam("code") if(code==null || code == ""){ //這里走讓用戶授權(quán)的操作 }else{ //這里走支付的操作,下面詳寫 }
在else中,使用前面下載的依賴(jweixin-module)來走支付流程
//這里是將code傳給后端來獲取openId const res2 = await api.sendCode(code); //然后將后端給的openId和訂單信息繼續(xù)給后端,在后端返回支付所需的一些參數(shù)在依賴中需要配置 //這兩個接口后端可以合并成一個的吧,我不太理解為啥分成兩個,繁瑣 const res3 = await api.weixinPayOrder(this.orderNum,res2.data); //下面就是使用這個依賴包了,具體配置什么意思網(wǎng)上復制來的,一查一大堆 jweixin.config({//初始化配置 debug: false, // 是否開啟調(diào)試模式,調(diào)用的所有api的返回值會在客戶端alert出來,若要查看傳入的參數(shù),可以在pc端打開,參數(shù)信息會通過log打出,僅在pc端時才會打印。 appId: appId, // 必填,公眾號的唯一標識 timestamp: res3.timeStamp, // 必填,生成簽名的時間戳 nonceStr: res3.nonceStr, // 必填,生成簽名的隨機串 signature:res3.paySign, // 必填,簽名, jsApiList: ['chooseWXPay'] // 必填,需要使用的JS接口列表 }); jweixin.ready(function() { jweixin.chooseWXPay({//下面這些參數(shù)都是在上面接口返回給我的 timestamp: res3.timeStamp, // 支付簽名時間戳,注意微信jssdk中的所有使用timestamp字段均為小寫。但最新版的支付后臺生成簽名使用的timeStamp字段名需大寫其中的S字符 nonceStr: res3.nonceStr, // 支付簽名隨機串,不長于 32 位 package: res3.package, // 統(tǒng)一支付接口返回的prepay_id參數(shù)值,提交格式如:prepay_id=***) signType: res3.signType, // 簽名方式,默認為'SHA1',使用新版支付需傳入'MD5' paySign: res3.paySign, // 支付簽名 success: function(res) { // 支付成功后的回調(diào)函數(shù) uni.showToast({ icon: 'none', title: '支付成功', duration: 4000 }); uni.navigateTo({ url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=yes`, }); }, cancel: function(r) {}, fail: function(res) { console.log('payfail') } }); }); jweixin.error(function(res) { uni.showToast({ icon: 'none', title: '支付失敗了', duration: 4000 }); uni.navigateTo({ url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=no`, }); });
走到這里h5的微信支付也完成了,其實大部分邏輯在后端,前端只管調(diào)接口就完事了,不明白接口什么意思可以去問問后端就行,是不是非常簡單,如果看官方文檔那樣詳細的流程圖其實很多流程在后端走了,想要更多的學習的話那就去掌握后端知識咯
總結(jié)
到此這篇關于前端H5微信支付寶支付實現(xiàn)方法的文章就介紹到這了,更多相關前端H5微信支付寶支付內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
vue中for循環(huán)更改數(shù)據(jù)的實例代碼(數(shù)據(jù)變化但頁面數(shù)據(jù)未變)
這篇文章主要介紹了vue中for循環(huán)更改數(shù)據(jù)的實例代碼(數(shù)據(jù)變化但頁面數(shù)據(jù)未變)的相關資料,需要的朋友可以參考下2017-09-09