2017-02-18日更新
1.1 此文章是继iOS-微信支付(一)前戏之后的文章,有疑问,请回顾前一篇文章 1.2 微信支付签名、加密都在服务器端做,此篇文章只调用服务器接口获取(partnerId、prepayId、nonceStr、timeStamp、sign)数据,然后直接调起微信支付。(网上看到好多大神,把签名、加密等方法都放在app端做,不明白目的是什么,其实微信也是提倡把这些事情都放在服务端去做,参见APP端开发步骤)
01-app端开发步骤.png
1.3 微信的DEMO真的很傻逼,把N多功能都集成到这个DEMO里面,不知为何。而且2017-02-18下载的DEMO(有xcworkspace),居然没有(V3&V4支付流程实现)这段代码了,幸好之前的保留了一份,缺少的代码见下图:
02-微信DEMO缺少代码示例.png
1.4 我的工程中因为集成了友盟第三方(登录、分享)等功能,所以微信支付的SDK就可以不用再导入到项目中了(导入也会重复报错的)需要下载SDK的同学点击这里:微信支付SDK(点击iOS开发工具包即可下载)
03-微信支付SDK.png
1.5 建议写代码之前,先看下微信支付的业务逻辑,可以少走一些弯路微信支付业务流程
04-微信支付业务流程图.png
04-微信支付业务逻辑简化版.png
2.1 项目设置APPID
05-设置项目APPID.png
2.2 导入微信支付SDK(我的工程中因为集成了友盟第三方(登录、分享)等功能,所以微信支付的SDK就可以不用再导入到项目中了)
06-微信支付SDK介绍.png
2.3 工程中链接上(因为我的项目里面用到的东西比较多,所以我引入的库也相对多一些,大家可以不用导入我这么多的)
SystemConfiguration.framework libz.dylib libsqlite3.0.dylib libc++.dylib Security.framework CoreTelephony.framework CFNetwork.framework
07-引入一些库.png
2.4 注册微信的APPID - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // 向微信注册wxd930ea5d5a258f4f [WXApi registerApp:@"wxb4ba3c02aa476ea1" withDescription:@"demo 2.0"];
return YES;
}
2.5 将WXApiManager.h
和WXApiManager.m
拖入到工程里面(能用工具类,最好用工具类,不然AppDelegate里面代码会越来越多)
08-将WXApiManager拖入到项目中.png
2.6 调起微信支付
/**
* 点击微信支付按钮,走下面的方法
* 去服务器请求数据(partnerId、prepayId、nonceStr、timeStamp)
*/
[[HQNetworkTools sharedTools] requestPingTaiDataWithRootCode:rootCode parameters:para finished:^(id result, NSError *error) {
if (error) {
HQLog(@"%@", error);
} else {
[MBProgressHUD hideHUD];
int success = [result[@"success"] intValue];
if (success == 1) {
HQBaseModel *model = [HQBaseModel mj_objectWithKeyValues:result[@"data"]];
if ([payType_str isEqualToString:@"1"]) { // wepay
PayReq *request = [[PayReq alloc] init];
request.partnerId = model.partnerid;
request.prepayId = model.prepayid;
request.package = @"Sign=WXPay";
request.nonceStr = model.noncestr;
// 注意时间戳一定是10位的(最开始我们后台返回的就是13位的,一直报错),而且后台返回的是String,要转换类型
request.timeStamp = [model.timestamp intValue];
request.sign = model.sign;
/**
* 调起微信支付的方法
*/
[WXApi sendReq:request];
} else if ([payType_str isEqualToString:@"2"]) { // alipay
}
} else {
[MBProgressHUD showText:result[@"errorInfo"]];
}
}
}];
2.7 处理微信支付回调AppDelegate
里操作
2.7.1 首先在AppDelegate
里面导入头文件
#import <WXApi.h>
#import "WXApiManager.h"
2.7.2 其次,处理微信回调,调用handleOpenURL
方法,因为随着iOS的升级,有的方法废弃了,这里建议,下面三个方法里面都增加上,以防万一
#pragma mark - 设置微信回调
// 支持所有iOS系统(被废弃的方法.但是在低版本中会用到.建议写上)
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
{
/**
* 微信支付回调
*/
return [WXApi handleOpenURL:url delegate:[WXApiManager sharedManager]];
}
// 仅支持iOS9以上系统,iOS8及以下系统不会回调
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey, id> *)options
{
/**
* 微信支付回调
*/
return [WXApi handleOpenURL:url delegate:[WXApiManager sharedManager]];
}
// 支持目前所有iOS系统
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url
{
/**
* 微信支付回调
*/
return [WXApi handleOpenURL:url delegate:[WXApiManager sharedManager]];
}
2.7.3 在WXApiManager
中处理支付成功或者失败的回调(我这里是给支付控制器发送通知,然后跳转到"已付款订单页面"或者"待付款订单页面",详见下一篇文章iOS-微信支付(三)结局)
#pragma mark - WXApiDelegate
- (void)onResp:(BaseResp *)resp
{
if([resp isKindOfClass:[PayResp class]]) {
// 支付返回结果,实际支付结果需要去微信服务器端查询
NSString *strMsg = [NSString stringWithFormat:@"支付结果"];
switch (resp.errCode) {
case WXSuccess:
strMsg = @"支付结果:成功!";
HQLog(@"支付成功-PaySuccess,retcode = %d", resp.errCode);
/** 发送支付成功的通知 */
[[NSNotificationCenter defaultCenter] postNotificationName:HQPaySuccessNotification object:nil userInfo:nil];
break;
default:
strMsg = [NSString stringWithFormat:@"支付失败"];
HQLog(@"错误,retcode = %d, retstr = %@", resp.errCode,resp.errStr);
/** 发送支付失败的通知 */
[[NSNotificationCenter defaultCenter] postNotificationName:HQCancelPayNotification object:nil userInfo:nil];
break;
}
}
}