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

来源:http://www.cnblogs.com/24la/archive/2016/04/25/wxpay-jsapi-refund.html
-Advertisement-
Play Games

微信支付介面文檔:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_1 首先你得知道這個jsapi是不能離開微信進行調用支付的,明白了這個道理我們好下手,頁面是在微信內顯示並通過jsapi調用微信支付組件進行支付。 可以看看我們上一 ...


微信支付介面文檔: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');
                }
            }
        }
    }

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

以上是這幾天摸索出來的東西,分享給大家。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、接收用戶輸入: input:接收用戶輸入的是合法的python表達式,比如字元串。 raw_input:把所有的輸入當做原始數據(raw data)。 除非對input有特別的需要,否則儘可能使用raw_input函數。 2、長字元串和原始字元串 長字元串常利用'\'經行轉義,例如: 這句話會打 ...
  • 原文地址:http://blog.csdn.net/morewindows/article/details/7421759 使用多線程其實是非常容易的,下麵這個程式的主線程會創建了一個子線程並等待其運行完畢,子線程就輸出它的線程ID號然後輸出一句經典名言——Hello World。整個程式的代碼非常 ...
  • 概述 Java語言中,提供了一套數據集合框架,其中定義了一些諸如List、Set等抽象數據類型,每個抽象數據類型的各個具體實現,底層又採用了不同的實現方式,比如ArrayList和LinkedList。 除此之外,Java對於數據集合的遍歷,也提供了幾種不同的方式。開發人員必須要清楚的明白每一種遍歷 ...
  • 程式中需USE COMOBJ單元 1.Q:如何得到機器上IIS中所有的WEB虛擬站點. A: var InstallPath: String; WebSite, WebServer, WebRoot: Variant; count: Integer; Flag: Boolean; begin Fla ...
  • 本文目的 PHP的全局錯誤處理,在開發項目的時候很有用,可以幫助開發者快速定位一些問題,提高工作效率。預設情況下,全局錯誤會直接輸出,但是最近開發時使用的一個框架庫對全局錯誤處理進行了設定,導致很多錯誤信息沒有輸出,在定位問題上有一定的耗時。所以,研究了一下此庫的實現,發現它設定了error_rep ...
  • 什麼是Queue集合? 答:Queue用於模擬隊列這種數據結構。隊列通常是指“先進先出(FIFO)”的容器。隊列的頭部保存在隊列中存放時間最長的元素,尾部保存存放時間最短的元素。新元素插入到隊列的尾部,取出元素會返回隊列頭部的元素。通常,隊列不允許隨機訪問隊列中的元素。 Queue介面中定義瞭如下的 ...
  • “這裡要用char類型”; “這裡要用int類型”; “其實實現這個方法只需要把另一個方法的返回值的類型和傳入參數的類型改成float類型就實現了”; “其實這個演算法只需要把以前寫的那個稍微改動一下就行了”; ……………… 學過面向對象語言的都知道GP這個概念,就是泛型程式設計,說的再明白點就是編寫 ...
  • Java併發編程系列【未完】: Java 併發編程:核心理論 Java併發編程:Synchronized及其實現原理 Java併發編程:Synchronized底層優化(輕量級鎖、偏向鎖) 一、重量級鎖 上篇文章中向大家介紹了Synchronized的用法及其實現的原理。現在我們應該知道,Synch ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...