1. 程式人生 > >Android移動開發-在Android應用裡整合QQ分享的實現

Android移動開發-在Android應用裡整合QQ分享的實現

QQ分享分為QQ好友分享與QQ空間分享同屬QQ互聯平臺上的QQ分享,該QQ互聯的網址為:https://connect.qq.com/ 。然後在該網址裡申請開發者應用id。

  • 建立工程並配置工程
  1. 新建工程並匯入SDK的jar檔案(SDK下載地址
    建立一個工程,並把open-sdk.jar檔案拷貝到libs目錄下並依賴到專案工程裡,如下圖所示:

這裡寫圖片描述

配置AndroidManifest
在應用的AndroidManifest.xml增加配置的節點下增加以下配置(注:不配置將會導致無法呼叫API);

<!-- QQ登入鑑權頁面 -->
<activity android:name="com.tencent.tauth.AuthActivity" android:launchMode="singleTask" android:noHistory="true"> <intent-filter> <action android:name="android.intent.action.VIEW" /> <category
android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" /> <data android:scheme="填寫tencent你的AppId" /> </intent-filter> </activity> <activity android:name
="com.tencent.connect.common.AssistActivity" android:configChanges="orientation|keyboardHidden|screenSize" android:screenOrientation="portrait" android:theme="@android:style/Theme.Translucent.NoTitleBar" />

通過以上兩個步驟,工程就已經配置完成了。接下來就可以在程式碼裡使用QQ互聯的SDK進行開發了。

  • layout/activity_share_qq.xml介面佈局程式碼如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="5dp" >

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"
        android:orientation="horizontal" >

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="分享標題:"
            android:textColor="@color/black"
            android:textSize="17sp" />

        <EditText
            android:id="@+id/et_share_title"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:padding="5dp"
            android:background="@drawable/editext_selector"
            android:text="CSDN部落格分享"
            android:textColor="@color/black"
            android:textSize="17sp" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp"
        android:orientation="horizontal" >

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="分享內容:"
            android:textColor="@color/black"
            android:textSize="17sp" />

        <EditText
            android:id="@+id/et_share_content"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:padding="5dp"
            android:background="@drawable/editext_selector"
            android:text="哈哈,這是我的CSDN部落格地址"
            android:textColor="@color/black"
            android:textSize="17sp" />
    </LinearLayout>

    <Button
        android:id="@+id/btn_share_qq"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="分享到QQ"
        android:textColor="@color/black"
        android:textSize="17sp" />

</LinearLayout>
  • layout/dialog_share.xml介面佈局程式碼如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:background="@color/white"
    android:orientation="vertical" >

    <GridView
        android:id="@+id/gv_share_channel"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:horizontalSpacing="0dp"
        android:verticalSpacing="0dp"
        android:listSelector="@null"
        android:numColumns="4"
        android:paddingTop="10dp"
        android:paddingBottom="10dp"
        android:stretchMode="columnWidth" />

    <TextView
        android:id="@+id/tv_share_cancel"
        android:layout_width="match_parent"
        android:layout_height="40dp"
        android:gravity="center"
        android:background="@color/blue_light"
        android:text="取 消"
        android:textColor="@color/black"
        android:textSize="17sp" />

</LinearLayout>
  • ShareQQActivity.java邏輯程式碼如下:
package com.fukaimei.shareqq;

import com.fukaimei.shareqq.adapter.ShareGridAdapter;
import com.fukaimei.shareqq.widget.ShareGridDialog;
import com.tencent.connect.common.Constants;
import com.tencent.tauth.Tencent;

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.EditText;

public class ShareQQActivity extends AppCompatActivity implements OnClickListener {
    private static final String TAG = "ShareQQActivity";
    private EditText et_share_title;
    private EditText et_share_content;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_share_qq);
        et_share_title = (EditText) findViewById(R.id.et_share_title);
        et_share_content = (EditText) findViewById(R.id.et_share_content);
        findViewById(R.id.btn_share_qq).setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        if (v.getId() == R.id.btn_share_qq) {
            ShareGridDialog dialog = new ShareGridDialog(this, null);
            dialog.setUrl("http://blog.csdn.net/fukaimei");
            dialog.setTitle(et_share_title.getText().toString());
            dialog.setContent(et_share_content.getText().toString());
            dialog.setImgUrl("");
            dialog.show();
        }
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        Log.d(TAG, "-->onActivityResult " + requestCode + " resultCode=" + resultCode);
        if (requestCode == Constants.REQUEST_LOGIN || requestCode == Constants.REQUEST_APPBAR) {
            Tencent.onActivityResultData(requestCode, resultCode, data, ShareGridAdapter.mLoginListener);
        } else if (requestCode == Constants.REQUEST_QQ_SHARE || requestCode == Constants.REQUEST_QZONE_SHARE) {
            Tencent.onActivityResultData(requestCode, resultCode, data, ShareGridAdapter.mShareListener);
        }
        super.onActivityResult(requestCode, resultCode, data);
    }

}
  • ShareGridAdapter.java邏輯程式碼如下:
