本篇在基於以前封裝的SocialSDK的項目上增長了QQ SDK的登陸受權和分享。介紹了QQ登陸分享的接入和使用注意事項。java
具體代碼項目Github地址:github.com/tsy12321/So…android
系列一 Android SDK的二次封裝和使用
系列二 源碼解析
系列三 微信SDK接入
系列四 QQ SDK接入
系列五 新浪微博 SDK接入bfd)git
QQ的官方接入文檔:github
wiki.open.qq.com/wiki/%E7%A7…json
詳細看裏面的這個部分:微信
在這裏先說回調通知,由於qq的登陸和分享的回調通知使用的是一個機制。即在onActivityResult中調用Tencent.onActivityResultData。網絡
因此咱們將onActivityResult也放入SocialApi中,activity的onActivityResult中直接調用SocialApi便可:app
mSocialApi.onActivityResult(requestCode, resultCode, data);複製代碼
而後在mSocialApi中再分別調用各個SSOHandler的onActivityResult,這樣就能夠在QQHandler中實現Tencent.onActivityResultData。ide
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
Tencent.onActivityResultData(requestCode, resultCode, data, null);
}複製代碼
先初始化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());
}
});複製代碼
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<String> 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<String> 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相關的登陸和分享。