示例代码在末尾。
示例代码在末尾。
今天用了Facebook的SDK,给app集成分享功能。本以为只要照着官方文档来,一定能成功,没想到,,确实成功了。
但是,且慢!!!
FB给的示例代码是这样的
FBSDKShareLinkContent *content = [[FBSDKShareLinkContent alloc] init];
content.contentURL = [NSURL URLWithString:@"https://developers.facebook.com"];
[FBSDKShareDialog showFromViewController:self withContent:content delegate:nil];
代码简介明了, 先创建一个分享内容对象content,再vc弹出一个分享对话框。
可是我发现在相同的环境下,同样的代码,却有不同的效果:有两台iPhone,都装了Facebook原生app。同样的SDK和代码,但有台手机调用的是原生的分享对话框,另一台调用的却是SafariViewController(webview)来加载分享对话框。为什么?
好吧,这能忍能忍。。不过是分享的方式不一样罢了,两台手机都能正常分享内容的。然而我有这么个需求:分享结束之后,我希望知道结果,是分享成功还是失败?或是取消了。
这个简单,[FBSDKShareDialog showFromViewController:withContent:delegate:];方法可设一个代理来响应对话框的回调,正好!该代理可以实现三个回调方法,
- (void)sharer:(id<FBSDKSharing>)sharer didCompleteWithResults:(NSDictionary *)results;
- (void)sharer:(id<FBSDKSharing>)sharer didFailWithError:(NSError *)error;
- (void)sharerDidCancel:(id<FBSDKSharing>)sharer;
鸡冻啊!这不正是我想要的吗?
测了一下,所有的方法都能正常回调,打完收工。
但是,又且慢!
我发现如果使用webview来分享,如果用户没有确认分享,甚至是页面都还没有加载完,只要点击了左上角的『完成』按钮,就会回调FBSDKSharingDelegate 的sharer:didCompleteWithResults:方法。而成功分享也是回调这个方法,很逗的,直接点完成应该算取消分享啊。。。
好吧好吧,我们还可以从回调方法提供的参数results中读取出详细的信息,可以发现当用户真的分享成功时,results中有个postId字段,而点击『完成』按钮时,results是空的。
然而我又不小心测了下原生分享,遇到了另一个矛盾的情况:原生分享成功时,回调的results参数居然也是空的空的空的!哦啦个去!
那么如果我能区分出对话框的类型,也就可以准确的判断出成功和取消了吧?
对了,回调的方法中还有个sharer参数,可以强转为FBSDKShareDialog对象,它有一个mode属性,嘿嘿,我又要成功了!
mode是个枚举,有Automatic、Navtive、ShareSheet、Browser、Web......还好我没有高兴得太早,我发现不管我用的是原生分享还是webview分享,mode都是Automatic,实在是郁闷。
总算的,我不想偷懒了,我想起了在官方文档里看到的另一段代码
// Example content. Replace with content from your app.
FBSDKShareLinkContent *content = [[FBSDKShareLinkContent alloc] init];
content.contentURL = [NSURL URLWithString:@"https://developers.facebook.com"];
FBSDKShareDialog *dialog = [[FBSDKShareDialog alloc] init];
dialog.fromViewController = self;
dialog.content = content;
dialog.mode = FBSDKShareDialogModeShareSheet;
[dialog show];
我写了这段多了几行,长了一点的代码,我可以手动指定用原生对话框分享啊!!!
到这里,只要指定dialog.mode = FBSDKShareDialogModeNative;安装了FB app的用户都会调出原生app分享。啊!总算美好了!
等等,我有个疑问。如果用户没有安装app呢?
如果用户没有安装app而我指定用native分享,SDK并不会去调用webview分享,而是直接回调失败的方法。
So sad. 不过可以在失败的回调里重新指定一下mode,再重新打开对话框。
那么又要回到之前那个郁闷的结果了:如果调用了成功的回调,
如果有postId则一定成功,
如果没有postId,是使用原生的,成功,
如果没有postId,是使用webview的,失败
好在,这次,再读取sharer的mode时,能区别出原生或webview,至此,是真的真的真的完美了!
代码如下
- (void)facebookShareWithMessage:(id)message {
NSData *data = [message dataUsingEncoding:NSUTF8StringEncoding];
NSDictionary *dictionary = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
NSString *contentUrlString = dictionary[@"content_url"];
NSString *imageUrlString = dictionary[@"image_url"];
NSString *description = dictionary[@"description"];
NSString *title = dictionary[@"title"];
NSString *quote = dictionary[@"quote"];
FBSDKShareLinkContent *content = [[FBSDKShareLinkContent alloc] init];
content.contentURL = [NSURL URLWithString:contentUrlString];
content.imageURL = [NSURL URLWithString:imageUrlString];
content.contentDescription = description;
content.contentTitle = title;
content.quote = quote;
FBSDKShareDialog *dialog = [[FBSDKShareDialog alloc] init];
dialog.shareContent = content;
dialog.fromViewController = self;
dialog.delegate = self;
dialog.mode = FBSDKShareDialogModeNative;
[dialog show];
}
#pragma mark - FaceBook Share Delegate
- (void)sharer:(id<FBSDKSharing>)sharer didCompleteWithResults:(NSDictionary *)results {
NSString *postId = results[@"postId"];
FBSDKShareDialog *dialog = (FBSDKShareDialog *)sharer;
if (dialog.mode == FBSDKShareDialogModeBrowser && (postId == nil || [postId isEqualToString:@""])) {
// 如果使用webview分享的,但postId是空的,
// 这种情况是用户点击了『完成』按钮,并没有真的分享
NSLog(@"Cancel");
} else {
NSLog(@"Success");
}
}
- (void)sharer:(id<FBSDKSharing>)sharer didFailWithError:(NSError *)error {
FBSDKShareDialog *dialog = (FBSDKShareDialog *)sharer;
if (error == nil && dialog.mode == FBSDKShareDialogModeNative) {
// 如果使用原生登录失败,但error为空,那是因为用户没有安装Facebook app
// 重设dialog的mode,再次弹出对话框
dialog.mode = FBSDKShareDialogModeBrowser;
[dialog show];
} else {
NSLog(@"Failure");
}
}
- (void)sharerDidCancel:(id<FBSDKSharing>)sharer {
NSLog(@"Cancel");
}