package com.fukaimei.shareqq.adapter;

import java.util.ArrayList;

import org.json.JSONObject;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import android.util.Patterns;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.Toast;

import com.fukaimei.shareqq.R;
import com.fukaimei.shareqq.bean.ShareChanels;
import com.fukaimei.shareqq.util.CacheUtil;
import com.tencent.connect.common.Constants;
import com.tencent.connect.share.QQShare;
import com.tencent.connect.share.QzoneShare;
import com.tencent.tauth.IUiListener;
import com.tencent.tauth.Tencent;
import com.tencent.tauth.UiError;

public class ShareGridAdapter extends BaseAdapter implements OnItemClickListener {
    private final static String TAG = "ShareGridAdapter";
    private LayoutInflater mInflater;
    private static Context mContext;
    private Handler mHandler;
    private String mUrl;
    private String mTitle;
    private static String mContent;
    private static String mImageUrl;
    private ArrayList<ShareChanels> mChannelList;

    private final String QQ_APPID = "填寫QQ互聯平臺裡申請到的應用id"; // 這裡替換為開發者在QQ互聯平臺申請的應用id
    private static Tencent mTencent;

    private int QQ = 0;
    private int QZONE = 1;
    private int WEIBO = 2;
    private int[] mShareIcons = {R.drawable.logo_qq, R.drawable.logo_qzone,
            R.drawable.logo_tencentweibo};

