Android 使用微信登录、分享功能

具体的文档详情微信官网上介绍(微信官网文档),本人直接按照项目部署步骤进行讲解:

第一步:申请你的AppID;

第二步:依赖

dependencies {
compile 'com.tencent.mm.opensdk:wechat-sdk-android-with-mta:1.0.2'
}

或者:

dependencies {
compile 'com.tencent.mm.opensdk:wechat-sdk-android-without-mta:1.0.2'
}

这两个依赖包的区别是前者包含统计功能,后者没有。

第三步:在AndroidManifest.xml中加入必须的权限

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

第四步:注册到微信,微信的SDK在使用之前需要调用代码进行注册,注册可以是在调用微信SDK的Activity,但最好还是放到入口Activity或者Application中;

WxApplication.java

public static String APP_ID="你的APPID";
public static IWXAPI api;
public void onCreate() {
super.onCreate();
api = WXAPIFactory.createWXAPI(this, APP_ID, true);
api.registerApp(APP_ID);
}

第五步:微信登录。

  5.1 点击你的微信登录按钮,进行微信授权,首先需要获取一个code值,在包名下新建一个包,这里必须这样,不能随意命名,比如你的项目包名是:come.shenhua.wx则新建的包为come.shenhua.wx.wxapi在该包下创建一个WXEntryActivity继承IWXAPIEventHandler;(注意Activity的命名也不能随意更改)

LoginActivity.java

  

private void onClickWeChatLogin() {
if (WxApplicaiton.api == null) {
WxApplication.api = WXAPIFactory.createWXAPI(this, WxApplication.APP_ID, true);
}
if (!WxApplication.api.isWXAppInstalled()) {
ToastUtils.toast("您手机尚未安装微信,请安装后再登录");
return;
}
WxApplication.api.registerApp(WxApplication.APP_ID);
SendAuth.Req req = new SendAuth.Req();
req.scope = "snsapi_userinfo";
req.state = "wechat_sdk_xb_live_state";//官方说明:用于保持请求和回调的状态,授权请求后原样带回给第三方。该参数可用于防止csrf攻击(跨站请求伪造攻击),建议第三方带上该参数,可设置为简单的随机数加session进行校验
WxApplication.api.sendReq(req);
}

  在AndroidManifest.xml中加入,(android:exported="true" 一定不能少)

<activity
android:name=".wxapi.WXEntryActivity"
android:exported="true" />
WXEntryActivity.java
public class WXEntryActivity extends AppCompatActivity implements IWXAPIEventHandler {
public static final String TAG = WXEntryActivity.class.getSimpleName();
public static String code;
public static BaseResp resp = null; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_wxentry);
boolean handleIntent = XBLiveApplication.api.handleIntent(getIntent(), this);
     //下面代码是判断微信分享后返回WXEnteryActivity的,如果handleIntent==false,说明没有调用IWXAPIEventHandler,则需要在这里销毁这个透明的Activity;
if(handleIntent==false){
Log.d(TAG, "onCreate: "+handleIntent);
finish();
}
} @Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
setIntent(intent);
XBLiveApplication.api.handleIntent(intent, this);
} @Override
public void onReq(BaseReq baseReq) {
Log.d(TAG, "onReq: ");
finish();
} @Override
public void onResp(BaseResp baseResp) {
if (baseResp != null) {
resp = baseResp;
code = ((SendAuth.Resp) baseResp).code; //即为所需的code
}
switch (baseResp.errCode) {
case BaseResp.ErrCode.ERR_OK:
Log.d(TAG, "onResp: 成功");
finish();
break;
case BaseResp.ErrCode.ERR_USER_CANCEL:
Log.d(TAG, "onResp: 用户取消");
finish();
break;
case BaseResp.ErrCode.ERR_AUTH_DENIED:
Log.d(TAG, "onResp: 发送请求被拒绝");
finish();
break;
}
}

resp和code定义成全局变量,在WXEntryWActivity页面销毁后登录页面会走onResume()方法,在这里开始其他操作

  5.2 拿到code后在登录页面通过code获取access_token,官网的说法是:“access_token是调用授权关系接口的调用凭证,由于access_token有效期(目前为2个小时)较短,当access_token超时后,可以使用refresh_token进行刷新”,所以拿到access_token后再次请求进行refresh_token,刷新后拿到access_token即可发送请求获取微信用户信息了;

  获取access_token的URL:AccessTokenUrl = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=你的appid&secret=你的AppSecret&code=授权获取到的code&grant_type=authorization_code";

  刷新access_token的URL:Refresh_tokenUrl = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid=你的appid&grant_type=refresh_token&refresh_token=上一步请求到的access_token;

  获取用户信息URL:UserInfoUrl = "https://api.weixin.qq.com/sns/userinfo?access_token=刷新的access_token&openid=刷新access_token时获取的openid;


 @Override
