編輯:關於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
/** * 微信分享,登陸,支付 * @author ansen * @create time 2015-08-29 */ 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.
/** * 微信登陸分享回調Activity * @author ansen * @create time 2015-05-25 */ 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中進行傳參
/** * 微信支付回調Activity * @author ansen * @create time 2015-08-29 */ 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
private class WXEntryReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent){ MLog.i("WXEntryReceiver", "接收微信登陸廣播"); if(MLog.debug) showToast("接收微信登陸廣播"); if(intent.getAction().equals(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN)){ int errCode = intent.getExtras().getInt("errCode"); if(MLog.debug) System.out.println("獲取錯誤碼:"+errCode); if(errCode==BaseResp.ErrCode.ERR_USER_CANCEL||errCode==BaseResp.ErrCode.ERR_AUTH_DENIED){ requestDataFinish(); }else{ String code = intent.getExtras().getString("code"); xinTestPresenter.getAccessToken(code); } } } }
2).定義成員變量
private WXEntryReceiver wxEntryReceiver=null;
3).在oncreate中注冊廣播
//微信登陸廣播 wxEntryReceiver= new WXEntryReceiver(); LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this); IntentFilter filter = new IntentFilter(); filter.addAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN); lbm.registerReceiver(wxEntryReceiver,filter);
4).調用微信登陸
WeiXinPresenter xinTestPresenter=new WeiXinPresenter(this); xinTestPresenter.login();
在Activity中調用微信分享跟調用微信支付的代碼我就不貼出來了,我這篇博客只是給大家一個參考的地方,遇到問題還是建議第一時間看官方文檔.
說說我在做微信登陸碰到的問題
1.微信登陸、分享、支付 回調的activity 包名跟類名一定要嚴格按照要求去寫
2.接收回調的是activity 一定要在AndroidManifest.xml中注冊
3.WeiXinPresenter中有兩個常量 APP_ID跟SECRET 要去微信申請的時候才有的.你們copy代碼的時候要給這兩個常量賦值
4.可能訪問網絡神馬的還需要一些權限 記得在AndroidManifest.xml添加權限
5.調用微信的登陸、分享、支付 你的安裝包一定要有簽名,簽名信息一定要跟你在微信官網上申請時簽名信息一致
6.微信沒有客服支持。。。。。如果出了問題看官方demo 或者 官方API
7.微信sdk經常升級,如果你開發的時候有最新的就用最新的吧.....
public void drawArc(RectF oval, float startAngle, float sweepAngle, boolean useCenter
一:概述Android4.4以及Android5.0之後,Android上很多APP都出現了如下效果: 簡單的說就是狀態欄顏色出現了變化,能夠跟APP界面顏色相
隨著移動互聯網的快速發展,它已經和我們的生活息息相關了,在公交地鐵裡面都能看到很多人的人低頭看著自己的手機屏幕,從此“低頭族”一詞就產生了,作為一名移動行業的開發人員,我
本節引言: 最近一段時間因為工作上的事以及面試等等,耽誤了博客的更新,這裡道歉下~ 今天下午去追夢網絡面試了一趟,全齊大神給小弟我上了一課,增長了