前端H5微信支付宝支付实现方法(uniapp为例)

 更新时间:2024年04月01日 11:26:14   作者:燕燕的猪  
最近上线一个项目,手机网站进行调起支付宝App支付,做起来还是满顺手的,在此做个记录,这篇文章主要给大家介绍了关于前端H5微信支付宝支付实现方法的相关资料,需要的朋友可以参考下
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

以uniapp项目为例

支付宝的

首先是一个支付类型选择页面,在选择支付宝支付后,跳转到一个空白页,用于支付宝支付的中转页面。

在点击立即支付之后,直接跳转至自行设置好的空白页就好,并把你生成订单所需要的数据一并带过去。

uni.navigateTo({
  url: `/pages/alipaying/alipay?orderNum=${this.orderNum}&price=${this.orderinfo.price}`,
});

接下来就是重点操作了(其实非常简单),在跳转这个页面之后直接调用后端接口,将数据传给后端生成订单,并且还需要两个地址,一个是取消支付的回调地址(quitUrl),一个是支付完之后的回调地址(returnUrl),后端会根据这些信息生成一个form表单返回给你,我们只需要将这个表单渲染到这个空白页面,并进行提交表单的操作,就会跳转至支付宝的页面去进行支付操作,在取消或支付完成会跳转至之前传给后端规定好的地址,在那里去调后端接口查询支付结果或者重新支付等其他操作就行了。
空白页面内容如下:

<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){//这是弄了个变量来判断是否已经进入了支付,防止用户直接左滑的返回一直在触发支付
			this.alipay()
		}
	},
	data() {
	  return {
		orderNum:"",
		price:"",
		html:"",
		couponId:""
	  };
	},
	methods: {
		async alipay(){
			//这里是获取支付宝返回的表单用的接口,那两个地址我在api里面写了默认值,所以这个地方就不传了,接口详细内容就和普通调接口一样,该传传,该接接
			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>

不过这里的实现支持正常浏览器h5,但是微信内置浏览器的话会提示在浏览器打开该链接,非常的繁琐,查了一下午解决方案,就是没有解决方案(可能也是我没找到),索性直接微信内置浏览器的话就不显示支付宝支付,反正需求很奇怪,都打开微信了还要支付宝干啥,对吧

微信的

微信支付比支付宝支付稍微麻烦一点点,但也不难,对于前端来说调接口就完事了。

微信的话有两种,一种外部浏览器,一种微信内置浏览器

所以首先需要一个方法来判断浏览器的类型

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;
		}
	},

是微信内置就返回true,不是返回false

先说外部浏览器的情况,在立即支付之后去判断一下浏览器是什么,返回false就走下面代码,调接口就完了

//把生成订单需要的数据传给后端,然后后端会返回一个地址
const res2 = await api.weixinPayOrder(this.orderNum,this.couponId);
const name = "_self";
//因为微信支付没有传什么回调地址,在支付完之后还会回到这个页面,所以防止重复触发这里设置一个值来控制
utils.setStorage("weixinCallback", 1);
//打开后端返回的地址,就可以去微信支付了
window.open(res2.mweb_url, name);

在页面展示的时候onShow钩子,判断weixinCallback这个值来执行不同的操作,如果是1说明已经发起过支付,那么需要一个执行另一个方法来调用查询支付结果的接口来跳转页面,下面是查询结果的方法

