微信支付JSAPI模式及退款CodeIgniter集成篇

时间:2024-02-21 08:42:07

微信支付接口文档:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_1

首先你得知道这个jsapi是不能离开微信进行调用支付的,明白了这个道理我们好下手,页面是在微信内显示并通过jsapi调用微信支付组件进行支付。

可以看看我们上一篇文章,主要是Native扫码支付模式二

我们仍然继续使用wechatpay.php这个支付集成类,简单方便好理解,不过如果应用jsapi的话这个类有个bug

在我们构造jsapi需要的参数时有个时间戳,我们用time()生成的,会报微信支付调用JSAPI缺少参数:timeStamp

修改如下:

	/**
	 * 获取js支付使用的第二个参数
	 */
	public function get_package($prepay_id) {
		$data = array();
		$data["appId"] = $this->_config["appid"];
          //改动地方,把它变成字符串
		$time=time();
		$data["timeStamp"] = "\"".$time."\"";
		$data["nonceStr"]  = $this->get_nonce_string();
		$data["package"]   = "prepay_id=$prepay_id";
		$data["signType"]  = "MD5";
		$data["paySign"]   = $this->sign($data);
		return $data;
	}

 其实这个方法就是获取jsapi的支付参数了

一、微信JSAPI支付

不能忘记配置授权目录,调用jsapi我是在http://xxx.com/index.php/home下我配置了这个

首先我们还是要调用统一下单接口,获取我们要的参数(如果此类的配置放置位置等不会的请参考上篇文章),此为pay方法,在调用统一下单接口的时候我们需要知道需要哪些参数

1、要获取openid,这个我是项目用了一个微信API的类库,https://github.com/dodgepudding/wechat-php-sdk,主要是用了这里面的方法

