Swift3.0集成微信支付SDK

2017-05-03  本文已影响295人  慕诩
PS:我是小白,有不正确的请多指教

最近公司的项目需要做微信和支付宝支付功能,我的项目是用Swift3.0做的,记录一下我的集成过程以及遇到的坑。
先做的是微信支付,不得不吐槽,微信支付的官方文档有点坑,好久都没更新了,遇到了不少坑。当然也许只我技术不行。调起支付的参数出了签名(sign)外别的我们都是后台弄好给我们。其余关于在微信开放平台注册等就不说了

首先是集成SDK,根据微信支付官方文档我用的是cocoapods,没有什么问题,根据前两步项目设置APPID和在appdelegate里面注册APPID(WXApi.registerApp(weixinAppId, enableMTA: true)
然后接着在appdelegate里设置回调

 //ios9之前
    func application(_ application: UIApplication, handleOpen url: URL) -> Bool {
        if url.host == "safepay"{
//这个是支付宝用的
            aliPayCallBackMethod(url: url)
            return true
        }
        return WXApi.handleOpen(url, delegate: self)
    }
    
    //ios9之后
    func application(_ application: UIApplication, open url: URL, sourceApplication: String?, annotation: Any) -> Bool {
        if url.host == "safepay"{
            aliPayCallBackMethod(url: url)
            return true
        }
         return WXApi.handleOpen(url, delegate: self)
    }
    
    func application(_ app: UIApplication, open url: URL, options: [UIApplicationOpenURLOptionsKey : Any] = [:]) -> Bool {
        if url.host == "safepay"{
           aliPayCallBackMethod(url: url)
            return true
        }
        return WXApi.handleOpen(url, delegate: self)
    }

上面三个方法我用到时候发现之后调用第三个,但是我都写上去了。
然后设置appdelegate遵守WXApiDelegate协议
代理方法是支付结果的一个回调

 func onResp(_ resp: BaseResp!) {
        var detailMessage = String()
        if resp.isKind(of: PayResp.self){
            //支付返回的结果
            switch resp.errCode {
            case WXSuccess.rawValue:
                detailMessage = "支付结果:成功!"
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "success")
                break
            default:
                detailMessage = "支付结果:失败!" + "errCode\(resp.errCode)" + "errStr\(resp.errStr)"
                MyLog(message: detailMessage)
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "fail")
                break
            }
        }
    }

这样appdelegate里的工作就结束了,接下来就是点击支付按钮进行支付的那个控制器里进行下面的工作了。
刚才那个支付结果的回调结果发的通知在这里接收。注册通知

override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        NotificationCenter.default.addObserver(self, selector: #selector(getWeiXinPayResultNotificationMethod(noti:)), name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: nil)
    }
    
    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        NotificationCenter.default.removeObserver(self)
    }

接着就是点击微信支付按钮的点击事件了

 @objc fileprivate func submitBtnClick(){
              //微信
                if !(WXApi.isWXAppInstalled()) || !(WXApi.isWXAppSupport()){
                MySVProgressHUD.show(message: "您未安装微信或者版本不支持", status: .error)
                    return
                }
                 weixinPayMethod()   
    }

