站内搜索

Thinkphp整合微信支付功能详解

微信支付现在在我们的生活中普遍皆是,在开发当中微信小程序也非常的火,本文我们继续给大家讲解关于Thinkphp整合微信支付功能。

1、微信公众号:

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

2、微信支付平台:

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

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

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

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

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

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

代码:

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

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

发起支付控制器Wxpay:

<?phpnamespace 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的页面:

<!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>

然后就是类文件啦:

11.png

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

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

以上内容就是Thinkphp整合微信支付功能,希望能帮助到大家。

相关推荐:

php如何实现微信小程序支付及退款

微信小程序开发入门实例

微信小程序网络请求的封装

微信小程序实现密码输入的实例

微信小程序机器人自动客服功能

以上就是Thinkphp整合微信支付功能详解的详细内容,更多请关注php中文网其它相关文章!

  • 上一篇:微信API接口
  • 下一篇:5 分钟实现微信支付接入教程