由于微信小程序支付方式的改版,之前借助官方的云函数实现支付的模式,新注册的小程序无法使用了,所以今天就再给大家讲一个简单的,几行代码既可以实现小程序支付功能。
老规矩,先看本节效果图
我们实现这个支付功能完全是借助小程序云开发实现的,不用搭建自己的服务器,不用买域名,不用备案域名,不用支持https。只需要一个简单的云函数,就可以轻松的实现微信小程序支付功能。
核心代码就下面这些
一,创建一个云开发小程序
关于如何创建云开发小程序,这里我就不再做具体讲解。不知道怎么创建云开发小程序的同学,可以去翻看我之前的文章,或者看下我录制的视频:https://edu.csdn.net/course/play/9604/204528
创建云开发小程序有几点注意的
1,一定不要忘记在app.js里初始化云开发环境。
2,创建完云函数后,一定要记得上传
二, 创建支付的云函数
1,创建云函数pay
三,引入三方依赖tenpay
我们这里引入三方依赖的目的,是创建我们支付时需要的一些参数。我们安装依赖是使用里npm 而npm必须安装node,关于如何安装node,我这里不做讲解,百度一下,网上一大堆。
1,首先右键pay,然后选择在终端中打开
2,我们使用npm来安装tenpay依赖。
在命令行里执行 npm i tenpay
安装完成后,我们的pay云函数会多出一个package.json 文件
并且可以看到我们安装了tenpay这个三方类库
到这里我们的tenpay依赖就安装好了。
四,编写云函数pay
完整代码如下
//云开发实现支付
const cloud = require('wx-server-sdk')
cloud.init()
//1,引入支付的三方依赖
const tenpay = require('tenpay');
//2,配置支付信息
const config = {
appid: '你的小程序appid',
mchid: '你的微信商户号',
partnerKey: '微信支付安全密钥',
notify_url: '支付回调网址,这里可以先随意填一个网址',
spbill_create_ip: '127.0.0.1' //这里填这个就可以
};
exports.main = async(event, context) => {
const wxContext = cloud.getWXContext()
let {
orderid,
money
} = event;
//3,初始化支付
const api = tenpay.init(config);
let result = await api.getPayParams({
out_trade_no: orderid,
body: '商品简单描述',
total_fee: money, //订单金额(分),
openid: wxContext.OPENID //付款用户的openid
});
return result;
}
注意把appid,mchid,partnerKey换成你自己的。
这里再强调下 partnerKey 我们这里使用apiv2的密匙,而不是apiv3的密匙。APIv2是一串32位的数字和字母的组合。自己随意设置即可。设置好一定要记牢。
到这里我们获取小程序支付所需参数的云函数代码就编写完成了。
不要忘记上传这个云函数。
出现下图就代表上传成功
五,写一个简单的页面,用来提交订单,调用pay云函数。
这个页面很简单,
1,自己随便编写一个订单号(这个订单号要大于6位)
2,自己随便填写一个订单价(单位是分)
3,点击按钮,调用pay云函数。获取支付所需参数。
下图是官方支付api所需要的一些必须参数。
下图是我们调用pay云函数获取的参数,和上图所需要的是不是一样。
![
六,调用wx.requestPayment实现支付
下图是官方的示例代码
这里不在做具体讲解了,把完整代码给大家贴出来
// pages/pay/pay.js
Page({
//提交订单
formSubmit: function(e) {
let that = this;
let formData = e.detail.value
console.log('form发生了submit事件,携带数据为:', formData)
wx.cloud.callFunction({
name: "pay",
data: {
orderid: "" + formData.orderid,
money: formData.money
},
success(res) {
console.log("提交成功", res.result)
that.pay(res.result)
},
fail(res) {
console.log("提交失败", res)
}
})
},
//实现小程序支付
pay(payData) {
//官方标准的支付方法
wx.requestPayment({
timeStamp: payData.timeStamp,
nonceStr: payData.nonceStr,
package: payData.package, //统一下单接口返回的 prepay_id 格式如:prepay_id=***
signType: 'MD5',
paySign: payData.paySign, //签名
success(res) {
console.log("支付成功", res)
},
fail(res) {
console.log("支付失败", res)
},
complete(res) {
console.log("支付完成", res)
}
})
}
})
到这里,云开发实现小程序支付的功能就完整实现了。
七,实现效果
1,调起支付键盘
2,支付完成
3,log日志,可以看出不同支付状态的回调
上图是支付成功的回调,我们可以在支付成功回调时,改变订单支付状态。
下图是支付失败的回调,
下图是支付完成的状态。
到这里我们就轻松的实现了微信小程序的支付功能了。是不是很简单啊。
如果感觉图文不是很好理解,我后面会录制视频讲解。
八,(选看)支付成功回调
如果有你遇到下面的问题再来学习这个,没有遇到的话忽略即可。
问题:有的用户支付后如果不点击完成,在wx.requestPayment支付后就无法监听到支付成功的状态。
如果遇到这个问题再通过下面这个方法解决
就是在我们前面写云函数里有一个回调地址。
通常这个地址是一个url,但是我们没有自己的服务器,所以还是要借助云函数来实现的。其实云函数有一个http调用的。我们只需要给对应的云函数开通http调用就行。
1,http调用云函数
需要我们登录自己的腾讯云,搜索云开发
进入后,我们需要切换回旧版云开发控制台,因为新版找不到设置http调用的地方了。
然后点击进入我们的云开发环境
点击访问服务,新建一个http访问服务。
域名使用自带的,关联资源选择云函数。
然后关联我们支付成功后需要回调的云函数,我这里随便创建一个test22云函数。
触发路径填写 /云函数名就行,添加好需要等几分钟。
下面就是我们回调云函数对应的http调用路径了,点击下就可以看到完整的路径了。
我们去复制下,粘贴到我们的云函数里就行
这样在支付成功后,不管用户点不点完成,都会回调这个云函数。我们在这个回调云函数里做我们的订单处理就行。比如把订单从待支付状态改为已支付状态。
2,编写支付成功回调云函数
代码如下
// 云函数入口文件
const cloud = require('wx-server-sdk')
cloud.init({
env: cloud.DYNAMIC_CURRENT_ENV
})
// 云函数入口函数
exports.main = async (event, context) => {
cloud.database().collection('test').add({
data: event
})
//订单号 event.outTradeNo
return await cloud.database().collection('order').doc(event.outTradeNo).update({
data: {
status: 1
}
}).then(res => {
return {
errcode: 0,
errmsg: '支付成功'
}
}).catch(res => {
return res
})
}
3,创建订单表
我们正规的支付,应该是先创建一个订单表,支付前先生成订单数据,然后用订单数据的_id作为支付的outTradeNo就行
我们借助这个订单的id就可以去调用支付的云函数了,调用成功后返回我们支付所需要的参数
然后我们调用支付,支付成功后就会给我们设置的云函数test22返回如下信息
我们在支付成功后要改变订单状态为已支付。需要用到订单号。而这个订单号在支付成功后返回给我们了,就存在这个body字段里,不过进行了加密,而我们要做的就是解析这个body拿到订单id。
可以看到成功的从body里解析出了订单id
有了这个订单id。我们就可以在test22云函数里修改订单状态为已支付了。
4,项目代码
首先支付的云函数pay的代码如下
记得先通过 npm i tenpay 安装依赖包
// 云函数入口文件
const cloud = require('wx-server-sdk')
cloud.init({
env: cloud.DYNAMIC_CURRENT_ENV
}) // 使用当前云环境
//1,引入依赖
const tenpay = require('tenpay');
//2,配置参数
const config = {
appid: '小程序的appid',
mchid: '商户号',
partnerKey: 'APIv2密匙', //apiv2密匙
notify_url: '你的回调地址',
spbill_create_ip: '127.0.0.1'
};
exports.main = async (event, context) => {
const wxContext = cloud.getWXContext()
//3,初始化
const api = tenpay.init(config);
//4,获取支付参数
let result = await api.getPayParams({
out_trade_no: event.orderid,
body: '商品简单描述',
total_fee: event.money,
openid: wxContext.OPENID
});
return result
}
其次是回调云函数test22的代码如下
同样要先通过npm i xml2js安装解析xml的依赖包。
// paymentCallback/index.js
const cloud = require('wx-server-sdk');
const xml2js = require('xml2js');
const Buffer = require('buffer/').Buffer;
cloud.init({
env: cloud.DYNAMIC_CURRENT_ENV
});
exports.main = async (event, context) => {
// 解码 Base64 字符串
const body = event.body;
const decodedXml = Buffer.from(body, 'base64').toString('utf8');
// 解析 XML 数据
const parser = new xml2js.Parser();
let parsedData;
try {
parsedData = await parser.parseStringPromise(decodedXml);
} catch (error) {
console.error('解析 XML 失败', error);
return buildXmlResponse('FAIL', '解析 XML 失败');
}
const result = parsedData.xml;
// 检查返回码和结果码
if (result.return_code[0] === 'SUCCESS' && result.result_code[0] === 'SUCCESS') {
// 支付成功,处理业务逻辑
const outTradeNo = result.out_trade_no[0];
console.log('支付成功,订单号:', outTradeNo);
// 例如,更新订单状态
await updateOrderStatus(outTradeNo);
return buildXmlResponse('SUCCESS', 'OK');
} else {
// 支付失败或其他情况
console.log('支付失败', result);
return buildXmlResponse('FAIL', '支付失败');
}
};
async function updateOrderStatus(orderId) {
// 假设你使用云数据库来存储订单信息
const db = cloud.database();
try {
await db.collection('aorder').doc(orderId).update({
data: {
status: '已支付'
}
});
} catch (error) {
console.error('更新订单状态失败', error);
}
}
// 需要返回给微信xml格式的,才可以避免微信一直回调
function buildXmlResponse(returnCode, returnMsg) {
const response = {
return_code: returnCode,
return_msg: returnMsg
};
return new xml2js.Builder().buildObject({
xml: response
});
}
然后就是调用pay云函数的小程序端代码如下
/**
* 作者:编程小石头
* v 2501902696
*/
Page({
tenpay() {
wx.cloud.database().collection('aorder')
.add({
data: {
name: '商品订单22',
status: '待支付',
money: 1 //这里假设商品是1分钟
}
}).then(res => {
console.log('订单添加成功', res._id)
let orderid = res._id
wx.cloud.callFunction({
name: "pay",
data: {
orderid: orderid,
money: 1
},
}).then(res => {
console.log("调用pay成功", res.result)
this.pay(res.result, orderid)
}).catch(res => {
console.log("调用pay失败", res)
})
})
},
//实现小程序支付
pay(payData, orderid) {
//官方标准的支付方法
wx.requestPayment({
timeStamp: payData.timeStamp,
nonceStr: payData.nonceStr,
package: payData.package,
signType: 'MD5',
paySign: payData.paySign, //签名
success(res) {
console.log("支付成功", res)
wx.cloud.database().collection('aorder').doc(orderid)
.update({
data: {
status: '已支付'
}
})
},
fail(res) {
console.log("支付失败", res)
},
complete(res) {
console.log("支付完成", res)
}
})
},
})
视频讲解
https://edu.csdn.net/course/detail/25701
源码地址:
https://github.com/qiushi123/xiaochengxu_demos
014云开发实现小程序支付,就是我们的源码,如果你导入源码或者学习过程中有任何问题,都可以留言或者私信我。大家一起学习,一起进步。