扫码支付模式一又称静态链接支付,开发前,商户必须在公众平台后台设置支付回调URL。该URL的功能是接收用户扫码后微信支付系统回调的product_id和openid。
扫码支付的接口类定义如下。
1 /** 2 * 请求商家获取商品信息接口 3 */ 4 class NativeCall_pub extends Wxpay_server_pub 5 { 6 /** 7 * 生成接口参数XML 8 */ 9 function createXml10 {11 if($this->returnParameters["return_code"] == "SUCCESS"){12 $this->returnParameters["appid"] = WxPayConf_pub::APPID;// 公众账号ID13 $this->returnParameters["mch_id"] = WxPayConf_pub::MCHID;// 商户号14 $this->returnParameters["nonce_str"] = $this->createNoncestr;// 随机字符串15 $this->returnParameters["sign"] = $this->getSign($this->returnParameters);// 签名16 }17 return $this->arrayToXml($this->returnParameters);18 }19 20 /**21 * 获取product_id22 */23 function getProductId24 {25 $product_id = $this->data["product_id"];26 return $product_id;27 }28 }
扫码支付模式一生成二维码的流程如下。
首先设置支付相关参数,其中需要自己指定的参数是产品ID。其他由系统自动获取或自动生成。其代码如下。
$this->parameters["appid"] = WxPayConf_pub::APPID; // 公众账号ID$this->parameters["mch_id"] = WxPayConf_pub::MCHID; // 商户号$time_stamp = time;$this->parameters["time_stamp"] = "$time_stamp"; // 时间戳$this->parameters["nonce_str"] = $this->createNoncestr; // 随机字符串$product_id = WxPayConf_pub::APPID."static"; // 自定义商品ID$nativeLink->setParameter("product_id","$product_id"); // 商品ID
生成之后,获得的数组如下。
object(NativeLink_pub)[1] public 'parameters' => array (size=5) 'product_id' => string 'wxdbfd43c561acxxxxstatic' (length=24) 'appid' => string 'wxdbfd43c561acxxxx' (length=18) 'mch_id' => string '10012345' (length=8) 'time_stamp' => string '1419733441' (length=10) 'nonce_str' => string 'no6qegpf11rn13nyl2q9izsk60be7fxc' (length=32)
再使用签名算法,将上述数据生成签名,得到sign值,结果如下。
object(NativeLink_pub)[1] public 'parameters' => array (size=6) 'product_id' => string 'wxdbfd43c561acxxxxstatic' (length=24) 'appid' => string 'wxdbfd43c561acxxxx' (length=18) 'mch_id' => string '10012345' (length=8) 'time_stamp' => string '1419733441' (length=10) 'nonce_str' => string 'no6qegpf11rn13nyl2q9izsk60be7fxc' (length=32) 'sign' => string '546CD81B0B66F57DC27BFEECEA1FB218' (length=32)
基于上述参数,将生成二维码的链接地址,生成代码如下。
// 获取链接$product_url = $nativeLink->getUrl;
生成二维码的链接如下。
weixin:// wxpay/bizpayurl?appid=wxdbfd43c561acxxxx&mch_id=10012345&nonce_str=no6qegpf11rn13nyl2q9izsk60be7fxc&product_id=wxdbfd43c561acxxxxstatic&sign=546CD81B0B66F57DC27BFEECEA1FB218&time_stamp=1419733441
将上述链接使用二维码生成接口,就可以生成一个模式一下的微信支付二维码,如图17-6所示。
图17-6 微信支付模式一的二维码
当用户扫描上述支付二维码时,回调接口URL将接收到来自微信服务器推送的静态Native支付链接的通知,接收通知的代码如下。
// 使用Native通知接口$nativeCall = new NativeCall_pub;// 接收微信请求$xml = $GLOBALS['HTTP_RAW_POST_DATA'];
该代码接收的XML通知数据如下。
<xml> <appid><![CDATA[wxdbfd43c561acxxxx]]></appid> <openid><![CDATA[oc-XIjh32OByBiak_gSZ6JOqGFx8]] ></openid> <mch_id><![CDATA[10012345]]></mch_id> <is_subscribe><![CDATA[Y]]></is_subscribe> <nonce_str><![CDATA[PvLH3nsJjQCvwnYY]]></nonce_str> <product_id><![CDATA[wxdbfd43c561acxxxxstatic]]></product_id> <sign><![CDATA[F1CBDE07E3B5AE6EAF4D4033368264EC]]></sign></xml>
统一支付将提取product_id参数的值,并填充其他支付参数,然后请求统一下单接口,代码如下。
// 提取product_id$product_id = $nativeCall->getProductId;$unifiedOrder = new UnifiedOrder_pub;$this->parameters["appid"] = WxPayConf_pub::APPID; // 公众账号ID$this->parameters["mch_id"] = WxPayConf_pub::MCHID; // 商户号$this->parameters["spbill_create_ip"] = $_SERVER['REMOTE_ADDR']; // 终端IP$this->parameters["nonce_str"] = $this->createNoncestr; // 随机字符串$this->parameters["sign"] = $this->getSign($this->parameters); // 签名$unifiedOrder->setParameter("body","贡献一分钱"); // 商品描述// 自定义订单号,此处仅作举例$timeStamp = time;$out_trade_no = WxPayConf_pub::APPID."$timeStamp";$unifiedOrder->setParameter("out_trade_no","$out_trade_no"); // 商户订单号 $unifiedOrder->setParameter("total_fee","1"); // 总金额$unifiedOrder->setParameter("notify_url",WxPayConf_pub::NOTIFY_URL); // 通知地址$unifiedOrder->setParameter("trade_type","NATIVE"); // 交易类型$unifiedOrder->setParameter("product_id","$product_id"); // 用户标识// 非必填参数,商户可根据实际情况选填// $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"); // 用户标识// 获取prepay_id$prepay_id = $unifiedOrder->getPrepayId;
统一支付将返回如下XML数据
<xml> <return_code><![CDATA[SUCCESS]]></return_code> <return_msg><![CDATA[OK]]></return_msg> <appid><![CDATA[wxdbfd43c561acxxxx]]></appid> <mch_id><![CDATA[10012345]]></mch_id> <nonce_str><![CDATA[JLQ67G1EhjfZvlKv]]></nonce_str> <sign><![CDATA[7A4F2751F955C32EB65063CC9E3EAB57]]></sign> <result_code><![CDATA[SUCCESS]]></result_code> <prepay_id><![CDATA[wx2014122820020936799023550244567827]]></prepay_id> <trade_type><![CDATA[NATIVE]]></trade_type> <code_url><![CDATA[weixin:// wxpay/bizpayurl?sr=yQtNpvo]]></code_url></xml>
上述数据中包含重要的prepay_id。提取出该参数,然后回调接口生成一个响应的XML数据,代码如下。
// 设置返回码// 设置必填参数// appid已填,商户无须重复填写// mch_id已填,商户无须重复填写// noncestr已填,商户无须重复填写// sign已填,商户无须重复填写$nativeCall->setReturnParameter("return_code","SUCCESS"); // 返回状态码$nativeCall->setReturnParameter("result_code","SUCCESS"); // 业务结果$nativeCall->setReturnParameter("prepay_id","$prepay_id");// 预支付ID// 将结果返回微信$returnXml = $nativeCall->returnXml;echo $returnXml;
而生成的XML数据如下。
<xml> <return_code><![CDATA[SUCCESS]]></return_code> <result_code><![CDATA[SUCCESS]]></result_code> <prepay_id><![CDATA[wx2014122820020936799023550244567827]]></prepay_id> <appid><![CDATA[wxdbfd43c561acxxxx]]></appid> <mch_id>10012345</mch_id> <nonce_str><![CDATA[e2bpc9fz3ykc2tcpipyvnb1l2qf8my3d]]></nonce_str> <sign><![CDATA[32C698EA795C0FBCDBCED622D1E01168]]></sign></xml>
这个XML数据回显给微信服务器后,用户的微信客户端将会显示出支付界面,如图17-7所示。
图17-7 微信支付界面
当用户点击“立即支付”按钮后,将会弹出输入密码插件,用户输入支付密码后,一个支付过程就完成了。