ホームページ >WeChat アプレット >WeChatの開発 >ThinkphpがWeChat決済機能を統合

ThinkphpがWeChat決済機能を統合

高洛峰
高洛峰オリジナル
2017-02-21 15:01:332023ブラウズ

この記事は主に、WeChat 決済機能を統合した Thinkphp に関する関連情報を紹介しています。非常に優れており、困っている友人は参考にしていただけます。

まず、この記事は次のことについてお伝えしたいと思います。 WeChat 支払い (サービスプロバイダーの販売者ではなく、一般の販売者による統一注文用の JSPI) WeChat 支払い:

ThinkphpがWeChat決済機能を統合ThinkphpがWeChat決済機能を統合ThinkphpがWeChat決済機能を統合ThinkphpがWeChat決済機能を統合ThinkphpがWeChat決済機能を統合

実際、私は SDK を自分で統合することに失敗し、ブロガーによって統合されたコードを使用しました。ここに書きました メモを取る:

準備:

1. WeChatパブリックアカウント:

固有のappid、appscrect、interface権限のうち、ユーザーID情報の権限を取得できるドメイン名を設定します(各ユーザー)別の公開アカウントがあります。一意の ID が存在します。この ID を使用して、ユーザーの WeChat アカウントの基本情報を取得します。詳細については、WeChat 開発者ドキュメントを参照してください)、WeChat 支払いボタンに WeChat 支払い承認ディレクトリを設定します。 WeChat アカウントはテスト ホワイトリストです (WeChat 開発者ツールを使用する場合に必要です)

2. WeChat 支払いプラットフォーム:

