PS:我是小白耘擂,有不正確的請多指教
最近公司的項目需要做微信和支付寶支付功能蓖租,我的項目是用Swift3.0做的,記錄一下我的集成過程以及遇到的坑饭玲。
先做的是微信支付侥祭,不得不吐槽,微信支付的官方文檔有點坑茄厘,好久都沒更新了矮冬,遇到了不少坑。當(dāng)然也許只我技術(shù)不行次哈。調(diào)起支付的參數(shù)出了簽名(sign)外別的我們都是后臺弄好給我們胎署。其余關(guān)于在微信開放平臺注冊等就不說了
首先是集成SDK,根據(jù)微信支付官方文檔我用的是cocoapods,沒有什么問題,根據(jù)前兩步項目設(shè)置APPID和在appdelegate里面注冊APPID(WXApi.registerApp(weixinAppId, enableMTA: true)
)
然后接著在appdelegate里設(shè)置回調(diào)
//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)
}
上面三個方法我用到時候發(fā)現(xiàn)之后調(diào)用第三個窑滞,但是我都寫上去了琼牧。
然后設(shè)置appdelegate遵守WXApiDelegate協(xié)議
代理方法是支付結(jié)果的一個回調(diào)
func onResp(_ resp: BaseResp!) {
var detailMessage = String()
if resp.isKind(of: PayResp.self){
//支付返回的結(jié)果
switch resp.errCode {
case WXSuccess.rawValue:
detailMessage = "支付結(jié)果:成功!"
NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "success")
break
default:
detailMessage = "支付結(jié)果:失敗!" + "errCode\(resp.errCode)" + "errStr\(resp.errStr)"
MyLog(message: detailMessage)
NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "fail")
break
}
}
}
這樣appdelegate里的工作就結(jié)束了,接下來就是點擊支付按鈕進行支付的那個控制器里進行下面的工作了哀卫。
剛才那個支付結(jié)果的回調(diào)結(jié)果發(fā)的通知在這里接收巨坊。注冊通知
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)
//下面這個方法就是向自己的服務(wù)器請求發(fā)起支付需要的數(shù)據(jù)
getWeiXinPayNeedDataMethod()
}
//請求數(shù)據(jù),解析數(shù)據(jù)
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]){
// ##這里我把數(shù)據(jù)處理的代碼放到WeiXinPayManager里面去了,TimesOperations.getNetworkTime()是獲取當(dāng)前時間戳共啃,在這里獲取一次就行了占调,帶到后面去后面會用到
WeiXinPayManager.weiXinPayMethodWithDic(dic: dic, payTimeStr: TimesOperations.getNetworkTime())
}
這個控制器里還有一個剛才接收到接受支付結(jié)果通知的觀察者方法
@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
//進行一些支付完成后的操作暂题,比如刷新數(shù)據(jù),移除一些遮罩View等
}
sureAction.setValue(RGB(R: 107, G: 107, B: 199), forKey: "titleTextColor")
alertVC.addAction(sureAction)
self.present(alertVC, animated: true, completion: nil)
}
接下來就是數(shù)據(jù)處理的部分了究珊,坑就在這里薪者,主要是生成一個簽名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: "參數(shù)出錯啦", 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)
}
到這差不多就可以經(jīng)行支付了。當(dāng)然還有一個重要的Sign的生成還沒寫
關(guān)于生成Sign,官方文檔是這么寫的:
第一步設(shè)所有發(fā)送或者接收到的數(shù)據(jù)為集合M瑟蜈,將集合M內(nèi)非空參數(shù)值的參數(shù)按照參數(shù)名ASCII碼從小到大排序(字典序)烟逊,使用URL鍵值對的格式(即key1=value1&key2=value2…)拼接成字符串stringA。
雖然下面也列舉了一些參數(shù)铺根,但是這里并沒有真正告訴我們要傳哪些宪躯,根據(jù)他列舉的參數(shù)我搞了好久,毛用沒有位迂,都是只出現(xiàn)一個白色確定按鈕访雪,這個就說明是這個簽名Sign錯了,后來還是安卓的同事提示慢慢試出來了掂林,注意參數(shù)是"appid","noncestr","prepayid"臣缀,package,timestamp泻帮,partnerid精置,這6個,其中partnerid是商戶id,可以讓后臺返回锣杂,也可以在自己申請的微信支付平臺賬號中找到脂倦,直接寫死,并且所有的參數(shù)不用加下劃線元莫,不要像官方文檔那樣使用mch_id這種拼接赖阻,注意mch_id用不上,下面就是要將參數(shù)按照從小到大排序(文檔要求的)然后拼接了柒竞,(sign也可以服務(wù)端生成,服務(wù)端調(diào)用統(tǒng)一下單接口后獲取到預(yù)支付id,再進行二次簽名得到我們需要的sign:參考(https://blog.csdn.net/sinat_17820239/article/details/51334181?utm_source=blogxgwz9))因為我是直接從后臺給的數(shù)據(jù)里去取參數(shù)播聪,需要過濾不需要的key,也許把問題弄復(fù)雜了朽基,反正這里要做的就是將這些Key從小到大排序,對應(yīng)的Value也是一樣布隔,讓后按照”key1=value1&key2=value2…“拼接。從小到大排序我用的是swift中的方法
var sortedKeyArray = newKeys.sorted { ( key1 : String, key2 : String) -> Bool in return key1 < key2 }
每個人拼接的方法思路不一樣稼虎,我就不獻丑了衅檀,好了,假設(shè)把這6個Key和對應(yīng)的Value按照URL鍵值對的格式(即key1=value1&key2=value2…)拼接成了stringA霎俩,接著拼接行一個申請得到的weixinKey,這個key在微信支付開發(fā)者賬號里找
fileprivate func getWeiXinSignStrMethod(dic : [String : Any],payTimeStr : String?)->String?{
//1.先自己拼接得到了stringA哀军,假設(shè)是stringA = "appid="123"&noncestr="123&package="Sign=WXPay"&partnerid="123"&prepayid="123"×tamp= 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微信支付就做完了,技術(shù)有限柳击,有做的不對的請指教猿推!下篇記錄關(guān)于Swift支付寶支付的集成