服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - PHP教程 - Thinkphp整合微信支付功能

Thinkphp整合微信支付功能

2021-04-07 16:52mrr PHP教程

这篇文章主要介绍了Thinkphp整合微信支付功能的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下

先上效果图:我要告诉你我这一篇文章写的是微信支付之中的(普通商户而非服务商商户的统一下单jspi)微信支付:

Thinkphp整合微信支付功能Thinkphp整合微信支付功能Thinkphp整合微信支付功能Thinkphp整合微信支付功能Thinkphp整合微信支付功能

其实自己整合sdk失败了,用了一个博客博主整合的代码,在这里写一下笔记:

前面准备:

1、微信公众号:

    独特的appid、appscrect、接口权限之中设置可以获取用户id信息权限的域名(每个用户对于不同公众都会有一个特有id,通过这个id获取用户微信账号基本信息、详情看微信开发者文档)、在微信支付按钮出设置微信支付授权目录(写到发起请求的控制器那一层)、设置开发者微信账号为测试白名单(用微信开发者工具的时候需要)

2、微信支付平台:

  商户平台登陆账号、支付密钥(随时可以自行设置,只能有一个)、

3、整合进去thinkphp之中逻辑:

  前端微信支付按钮设置点击调用支付发起控制器方法、

  控制器运行,引用微信支付类、获取用户openid、获取订单数据、拼接出所有普通商户预支付jsp需要的数据,display出那个自定义的支付页面、

  在支付页面点击支付、调用微信提供的jspi发起支付的scripet函数发起支付、

  支付完成以后页面会重定向到(在自定义支付页面的script函数里设置的跳转目录{:u('controller/function)}),并且异步(静默)设置的异步处理订单逻辑(记录支付时间啦、标记为已经支付啦、标记是微信支付啦)之类的、

代码:

  我的订单页面的微信支付按钮:

?
1
<a href="{:u('wxpay/js_api_start',array('order_key_num'=>$v['order_key_num]))}"> 微信支付</a>

发起支付控制器wxpay:

?
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
<?php
namespace home\controller;
use think\controller;
//微信支付类
class wxpaycontroller extends controller {
 //获取access_token过程中的跳转uri,通过跳转将code传入jsapi支付页面
 public function js_api_start(){
  if(!empty($_get['order_key_num'])){
   // session(array('pay_now_id'=>$_get['order_key_num'],'expire'=>3600));
   s('pay_now_id',$_get['order_key_num'],3600);
  }
  vendor('weixinpay.wxpaypubhelper');
  //使用jsapi接口
  $jsapi = new \jsapi_pub();
  //=========步骤1:网页授权获取用户openid============
  //通过code获得openid
   if($_get['code'] == ''){
   //跳转
    $redirect_uri = 'https://当前域名+模块+控制器+方法';
    $url = 'https://open.weixin.qq.com/connect/oauth2/authorize
    ?appid=公众号特有idredirect_uri='.$redirect_uri.'&response_type=code&scope=snsapi_base&state=state#wechat_redirect';
    header("location: $url");
    exit();
   }else{
   //获取openid
   $url = 'https://api.weixin.qq.com/sns/oauth2/access_token
   ?appid=公众号id&secret=公众号scrept&code='.$_get['code'].'&grant_type=authorization_code';
   $openid_arr = json_decode(file_get_contents($url),true);
  }
  $openid=$openid_arr['openid'];
  $pay_now_id = s('pay_now_id');
  if($pay_now_id){
   $id=$pay_now_id;
   $o = d('order_info');
   $order_info = $o->where('order_id = %d',$id)->find();
   if(empty($order_info['paycode'])){
    $order_info['paycode'] = 'weixin';
   }
   if($order_info['is_pay']){
    $this->error('当前订单已经支付');
   }
  }else{
   $this->error("不存在当前订单编号!");
  }
   $res = array(
   'order_sn' => $order_info['order_sn'],
   'order_amount' => $order_info['pay_money']
   );
  //=========步骤2:使用统一支付接口,获取prepay_id============
  //使用统一支付接口
  $unifiedorder = new \unifiedorder_pub();
  //设置统一支付接口参数
  //设置必填参数
  //appid已填,商户无需重复填写
  //mch_id已填,商户无需重复填写
  //noncestr已填,商户无需重复填写
  //spbill_create_ip已填,商户无需重复填写
  //sign已填,商户无需重复填写
  $total_fee = $order_info['pay_money']*100;
  // $total_fee = $res['order_amount'];
  //$total_fee = 1;
  // var_dump($order_info['pay_money']);die;
  $body = "订单支付";
  $unifiedorder->setparameter("openid", "$openid");//用户标识
  $unifiedorder->setparameter("body", '商品采购');//商品描述
  //自定义订单号,此处仅作举例
  $unifiedorder->setparameter("out_trade_no", $order_info['order_sn']);//商户订单号
  $unifiedorder->setparameter("total_fee", $total_fee);//总金额
  //$unifiedorder->setparameter("attach", "order_sn={$res['order_sn']}");//附加数据
  $unifiedorder->setparameter("notify_url", \wxpayconf_pub::notify_url);//通知地址
  $unifiedorder->setparameter("trade_type", "jsapi");//交易类型
  //非必填参数,商户可根据实际情况选填
  //$unifiedorder->setparameter("sub_mch_id","xxxx");//子商户号
  //$unifiedorder->setparameter("device_info","xxxx");//设备号
  //$unifiedorder->setparameter("attach","xxxx");//附加数据
  //$unifiedorder->setparameter("time_start","xxxx");//交易起始时间
  //$unifiedorder->setparameter("time_expire","xxxx");//交易结束时间
  //$unifiedorder->setparameter("goods_tag","xxxx");//商品标记
  //$unifiedorder->setparameter("openid","xxxx");//用户标识
  //$unifiedorder->setparameter("product_id","xxxx");//商品id
  $prepay_id = $unifiedorder->getprepayid();
  // var_dump($prepay_id);die;
  //=========步骤3:使用jsapi调起支付============
  $jsapi->setprepayid($prepay_id);
  $jsapiparameters = $jsapi->getparameters();
  $wxconf = json_decode($jsapiparameters, true);
  if ($wxconf['package'] == 'prepay_id=') {
   $this->error('当前订单存在异常!');
  }
  $this->assign('res', $res);
  $this->assign('jsapiparameters', $jsapiparameters);
  $this->display('jsapi');
 }
 //异步通知url,商户根据实际开发过程设定
 public function notify_url() {
  vendor('weixinpay.wxpaypubhelper');
  //使用通用通知接口
  $notify = new \notify_pub();
  //存储微信的回调
  $xml = $globals['http_raw_post_data'];
  $notify->savedata($xml);
  //验证签名,并回应微信。
  //对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,
  //微信会通过一定的策略(如30分钟共8次)定期重新发起通知,
  //尽可能提高通知的成功率,但微信不保证通知最终能成功。
  if($notify->checksign() == false){
   $notify->setreturnparameter("return_code", "fail");//返回状态码
   $notify->setreturnparameter("return_msg", "签名失败");//返回信息
  }else{
   $notify->setreturnparameter("return_code", "success");//设置返回码
  }
  $returnxml = $notify->returnxml();
  //==商户根据实际情况设置相应的处理流程,此处仅作举例=======
  //以log文件形式记录回调信息
  //$log_name = "notify_url.log";//log文件路径
  //$this->log_result($log_name, "【接收到的notify通知】:\n".$xml."\n");
  $parameter = $notify->xmltoarray($xml);
  //$this->log_result($log_name, "【接收到的notify通知】:\n".$parameter."\n");
  if($notify->checksign() == true){
   if ($notify->data["return_code"] == "fail") {
    //此处应该更新一下订单状态,商户自行增删操作
    //$this->log_result($log_name, "【通信出错】:\n".$xml."\n");
    //更新订单数据【通信出错】设为无效订单
    echo 'error';
   }
   else if($notify->data["result_code"] == "fail"){
    //此处应该更新一下订单状态,商户自行增删操作
    //$this->log_result($log_name, "【业务出错】:\n".$xml."\n");
    //更新订单数据【通信出错】设为无效订单
    echo 'error';
   }
   else{
    //$this->log_result($log_name, "【支付成功】:\n".$xml."\n");
    //我这里用到一个process方法,成功返回数据后处理,返回地数据具体可以参考微信的文档
    if ($this->process($parameter)) {
     //处理成功后输出success,微信就不会再下发请求了
     echo 'success';
    }else {
     //没有处理成功,微信会间隔的发送请求
     echo 'error';
    }
   }
  }
 }
 //订单处理
 private function process($parameter) {
  //此处应该更新一下订单状态,商户自行增删操作
  /*
  * 返回的数据最少有以下几个
  * $parameter = array(
   'out_trade_no' => xxx,//商户订单号
   'total_fee' => xxxx,//支付金额
   'openid' => xxxxx,//付款的用户id
  );
  */
  $data = array(
      'order_sn'=>$parameter['out_trade_no'],
      'des'=>('订单交易:'.$parameter['out_trade_no']),
      'money'=>$parameter['total_fee'],
     );
  orderhandlestarysdgdss($data);//这是一个common方法,他会将该订单状态设置为已支付之类的
  return true;
 }
}
?>

发起支付后拼接预支付数据参数(参数列表看微信普通商户开发者文档——微信支付——统一下单)display的页面:

?
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
<!doctype html>
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=utf-8">
<meta http-equiv="x-ua-compatible" content="ie=edge">
<meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0;" name="viewport" />
<meta name="format-detection" content="telephone=no"/>
<title>下</title>
<meta http-equiv="content-security-policy" content="upgrade-insecure-requests">
<meta name="keyword" content="">
<meta name="description" content="">
<script type="text/javascript">
var order_sn = "{$res['order_sn']}";
//调用微信js api 支付
function jsapicall(){
 weixinjsbridge.invoke(
  'getbrandwcpayrequest',
  <?php echo $jsapiparameters; ?>,
  function(res){
   //如果支付成功
   if (res.err_msg == 'get_brand_wcpay_request:ok') {
    //支付成功后跳转的地址
    location.href = "{:u('home/user/my_order')}";
   }else if (res.err_msg == 'get_brand_wcpay_request:cancel') {
    alert('请尽快完成支付哦!');
   }else if (res.err_msg == 'get_brand_wcpay_request:fail') {
    alert('支付失败');
   }else {
    alert('意外错误');
   }
   //weixinjsbridge.log(res.err_msg);
   //alert(res.err_code+res.err_desc+res.err_msg);
   /*if (res.err_msg == 'get_brand_wcpay_request:ok') {
    alert('支付成功');
   }else {
    alert('取消支付');
   }*/
  }
 );
}
function callpay(){
 if (typeof weixinjsbridge == "undefined"){
  if( document.addeventlistener ){
   document.addeventlistener('weixinjsbridgeready', jsapicall, false);
  }else if (document.attachevent){
   document.attachevent('weixinjsbridgeready', jsapicall);
   document.attachevent('onweixinjsbridgeready', jsapicall);
  }
 }else{
  jsapicall();
 }
}
</script>
<style>
*{font-family:'微软雅黑','microsoft yahei';}
body #head{position:relative;z-index:99999999999999;padding:0 10px;}
body .zh-head{padding:0 0 0 0;height:auto;}
.zh-head-conter{position:relative;height:40px;}
.zh-logo{position:absolute;left:50%;top:0;margin:0 0 0 -60px;float:none;width:auto;}
.zh-logo a{display:block;}
.zh-logo img{width:120px;height:40px;display:block;}
.heads_fix .zh-logo{}
#head{position:fixed!important;left:0;top:0;right:0;z-index:99999;background:#fff;border-bottom:1px solid #ddd;}
.zh-logo{height:40px;}
.flowpay{margin-top:25%;}
.flowpay dt{text-align:center;}
.flowpay strong.price{font-size:40px;}
.wxlogo{text-align:center;}
.wxlogo img{}
.flowpay dd{margin:0;padding:20px 0 10px 0;}
.flowpay dd input{margin:0 auto;padding:0;width:90%;height:45px;line-height:45px;border:0;border-radius:4px;background:#0cbc0a;color:#fff;font-size:17px;display:block;-webkit-appearance:none;-moz-appearance:none;appearance:none;outline:none;}
</style>
</head>
<body>
<!--头部开始-->
<div class="flowpay">
 <dl>
  <dt>
   <p class="wxlogo"><img src="__public__/home/images/1479953699138120.png" alt=""></p>
   本次订单需支付:¥<strong class="price">{$res['order_amount']}</strong> 元
  </dt>
  <dd>
   <input type="button" id="hhhhhh" onclick="callpay()" value="立即支付" />
  </dd>
 </dl>
</div>
<!--尾结束-->
</body>
</html>

然后就是类文件啦:

Thinkphp整合微信支付功能

那个cacert是证书存放目录;证书不一定需要的;

vendor文件夹在我的文件里面找找就可以。

以上所述是小编给大家介绍的thinkphp整合微信支付功能,希望对大家有所帮助如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!

延伸 · 阅读

精彩推荐