iOS 微信登录授权流程
2021-03-24 本文已影响0人
未来々人生
前言:最近项目中使用了微信第三方登录。做一下记录
准备工作:
微信开放平台账号申请,应用配置
确保你的应用在微信开发平台上注册创建并获得对应的接口,具体申请流程不困难,按流程走就可以了,就不再这里说了。(最好预留3个工作日的时间)
申请完之后,确认你的应用已有微信登录接口(这个功能是需要另外申请开通的),应用审核通过后,微信会分配给我们AppID和AppSecret,这两个备份下,晚些会用。
微信官方文档说明(文档其实已经说明的很明了,具体点击链接看)
大体流程
1.第三方发起微信授权登录请求,微信用户允许授权第三方应用后,微信会拉起应用或重定向到第三方网站,并且带上授权临时票据code参数;
2.通过code参数加上AppID和AppSecret等,通过API换取access_token;
3.通过access_token和openId进行接口调用,获取用户基本数据资源或帮助用户实现基本操作。
实现逻辑:
导入微信的开发工具包
如果没有微信支付的话只下载没有支付的工具包就可以了,最新的是有四个文件,下载后导入到项目中。
libWeChatSDK.a
、WechatAuthSDK.h
、WXApi.h
、 WXApiObject.h
- 开始接入
配置全局Define
#define KNotificationWXLOGIN_AUTHORIZED @"KNotificationWXLOGIN_AUTHORIZED"
#define KNotificationWXLOGIN_USERCANCELLED @"KNotificationWXLOGIN_USERCANCELLED"
1.在AppDelegate中导入#import "WXApi.h"并添加代理WXApiDelegate
这个方法要不添加的话,其他的所有回调是都不会触发的。
- (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray<id<UIUserActivityRestoring>> * _Nullable))restorationHandler{
return [WXApi handleOpenUniversalLink:userActivity delegate:self];
}
-(void)onResp:(BaseResp *)resp {
NSLog(@"resp %d",resp.errCode);
if ([resp isKindOfClass: [SendAuthResp class]]) {
SendAuthResp* authResp = (SendAuthResp*)resp;
if (authResp.errCode == 0) {
// 用户确认授权,发送全局通知,处理逻辑
[[NSNotificationCenter defaultCenter] postNotificationName: KNotificationWXLOGIN_AUTHORIZED
object: authResp];
} else {
// 用户取消授权,发送全局通知,处理逻辑
[[NSNotificationCenter defaultCenter] postNotificationName: KNotificationWXLOGIN_USERCANCELLED
object: nil];
}
}
}
2.在登录页面注册微信登录成功通知和失败通知,用来处理微信返回的逻辑.首先通过isWXAppInstalled判断手机是否安装了微信,安装正常操作,没有安装的话隐藏微信登录按钮。要不然苹果审核会有被拒的可能。
if ([WXApi isWXAppInstalled]) {
self.wechatLoginBtn.hidden = NO;
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:KNotificationWXLOGIN_USERCANCELLED object:nil] subscribeNext:^(NSNotification * _Nullable x) {
GLLog(@"cancle");
}];
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:KNotificationWXLOGIN_AUTHORIZED object:nil] subscribeNext:^(NSNotification * _Nullable x) {
[self wechat_loginAuthorized:x completion:^(BOOL success, NSDictionary *resp) {
if (success) {
//登录成功
}
}];
}];
} else {
self.wechatLoginBtn.hidden = YES;
}
3.在登录页面导入#import "WXApi.h"并添加代理WXApiDelegate
typedef void(^SuccessBlock)(BOOL success, NSDictionary *resp);
3.1点击唤起微信授权
- (void)weChatLoginClick{
[WXApi sendAuthReq:[self sendAuthRequest] viewController:self delegate:self completion:nil];
}
//开发者需要配合使用微信开放平台提供的 SDK 进行授权登录请求接入。正确接入 SDK 后并拥有相关授权域(scope)权限后,开发者移动应用会在终端本地拉起微信应用进行授权登录,微信用户确认后微信将拉起开发者移动应用,并带上授权临时票据(code)。
-(SendAuthReq *)sendAuthRequest
{
SendAuthReq *req = [[SendAuthReq alloc] init];
req.scope = @"snsapi_userinfo";
req.state = @"wx_oauth_authorization_state";
return req;
}
3.2接收微信返回通知,处理微信授权结果,授权成功后会返回一个NSDictionary,里面有我们需要的字段。
- (void)wechat_loginAuthorized:(NSNotification*)notification completion:(SuccessBlock)completion
{
//获取到code
SendAuthResp *resp = notification.object;
// _code = resp.code;
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
NSString *url = [NSString stringWithFormat:@"https://api.weixin.qq.com/sns/oauth2/access_token?appid=%@&secret=%@&code=%@&grant_type=%@",kAppKey_Wechat,kSecret_Wechat,resp.code,@"authorization_code"];
manager.requestSerializer = [AFJSONRequestSerializer serializer];
[manager.requestSerializer setValue:@"text/html; charset=utf-8" forHTTPHeaderField:@"Content-Type"];
NSMutableSet *mgrSet = [NSMutableSet set];
mgrSet.set = manager.responseSerializer.acceptableContentTypes;
[mgrSet addObject:@"text/html"];
[mgrSet addObject:@"text/plain"];
[mgrSet addObject:@"application/json"];
manager.responseSerializer.acceptableContentTypes = mgrSet;
[manager GET:url parameters:nil headers:nil progress:^(NSProgress * _Nonnull downloadProgress) {
} success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSLog(@"微信授权成功");
NSDictionary *resp = (NSDictionary*)responseObject;
NSLog(@"%@",resp);
NSString *openid = resp[@"openid"];
NSString *unionid = resp[@"unionid"];
NSString *accessToken = resp[@"access_token"];
NSString *refreshToken = resp[@"refresh_token"];
//调用获取用户详细信息方法,获取用户信息;
[self getWechatUserInfo:completion accessToken:accessToken openId:openid];
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
completion(NO,nil);
}];
}
3.3授权成功后,获取用户详细信息
-(void)getWechatUserInfo:(SuccessBlock)completion accessToken:(NSString *)accessToken openId:(NSString *)openId
{
//获取个人信息
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
NSString *url = [NSString stringWithFormat:@"https://api.weixin.qq.com/sns/userinfo?access_token=%@&openid=%@",accessToken,openId];
NSMutableSet *mgrSet = [NSMutableSet set];
mgrSet.set = manager.responseSerializer.acceptableContentTypes;
[mgrSet addObject:@"text/html"];
//因为微信返回的参数是text/plain 必须加上 会进入fail方法
[mgrSet addObject:@"text/plain"];
[mgrSet addObject:@"application/json"];
manager.responseSerializer.acceptableContentTypes = mgrSet;
[manager GET:url parameters:nil headers:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSLog(@"获取用户信息成功");
NSLog(@"%@",responseObject);
NSDictionary *resp = (NSDictionary*)responseObject;
completion(YES,resp);
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
NSLog(@"fail");
NSLog(@"%@",task.response);
completion(NO,nil);
}];
}
整个微信登录授权流程就是这个样子了。
原文来自https://www.jianshu.com/p/6d31d849305a