//微信支付
    fileprivate func weixinPayMethod(){
            MySVProgressHUD.show(message: "正在为您支付!", status: .status)

//下面这个方法就是向自己的服务器请求发起支付需要的数据
            getWeiXinPayNeedDataMethod()
    }

 //请求数据,解析数据
    fileprivate func getWeiXinPayNeedDataMethod(){
        let nameArr = []
        let valueArr = []
        NetWorkTools.postWithURLString(URLString: URL, nameArray: nameArr, valueArray: valueArr) { (dict, error) in
       '''省略。。。
                weakSelf?.dealDataWithDic(dic: orderInfo)
            }else{
            MySVProgressHUD.show(message: netError, status: .error)
            }
      }
  }

 fileprivate func dealDataWithDic(dic : [String : Any]){
// ##这里我把数据处理的代码放到WeiXinPayManager里面去了,TimesOperations.getNetworkTime()是获取当前时间戳,在这里获取一次就行了,带到后面去后面会用到
    WeiXinPayManager.weiXinPayMethodWithDic(dic: dic, payTimeStr: TimesOperations.getNetworkTime())
    }

这个控制器里还有一个刚才接收到接受支付结果通知的观察者方法
 @objc fileprivate func getWeiXinPayResultNotificationMethod(noti : NSNotification){
        let resultMessage = noti.object as! String
        if resultMessage == "success"{
            self.showAlertMessage(title: "恭喜", message: "您已成功支付啦!")
        }else{
            self.showAlertMessage(title: "提示", message: "支付失败!")
        }
    }
 fileprivate func showAlertMessage(title : String,message : String){
        let alertVC = UIAlertController(title: title, message: message, preferredStyle: .alert)
        let sureAction = UIAlertAction(title: "OK", style: .default) { (action) in
           //进行一些支付完成后的操作,比如刷新数据,移除一些遮罩View等
        }
        sureAction.setValue(RGB(R: 107, G: 107, B: 199), forKey: "titleTextColor")
        alertVC.addAction(sureAction)
        self.present(alertVC, animated: true, completion: nil)
    }

接下来就是数据处理的部分了,坑就在这里,主要是生成一个签名Sign麻烦,去到WeiXinPayManager类里面
通过传过去的字典,获取到partnerId,prepayId,nonceStr,我方法比较笨拙,别见笑,是直接去取得

    class func weiXinPayMethodWithDic(dic : [String : Any],payTimeStr : String?){
        let partnerId = dic["mch_id"] as? String
        let prepayId = dic["prepay_id"] as? String
        let nonceStr = dic["nonce_str"] as? String

//下面这个sign就是我们需要的,我是下面用了一个方法返回的,package是微信官方文档要求的这么写的值为"Sign=WXPay”
        let sign = WeiXinPayManager().getWeiXinSignStrMethod(dic: dic, payTimeStr: payTimeStr)

        if partnerId != nil,prepayId != nil,nonceStr != nil,payTimeStr != nil,sign != nil{
            let time = (payTimeStr! as NSString).intValue
            WeiXinPayManager().requestWeiXinPayMethod(partnerId: partnerId!, prepayId: prepayId!, package: "Sign=WXPay", nonceStr: nonceStr!, timeStamp: UInt32(time), sign: sign!)
        }else{
            MySVProgressHUD.show(message: "参数出错啦", status: .error)
        }
    }
    
    fileprivate func requestWeiXinPayMethod(partnerId : String,prepayId : String,package : String,nonceStr : String,timeStamp : UInt32,sign : String){
        MySVProgressHUD.dismiss()
        let request = PayReq()
        request.partnerId = partnerId
        request.prepayId = prepayId
        request.package = package
        request.nonceStr = nonceStr
        request.timeStamp = timeStamp
        request.sign = sign
        WXApi.send(request)
    }

到这差不多就可以经行支付了。当然还有一个重要的Sign的生成还没写
关于生成Sign,官方文档是这么写的:

第一步设所有发送或者接收到的数据为集合M,将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序),使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA。

虽然下面也列举了一些参数,但是这里并没有真正告诉我们要传哪些,根据他列举的参数我搞了好久,毛用没有,都是只出现一个白色确定按钮,这个就说明是这个签名Sign错了,后来还是安卓的同事提示慢慢试出来了,注意参数是"appid","noncestr","prepayid",package,timestamp,partnerid,这6个,其中partnerid是商户id,可以让后台返回,也可以在自己申请的微信支付平台账号中找到,直接写死并且所有的参数不用加下划线,不要像官方文档那样使用mch_id这种拼接,注意mch_id用不上,下面就是要将参数按照从小到大排序(文档要求的)然后拼接了,(sign也可以服务端生成,服务端调用统一下单接口后获取到预支付id,再进行二次签名得到我们需要的sign:参考(https://blog.csdn.net/sinat_17820239/article/details/51334181?utm_source=blogxgwz9))因为我是直接从后台给的数据里去取参数,需要过滤不需要的key,也许把问题弄复杂了,反正这里要做的就是将这些Key从小到大排序,对应的Value也是一样,让后按照”key1=value1&key2=value2…“拼接。从小到大排序我用的是swift中的方法
var sortedKeyArray = newKeys.sorted { ( key1 : String, key2 : String) -> Bool in return key1 < key2 }
每个人拼接的方法思路不一样,我就不献丑了,好了,假设把这6个Key和对应的Value按照URL键值对的格式(即key1=value1&key2=value2…)拼接成了stringA,接着拼接行一个申请得到的weixinKey,这个key在微信支付开发者账号里找

 fileprivate func getWeiXinSignStrMethod(dic : [String : Any],payTimeStr : String?)->String?{

//1.先自己拼接得到了stringA,假设是stringA = "appid="123"&noncestr="123&package="Sign=WXPay"&partnerid="123"&prepayid="123"&timestamp= payTimeStr!"

//2.接着拼接weixinKey
stringA.append("&key=")  
stringA.append(weixinKey)

//3.把新得到的stringA进行MD5加密,再把得到的MD5值中所有的字母变成大写的,使用NSString的方法uppercased
if MD5Hash.get32MD5Hash(stringA as String!) != nil{
            return (MD5Hash.get32MD5Hash(contentString as String!)! as NSString).uppercased
        }else{
            return nil
        }
}

至此,Swift微信支付就做完了,技术有限,有做的不对的请指教!下篇记录关于Swift支付宝支付的集成

上一篇 下一篇

猜你喜欢

热点阅读