本篇在基于之前封装的SocialSDK的项目上增加了QQ SDK的登录授权和分享。介绍了QQ登录分享的接入和使用注意事项。
具体代码项目Github地址:https://github.com/tsy12321/SocialSDKAndroid
0 系列文章
系列一 Android SDK的二次封装和使用
系列二 源码解析
系列三 微信SDK接入
系列四 QQ SDK接入
系列五 新浪微博 SDK接入
1 官方文档
QQ的官方接入文档:
http://wiki.open.qq.com/wiki/%E7%A7%BB%E5%8A%A8%E5%BA%94%E7%94%A8%E6%8E%A5%E5%85%A5wiki%E7%B4%A2%E5%BC%95
详细看里面的这个部分:
2 回调通知
在这里先说回调通知,因为qq的登录和分享的回调通知使用的是一个机制。即在onActivityResult中调用Tencent.onActivityResultData。
所以我们将onActivityResult也放入SocialApi中,activity的onActivityResult中直接调用SocialApi即可:
mSocialApi.onActivityResult(requestCode, resultCode, data);
然后在mSocialApi中再分别调用各个SSOHandler的onActivityResult,这样就可以在QQHandler中实现Tencent.onActivityResultData。
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
Tencent.onActivityResultData(requestCode, resultCode, data, null);
}
2 登录授权
2.1 Android 接入授权代码
先初始化mTencent:
this.mTencent = Tencent.createInstance(mConfig.appId, mContext);
授权登录时调用:
this.mTencent.login(this.mActivity, "all", new IUiListener() {
@Override
public void onComplete(Object o) {
if (null == o || ((JSONObject)o) == null) {
LogUtils.e("onComplete response=null");
mAuthListener.onError(mConfig.getName(), "onComplete response=null");
return;
}
JSONObject response = (JSONObject) o;
initOpenidAndToken(response);
mAuthListener.onComplete(mConfig.getName(), Utils.jsonToMap(response));
mTencent.logout(mActivity);
}
@Override
public void onError(UiError uiError) {
String errmsg = "errcode=" + uiError.errorCode + " errmsg=" + uiError.errorMessage + " errdetail=" + uiError.errorDetail;
LogUtils.e(errmsg);
mAuthListener.onError(mConfig.getName(), errmsg);
}
@Override
public void onCancel() {
mAuthListener.onCancel(mConfig.getName());
}
});
3 分享
QQ分享只能有网页分享、图片分享、音乐分享。
QQ空间分享只能有网页分享。
所以在判断到其他媒介分享时要直接返回错误。
if(this.mShareListener != null) {
this.mShareListener.onError(this.mConfig.getName(), "shareMedia error");
}
return ;
具体的分享代码很简单,参考文档就行(代码里也有,媒介判断机制跟微信一样)。在这我要提一点开发中遇到的坑。
在QQ分享时,当图片地址放的是本地图片地址时,对应的KEY应该是SHARE_TO_QQ_IMAGE_LOCAL_URL,如果是网络图片地址,对应的KEY是SHARE_TO_QQ_IMAGE_URL。
params.putInt(QQShare.SHARE_TO_QQ_KEY_TYPE, QQShare.SHARE_TO_QQ_TYPE_DEFAULT);
params.putString(QQShare.SHARE_TO_QQ_TITLE, shareWebMedia.getTitle());
params.putString(QQShare.SHARE_TO_QQ_SUMMARY, shareWebMedia.getDescription());
params.putString(QQShare.SHARE_TO_QQ_TARGET_URL, shareWebMedia.getWebPageUrl());
params.putString(QQShare.SHARE_TO_QQ_IMAGE_LOCAL_URL, path);
但是在QQ空间分享时,无论图片是本地地址还是网络地址KEY都只能设置为SHARE_TO_QQ_IMAGE_URL。
params.putInt(QzoneShare.SHARE_TO_QZONE_KEY_TYPE, QzoneShare.SHARE_TO_QZONE_TYPE_IMAGE_TEXT);
params.putString(QzoneShare.SHARE_TO_QQ_TITLE, shareWebMedia.getTitle());
params.putString(QzoneShare.SHARE_TO_QQ_SUMMARY, shareWebMedia.getDescription());
params.putString(QzoneShare.SHARE_TO_QQ_TARGET_URL, shareWebMedia.getWebPageUrl());
ArrayList path_arr = new ArrayList<>();
path_arr.add(path);
params.putStringArrayList(QzoneShare.SHARE_TO_QQ_IMAGE_URL, path_arr); //!这里是大坑 不能用SHARE_TO_QQ_IMAGE_LOCAL_URL
这个坑一开始我以为是本地图片的读取权限之类的,直到后面痛苦的读QQ SDK里面混淆过的代码才发现在QZoneShare这个文件中:
public void shareToQzone(final Activity var1, final Bundle var2, final IUiListener var3) {
...
ArrayList var7 = var2.getStringArrayList("imageUrl");
...
if(SystemUtils.compareQQVersion(var1, "4.6.0") >= 0) {
f.c("openSDK_LOG.QzoneShare", "shareToQzone() qqver greater than 4.6.0");
a.a(var1, var7, new AsynLoadImgBack() {
public void saved(int var1x, String var2x) {
}
public void batchSaved(int var1x, ArrayList var2x) {
if(var1x == 0) {
var2.putStringArrayList("imageUrl", var2x);
}
QzoneShare.this.a(var1, var2, var3);
}
});
}
}
发现上面,它只从“imageUrl”这个key里面取pic数据,QzoneShare.this.a(var1, var2, var3)这句就是具体调用起QQ空间分享的代码。
public static final String SHARE_TO_QQ_IMAGE_URL = "imageUrl";
public static final String SHARE_TO_QQ_IMAGE_LOCAL_URL = "imageLocalUrl";
所以,如果填写SHARE_TO_QQ_IMAGE_LOCAL_URL这个key,那var7里面肯定是null,a.a(var1, var7, new AsynLoadImgBack()这句就会执行到saved这个函数,也就执行不到batchSaved的QzoneShare.this.a(var1, var2, var3)函数,就无法调用起分享。
坑!!!!! 就这样了O。O。
总结,QQ SDK的设计感觉不如微信的感觉好。喷一口。
结尾
以上即实现了QQ的接入,当接入方需要接入QQ部分时,只需要同时将social_sdk.jar和qq相关的sdk包同时引用进项目,即可调用qq相关的登录和分享。