NoteZ_技术博客 NoteZ_技术博客
🏠 首页
  • 📚 Web技术
  • 📋 Npm笔记
  • 📑 Markdown
  • 📄 Git笔记
  • 📝 Nginx文档
  • 📓 Linux文档
  • 📖 技术文档
  • 📜 其他文档
  • 🧊 NodeJs
  • 🎡 Express
  • 🔥 Rust
  • 🎉 Koa2
  • 🍃 MongoDB
  • 🐬 MySql
  • 🥦 Oracle
  • 🍁 Python
  • 🍄 JavaScript
  • 🌰 CSS
  • 🧄 HTML
  • 🥑 Canvas
  • 🌽 Nuxt
  • 🍆 React
  • 🥜 Vue
  • 🧅 TypeScript
  • 🌶️ AI
  • 📘 分类
  • 📗 标签
  • 📙 归档
⚜️ 在线编辑 (opens new window)
  • 📁 站点收藏
  • 📦 前端组件库
  • 📊 数据可视化
  • 🌈 开源插件
  • 🎗️ 关于我
  • 🔗 友情链接
GitHub (opens new window)

NoteZ_技术博客

前端界的小学生
🏠 首页
  • 📚 Web技术
  • 📋 Npm笔记
  • 📑 Markdown
  • 📄 Git笔记
  • 📝 Nginx文档
  • 📓 Linux文档
  • 📖 技术文档
  • 📜 其他文档
  • 🧊 NodeJs
  • 🎡 Express
  • 🔥 Rust
  • 🎉 Koa2
  • 🍃 MongoDB
  • 🐬 MySql
  • 🥦 Oracle
  • 🍁 Python
  • 🍄 JavaScript
  • 🌰 CSS
  • 🧄 HTML
  • 🥑 Canvas
  • 🌽 Nuxt
  • 🍆 React
  • 🥜 Vue
  • 🧅 TypeScript
  • 🌶️ AI
  • 📘 分类
  • 📗 标签
  • 📙 归档
⚜️ 在线编辑 (opens new window)
  • 📁 站点收藏
  • 📦 前端组件库
  • 📊 数据可视化
  • 🌈 开源插件
  • 🎗️ 关于我
  • 🔗 友情链接
GitHub (opens new window)
  • Web技术

  • Git笔记

  • Linux文档

  • Markdown

  • Nginx文档

  • Npm笔记

  • 技术文档

  • 其他文档

    • 解决 bash_wget 未找到命令的解决办法
    • CentOS7 常见问题应对,如何升级make和gcc版本
    • CodeMirror代码编辑器实现自定义提示功能增强版(支持搜索、调用接口查询提示内容)
    • CSS 滚动条样式
    • d3 svg 基本图形绘制
    • d3.js中update,enter,exit的概念
    • dat.gui 基本使用方法
    • echarts 使用案例(demo)
    • jsplumb 中文基础教程
    • Linux服务器(centos7)使用LibreOffice将Word转换PDF文档出现中文乱码或方格解决方法
    • npm install 报错 npm ERR code UNABLE_TO_VERIFY_LEAF_SIGNATURE npm ERR errno UNABLE_TO_VERIFY_LEA 解决
    • Npm 清除缓存
    • Npm设置淘宝镜像
    • NPS内网穿透安装方法
    • ThreeJs 基础入门
    • unable to verify the first certificate 原因及解决方法
    • vue 安装node-sass报错解决方案(缺少python2.7支持)
    • windows下安装 stable-diffusion-webui 步骤
    • yarn的安装与使用
    • 关于微信支付 WeixinJSBridge.invoke 、 wx.chooseWXPay使用方法
      • 方式一
      • 方式二
      • 其他
      • uniapp案例
    • 内网穿透的几款工具汇总
    • 前端使用 swd-deploy 自动化部署项目到服务器
    • 常用工具集(utils.js)
    • 开源项目大杂烩
    • 微信小程序-APP生命周期与运行机制总结
    • 微信小程序踩坑之布局适配单位(rpx、px、vw、vh)
    • 服务器常用的状态码
    • 解决google浏览器翻译无法使用的问题
    • 解决使用 Gitalk 登录授权报错的问题
    • 解决在使用 stable-diffusion-webui 时,安装 gfpgan 失败的方案(windows下的操作)
    • 通过 js 进行 shapefile 文件解析渲染方法
    • 部署脚本 deploy.sh
    • Tauri打包慢或者报错问题解决方法
    • Ubuntu和Nginx搭配Certbot配置SSL证书https访问网站
    • Centos下yum无法正常使用
    • Linux 系统下通过 Let‘s Encrypt 生成免费 https 证书的步骤
    • Mongo 风格的查询对象映射到 SQL 查询的 Node.js 库 json-sql
    • CentOS7安装与卸载anaconda3基础步骤
  • 前端开发
  • 其他文档