販売者プラットフォームのログイン アカウント、支払いキー (あなたいつでも自分で設定できます (1 つだけあります)、

3. ロジックを thinkphp に統合します:

フロントエンド WeChat 支払いボタン設定をクリックして、支払い開始コントローラー メソッドを呼び出します。

コントローラー実行し、WeChat 支払いクラスを引用し、ユーザーの openid を取得し、注文データを取得し、すべての通常の販売者を結合します。 jsp に必要なデータを前払いし、カスタマイズされた支払いページを表示し、支払いページで支払いをクリックし、提供されている jspi スクリプト関数を呼び出します。 WeChat で支払いを開始します。

支払いが完了すると、ページは支払いページのスクリプト関数で設定された (ジャンプ ディレクトリを自己定義 {:U('controller/function)}) にリダイレクトされ、非同期処理の注文ロジックを非同期的に (サイレントに) 設定します (支払い時間を記録し、支払い済みとしてマークし、WeChat Pay としてマークします) など、

コード:

注文ページの WeChat 支払いボタン:

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

支払いコントローラー Wxpay の開始:

<?php
namespace Home\Controller;
use Think\Controller;
//微信支付类
class WxpayController extends Controller {
 //获取access_token过程中的跳转uri,通过跳转将code传入jsapi支付页面
 public function js_api_start(){
  if(!empty($_GET[&#39;order_key_num&#39;])){
   // session(array(&#39;pay_now_id&#39;=>$_GET[&#39;order_key_num&#39;],&#39;expire&#39;=>3600));
   S(&#39;pay_now_id&#39;,$_GET[&#39;order_key_num&#39;],3600);
  }
  vendor(&#39;Weixinpay.WxPayPubHelper&#39;);
  //使用jsapi接口
  $jsApi = new \JsApi_pub();
  //=========步骤1:网页授权获取用户openid============
  //通过code获得openid
   if($_GET[&#39;code&#39;] == &#39;&#39;){
   //跳转
    $redirect_uri = &#39;https://当前域名+模块+控制器+方法&#39;;
    $url = &#39;https://open.weixin.qq.com/connect/oauth2/authorize
    ?appid=公众号特有IDredirect_uri=&#39;.$redirect_uri.&#39;&response_type=code&scope=snsapi_base&state=STATE#wechat_redirect&#39;;
    header("Location: $url");
    exit();
   }else{
   //获取openid
   $url = &#39;https://api.weixin.qq.com/sns/oauth2/access_token
   ?appid=公众号ID&secret=公众号scrept&code=&#39;.$_GET[&#39;code&#39;].&#39;&grant_type=authorization_code&#39;; 
   $openid_arr = json_decode(file_get_contents($url),true);
  }
  $openid=$openid_arr[&#39;openid&#39;];
  $pay_now_id = S(&#39;pay_now_id&#39;);
  if($pay_now_id){
   $id=$pay_now_id;
   $o = D(&#39;order_info&#39;);
   $order_info = $o->where(&#39;order_id = %d&#39;,$id)->find();
   if(empty($order_info[&#39;paycode&#39;])){
    $order_info[&#39;paycode&#39;] = &#39;weixin&#39;;
   }
   if($order_info[&#39;is_pay&#39;]){
    $this->error(&#39;当前订单已经支付&#39;);
   }
  }else{
   $this->error("不存在当前订单编号!");
  }
   $res = array(
   &#39;order_sn&#39; => $order_info[&#39;order_sn&#39;],
   &#39;order_amount&#39; => $order_info[&#39;pay_money&#39;]
   );
  //=========步骤2:使用统一支付接口,获取prepay_id============
  //使用统一支付接口
  $unifiedOrder = new \UnifiedOrder_pub();
  //设置统一支付接口参数
  //设置必填参数
  //appid已填,商户无需重复填写
  //mch_id已填,商户无需重复填写
  //noncestr已填,商户无需重复填写
  //spbill_create_ip已填,商户无需重复填写
  //sign已填,商户无需重复填写
  $total_fee = $order_info[&#39;pay_money&#39;]*100;
  // $total_fee = $res[&#39;order_amount&#39;];
  //$total_fee = 1;
  // var_dump($order_info[&#39;pay_money&#39;]);die;
  $body = "订单支付";
  $unifiedOrder->setParameter("openid", "$openid");//用户标识
  $unifiedOrder->setParameter("body", &#39;商品采购&#39;);//商品描述
  //自定义订单号,此处仅作举例
  $unifiedOrder->setParameter("out_trade_no", $order_info[&#39;order_sn&#39;]);//商户订单号 
  $unifiedOrder->setParameter("total_fee", $total_fee);//总金额
  //$unifiedOrder->setParameter("attach", "order_sn={$res[&#39;order_sn&#39;]}");//附加数据 
  $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[&#39;package&#39;] == &#39;prepay_id=&#39;) {
   $this->error(&#39;当前订单存在异常!&#39;);
  }
  $this->assign(&#39;res&#39;, $res);
  $this->assign(&#39;jsApiParameters&#39;, $jsApiParameters);
  $this->display(&#39;jsapi&#39;);
 }
 //异步通知url,商户根据实际开发过程设定
 public function notify_url() {
  vendor(&#39;Weixinpay.WxPayPubHelper&#39;);
  //使用通用通知接口
  $notify = new \Notify_pub();
  //存储微信的回调
  $xml = $GLOBALS[&#39;HTTP_RAW_POST_DATA&#39;]; 
  $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 &#39;error&#39;;
   }
   else if($notify->data["result_code"] == "FAIL"){
    //此处应该更新一下订单状态,商户自行增删操作
    //$this->log_result($log_name, "【业务出错】:\n".$xml."\n");
    //更新订单数据【通信出错】设为无效订单
    echo &#39;error&#39;;
   }
   else{
    //$this->log_result($log_name, "【支付成功】:\n".$xml."\n");
    //我这里用到一个process方法,成功返回数据后处理,返回地数据具体可以参考微信的文档
    if ($this->process($parameter)) {
     //处理成功后输出success,微信就不会再下发请求了
     echo &#39;success&#39;;
    }else {
     //没有处理成功,微信会间隔的发送请求
     echo &#39;error&#39;;
    }
   }
  }
 }
 //订单处理
 private function process($parameter) {
  //此处应该更新一下订单状态,商户自行增删操作
  /*
  * 返回的数据最少有以下几个
  * $parameter = array(
   &#39;out_trade_no&#39; => xxx,//商户订单号
   &#39;total_fee&#39; => XXXX,//支付金额
   &#39;openid&#39; => XXxxx,//付款的用户ID
  );
  */
  $data = array(
      &#39;order_sn&#39;=>$parameter[&#39;out_trade_no&#39;],
      &#39;des&#39;=>(&#39;订单交易:&#39;.$parameter[&#39;out_trade_no&#39;]),
      &#39;money&#39;=>$parameter[&#39;total_fee&#39;],
     );
  orderhandlestarysdgdss($data);//这是一个common方法,他会将该订单状态设置为已支付之类的
  return true;
 }
}
?>

支払い開始後の前払いデータ パラメータの結合 (パラメータ リストについては、WeChat 一般販売者開発者ドキュメント - WeChat 支払い - 統合注文を参照してください) 表示ページ:

<!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[&#39;order_sn&#39;]}";
//调用微信JS api 支付
function jsApiCall(){
 WeixinJSBridge.invoke(
  &#39;getBrandWCPayRequest&#39;,
  <?php echo $jsApiParameters; ?>,
  function(res){
   //如果支付成功
   if (res.err_msg == &#39;get_brand_wcpay_request:ok&#39;) {
    //支付成功后跳转的地址
    location.href = "{:U(&#39;Home/User/my_order&#39;)}";
   }else if (res.err_msg == &#39;get_brand_wcpay_request:cancel&#39;) {
    alert(&#39;请尽快完成支付哦!&#39;);
   }else if (res.err_msg == &#39;get_brand_wcpay_request:fail&#39;) {
    alert(&#39;支付失败&#39;);
   }else {
    alert(&#39;意外错误&#39;);
   }
   //WeixinJSBridge.log(res.err_msg);
   //alert(res.err_code+res.err_desc+res.err_msg);
   /*if (res.err_msg == &#39;get_brand_wcpay_request:ok&#39;) {
    alert(&#39;支付成功&#39;);
   }else {
    alert(&#39;取消支付&#39;);
   }*/
  }
 );
}
function callpay(){
 if (typeof WeixinJSBridge == "undefined"){
  if( document.addEventListener ){
   document.addEventListener(&#39;WeixinJSBridgeReady&#39;, jsApiCall, false);
  }else if (document.attachEvent){
   document.attachEvent(&#39;WeixinJSBridgeReady&#39;, jsApiCall); 
   document.attachEvent(&#39;onWeixinJSBridgeReady&#39;, jsApiCall);
  }
 }else{
  jsApiCall();
 }
}
</script>
<style>
*{font-family:&#39;微软雅黑&#39;,&#39;Microsoft YaHei&#39;;}
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>
<!--头部开始-->
<p class="flowpay">
 <dl>
  <dt>
   <p class="wxLogo"><img src="__PUBLIC__/home/images/1479953699138120.png" alt=""></p>
   本次订单需支付:¥<strong class="price">{$res[&#39;order_amount&#39;]}</strong> 元
  </dt>
  <dd>
   <input type="button" id="hhhhhh" onclick="callpay()" value="立即支付" />
  </dd>
 </dl>
</p>
<!--尾结束-->
</body>
</html>

次にクラス ファイルがあります:

ThinkphpがWeChat決済機能を統合その cacert は証明書の保存ディレクトリです。証明書は必ずしも必要ではありません

[マイ ファイル] でベンダー フォルダーを探すだけです。

上記は編集者が紹介した Thinkphp 統合 WeChat 決済機能です。ご質問がございましたら、メッセージを残してください。編集者がすぐにご返答いたします。また、PHP 中国語 Web サイトをサポートしていただきありがとうございます。

WeChat 決済機能を統合した Thinkphp に関連するその他の記事については、PHP 中国語 Web サイトに注目してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。