此项目有朋友专门的对接了CodeIgniter框架的扩展类库,可以直接用,目录结构,我们直接上代码吧

    public function __construct()
    {
        parent::__construct();
        $this->load->library(\'CI_Wechat\');//由于我的项目是时刻都跟微信绑在一起,所以直接加载在构造函数里了,不用每个方法都加载了。
        $this->load->library(\'pagination\');
    }

 CI_Model内容大家看下上面的类库源码,还有里面如何配置的,下面我们看看如何获取openid

 

    function oauthurl()
    {
        $oauth_url = $this->ci_wechat->getOauthRedirect(base_url() . \'index.php/home/oauth\', 1);
        header(\'Location: \' . $oauth_url);
        exit();
    }

    function oauth()
    {
        if (!isset($_GET[\'code\'])) {

            //触发微信返回code码
            $baseUrl = urlencode(\'http://\' . $_SERVER[\'HTTP_HOST\'] . $_SERVER[\'PHP_SELF\'] . $_SERVER[\'QUERY_STRING\']);
            $url = $this->__CreateOauthUrlForCode($baseUrl);
            Header("Location: $url");
            exit();

        } else {
            $json = $this->ci_wechat->getOauthAccessToken();
            $openid = $json[\'openid\'];
            //注册用户,成功后可以抢单
            //return $this->_isRegistered($_SESSION[\'user\'][\'openid\']);
            return $openid;

        }
    }

 以上两个方法就是获取openid的,获取之后我是保存在session里的,我每个页面都判断是否获取了openid如果没有获取直接

$this->session->set_userdata(\'openid\', $this->oauth());

 这样保证一直能得到openid

2、构造JSAPI支付所需参数(统一下单的参数构造)

                $this->load->model(\'publist\');//获取订单信息
                $pub = $this->publist->GetList(array(\'id\' => $_SESSION[\'orderid\']));
                //微信支付配置的参数配置读取
                $this->load->config(\'wxpay_config\');
                $wxconfig[\'appid\']=$this->config->item(\'appid\');
                $wxconfig[\'mch_id\']=$this->config->item(\'mch_id\');
                $wxconfig[\'apikey\']=$this->config->item(\'apikey\');
                $wxconfig[\'appsecret\']=$this->config->item(\'appsecret\');
                $wxconfig[\'sslcertPath\']=$this->config->item(\'sslcertPath\');
                $wxconfig[\'sslkeyPath\']=$this->config->item(\'sslkeyPath\');
                $this->load->library(\'Wechatpay\',$wxconfig);
                //商户交易单号
                $out_trade_no = $pub->listno;
                $total_fee=$pub->fee;
                $openid=$_SESSION[\'openid\'];
                $param[\'body\']="黑人牙膏";
                $param[\'attach\']=$pub->id;
                $param[\'detail\']="黑人牙膏-".$out_trade_no;
                $param[\'out_trade_no\']=$out_trade_no;
                $param[\'total_fee\']=$total_fee*100;
                $param["spbill_create_ip"] =$_SERVER[\'REMOTE_ADDR\'];
                $param["time_start"] = date("YmdHis");
                $param["time_expire"] =date("YmdHis", time() + 600);
                $param["goods_tag"] = "黑人牙膏";
                $param["notify_url"] = base_url()."index.php/home/notify";
                $param["trade_type"] = "JSAPI";
                $param["openid"] = $openid;

                //统一下单,获取结果,结果是为了构造jsapi调用微信支付组件所需参数
                $result=$this->wechatpay->unifiedOrder($param);
                
                //如果结果是成功的我们才能构造所需参数,首要判断预支付id

                if (isset($result["prepay_id"]) && !empty($result["prepay_id"])) {
                    //调用支付类里的get_package方法,得到构造的参数
                    $data[\'parameters\']=json_encode($this->wechatpay->get_package($result[\'prepay_id\']));
                    $data[\'notifyurl\']=$param["notify_url"];
                    $data[\'fee\']=$total_fee;
                    $data[\'pubid\']=$_SESSION[\'orderid\'];

                    $this->load->view(\'home/header\');
                    //要有个页面将以上数据传递过去并展示给用户
                    $this->load->view(\'home/pay\', $data);
                    $this->load->view(\'home/footer\');
                }        

 3、支付页面,views视图pay.php

<?php
$jsApiParameters = $parameters;//参数赋值
?>

<script type="text/javascript">
    //调用微信JS api 支付
    function jsApiCall()
    {
        WeixinJSBridge.invoke(
            \'getBrandWCPayRequest\',
            <?php echo $jsApiParameters; ?>,
            function(res){
                WeixinJSBridge.log(res.err_msg);
                if(res.err_msg == "get_brand_wcpay_request:ok" ){
                    $.alert(\'支付成功\');
                    //我在这里选择了前台只要支付成功将单号传递更新数据
                    $.ajax({
                        url:\'<?php  echo $notifyurl.\'/\'.$pubid;?>\',
                        dataType:\'json\',
                        success : function(ret){
                            if(ret==1){
                                //成功后返回我的订单页面
                                location.href="<?php echo base_url().\'index.php/home/myorder\';?>";
                            }
                        }
                    });
                }else
                {
                    //$.alert(\'支付失败\');
                }
                //alert(res.err_code+res.err_desc+res.err_msg);
            }
        );
    }

    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>
<div class="hd">
    <h1 class="page_title">支付佣金</h1>
    <p class="page_desc">请认真核对佣金金额</p>
</div>
<div class="weui_cells">
    <div class="weui_cell">
        <div class="weui_cell_hd weui_cell_primary">
            该笔订单支付金额为<span style="color:#f00;font-size:50px"><?php echo $fee; ?></span>元钱
        </div>
    </div>
</div>
<button class="weui_btn weui_btn_primary" type="button" onclick="callpay()" >立即支付</button>

 以上代码可以用微信web开发者工具,使用方式自己看看吧,有了这个工具调试不再难

4、支付成功跳转页面,我们看notify方法

    function notify()
    {
        $id = $this->uri->segment(3);
        if (isset($_SESSION[\'openid\'])) {
            $this->load->model(\'publist\');//更新业务逻辑
            $rs = $this->publist->UpdateList(array(\'id\' => $id, \'feestatus\' => 1));
            if ($rs > 0) {
                echo 1;
                exit;
            } else {
                echo 0;
                exit;
            }
        }
    }

 这样我们的支付流程就彻底走完了。

二、当我们支付完之后,有些单子可以退单的,如何将款项也退回呢

以上场景要弄明白了

我们申请退款需要参数有哪些?我们看看支付类里的退款方法

	/**
	 * 申请退款 - 使用商户订单号
	 * @param $out_trade_no 商户订单号
	 * @param $out_refund_no 退款单号
	 * @param $total_fee 总金额(单位:分)
	 * @param $refund_fee 退款金额(单位:分)
	 * @param $op_user_id 操作员账号
	 * @return array
	 */
	public function refund($out_trade_no,$out_refund_no,$total_fee,$refund_fee,$op_user_id){
		$data = array();
		$data["appid"] = $this->_config["appid"];
		$data["mch_id"] = $this->_config["mch_id"];
		$data["nonce_str"] = $this->get_nonce_string();
		$data["out_trade_no"] = $out_trade_no;
		$data["out_refund_no"] = $out_refund_no;
		$data["total_fee"] = $total_fee;
		$data["refund_fee"] = $refund_fee;
		$data["op_user_id"] = $op_user_id;
		$result = $this->post(self::URL_REFUND, $data,true);

		return $result;
	}

 商户订单号,商户提供的退单号,付款金额,退款金额(不能退的比实际付款的多),操作员(一般商户号)

控制器内写退款方法

    //申请退款
    function refund($id="")
    {
        if($id==""){
            //方便我手动调用退单
            $id = $this->uri->segment(3);
        }
        if (isset($id) && $id != "") {
            $this->load->model(\'publist\');
            //1、取消订单可以退款。2、失败订单可以退款
            $pub = $this->publist->GetList(array(\'id\' => $id));
            if ($pub->liststatus == 3 || $pub->liststatus == 4) {
                $listno = $pub->listno;
                $fee = $pub->fee * 100;

                $this->load->config(\'wxpay_config\');
                $wxconfig[\'appid\']=$this->config->item(\'appid\');
                $wxconfig[\'mch_id\']=$this->config->item(\'mch_id\');
                $wxconfig[\'apikey\']=$this->config->item(\'apikey\');
                $wxconfig[\'appsecret\']=$this->config->item(\'appsecret\');
                $wxconfig[\'sslcertPath\']=$this->config->item(\'sslcertPath\');
                $wxconfig[\'sslkeyPath\']=$this->config->item(\'sslkeyPath\');
                $this->load->library(\'Wechatpay\',$wxconfig);

                if (isset($listno) && $listno != "") {
                    $out_trade_no = $listno;
                    $total_fee = $fee;
                    $refund_fee = $fee;
                    //自定义商户退单号
                    $out_refund_no=$wxconfig[\'mch_id\'].date("YmdHis");
                        $result=$this->wechatpay->refund($out_trade_no,$out_refund_no,$total_fee,$refund_fee,$wxconfig[\'mch_id\']);

                    log::DEBUG(json_encode($result));
                    if (isset($result["return_code"]) && $result["return_code"]="SUCCESS"&&isset($result["result_code"]) && $result["result_code"]="SUCCESS") {
                        echo "<script>$.toast(\'退款成功\')</script>";
                    }
                    //佣金状态更改为已退款
                    $this->publist->UpdateList(array(\'id\'=>$id,\'liststatus\'=>3,\'listoutno\'=>$out_refund_no));
                    redirect(\'home/myorder\');
                }
            }
        }
    }

 试试就好了,很快就可以接到退款消息

以上是这几天摸索出来的东西,分享给大家。