Android调用微信登陆、分享、支付
前言:用了微信sdk各种痛苦,感觉比qq sdk调用麻烦多了,回调过于麻烦,还必须要在指定包名下的actvity进行回调,所以我在这里写一篇博客,有这个需求的朋友可以借鉴一下,以后自己别的项目有用到也有个找资料的地方.
一.微信登陆分三个步骤:
1).微信授权登陆
2).根据授权登陆code 获取该用户token
3).根据token获取用户资料
4).接收微信的请求及返回值 如果你的程序需要接收微信发送的请求,或者接收发送到微信请求的响应结果,需要下面3步操作:
a. 在你的包名相应目录下新建一个wxapi目录,并在该wxapi目录下新增一个WXEntryActivity类,该类继承自Activity(例如应用程序的包名为net.sourceforge.simcpux,
则新添加的类如下图所示)
并在manifest文件里面加上exported属性,设置为true,例如:
b. 实现IWXAPIEventHandler接口,微信发送的请求将回调到onReq方法,发送到微信请求的响应结果将回调到onResp方法
c. 在WXEntryActivity中将接收到的intent及实现了IWXAPIEventHandler接口的对象传递给IWXAPI接口的handleIntent方法,示例如下图:
微信官网登陆教程:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list t=resource/res_list verify=1 id=open1419317851 token= lang=zh_CN
微信官网接入指南:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list t=resource/res_list verify=1 id=1417751808 token= lang=zh_CN
二.微信分享直接调用sdk就行,回调跟登陆回调的类是一样的,根据BaseResp的类型来区分是登陆还是分享。
三.微信支付
1).发送一个支付请求
2).接收微信支付的返回值(跟接收微信登陆.分享的返回值类似,我就不写详细操作步骤了)
官网参考地址:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=8_5
四.贴上代码进行讲解
我把微信登陆,分享,支付都封装到了一个类里面了,你们可以参考这个类.封装了6个方法,我对几个需要的方法介绍一下
1).构造方法:初始化对象的时候,顺便初始化微信对象,把app_id注册到微信
2).login() 发起一个登陆的请求 在微信登陆监听Actviity中获取code
3).getAccessToken(String code) 当你从监听Activity中获取了code之后就可以通过这个方法获取微信访问token了
4).getWeiXinUserInfo(final WeiXinToken obj) 获取微信用户信息 传入一个WeiXinToken对象,这个对象是第三步的返回值
5).share(final boolean friendsCircle,final VideoB videoB) 分享给朋友或者朋友圈 如果你有分享图片,图片过大的时候一定要经过压缩,微信官网说明图片不能大
于32kb
6).isWXAppInstalled() 检查微信是否安装
7).wxPay(final BaseActivity activity,String order_id,String payType) 微信支付 我们项目微信支付的一些参数保存在服务器上,所以我这边还请求了自己的
服务器,如果你们是放在本地,直接copy回调函数里面的代码即可.在微信支付监听Actviity中获取支付的状态码
PayReq类属性对应含义请参考微信官方文档:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=9_12
public class WeiXinPresenter extends Presenter{ public static final int IMAGE_SIZE=32768;//微信分享图片大小限制 public static final String APP_ID = "";//应用唯一标识,在微信开放平台提交应用审核通过后获得 public static final String SECRET="";//应用密钥AppSecret,在微信开放平台提交应用审核通过后获得 private IWXAPI wxAPI; private IView iView; private IUserController userController; @Override public IView getIView() { return iView; } public WeiXinPresenter(Context context){ if(context!=null context instanceof IView) iView =(IView) context; if(wxAPI==null){ wxAPI = WXAPIFactory.createWXAPI(context,APP_ID,true); wxAPI.registerApp(APP_ID); } if(null==userController) userController=ControllerFactory.getUserController(); } /** * 微信登陆(三个步骤) * 1.微信授权登陆 * 2.根据授权登陆code 获取该用户token * 3.根据token获取用户资料 * @param activity */ public void login(){ SendAuth.Req req = new SendAuth.Req(); req.scope = "snsapi_userinfo"; req.state = String.valueOf(System.currentTimeMillis()); wxAPI.sendReq(req); } /** * 获取微信访问token */ public void getAccessToken(String code){ if(!userController.isLogin()){//没有登陆的情况用第三方登陆 userController.getWeiXinAccessToken(APP_ID,SECRET,code,new RequestDataCallback WeiXinToken (){ @Override public void dataCallback(WeiXinToken obj){ if(obj!=null){ if(obj.getErrcode()==0){ if(MLog.debug) iView.showToast("授权用户唯一标识:"+obj.getOpenid()); getWeiXinUserInfo(obj); }else{ iView.showToast(obj.getErrmsg()); } }else{ } } }); }else{//用户已登陆 } } /** * 获取微信用户信息 */ private void getWeiXinUserInfo(final WeiXinToken obj){ userController.getWeiXinUserInfo(obj.getAccess_token(), obj.getOpenid(), new RequestDataCallback RegisterB () { @Override public void dataCallback(RegisterB registerB){ registerB.setAccess_token(obj.getAccess_token()); registerB.setToken_expire_at(obj.getExpires_in()); if(registerB.getErrcode()==0){ registerB.setThird_type_name(Constants.WEI_XIN); thirdLogin(registerB); }else{ iView.showToast(registerB.getErrmsg()); } } }); } /** * 调用我们自己的服务器进行登录 * @param registerB */ private void thirdLogin(RegisterB registerB){ userController.thirdAuth(registerB,new RequestDataCallback UserP (){ @Override public void dataCallback(UserP user){ if(checkCallbackData(user, true)){ if(user.getError()==user.ErrorNone){ iView.showToast(R.string.login_success); getAppController().sendLoginChangeIntent(); userController.saveLoginUser(user,FileUtil.getFilePath()); ((ILoginView)iView).toMain(); }else{ iView.showToast(user.getError_reason()); } } } }); } /** * 微信分享 * @param friendsCircle 是否分享到朋友圈 */ public void share(final boolean friendsCircle,final VideoB videoB){ new LoadPicThread(videoB.getCover_url(),new Handler(){ @Override public void handleMessage(Message msg) { byte[] bytes=(byte[]) msg.obj; if(bytes.length IMAGE_SIZE){ iView.showToast(R.string.image_no_big); return; } System.out.println("图片长度:"+bytes.length); WXVideoObject videoObject = new WXVideoObject();//视频类型 videoObject.videoUrl = videoB.getShare_url() + Constants.WEI_XIN + " share_from="+com.kaka.utils.Constants.ANDROID;// 视频播放url WXMediaMessage wxMessage = new WXMediaMessage(videoObject); wxMessage.title = videoB.getContent(); wxMessage.thumbData = bytes; SendMessageToWX.Req req = new SendMessageToWX.Req(); //transaction字段用于唯一标识一个请求 req.transaction = String.valueOf(videoB.getId() + System.currentTimeMillis()); req.message = wxMessage; req.scene = friendsCircle ? SendMessageToWX.Req.WXSceneTimeline : SendMessageToWX.Req.WXSceneSession; wxAPI.sendReq(req); } }).start(); } private class LoadPicThread extends Thread{ private String url; private Handler handler; public LoadPicThread(String url,Handler handler){ this.url=url; this.handler=handler; } @Override public void run(){ try { URL picurl = new URL(url); HttpURLConnection conn = (HttpURLConnection)picurl.openConnection(); // 获得连接 conn.setConnectTimeout(6000);//设置超时 conn.setDoInput(true); conn.setUseCaches(false);//不缓存 conn.connect(); Bitmap bmp=BitmapFactory.decodeStream(conn.getInputStream()); ByteArrayOutputStream output = new ByteArrayOutputStream(); bmp.compress(Bitmap.CompressFormat.JPEG, 100, output); int options = 100; while (output.toByteArray().length IMAGE_SIZE options != 10) { output.reset(); // 清空baos bmp.compress(Bitmap.CompressFormat.JPEG, options, output);// 这里压缩options%,把压缩后的数据存放到baos中 options -= 10; } bmp.recycle(); byte[] result = output.toByteArray(); output.close(); Message message=handler.obtainMessage(); message.obj=result; message.sendToTarget(); } catch (Exception e) { e.printStackTrace(); } } } //检查微信是否安装 public boolean isWXAppInstalled(){ return wxAPI.isWXAppInstalled(); } public void wxPay(final BaseActivity activity,String order_id,String payType){ activity.showProgress(""); ControllerFactory.getWalletsController().getPayments(order_id, payType, new RequestDataCallback PaymentsP () { @Override public void dataCallback(PaymentsP obj) { if(checkCallbackData(obj, true)){ if(obj.getError()==obj.ErrorNone){ PayReq req = new PayReq();//待修改 req.appId = obj.getAppid(); req.nonceStr=obj.getNoncestr(); req.packageValue=obj.getPackage_value(); req.sign=obj.getSign(); req.partnerId=obj.getPartnerid(); req.prepayId=obj.getPrepayid(); req.timeStamp=obj.getTimestamp(); wxAPI.registerApp(obj.getAppid()); wxAPI.sendReq(req); MLog.i("ansen", "开始进行微信支付.."); iView.showToast("开始进行微信支付.."); } }else{ iView.showToast(obj.getError_reason()); } activity.hideProgress(); } }); }
微信登陆以及分享 请求跟返回值的接收 我这边登陆.分享的状态都是发送广播出去,然后结束当前的Activity.
public class WXEntryActivity extends Activity implements IWXAPIEventHandler{ private IWXAPI wxAPI; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); if(MLog.debug) System.out.println("WXEntryActivity onCreate"); wxAPI = WXAPIFactory.createWXAPI(this,WeiXinPresenter.APP_ID,true); wxAPI.registerApp(WeiXinPresenter.APP_ID); wxAPI.handleIntent(getIntent(), this); } @Override protected void onNewIntent(Intent intent){ super.onNewIntent(intent); wxAPI.handleIntent(getIntent(),this); if(MLog.debug) System.out.println("WXEntryActivity onNewIntent"); } @Override public void onReq(BaseReq arg0) { if(MLog.debug) System.out.println("WXEntryActivity onReq:"+arg0); if(MLog.debug) Toast.makeText(this, "onReq 方法运行", 0).show(); } @Override public void onResp(BaseResp resp){ MLog.d("ansen", "onResp....."); if(MLog.debug) Toast.makeText(this,"onResp 方法运行", 0).show(); if(resp.getType()==ConstantsAPI.COMMAND_SENDMESSAGE_TO_WX){//分享 switch (resp.errCode){ case BaseResp.ErrCode.ERR_OK: if(MLog.debug) Toast.makeText(WXEntryActivity.this, "分享成功!", Toast.LENGTH_SHORT).show(); break; case BaseResp.ErrCode.ERR_USER_CANCEL: // Toast.makeText(WXEntryActivity.this, "分享取消!", Toast.LENGTH_SHORT).show(); break; case BaseResp.ErrCode.ERR_AUTH_DENIED: break; } Intent intent = new Intent(); intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_SHARE); LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this); lbm.sendBroadcast(intent); }else if(resp.getType()==ConstantsAPI.COMMAND_SENDAUTH){//登陆发送广播 SendAuth.Resp authResp = (Resp) resp; String code = authResp.code; Intent intent = new Intent(); intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN); intent.putExtra("errCode", authResp.errCode); if (authResp.errCode == BaseResp.ErrCode.ERR_OK){//用户同意 intent.putExtra("code", code); } if(MLog.debug) Toast.makeText(this, "WXEntryActivity 发送登陆广播!!!!", 0).show(); if (android.os.Build.VERSION.SDK_INT = 12) { intent.setFlags(32);//3.1以后的版本需要设置Intent.FLAG_INCLUDE_STOPPED_PACKAGES } LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this); lbm.sendBroadcast(intent); } finish(); }
微信支付 请求跟返回值的接收 微信支付也是发送广播,如果你们还有需求判断支付成功或者失败,可以在广播的intent中进行传参
*/ public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler{ private IWXAPI wxAPI; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); wxAPI = WXAPIFactory.createWXAPI(this, WeiXinPresenter.APP_ID); wxAPI.handleIntent(getIntent(), this); } @Override protected void onNewIntent(Intent intent){ super.onNewIntent(intent); setIntent(intent); wxAPI.handleIntent(intent, this); } @Override public void onReq(BaseReq arg0) { } @Override public void onResp(BaseResp resp) { MLog.i("微信支付回调..", "ansen onResp"); if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX){//微信支付回调 if(resp.errCode==BaseResp.ErrCode.ERR_OK){//微信支付成功 Intent intent = new Intent(); intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_PAY); LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this); lbm.sendBroadcast(intent); //成功 // Toast.makeText(this,R.string.wxpay_success, 0).show(); }else{ // Toast.makeText(this,R.string.wxpay_success, 0).show(); } } finish(); }
强调一点,一定要注意 接收微信的请求及返回值 的包名跟类名,包名是应用程序的包名+".wxapi" 类名必须是微信指定的类名 并且这两个Activity一定要在AndroidManifest.xml中注册,上传一张是我做的app中包名跟类名的截图
如何在activity中调用微信登陆
1).登陆广播监听内部类 如果接收到了广播就去获取微信token
if(errCode==BaseResp.ErrCode.ERR_USER_CANCEL||errCode==BaseResp.ErrCode.ERR_AUTH_DENIED){ requestDataFinish(); }else{ String code = intent.getExtras().getString("code"); xinTestPresenter.getAccessToken(code); } } }
filter.addAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN); lbm.registerReceiver(wxEntryReceiver,filter);
在Activity中调用微信分享跟调用微信支付的代码我就不贴出来了,我这篇博客只是给大家一个参考的地方,遇到问题还是建议第一时间看官方文档.
说说我在做微信登陆碰到的问题
1.微信登陆、分享、支付 回调的activity 包名跟类名一定要严格按照要求去写
2.接收回调的是activity 一定要在AndroidManifest.xml中注册
3.WeiXinPresenter中有两个常量 APP_ID跟SECRET 要去微信申请的时候才有的.你们copy代码的时候要给这两个常量赋值
4.可能访问网络神马的还需要一些权限 记得在AndroidManifest.xml添加权限
5.调用微信的登陆、分享、支付 你的安装包一定要有签名,签名信息一定要跟你在微信官网上申请时签名信息一致
6.微信没有客服支持。。。。。如果出了问题看官方demo 或者 官方API
7.微信sdk经常升级,如果你开发的时候有最新的就用最新的吧.....
说了那么多,感觉说了一大堆废话......希望能帮到大家....有神马问题可以给我留言....
推荐下自己创建的android QQ群:202928390 欢迎大家的加入.
我在csdn上上传了一个微信sdk的jar包,有需要的可以去下载
Android 银联控件支付开发流程 项目中要用到支付功能,需要支付宝支付、微信支付、银联支付,所以打算总结一下,方便以后的查阅,也方便大家, 用到的地方避免再次被坑。 今天我们就主要介绍一下银联控件支付,其他支付也写了对应教程,并且给出了连接。
Android 支付宝支付开发流程 项目中要用到支付功能,需要支付宝支付、微信支付、银联支付,所以打算总结一下,方便以后的查阅,也方便大家, 用到的地方避免再次被坑。 今天我们就主要介绍一下支付宝支付,其他支付也给出了对应的连接。
Android ONE store支付 什么是ONE store应用内支付(IAP) 一家商店应用内部支付(下称IAP)是一家商店使用的,支付服务,开发者销售手机应用程序的应用内部商品时,利用一个商店的验证和支付系统完成向用户支付费用,解决等价流程。一店服务(一店服务,OSS)替代开发的应用商品,OSS与一店总服务器连接执行支付工作,用于响应用户的应用内部商品购买请求。
Android MyCard支付 MyCard支付: 1、通过商务联系MyCard进行技术对接,获得最新MyCardPaySDK.jar和接入文档 2、导入MyCardPaySDK.jar 3、根据MyCard提供文档设置AndroidManifest.xml 4、根据自身需求确定是走SDK支付方式还是WebView支付方式
Android微信支付和支付宝支付快速接入 创建应用并获取APPID 开发者中心中创建您的应用,会为您生成应用唯一标识(APPID),并且可以申请开通开放产品使用权限,通过APPID您的应用才能调用开放产品的接口能力。在“我的应用”中按下图方式查看APPID。
ansen_666 有三年android开发经验,开发过加密.短视频.直播app,一直在互联网公司工作,目前就职于上海翼成科技,担任android开发组长。
相关文章
- Android基于DataBinding封装RecyclerView实现快速列表开发
- android 浏览器 开发,Android 浏览器的开发实例分享
- android okio使用方法,Android 开源框架 Okio 原理剖析「建议收藏」
- android autosize原理,Android屏幕适配头条:autosize的原理
- android toast显示时间,Android Toast自定义显示时间「建议收藏」
- android系统中toast是什么_Android个人资料简单布局
- android跳转到相册需要权限,Android打开相册获取图片路径[通俗易懂]
- android进程间通信的方式_Android进程注入
- 【Android Protobuf 序列化】Protobuf 服务器与客户端通信 ( TCP 通信中使用 Protobuf )
- 【错误记录】编译 Android 版本的 ijkplayer 报错 ( You must define ANDROID_NDK before starting. | 下载指定版本 NDK )
- 【错误记录】编译 Android 版本的 ijkplayer 报错 ( ./init-android.sh: 第 37 行: cd: android/contrib/: 没有那个文件或目录 )
- 【错误记录】Android 应用安全检测漏洞修复 ( StrandHogg 漏洞 | 设置 Activity 组件 android:taskAffinity=““ )
- Android开发学习总结(四)——Eclipse在线安装ADT插件详解手机开发
- 教你自己写Android第三方库详解手机开发
- Android中的ProgressBar的android:indeterminate详解手机开发
- Android Linux开发之路(安卓linux开发)
- 微信 v6.3.5 for Android 正式版发布
- 安卓编年史(25):Android 4.4,奇巧——更完美,更少的内存占用(2)
- 安卓编年史(28):Android 5.0 Lollipop——有史以来最重要的安卓版本(2)
- 手把手教你当微信运动第一名 – 利用Android Hook进行微信运动作弊
- Android Pay登陆香港,但可能拼不过微信和苹果
- Android第三方应用接入微信平台研究情况分享(一)
- Android九宫格的实现方法
- android获取音乐文件的内置专辑图片实现思路及代码
- Android中将View的内容保存为图像的简单实例