详解Android微信登录与分享
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微信登录与分享的更多相关文章
- Android微信登录、分享、支付
转载需要著名出处: http://blog.csdn.net/lowprofile_coding/article/details/78004224 之前写过微信登录分享支付第一版: http://bl ...
- android微信登录,分享
这几天开发要用到微信授权的功能,所以就研究了一下.可是微信开放平台接入指南里有几个地方写的不清不楚.在此总结一下,以便需要的人. 很多微信公众平台的应用如果移植到app上的话就需要微信授权登陆了. 目 ...
- 详解Android Activity---启动模式
相关的基本概念: 1.任务栈(Task) 若干个Activity的集合的栈表示一个Task. 栈不仅仅只包含自身程序的Activity,它也可以跨应用包含其他应用的Activity,这样有利于 ...
- 详解Android Activity启动模式
相关的基本概念: 1.任务栈(Task) 若干个Activity的集合的栈表示一个Task. 栈不仅仅只包含自身程序的Activity,它也可以跨应用包含其他应用的Activity,这样有利于 ...
- (转载)实例详解Android快速开发工具类总结
实例详解Android快速开发工具类总结 作者:LiJinlun 字体:[增加 减小] 类型:转载 时间:2016-01-24我要评论 这篇文章主要介绍了实例详解Android快速开发工具类总结的相关 ...
- iOS开发笔记14:微博/微信登录与分享、微信/支付宝支付
产品中接入了微博/微信的第三方登录分享功能.微信和支付宝的第三方支付功能,之前在开发过程中涉及到这些部分,于是抽空将接入过程梳理了一遍. 1.微博.微信.支付宝SDK相关接入设置 (1)微博SDK S ...
- 详解Android首选项框架ListPreference
详解Android首选项框架ListPreference 原文地址 探索首选项框架 在深入探讨Android的首选项框架之前,首先构想一个需要使用首选项的场景,然后分析如何实现这一场景.假设你正在编写 ...
- 详解android:scaleType属性
详解android:scaleType属性 转自:http://blog.csdn.net/encienqi/article/details/7913262 http://juliaailse. ...
- adb shell 命令详解,android
http://www.miui.com/article-275-1.html http://noobjava.iteye.com/blog/1914348 adb shell 命令详解,android ...
随机推荐
- 20155301 滕树晨linux基础——linux进程间通信(IPC)机制总结
20155301 滕树晨linux基础--linux进程间通信(IPC)机制总结 共享内存 共享内存是在多个进程之间共享内存区域的一种进程间的通信方式,由IPC为进程创建的一个特殊地址范围,它将出现在 ...
- 2017-2018-2 20155315《网络对抗技术》Exp1:PC平台逆向破解
实验目的 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入的字符串. 该程序同时包含另一个代码片段,getShe ...
- Noip前的大抱佛脚----文章索引
Noip前的大抱佛脚----赛前任务 Noip前的大抱佛脚----考场配置 Noip前的大抱佛脚----数论 Noip前的大抱佛脚----图论 Noip前的大抱佛脚----动态规划 Noip前的大抱佛 ...
- 11 [异常]-try...except
1.什么是异常 异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止),在python中,错误触发的异常如下 2.错误 错误分成 ...
- Restful和WeBAPI学习笔记
1.restful是基于无状态的,所谓无状态就是说客户端和服务端的每次通话都是独立的,不存在session和cookie之类的保存状态的机制,基于该协议可实现简单的curd操作, 其操作分为get\p ...
- springmvc框架开发中解决产生的乱码情况
一:解决post请求方式产生的乱码情况 示例代码如下: <!-- 解决post乱码 --> <filter> <filter-name>CharacterEncod ...
- python 思维导图-linux命令
linux基础 linux关于文件/目录的终端命令 linux关于文件内容处理的终端命令
- SICP读书笔记 3.3
SICP CONCLUSION 让我们举起杯,祝福那些将他们的思想镶嵌在重重括号之间的Lisp程序员 ! 祝我能够突破层层代码,找到住在里计算机的神灵! 目录 1. 构造过程抽象 2. 构造数据抽象 ...
- Prometheus 添加报警规则
https://prometheus.io/docs/prometheus/latest/migration/
- 《The Mythical Man-Month(人月神话)》读后感(2)
第10章 未雨绸缪 在化学领域中,在实验室可以进行的反应过程,并不能在工厂中一步实现.一个被称为“ 实验性工厂(pilot planet)”的中间步骤是非常必要的,它会为提高产量和在缺乏保护的环境下运 ...