社會化登入分享-QQ SDK接入

Tsy遠發表於2019-03-04

本篇在基於之前封裝的SocialSDK的專案上增加了QQ SDK的登入授權和分享。介紹了QQ登入分享的接入和使用注意事項。

具體程式碼專案Github地址:github.com/tsy12321/So…

0 系列文章

系列一 Android SDK的二次封裝和使用
系列二 原始碼解析
系列三 微信SDK接入
系列四 QQ SDK接入
系列五 新浪微博 SDK接入bfd)

1 官方文件

QQ的官方接入文件:

wiki.open.qq.com/wiki/%E7%A7…

詳細看裡面的這個部分:

QQ 官方文件
QQ 官方文件

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<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相關的登入和分享。

相關文章