一、什么是JSAPI支付
JSAPI支付是指商户通过调用微信支付提供的JSAPI接口,在支付场景中调起微信支付模块完成收款。
应用场景有:
线下场所:调用接口生成二维码,用户扫描二维码后在微信浏览器中打开页面后完成支付
公众号场景:用户在微信公众账号内进入商家公众号,打开某个主页面,完成支付
PC网站场景:在网站中展示二维码,用户扫描二维码后在微信浏览器中打开页面后完成支付
我们这里采用的是PC网站场景。
二、准备工作
1、申请入口:登录商户平台-->产品中心-->我的产品-->支付产品-->jsapi支付
2、浏览开发文档:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_3,这里面几乎介绍了全部流程。
3、在微信公众平台和微信商户平台拿到或配置一下参数:
appid:公众平台的appid
商户号:公众平台的商户号
appsecrut:商户平台里的签名密钥(这个要保存好,现在不支持查看,忘了就得重新配置了)
商户支付密钥Key:api_key 审核通过后,在微信发送的邮件中查看
a、微信商户平台(pay.weixin.qq.com)设置您的JSAPI支付支付目录,设置路径:商户平台-->产品中心-->开发配置
b、在公众平台设置网页授权域名(开发JSAPI支付时,在统一下单接口中要求必传用户openid,而获取openid则需要您在公众平台设置获取openid的域名,只有被设置过的域名才是一个有效的获取openid的域名,否则将获取失败)
三、下单流程:
jsapia支付和H5支付的流程:
H5支付 调用https://api.mch.weixin.qq.com/pay/unifiedorder 接口
当返回值return_code 和result_code都为SUCCESS的时候,使用其中的参数mweb_url :为拉起微信支付收银台的中间页面,可通过访问该url来拉起微信客户端,完成支付,mweb_url的有效期为5分钟。
jsapi支付:调用https://api.mch.weixin.qq.com/pay/unifiedorder 接口
(商户系统先调用该接口在微信支付服务后台生成预支付交易单,返回正确的预支付交易会话标识后再按Native、JSAPI、APP等不同场景生成交易串调起支付)
回到这里:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_7&index=6 在微信浏览器里面打开H5网页中执行JS调起支付
在jsapi预支付的过程中和H5支付的区别为:
1,trade_type不同->MWEB-H5支付,JSAPI -JSAPI支付
2,jsapi支付的时候需要传递openid,用户在商户appid下的唯一标识。openid如何获取参考:https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421140842
首先需要获取到code值:
(补充以下链接参数访问之后,地址发生改变,其中就有code值,,,其中code值无法直接将地址从地址栏中粘贴出来,alert("页面首部展示路径 : "+$location.absUrl())会看出来有值)
参考链接(请在微信客户端中打开此链接体验):
scope为snsapi_base
https://open.weixin.qq.com/connect/oauth2/authorize?appid=wx520c15f417810387&redirect_uri=https%3A%2F%2Fchong.qq.com%2Fphp%2Findex.php%3Fd%3D%26c%3DwxAdapter%26m%3DmobileDeal%26showwxpaytitle%3D1%26vb2ctag%3D4_2030_5_1194_60&response_type=code&scope=snsapi_base&state=123#wechat_redirect
获取到code之后即可根据code值获取到openid
public String getOpenId(String code, String appId){
String openId ="";
String secret =Config.getValue("weixin.appsecret");
log.error("appid:______________"+appId);
log.error("SECRET:______________"+secret);
String url = "https://api.weixin.qq.com/sns/oauth2/access_token?" + "appid=" + appId
+ "&secret=" + secret
+ "&code="+ code
+ "&grant_type=authorization_code";
//http连接
HttpClient client = new HttpClient();
GetMethod method = new GetMethod(url);
client.getParams().setContentCharset("UTF-8");
method.setRequestHeader("ContentType","application/x-www-form-ACCESS_TOKEN_OPENIDencoded;charset=UTF-8");
try {
client.executeMethod(method);
String SubmitResult =method.getResponseBodyAsString();
log.debug("------openId请求----------------"+SubmitResult);
Map map=new HashMap();
map = (Map)(new ObjectMapper()).readValue(SubmitResult, Map.class);
openId = (String) map.get("openid"); //获取openID
String errcode = String.valueOf((Integer) map.get("errcode")); //获取错误码
log.error("【获取openId的具体内容】 : "+SubmitResult);
log.debug("【openId请求---errcode】 : "+errcode);
log.debug("【openId请求---openId】 : "+openId);
} catch (HttpException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return openId;
}