    public ShareGridAdapter(final Context context, Handler handler, String url,
                            String title, String content, final String imageUrl,
                            ArrayList<ShareChanels> channelList) {
        mInflater = LayoutInflater.from(context);
        mContext = context;
        mHandler = handler;
        mUrl = url;
        mTitle = title;
        mContent = content;
        if (imageUrl != null && Patterns.WEB_URL.matcher(imageUrl).matches()) {
            new Thread(new Runnable() {
                public void run() {
                    try {
                        mImageUrl = CacheUtil.getImagePath(imageUrl,
                                CacheUtil.getFileCache(context));
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }).start();
        } else {
            mImageUrl = imageUrl;
        }

        if (channelList == null) {
            mChannelList = new ArrayList<ShareChanels>();
            mChannelList.add(new ShareChanels("QQ好友", QQ));
            mChannelList.add(new ShareChanels("QQ空間", QZONE));
        } else {
            mChannelList = channelList;
        }

        // 建立一個QQ例項,用於QQ分享、QQ空間分享
        mTencent = Tencent.createInstance(QQ_APPID, mContext);
    }

    @Override
    public int getCount() {
        return mChannelList.size();
    }

    @Override
    public Object getItem(int position) {
        return mChannelList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder = null;
        if (convertView == null) {
            holder = new ViewHolder();
            convertView = mInflater.inflate(R.layout.item_share, null);
            holder.tv_share_name = (TextView) convertView
                    .findViewById(R.id.tv_share_name);
            holder.iv_share_icon = (ImageView) convertView
                    .findViewById(R.id.iv_share_icon);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        ShareChanels item = mChannelList.get(position);
        holder.tv_share_name.setText(item.channelName);
        holder.tv_share_name.setPadding(0, 0, 0, 0);
        holder.iv_share_icon.setImageResource(mShareIcons[item.channelType]);
        return convertView;
    }

    public final class ViewHolder {
        public TextView tv_share_name;
        public ImageView iv_share_icon;
    }

    public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
        ShareChanels item = mChannelList.get(arg2);
        mHandler.sendEmptyMessageDelayed(0, 1500);
        if (item.channelType == QQ) {
            mShareListener = new ShareQQListener(mContext, item.channelName);
            Bundle params = new Bundle();
            params.putInt(QQShare.SHARE_TO_QQ_KEY_TYPE, QQShare.SHARE_TO_QQ_TYPE_DEFAULT);
            params.putString(QQShare.SHARE_TO_QQ_TITLE, mTitle);
            params.putString(QQShare.SHARE_TO_QQ_SUMMARY, mContent);
            params.putString(QQShare.SHARE_TO_QQ_TARGET_URL, mUrl);
            params.putString(QQShare.SHARE_TO_QQ_IMAGE_URL, mImageUrl);
            params.putString(QQShare.SHARE_TO_QQ_APP_NAME, mContext.getPackageName());
            mTencent.shareToQQ((Activity) mContext, params, mShareListener);
        } else if (item.channelType == QZONE) {
            mShareListener = new ShareQQListener(mContext, item.channelName);
            ArrayList<String> urlList = new ArrayList<String>();
            urlList.add(mImageUrl);
            Log.d(TAG, "mImageUrl=" + mImageUrl);
            Bundle params = new Bundle();
            params.putInt(QzoneShare.SHARE_TO_QZONE_KEY_TYPE, QzoneShare.SHARE_TO_QZONE_TYPE_IMAGE_TEXT);
            params.putString(QzoneShare.SHARE_TO_QQ_TITLE, mTitle);
            params.putString(QzoneShare.SHARE_TO_QQ_SUMMARY, mContent);
            params.putString(QzoneShare.SHARE_TO_QQ_TARGET_URL, mUrl);
            params.putStringArrayList(QzoneShare.SHARE_TO_QQ_IMAGE_URL, urlList);
            Log.d(TAG, "begin shareToQzone");
            mTencent.shareToQzone((Activity) mContext, params, mShareListener);
            Log.d(TAG, "end shareToQzone");
        } else if (item.channelType == WEIBO) {
            // 騰訊微博分享需要QQ登入授權
            mTencent.login((Activity) mContext, "all", mLoginListener);
        }
    }

    public static final int REQUEST_ADD_PIC_T = 1001;
    public static IUiListener mLoginListener = new IUiListener() {

        @Override
        public void onComplete(Object object) {
            Log.d(TAG, "登入完成:" + object.toString());
            try {
                JSONObject jsonObject = (JSONObject) object;
                String token = jsonObject.getString(Constants.PARAM_ACCESS_TOKEN);
                String expires = jsonObject.getString(Constants.PARAM_EXPIRES_IN);
                String openId = jsonObject.getString(Constants.PARAM_OPEN_ID);
            } catch (Exception e) {
                Log.d(TAG, "登入異常:" + e.getMessage());
            }
        }

        @Override
        public void onError(UiError error) {
            Log.d(TAG, "登入失敗:" + error.errorMessage);
        }

        @Override
        public void onCancel() {
            Log.d(TAG, "登入取消");
        }
    };

    public static ShareQQListener mShareListener;

    private static class ShareQQListener implements IUiListener {
        private Context context;
        private String channelName;

        public ShareQQListener(final Context context, final String channelName) {
            this.context = context;
            this.channelName = channelName;
        }

        @Override
        public void onComplete(Object object) {
            Toast.makeText(context, channelName + "分享完成:" + object.toString(), Toast.LENGTH_LONG).show();
        }

        @Override
        public void onError(UiError error) {
            Toast.makeText(context, channelName + "分享失敗:" + error.errorMessage, Toast.LENGTH_LONG).show();
        }

        @Override
        public void onCancel() {
            Toast.makeText(context, channelName + "分享取消", Toast.LENGTH_LONG).show();
        }
    }

}
  • ShareGridDialog.java邏輯程式碼如下:
package com.fukaimei.shareqq.widget;

import java.util.ArrayList;

import android.app.Dialog;
import android.content.Context;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import android.view.View.OnClickListener;
import android.view.ViewGroup.LayoutParams;
import android.widget.GridView;
import android.widget.TextView;

import com.fukaimei.shareqq.R;
import com.fukaimei.shareqq.adapter.ShareGridAdapter;
import com.fukaimei.shareqq.bean.ShareChanels;

public class ShareGridDialog implements OnClickListener {
    private final static String TAG = "ShareDialog";
    private Dialog dialog;
    private View view;
    private Context mContext;
    private GridView gv_share_channel;
    private TextView tv_share_cancel;
    private ShareGridAdapter shareItemAdapter;

    private String mUrl;
    private String mTitle;
    private String mContent;
    private String mImgUrl;
    private ArrayList<ShareChanels> mChannelList;

    public ShareGridDialog(final Context context, ArrayList<ShareChanels> channelList) {
        mContext = context;
        mChannelList = channelList;
        view = LayoutInflater.from(context).inflate(R.layout.dialog_share, null);
        dialog = new Dialog(context, R.style.dialog_layout_bottom);

        Window dialogWindow = dialog.getWindow();
        dialogWindow.setGravity(Gravity.BOTTOM);

        gv_share_channel = (GridView) view.findViewById(R.id.gv_share_channel);
        tv_share_cancel = (TextView) view.findViewById(R.id.tv_share_cancel);
        tv_share_cancel.setOnClickListener(this);
    }

    private Handler mHandler = new Handler() {
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            if (msg.what == 0) {
                dismiss();
            } else if (msg.what == 9) {
                Log.d(TAG, (String) msg.obj);
                // Toast.makeText(mContext, (String)msg.obj,
                // Toast.LENGTH_LONG).show();
            }
        }
    };

    public String getUrl() {
        return mUrl;
    }

    public void setUrl(String url) {
        mUrl = url;
    }

    public String getTitle() {
        return mTitle;
    }

    public void setTitle(String title) {
        mTitle = title;
    }

    public String getContent() {
        return mContent;
    }

    public void setContent(String content) {
        mContent = content;
    }

    public String getImgUrl() {
        return mImgUrl;
    }

    public void setImgUrl(String imgUrl) {
        mImgUrl = imgUrl;
    }

    public void setSysAlert() {
        dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
    }

    public void setCancelableOnTouchOutside(boolean flag) {
        dialog.setCanceledOnTouchOutside(flag);
    }

    public void show() {
        shareItemAdapter = new ShareGridAdapter(mContext, mHandler, mUrl,
                mTitle, mContent, mImgUrl, mChannelList);
        gv_share_channel.setAdapter(shareItemAdapter);
        gv_share_channel.setOnItemClickListener(shareItemAdapter);
        dialog.getWindow().setContentView(view);
        dialog.getWindow().setLayout(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
        dialog.show();
    }

    public void dismiss() {
        if (dialog != null && dialog.isShowing()) {
            dialog.dismiss();
        }
    }

    public boolean isShowing() {
        if (dialog != null)
            return dialog.isShowing();
        return false;
    }

    public void setCancelable(boolean flag) {
        dialog.setCancelable(flag);
    }

    @Override
    public void onClick(View v) {
        if (v.getId() == R.id.tv_share_cancel) {
            dismiss();
        }
    }
}
  • 注意:由於該程式不僅需要訪問網路,而且還要啟動QQ。因此還需要在清單檔案AndroidManifest.xml檔案中授權該程式訪問網路、QQ、QQ空間所需許可權的許可權:
    <!-- 上網 -->
    <uses-permission android:name="android.permission.INTERNET" />

    <!-- QQ、QQ空間所需許可權 -->
    <uses-permission android:name="android.permission.GET_TASKS" />
    <uses-permission android:name="android.permission.SET_DEBUG_APP" />
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
    <uses-permission android:name="android.permission.GET_ACCOUNTS" />
    <uses-permission android:name="android.permission.USE_CREDENTIALS" />
    <uses-permission android:name="android.permission.MANAGE_ACCOUNTS" />
  • Demo程式執行效果介面截圖如下:

這裡寫圖片描述 這裡寫圖片描述