NoteZ
2023-10-24
目录

关于微信支付 WeixinJSBridge.invoke 、 wx.chooseWXPay使用方法

# 方式一

微信支付,点击进入微信公众号官方文档【微信公众号】 (opens new window)

  • 此支付方式需要在HTML页面中引入JS文件,即`【 jweixin-1.6.0.js 】``;
  • 此JS文件链接支持http和https两种形式,引用时需要匹配当前项目的请求-响应协议(即http、https);
  • 该JS文件支持使用AMD/CMD标准模块加载方式加载。
let WeChatPay = function() {
	// 2、引入js后、获取公众号校验信息
	let timestamp = '',
		nonceStr = '',
		signature = '';
	let v = {
		// 用于换取微信校验信息的参数:要求不可以包含 “#” 号
		url: location.split('#')[0]
	};

	// 3、通过config接口注入权限验证配置(需要同步进行,在获取到校验信息后方可注入config,否则校验失败!)
	wx.config({
		debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
		appId: '', // 必填,公众号的唯一标识
		timestamp: , // 必填,生成签名的时间戳
		nonceStr: '', // 必填,生成签名的随机串
		signature: '', // 必填,签名
		jsApiList: ["checkJsApi", "chooseWXPay", "updateAppMessageShareData", "updateTimelineShareData"] // 必填,需要使用的JS接口列表
	});

	axios.post('/wx/pay/orderPay_XXXX', data).then(res => {
		// 支付成功状态
		if (res.code == 200) {
			// 获取支付必备的参数
			let {
				nonceStr,
				package,
				signType,
				paySign
			} = res.data;
			// 4、通过ready接口处理成功验证
			wx.ready(function() {
				/* 微信支付 */
				wx.chooseWXPay({
					timestamp: 0, // 支付签名时间戳,注意微信jssdk中的所有使用timestamp字段均为小写。但最新版的支付后台生成签名使用的timeStamp字段名需大写其中的S字符
					nonceStr: nonceStr, // 支付签名随机串,不长于 32 位
					package: package, // 统一支付接口返回的prepay_id参数值,提交格式如:prepay_id=\*\*\*)
					signType: signType, // 签名方式,默认为'SHA1',使用新版支付需传入'MD5'
					paySign: paySign, // 支付签名
					success: function(res) {
						// 前端判断返回方式,微信团队郑重提示:不保证绝对可靠,切记!
						if (res.errMsg == 'chooseWXPay:ok') {
							// 【支付成功】 
						} else if (res.errMsg == 'chooseWXPay:cancel') {
							// 【支付取消】:用户取消支付不会进入这个判断,而是进入complate和cancel函数
						} else {

						}
					},
					complete: function(res) {
						// 接口调用完成时执行的回调函数,无论成功或失败都会执行
						if (res.errMsg == 'chooseWXPay:ok') {
							// 【支付成功】:支付成功提示页面,点击完成按钮之后
							wx.closeWindow(); /* 关闭微信窗口,调用时需要在config中进行校验 */
						} else if (res.errMsg == 'chooseWXPay:cancel') {
							// 【支付取消】
						} else {

						}
						/**
						 * iOS和Android支付成功点击“完成”后都会进入success和complete函数,都返回'chooseWXPay:ok'
						 * (也有人说Android支付成功不进入success函数,)
						 * 原因是【iOS和Android返回数据不同。支付成功后Android返回 {"errMsg":"getBrandWCPayRequest:ok"},iOS返回{"err_Info":"success","errMsg":"chooseWXPay:ok"},故Android找不到success方法,导致失败】
						 * */
					},
					fail: function(err) {
						// 接口调用失败
					},
					cancel: function(err) {
						// 用户点击取消时的回调函数:用户取消支付后实际上进入cancel 和 complate函数
					}
				});
			});
		}
	}).catch(err => {
		console.log('支付失败:', err);
	});
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78

接口说明

不要尝试在trigger中使用ajax异步请求修改本次分享的内容,因为客户端分享操作是一个同步操作,这时候使用ajax的回包会还没有返回。

# 方式二

微信支付【 JSAPI 支付】(常用支付方式)

function onBridgeReady() {
	WeixinJSBridge.invoke(
		'getBrandWCPayRequest', {
			"appId": "appId", //公众号名称,由商户传入     
			"timeStamp": "timeStamp", //时间戳,自1970年以来的秒数     
			"nonceStr": "nonceStr", //随机串     
			"package": "package",
			"signType": "MD5", //微信签名方式:     
			"paySign": "paySign" //微信签名 
		},
		function(res) {
			// 支付成功
			if (res.err_msg == "get_brand_wcpay_request:ok") {
				// 使用以上方式判断前端返回,微信团队郑重提示:
				//res.err_msg将在用户支付成功后返回ok,但并不保证它绝对可靠。
			}
			// 支付过程中用户取消
			if (res.err_msg == "get_brand_wcpay_request:cancel") {

			}
			// 支付失败
			if (res.err_msg == "get_brand_wcpay_request:fail") {

			}
			/**
			 * 其它
			 * 1、请检查预支付会话标识prepay_id是否已失效
			 * 2、请求的appid与下单接口的appid是否一致
			 * */
			if (res.err_msg == "调用支付JSAPI缺少参数:total_fee") {

			}
		});
}
// 检测支付环境中的 WeixinJSBridge
if (typeof WeixinJSBridge == "undefined") {
	if (document.addEventListener) {
		document.addEventListener('WeixinJSBridgeReady', onBridgeReady, false);
	} else if (document.attachEvent) {
		document.attachEvent('WeixinJSBridgeReady', onBridgeReady);
		document.attachEvent('onWeixinJSBridgeReady', onBridgeReady);
	}
} else {
	onBridgeReady();
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45

SAPI调起支付注意事项

在微信浏览器里打开H5网页中执行JS调起支付; WeixinJSBridge是微信浏览器内置对象,在其他浏览器中无效; 调用支付传递的参数注意区分大小写。

# 其他

WeixinJSBridge.invoke 、 wx.chooseWXPay 的区别

  • WeixinJSBridge.invoke()出现的版本更早 无需引用jssdk 无需wx.config方法注入 需要参数appId 使用回调 有详细的说明
  • 而wx.choosewxpay()出现的版本比较晚 需要jssdk注入 不需要参数appId 使用回调 只有SUCCESS 和 FAIL没有具体的说明
  • WeixinJSBridge.invoke()是微信浏览器的内置方法 其实wx.choosewxpay()在引用的微信jssdk文件中 也调用了WeixinJSBridge.invoke() 是对WeixinJSBridge.invoke() 的再次封装

综上所诉 这是微信前后设计的不同方法的支付 还是weixinjsbridge更方便一些 有具体的失败回调

# uniapp案例

<template>
	<view class="donation_amount">
		<view class="donation_bg">
			<u-navbar title="充值中心" @leftClick="leftClick" :titleStyle="titleStyle" placeholder fixed
				bgColor="transparent"></u-navbar>
			<view class="recharge-btn"><u-button type="primary" text="立即充值"
					@click="$u.debounce(rechargePay, 500)"></u-button>
			</view>

			<u-modal width="580rpx" :show="payYejShow" @cancel="paygoconfirm" @confirm="paygoconfirm"
				:showCancelButton="true" cancelText="取消" confirmText="已完成支付">
				<view class=""
					style="display: flex;flex-direction: column;align-items: center;justify-content: center;">
					<u-loading-icon size="40"></u-loading-icon>
					<view class="" style="text-align: center;font-size: 28rpx;color: #666;margin-top: 28rpx;">
						请确认微信支付是否已完成
					</view>
				</view>
			</u-modal>
			<u-toast ref="uToast"></u-toast>
		</view>
</template>

<script>
	import {
		createRechargeOrder,
		ipGoodsPay
	} from '@/common/api.js';

	import {
		mapState
	} from 'vuex';
	export default {
		data() {
			return {
				titleStyle: {
					color: '#666'
				},
				amountList: [],
				ratingTageIndex: 0,
				value: '微信支付',
				amount: '',
				rechargeId: null,
				rechargeOrderId: '',
				wxShow: {},
				payYejShow: false,
				paySkipShow: false
			};
		},
		computed: {
			...mapState([''])
		},
		onLoad(option) {

			if (!!option.paySkipShow) {
				this.paySkipShow = option.paySkipShow;
			}
		},
		onShow() {
			this.wxShow = window.navigator.userAgent.toLowerCase();

			if (window.location.href.indexOf('#reloaded') == -1 && JSON.parse(this.paySkipShow)) {
				this.payYejShow = true;
				const url = location.href + '#reloaded';
				window.history.replaceState(null, '', url);
				// this.$router.go(0);
			} else {

			}
		},
		methods: {
			paygoconfirm() {
				this.payYejShow = false;
			},
			leftClick() {
				uni.$u.route({
					type: 'redirect',
					url: '/pages/IpModule/catch-doll'
				});
			},
			// 创建订单
			createRechargeOrderApi() {
				if (uni.$u.test.isEmpty(this.rechargeId)) {
					this.rechargeId = this.amountList[this.ratingTageIndex].id

				}
				createRechargeOrder({
						id: this.rechargeId
					}).then(res => {
						if (res.code == 0) {
							this.rechargeOrderId = res.result.orderId;
							this.toPay();
						}
					})
					.finally(() => {});
			},
			rechargePay() {
				this.createRechargeOrderApi()
			},
			toPay() {
				if (this.wxShow.search(/MicroMessenger/i) > -1) {
					this.ipGoodsPayApi('jsapi');
				} else {
					this.ipGoodsPayApi('h5');
				}
			},
			wxpay({
				appId,
				timestamp,
				nonceStr,
				signType,
				paySign,
				orderdetail
			}) {
				WeixinJSBridge.invoke(
					'getBrandWCPayRequest', {
						appId, //公众号名称,由商户传入
						timeStamp: timestamp, //时间戳,自1970年以来的秒数
						nonceStr, //随机串
						package: orderdetail,
						signType, //微信签名方式:
						paySign //微信签名
					},
					res => {
						if (res.err_msg == 'get_brand_wcpay_request:cancel') {
							// timer = null;
							this.$refs.uToast.show({
								message: '支付失败',
								type: 'error',
								duration: '2000'
							});
						}
						if (res.err_msg == 'get_brand_wcpay_request:ok') {
							// this.$u.vuex('$dollOrderId', this.dollOrderId);

						}
					}
				);
			},
			ipGoodsPayApi(payMethod) {


				uni.showLoading({
					title: '支付中...',
					mask: true
				});
				ipGoodsPay({
						orderId: this.rechargeOrderId,
						orderSource: 8,
						payMethod: payMethod,
					})
					.then(res => {
						if (res.code == 0) {
							if (this.wxShow.search(/MicroMessenger/i) > -1) {
								let wxJsApiParam = res.result.jsApi;
								this.wxpay(wxJsApiParam);
							} else {
								let tourl = '';
								const hrefUrl = window.location.href;
								hrefUrl.indexOf('test') != -1 ?
									(tourl = encodeURIComponent(
										`https://xxx.xxx.xxx.com/xxx/xxx?orderId=${this.rechargeOrderId}&paySkipShow=true`
									)) :
									(tourl = encodeURIComponent(
										`https://xxx.xxx.xxx.com/xxx/xxx?orderId=${this.rechargeOrderId}&paySkipShow=true`
									));
								this.$u.vuex('$payjump', {
									url: '/pages/secondary/vouche-center'
								});
								if (res.code == 0) {
									window.location.href = `${res.result.payUrl}&redirect_url=${tourl}`;
									// this.$u.vuex('$dollOrderId', this.rechargeOrderId);
								} else {
									this.$refs.uToast.show({
										message: res.resultMsg,
										type: 'error',
										duration: '2000'
									});
									// uni.$u.route({
									// 	type: 'redirect',
									// 	url: '/pages/secondary/exchange-record'
									// });
								}
							}
						}
					})
					.finally(() => {
						uni.hideLoading();

					});
			},

		}
	};
</script>

<style lang="scss" scoped>
	::v-deep .u-button--primary {
		background: #F75348 !important;
		border: none !important;
		border-radius: 16rpx;
	}



	.donation_bg {
		width: 100%;
		background: #FFCFCC;
		position: relative;
	}


	.donation_amount {




		.recharge-btn {
			width: 100%;
			height: 120rpx;
			position: fixed;
			bottom: 0;
			left: 0;
			box-sizing: border-box;
			padding: 0 30rpx;
		}

		.recharge-remark {
			margin: 20px;
			// margin-top: 520rpx;
			font-size: 24rpx;
			color: #999999;

			.title {
				font-size: 28rpx;
			}
		}
	}
</style>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
#Weixin#js
上次更新: 2024/01/30, 00:35:17
yarn的安装与使用
内网穿透的几款工具汇总

← yarn的安装与使用 内网穿透的几款工具汇总→

最近更新
01
Gitea数据备份与还原
03-10
02
Linux 中使用 rsync 同步文件目录教程
03-10
03
Linux 使用 rsync 互相传输同步文件的简单步骤
03-08
更多文章>
Theme by Vdoing | Copyright © 2019-2025 NoteZ,All rights reserved | 冀ICP备2021027292号-1
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式