protected void onResume() {
Log.d(TAG, "onResume: 1");
super.onResume();
//这里的判断是为了区分如果不是不是从WXEntryActivity页面销毁重启的,不走下面的代码
if (WXEntryActivity.resp != null) {
if (WXEntryActivity.resp.getType() == ConstantsAPI.COMMAND_SENDAUTH) {
Thread thread = new Thread(downloadRun);
thread.start();
try {
thread.join();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
} public Runnable downloadRun = new Runnable() { @Override
public void run() {
WXGetAccessToken();
}
}; //根据code获取access_token,这里用第三方volley框架进行post请求
private void WXGetAccessToken() {
String getAccessTokenUrl = AccessTokenUrl;
StringRequest request = new StringRequest(Request.Method.GET, getAccessTokenUrl, new Response.Listener<String>() {
@Override
public void onResponse(String response) {
Log.d(TAG, "onResponse: " + response);
Gson gson = new Gson();
WeChatLoginTokenModel tokenModel = gson.fromJson(response, WeChatLoginTokenModel.class);
refresh_token = tokenModel.getRefresh_token();
WXGetRefreshAccessToken();
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) { }
});
request.setTag("weChatGetToken");
XBLiveApplication.getHttpRequestQueue().add(request);
} //拿到access_token后再次发送请求进行刷新access_token,拿到access_refresh和openid
private void WXGetRefreshAccessToken() {
String getRefresh_tokenUrl =Refresh_tokenUrl ;
StringRequest request = new StringRequest(Request.Method.GET, getRefresh_tokenUrl, new Response.Listener<String>() {
@Override
public void onResponse(String response) {
Gson gson = new Gson();
WeChatLoginTokenModel tokenModel = gson.fromJson(response, WeChatLoginTokenModel.class);
String access_token = tokenModel.getAccess_token();
String openid = tokenModel.getOpenid();
WXGetUserInfo(access_token, openid);
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) { }
});
request.setTag("weChatGetRefreshToken");
XBLiveApplication.getHttpRequestQueue().add(request);
} //获取用户信息
private void WXGetUserInfo(String access_token, String openid) {
String getUserInfoUrl =UserInfoUr;
StringRequest request = new StringRequest(Request.Method.GET, getUserInfoUrl, new Response.Listener<String>() {
@Override
public void onResponse(String response) {
Log.d(TAG, "onResponse3: " + response);
Gson gson = new Gson();
WeChatUserInfoModel userInfoModel = gson.fromJson(response, WeChatUserInfoModel.class);
String city = userInfoModel.getCity();
String userName = userInfoModel.getNickname();
String userIcon = userInfoModel.getHeadimgurl();
int sex = userInfoModel.getSex();//1:男;0:女;2:未知
String province = userInfoModel.getProvince();
openId_WX = userInfoModel.getOpenid();
ToastUtils.toast("登录成功");
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) { }
});
    //下面代码是volley框架注册,请忽视
request.setTag("weChatGetUserInfo");
XBLiveApplication.getHttpRequestQueue().add(request);
}

注意一点:在做测试时,微信的应用签名必须要保持一致,比如你在官网填写的是没有打包签名的应用签名,你测试的APP也没有打包签名,那么是没有问题,但是如果你微信官网上填写的是已经上线经过打包签名的应用签名,那你在做测试时必须要用打包签名的应用去做测试。

第六步:微信分享到朋友圈、微信好友、添加微信收藏:

分享收藏基本完全一样,也是需要先进行注册的,如果之前已经进行了注册就不需要注册,否则在调用分享的Activity中或者全局Application中进行注册;scene 有一下三种:

SendMessageToWX.Req.WXSceneSession;分享到微信好友
SendMessageToWX.Req.WXSceneTimeline;分享到微信朋友圈
SendMessageToWX.Req.WXSceneFavorite;添加到微信收藏夹

  分享纯文本内容:

              // 初始化一个WXTextObject对象
WXTextObject textObj = new WXTextObject();
textObj.text = text; // 用WXTextObject对象初始化一个WXMediaMessage对象
WXMediaMessage msg = new WXMediaMessage();
msg.mediaObject = textObj;
// 发送文本类型的消息时,title字段不起作用
// msg.title = "Will be ignored";
msg.description = text; // 构造一个Req
SendMessageToWX.Req req = new SendMessageToWX.Req();
req.transaction = buildTransaction("text"); // transaction字段用于唯一标识一个请求
req.message = msg;
req.scene = isTimelineCb.isChecked() ? SendMessageToWX.Req.WXSceneTimeline : SendMessageToWX.Req.WXSceneSession; // 调用api接口发送数据到微信
api.sendReq(req);

分享音乐:

                WXMusicObject music = new WXMusicObject();
music.musicUrl="http://staff2.ustc.edu.cn/~wdw/softdown/index.asp/0042515_05.ANDY.mp3"; WXMediaMessage msg = new WXMediaMessage();
msg.mediaObject = music;
msg.title = "Music Title ";
msg.description = "Music Album "; Bitmap thumb = BitmapFactory.decodeResource(getResources(), R.drawable.send_music_thumb);
msg.thumbData = Util.bmpToByteArray(thumb, true); SendMessageToWX.Req req = new SendMessageToWX.Req();
req.transaction = buildTransaction("music");
req.message = msg;
req.scene = isTimelineCb.isChecked() ? SendMessageToWX.Req.WXSceneTimeline : SendMessageToWX.Req.WXSceneSession;
api.sendReq(req);

分享网页:

                        WXWebpageObject webpage = new WXWebpageObject();
webpage.webpageUrl = "http://www.qq.com";
WXMediaMessage msg = new WXMediaMessage(webpage);
msg.title = "WebPage Title ";
msg.description = "WebPage Description ";
Bitmap bmp = BitmapFactory.decodeResource(getResources(), R.drawable.send_music_thumb);
Bitmap thumbBmp = Bitmap.createScaledBitmap(bmp, THUMB_SIZE, THUMB_SIZE, true);
bmp.recycle();
msg.thumbData = Util.bmpToByteArray(thumbBmp, true);
SendMessageToWX.Req req = new SendMessageToWX.Req();
req.transaction = buildTransaction("webpage");
req.message = msg;
req.scene = mTargetScene;
api.sendReq(req);

具体可参照官网Demo。(可分享纯文本内容,照片,音乐,视频,网页,链接等等)

  楼主在测试时发现,如果我分享的网页要显示一张图片,官方Demo给的是一张本地照片,而我的是图片地址,我做了一下处理:

  点击分享时先加载图片,写了一个while循环,直到拿到bitmap为止;

    case R.id.line_share_wechat:
if (!api.isWXAppInstalled()) {
ToastUtils.toast("您手机尚未安装微信,请安装后再登录");
return;
}
if (api == null) {
api = WXAPIFactory.createWXAPI(this,APP_ID, true);
}
mTargetScene = SendMessageToWX.Req.WXSceneSession;//share to wx
gotoDownLoadSharePic();break;

  

private void gotoDownLoadSharePic() {
new Thread() {
@Override
public void run() {
URL url = null;
try {
url = new URL(thumb);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setDoInput(true);
conn.connect();
InputStream inputStream = conn.getInputStream();
bmp = BitmapFactory.decodeStream(inputStream); } catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
while (bmp == null) {
Log.d(TAG, "onClick: bmp=" + bmp);
}
goToShareToWeChat(mTargetScene);
} private void goToShareToWeChat(int targetScene) {
WXWebpageObject mWebPage = new WXWebpageObject();
mWebPage.webpageUrl = Constants.SHARE_QQ_URL + vid;
msg = new WXMediaMessage(mWebPage);
if (targetScene == SendMessageToWX.Req.WXSceneTimeline || targetScene == SendMessageToWX.Req.WXSceneFavorite) {
msg.title = mTitleContent;
} else {
msg.title = name;
msg.description = mTitleContent;
}
msg.setThumbImage(bmp);
SendMessageToWX.Req req = new SendMessageToWX.Req();
req.transaction = buildTransaction("webpage");
req.message = msg;
req.scene = targetScene;
api.sendReq(req);
}

详解Android微信登录与分享的更多相关文章

  1. Android微信登录、分享、支付

    转载需要著名出处: http://blog.csdn.net/lowprofile_coding/article/details/78004224 之前写过微信登录分享支付第一版: http://bl ...

  2. android微信登录,分享

    这几天开发要用到微信授权的功能,所以就研究了一下.可是微信开放平台接入指南里有几个地方写的不清不楚.在此总结一下,以便需要的人. 很多微信公众平台的应用如果移植到app上的话就需要微信授权登陆了. 目 ...

  3. 详解Android Activity---启动模式

    相关的基本概念: 1.任务栈(Task)   若干个Activity的集合的栈表示一个Task.   栈不仅仅只包含自身程序的Activity,它也可以跨应用包含其他应用的Activity,这样有利于 ...

  4. 详解Android Activity启动模式

    相关的基本概念: 1.任务栈(Task)   若干个Activity的集合的栈表示一个Task.   栈不仅仅只包含自身程序的Activity,它也可以跨应用包含其他应用的Activity,这样有利于 ...

  5. (转载)实例详解Android快速开发工具类总结

    实例详解Android快速开发工具类总结 作者:LiJinlun 字体:[增加 减小] 类型:转载 时间:2016-01-24我要评论 这篇文章主要介绍了实例详解Android快速开发工具类总结的相关 ...

  6. iOS开发笔记14:微博/微信登录与分享、微信/支付宝支付

    产品中接入了微博/微信的第三方登录分享功能.微信和支付宝的第三方支付功能,之前在开发过程中涉及到这些部分,于是抽空将接入过程梳理了一遍. 1.微博.微信.支付宝SDK相关接入设置 (1)微博SDK S ...

  7. 详解Android首选项框架ListPreference

    详解Android首选项框架ListPreference 原文地址 探索首选项框架 在深入探讨Android的首选项框架之前,首先构想一个需要使用首选项的场景,然后分析如何实现这一场景.假设你正在编写 ...

  8. 详解android:scaleType属性

    详解android:scaleType属性 转自:http://blog.csdn.net/encienqi/article/details/7913262    http://juliaailse. ...

  9. adb shell 命令详解,android

    http://www.miui.com/article-275-1.html http://noobjava.iteye.com/blog/1914348 adb shell 命令详解,android ...

随机推荐

  1. 20155301 滕树晨linux基础——linux进程间通信(IPC)机制总结

    20155301 滕树晨linux基础--linux进程间通信(IPC)机制总结 共享内存 共享内存是在多个进程之间共享内存区域的一种进程间的通信方式,由IPC为进程创建的一个特殊地址范围,它将出现在 ...

  2. 2017-2018-2 20155315《网络对抗技术》Exp1:PC平台逆向破解

    实验目的 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入的字符串. 该程序同时包含另一个代码片段,getShe ...

  3. Noip前的大抱佛脚----文章索引

    Noip前的大抱佛脚----赛前任务 Noip前的大抱佛脚----考场配置 Noip前的大抱佛脚----数论 Noip前的大抱佛脚----图论 Noip前的大抱佛脚----动态规划 Noip前的大抱佛 ...

  4. 11 [异常]-try...except

    1.什么是异常 异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止),在python中,错误触发的异常如下 2.错误 错误分成 ...

  5. Restful和WeBAPI学习笔记

    1.restful是基于无状态的,所谓无状态就是说客户端和服务端的每次通话都是独立的,不存在session和cookie之类的保存状态的机制,基于该协议可实现简单的curd操作, 其操作分为get\p ...

  6. springmvc框架开发中解决产生的乱码情况

    一:解决post请求方式产生的乱码情况 示例代码如下: <!-- 解决post乱码 --> <filter> <filter-name>CharacterEncod ...

  7. python 思维导图-linux命令

    linux基础 linux关于文件/目录的终端命令 linux关于文件内容处理的终端命令

  8. SICP读书笔记 3.3

    SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...

  9. Prometheus 添加报警规则

    https://prometheus.io/docs/prometheus/latest/migration/

  10. 《The Mythical Man-Month(人月神话)》读后感(2)

    第10章 未雨绸缪 在化学领域中,在实验室可以进行的反应过程,并不能在工厂中一步实现.一个被称为“ 实验性工厂(pilot planet)”的中间步骤是非常必要的,它会为提高产量和在缺乏保护的环境下运 ...