IOS 第三方登录集成
微信登录授权开发
1,到微信开发平台注册相关APP,现在是等待审核成功后才能获取到对应的key和secret;获取成功后需要单独申请开通登录和支付接口,如图

2,和QQ类似,需要填写Url Schemes,如demo中的wxd930ea5d5a258f4f ,然后引入相应framework;
3,在AppDelegate中注册和实现授权后的回调函数,代码如下:
//向微信注册
[WXApi registerApp:kWXAPP_ID withDescription:@"weixin"];
//授权后回调 WXApiDelegate
-(void)onResp:(BaseReq *)resp
{
/*
ErrCode ERR_OK = 0(用户同意)
ERR_AUTH_DENIED = -4(用户拒绝授权)
ERR_USER_CANCEL = -2(用户取消)
code 用户换取access_token的code,仅在ErrCode为0时有效
state 第三方程序发送时用来标识其请求的唯一性的标志,由第三方程序调用sendReq时传入,由微信终端回传,state字符串长度不能超过1K
lang 微信客户端当前语言
country 微信用户当前国家信息
*/
SendAuthResp *aresp = (SendAuthResp *)resp;
if (aresp.errCode== 0) {
NSString *code = aresp.code;
NSDictionary *dic = @{@"code":code};
}
}
//和QQ,新浪并列回调句柄
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
{
return [TencentOAuth HandleOpenURL:url] ||
[WeiboSDK handleOpenURL:url delegate:self] ||
[WXApi handleOpenURL:url delegate:self];;
}
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url
{
return [TencentOAuth HandleOpenURL:url] ||
[WeiboSDK handleOpenURL:url delegate:self] ||
[WXApi handleOpenURL:url delegate:self];;
}
4,微信登录授权比较复杂,相比QQ,新浪多了几步,简单说就是需要三步,第一步,获取code,这个用来获取token,第二步,就是带上code获取token,第三步,根据第二步获取的token和openid来获取用户的相关信息;
下面用代码来实现:
第一步:code
- (IBAction)weixinLogin:(id)sender
{
[self sendAuthRequest];
}
-(void)sendAuthRequest
{
SendAuthReq* req =[[SendAuthReq alloc ] init];
req.scope = @"snsapi_userinfo,snsapi_base";
req.state = @"0744" ;
[WXApi sendReq:req];
}
这里获取后会调用之前在AppDelegate里面的对应oauthResp回调,获得得到的code。
第二步:token和openid
-(void)getAccess_token
{
//https://api.weixin.qq.com/sns/oauth2/access_token?appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code
NSString *url =[NSString stringWithFormat:@"https://api.weixin.qq.com/sns/oauth2/access_token?appid=%@&secret=%@&code=%@&grant_type=authorization_code",kWXAPP_ID,kWXAPP_SECRET,self.wxCode.text];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSURL *zoneUrl = [NSURL URLWithString:url];
NSString *zoneStr = [NSString stringWithContentsOfURL:zoneUrl encoding:NSUTF8StringEncoding error:nil];
NSData *data = [zoneStr dataUsingEncoding:NSUTF8StringEncoding];
dispatch_async(dispatch_get_main_queue(), ^{
if (data) {
NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
/*
{
"access_token" = "OezXcEiiBSKSxW0eoylIeJDUKD6z6dmr42JANLPjNN7Kaf3e4GZ2OncrCfiKnGWiusJMZwzQU8kXcnT1hNs_ykAFDfDEuNp6waj-bDdepEzooL_k1vb7EQzhP8plTbD0AgR8zCRi1It3eNS7yRyd5A";
"expires_in" = 7200;
openid = oyAaTjsDx7pl4Q42O3sDzDtA7gZs;
"refresh_token" = "OezXcEiiBSKSxW0eoylIeJDUKD6z6dmr42JANLPjNN7Kaf3e4GZ2OncrCfiKnGWi2ZzH_XfVVxZbmha9oSFnKAhFsS0iyARkXCa7zPu4MqVRdwyb8J16V8cWw7oNIff0l-5F-4-GJwD8MopmjHXKiA";
scope = "snsapi_userinfo,snsapi_base";
}
*/
self.access_token.text = [dic objectForKey:@"access_token"];
self.openid.text = [dic objectForKey:@"openid"];
}
});
});
}
利用GCD来获取对应的token和openID.
第三步:userinfo
-(void)getUserInfo
{
// https://api.weixin.qq.com/sns/userinfo?access_token=ACCESS_TOKEN&openid=OPENID
NSString *url =[NSString stringWithFormat:@"https://api.weixin.qq.com/sns/userinfo?access_token=%@&openid=%@",self.access_token.text,self.openid.text];
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSURL *zoneUrl = [NSURL URLWithString:url];
NSString *zoneStr = [NSString stringWithContentsOfURL:zoneUrl encoding:NSUTF8StringEncoding error:nil];
NSData *data = [zoneStr dataUsingEncoding:NSUTF8StringEncoding];
dispatch_async(dispatch_get_main_queue(), ^{
if (data) {
NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
/*
{
city = Haidian;
country = CN;
headimgurl = "http://wx.qlogo.cn/mmopen/FrdAUicrPIibcpGzxuD0kjfnvc2klwzQ62a1brlWq1sjNfWREia6W8Cf8kNCbErowsSUcGSIltXTqrhQgPEibYakpl5EokGMibMPU/0";
language = "zh_CN";
nickname = "xxx";
openid = oyAaTjsDx7pl4xxxxxxx;
privilege = (
);
province = Beijing;
sex = 1;
unionid = oyAaTjsxxxxxxQ42O3xxxxxxs;
}
*/
self.nickname.text = [dic objectForKey:@"nickname"];
self.wxHeadImg.image = [UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:[dic objectForKey:@"headimgurl"]]]];
}
});
});
}
执行到这一步就算完成了整个授权登录的功能,能把昵称和头像显示出来,剩下的就是及时刷新你的token,详情可参考开发文档。
QQ登录授权开发:
1,按照开发文档导入SDK,然后把注册成功后获取到的Key加入到Url Schemes中,例如:tencent1101737816,注意这些文档写的很清楚;
2,在AppDelegate.m中实现下面方法,注意,直接复制代码,两者写其一即可,当然了别忘了引入头文件,这是用作返回APP时候用的回调函数
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
return [TencentOAuth HandleOpenURL:url];
}
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url {
return [TencentOAuth HandleOpenURL:url];
}
3,注册并授权,注意授权这里会出现问题
_tencentOAuth = [[TencentOAuth alloc] initWithAppId:kQQ_KEY andDelegate:self]; //注册
NSArray *_permissions = [NSArray arrayWithObjects:kOPEN_PERMISSION_GET_INFO, kOPEN_PERMISSION_GET_USER_INFO, kOPEN_PERMISSION_GET_SIMPLE_USER_INFO, nil nil]; [_tencentOAuth authorize:_permissions inSafari:NO]; //授权
注意:在官方文档中提到授权代码是:
_permissions = [[NSArray arrayWithObjects:@"get_user_info", @"add_t", nil] retain]; //官方授权代码
如果你写成这样,你会发现始终登录不了,提示权限不够,非常坑爹,正确授权代码如上面的写法,切记
4,实现登录成功与否的回调函数,实现协议TencentSessionDelegate,代码如下:
- (void)tencentDidLogin {
_labelTitle.text = @"登录完成";
if (_tencentOAuth.accessToken && 0 != [_tencentOAuth.accessToken length]){
// 记录登录用户的OpenID、Token以及过期时间
_labelAccessToken.text = _tencentOAuth.accessToken;
}else{
_labelAccessToken.text = @"登录不成功 没有获取accesstoken";
}
}
-(void)tencentDidNotLogin:(BOOL)cancelled {
if (cancelled){
_labelTitle.text = @"用户取消登录";
}else{
_labelTitle.text = @"登录失败";
}
}
-(void)tencentDidNotNetWork {
_labelTitle.text=@"无网络连接,请设置网络";
}
5,如果代码能走到登录完成这里,并且获得了accesstoken,那说明我们的登录流程就走完了,剩下的旧是获取用户具体的信息了,如昵称,城市等等;
6,如何获取昵称呢,文档提供了这样一个方法
[_tencentOAuth getUserInfo];
但是返回类型是个BOOL类似,很多朋友就郁闷了,这是什么情况,经过看代码,发现如果开发者调用了这个方法,如果成功则会实现下面协议TencentSessionDelegate里面的回调函数:
-(void)getUserInfoResponse:(APIResponse *)response {
// NSLog(@"respons:%@",response.jsonResponse);
self.name.text = [response.jsonResponse objectForKey:@"nickname"];
}
这样就能得到我们想要的昵称等信息,到这里就算完成了我们的QQ登录授权了,剩下的就是发给后台这些登录信息,并且保持授权成功后的token等,看官方文档说明。
新浪微博登录授权开发:
1,同样和QQ一样,配置Url Schemes,如:wb2045436852,然后根据官方文档接入SDK等信息;
2,同样在AppDelegate.m 中实现下面回调:
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
return [TencentOAuth HandleOpenURL:url] || [WeiboSDK handleOpenURL:url delegate:self];
}
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url {
return [TencentOAuth HandleOpenURL:url] || [WeiboSDK handleOpenURL:url delegate:self];
}
3,关于新浪微博的成功登录回调是写在该类中来实现,因为这里写入了self,如果你需要到别的类中实现,可在这里做修改即可
[WeiboSDK handleOpenURL:url delegate:self];
实现的协议为:WeiboSDKDelegate,协议方法:
-(void)didReceiveWeiboResponse:(WBBaseResponse *)response {
if ([response isKindOfClass:WBAuthorizeResponse.class]) {
if ((int)response.statusCode == 0) {
NSDictionary *dic = @{@"userID":[(WBAuthorizeResponse *)response userID], @"accessToken" :[(WBAuthorizeResponse *)response accessToken]};
}
}
}
如果能得到这个userID,token就说明登录成功了。
4,登录代码:
- (IBAction)sinaLogin:(id)sender {
[WeiboSDK enableDebugMode:YES];
[WeiboSDK registerApp:kSINA_KEY];
WBAuthorizeRequest *request = [WBAuthorizeRequest request];
request.redirectURI = kRedirectURI; //此处是微博开放平台上面配置的回调URL
request.scope = @"all";
request.userInfo = @{@"myKey": @"myValue"};
[WeiboSDK sendRequest:request];
}
5.获取用户信息
在获取到userID,token之后可以进一步获取用户信息
-(void)getWBUserInfo
{
NSString * oathString = @"https://api.weibo.com/2/users/show.json";
NSDictionary *parameters = @{@"uid": openid,@"access_token":access_token,@"source":wbAPIKey};
[WBHttpRequest requestWithAccessToken:access_token
url:oathString
httpMethod:@"GET"
params:parameters
delegate:self
withTag:@"userInfo"];
}
微博返回用户信息之后会回调这个函数
- (void)request:(WBHttpRequest *)request didFinishLoadingWithDataResult:(NSData *)data
{
NSDictionary * userInfo = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];//转换数据格式
*用户信息
*
}
浙公网安备 33010602011771号