getOrderDetail() {
      clearTimeout(this.timer);
      this.timer = setTimeout(() => {
        let initTime = +new Date();
        let loop = () => {
        //这里调用后端给的查询支付结果的接口,
          api.weixinPayOrderBack(this.orderNum).then((res) => {
            if (res.status == 1) {
            //支付成功跳转
              uni.navigateTo({
                url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=yes`,
              });
            } else {
              let now = +new Date();
              if (now - initTime < 45000) {
                loop();
              } else {
              //支付失败跳转
                uni.navigateTo({
                  url: `/pages/payWeixinSuccess/payWeixinSuccess?orderNum=${this.orderNum}&result=no`,
                });
              }
            }
          });
        };
        loop();
      }, 2000);
    },

这样其实就完成了外部浏览器的微信支付前端部分,可能还有不足或bug,至少现在我没发现。

接下来是微信内置浏览器支付,首先需要一个(jweixin-module),下载这个依赖npm、yarn都行

在调用isWeiXin()为true时,表示是微信内置浏览器

首先,调用微信官方提供的链接获取用户授权信息code

//应用的appid
const appId = "wxxxxxxxxxxxx";
//重定向的地址,重定向回当前页面
const local = window.location.href
//打开微信官方提供的链接,传入appid和回调地址,在用户确认授权后会在回调地址后面拼接上code
window.location.href = 'https://open.weixin.qq.com/connect/oauth2/authorize?appid='+appId+'&redirect_uri=' + encodeURIComponent(local) + '&response_type=code&scope=snsapi_base&state=1#wechat_redirect';

在授权后回到这个页面看看地址上有没有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,有则准备进行支付操作,没有则去到让用户授权

const code = this.getUrlParam("code")
if(code==null || code == ""){
	//这里走让用户授权的操作
}else{
	//这里走支付的操作,下面详写
}

在else中,使用前面下载的依赖(jweixin-module)来走支付流程

//这里是将code传给后端来获取openId
const res2 = await api.sendCode(code);
//然后将后端给的openId和订单信息继续给后端,在后端返回支付所需的一些参数在依赖中需要配置
//这两个接口后端可以合并成一个的吧,我不太理解为啥分成两个,繁琐
const res3 = await api.weixinPayOrder(this.orderNum,res2.data);
//下面就是使用这个依赖包了,具体配置什么意思网上复制来的,一查一大堆
jweixin.config({//初始化配置
	debug: false, // 是否开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
	appId: appId, // 必填,公众号的唯一标识
	timestamp: res3.timeStamp, // 必填,生成签名的时间戳
	nonceStr: res3.nonceStr, // 必填,生成签名的随机串
	signature:res3.paySign, // 必填,签名,
	jsApiList: ['chooseWXPay'] // 必填,需要使用的JS接口列表
});
jweixin.ready(function() {
	jweixin.chooseWXPay({//下面这些参数都是在上面接口返回给我的
		timestamp: res3.timeStamp, // 支付签名时间戳,注意微信jssdk中的所有使用timestamp字段均为小写。但最新版的支付后台生成签名使用的timeStamp字段名需大写其中的S字符
		nonceStr: res3.nonceStr, // 支付签名随机串,不长于 32 位
		package: res3.package, // 统一支付接口返回的prepay_id参数值,提交格式如:prepay_id=***)
		signType: res3.signType, // 签名方式,默认为'SHA1',使用新版支付需传入'MD5'
		paySign: res3.paySign, // 支付签名
		success: function(res) {
			// 支付成功后的回调函数
			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的微信支付也完成了,其实大部分逻辑在后端,前端只管调接口就完事了,不明白接口什么意思可以去问问后端就行,是不是非常简单,如果看官方文档那样详细的流程图其实很多流程在后端走了,想要更多的学习的话那就去掌握后端知识咯

总结

到此这篇关于前端H5微信支付宝支付实现方法的文章就介绍到这了,更多相关前端H5微信支付宝支付内容请搜索程序员之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员之家!

相关文章

  • 为什么要使用Vuex的介绍

    为什么要使用Vuex的介绍

    今天小编就为大家分享一篇关于为什么要使用Vuex的介绍,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • vue中关于v-for循环key值问题的研究

    vue中关于v-for循环key值问题的研究

    这篇文章主要介绍了vue中关于v-for循环key值问题的研究,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • vue常用的数字孪生可视化的自适应方案

    vue常用的数字孪生可视化的自适应方案

    这篇文章主要为大家介绍了vue常用的数字孪生可视化的自适应方案详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • Vue 防止短时间内连续点击后多次触发请求的操作

    Vue 防止短时间内连续点击后多次触发请求的操作

    这篇文章主要介绍了Vue 防止短时间内连续点击后多次触发请求的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-11-11
  • vue实现通讯录功能

    vue实现通讯录功能

    这篇文章主要为大家详细介绍了vue实现通讯录功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • vue两个组件间值的传递或修改方式

    vue两个组件间值的传递或修改方式

    这篇文章主要介绍了vue两个组件间值的传递或修改的实现代码,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-07-07
  • 如何使用elementUI组件实现表格的分页及搜索功能

    如何使用elementUI组件实现表格的分页及搜索功能

    最近在使用element-ui的表格组件时,遇到了搜索框功能的实现问题,这篇文章主要给大家介绍了关于如何使用elementUI组件实现表格的分页及搜索功能的相关资料,需要的朋友可以参考下
    2023-03-03
  • 关于Vue中过滤器的必懂小知识

    关于Vue中过滤器的必懂小知识

    vue过滤器可以在不改变原始数据,只是对数据进行加工处理后返回过滤后的数据再进行调用处理,下面这篇文章主要给大家介绍了关于Vue中过滤器必懂小知识的相关资料,需要的朋友可以参考下
    2021-10-10
  • 深入探索Vue3.x中的七大高级用法

    深入探索Vue3.x中的七大高级用法

    Vue3.x 自发布以来,凭借其性能的显著提升和更加灵活的组合式 API,已经成为了现代前端开发的重要工具之一,除了基本用法外,Vue3.x 还提供了许多高级功能,本文将和大家一起深入探索Vue3.x中的七大高级用法,需要的朋友可以参考下
    2024-03-03
  • vue-router解决相同路径跳转报错的问题

    vue-router解决相同路径跳转报错的问题

    这篇文章主要介绍了vue-router解决相同路径跳转报错的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-04-04

最新评论

?


http://www.vxiaotou.com