├── .gitignore
├── .idea
├── .name
├── compiler.xml
├── copyright
│ └── profiles_settings.xml
├── encodings.xml
├── gradle.xml
├── misc.xml
├── modules.xml
├── runConfigurations.xml
└── vcs.xml
├── README.md
├── README_CN.md
├── README_EN.md
├── app
├── .gitignore
├── build.gradle
├── proguard-rules.pro
└── src
│ ├── androidTest
│ └── java
│ │ └── io
│ │ └── github
│ │ └── mayubao
│ │ └── android_pay
│ │ └── ApplicationTest.java
│ ├── main
│ ├── AndroidManifest.xml
│ ├── java
│ │ └── io
│ │ │ └── github
│ │ │ └── mayubao
│ │ │ └── android_pay
│ │ │ ├── MainActivity.java
│ │ │ ├── PayAliPayActivity.java
│ │ │ ├── PayWechatActivity.java
│ │ │ └── TestOnlineActivity.java
│ └── res
│ │ ├── layout
│ │ ├── activity_main.xml
│ │ ├── activity_pay_ali_pay.xml
│ │ ├── activity_pay_wechat.xml
│ │ ├── activity_test_online.xml
│ │ ├── content_main.xml
│ │ ├── content_pay_ali_pay.xml
│ │ ├── content_pay_wechat.xml
│ │ └── content_test_online.xml
│ │ ├── menu
│ │ └── menu_main.xml
│ │ ├── mipmap-hdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-mdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xhdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xxhdpi
│ │ └── ic_launcher.png
│ │ ├── mipmap-xxxhdpi
│ │ └── ic_launcher.png
│ │ ├── values-v21
│ │ └── styles.xml
│ │ ├── values-w820dp
│ │ └── dimens.xml
│ │ └── values
│ │ ├── colors.xml
│ │ ├── dimens.xml
│ │ ├── strings.xml
│ │ └── styles.xml
│ └── test
│ └── java
│ └── io
│ └── github
│ └── mayubao
│ └── android_pay
│ └── ExampleUnitTest.java
├── build.gradle
├── gradle.properties
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
├── pay-library
├── .gitignore
├── build.gradle
├── libs
│ ├── alipaySDK-20150818.jar
│ └── libammsdk.jar
├── proguard-rules.pro
└── src
│ ├── androidTest
│ └── java
│ │ └── io
│ │ └── github
│ │ └── mayubao
│ │ └── pay_library
│ │ └── ApplicationTest.java
│ ├── main
│ ├── AndroidManifest.xml
│ ├── java
│ │ └── io
│ │ │ └── github
│ │ │ └── mayubao
│ │ │ └── pay_library
│ │ │ ├── AliPayAPI.java
│ │ │ ├── AliPayReq.java
│ │ │ ├── AliPayReq2.java
│ │ │ ├── MainActivity.java
│ │ │ ├── PayAPI.java
│ │ │ ├── WechatPayAPI.java
│ │ │ ├── WechatPayReq.java
│ │ │ └── alipay
│ │ │ ├── Base64.java
│ │ │ ├── ExternalFragment.java
│ │ │ ├── PayResult.java
│ │ │ └── SignUtils.java
│ └── res
│ │ ├── layout
│ │ ├── pay_external.xml
│ │ └── pay_main.xml
│ │ ├── mipmap-hdpi
│ │ ├── msp_demo_title.png
│ │ ├── msp_demo_title_bg.png
│ │ └── msp_icon.png
│ │ └── values
│ │ └── strings.xml
│ └── test
│ └── java
│ └── io
│ └── github
│ └── mayubao
│ └── pay_library
│ └── ExampleUnitTest.java
└── settings.gradle
/.gitignore:
--------------------------------------------------------------------------------
1 | *.iml
2 | .gradle
3 | /local.properties
4 | /.idea/workspace.xml
5 | /.idea/libraries
6 | .DS_Store
7 | /build
8 | /captures
9 |
--------------------------------------------------------------------------------
/.idea/.name:
--------------------------------------------------------------------------------
1 | Android-Pay
--------------------------------------------------------------------------------
/.idea/compiler.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
--------------------------------------------------------------------------------
/.idea/copyright/profiles_settings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/.idea/encodings.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/.idea/gradle.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
18 |
19 |
--------------------------------------------------------------------------------
/.idea/misc.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 | 1.8
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
--------------------------------------------------------------------------------
/.idea/modules.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/.idea/runConfigurations.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/.idea/vcs.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Android-Pay
2 | [中文](https://github.com/mayubao/Android-Pay/blob/master/README_CN.md)
3 |
4 | A pay library for Android, and which support Wechat pay and Ali pay.
5 | And developer can easily use Wechat pay in two lines of code.
6 | And developer can easily use Ali pay in three lines of code.
7 |
8 | ## Setup
9 |
10 | ### gradle
11 | add these code in the file **build.gradle** as follow:
12 |
13 | ```xml
14 | dependencies {
15 | //add pay library
16 | compile 'io.github.mayubao:pay_library:1.0.1'
17 | }
18 | ```
19 |
20 | ### maven
21 |
22 | ```xml
23 |
24 | io.github.mayubao
25 | pay_library
26 | 1.0.1
27 | pom
28 |
29 | ```
30 |
31 | ## Usage
32 |
33 | ### Wechat Pay
34 |
35 | ```java
36 | //1.create request for wechat pay
37 | WechatPayReq wechatPayReq = new WechatPayReq.Builder()
38 | .with(this) //activity instance
39 | .setAppId(appid) //wechat pay AppID
40 | .setPartnerId(partnerid)//wechat pay partner id
41 | .setPrepayId(prepayid)//pre pay id
42 | // .setPackageValue(wechatPayReq.get)//"Sign=WXPay"
43 | .setNonceStr(noncestr)
44 | .setTimeStamp(timestamp)//time stamp
45 | .setSign(sign)//sign
46 | .create();
47 | //2. send the request with wechat pay
48 | PayAPI.getInstance().sendPayRequest(wechatPayReq);
49 |
50 |
51 | //set the callback for wechat pay
52 | //wechatPayReq.setOnWechatPayListener(new OnWechatPayListener);
53 |
54 |
55 | ```
56 |
57 | >Notes:WechatPayReq have no method to set the money, because the money info is include in the parameter 'prepayid'.
58 |
59 | ### Ali Pay
60 |
61 |
62 | #### First way(**Not Recommend**, and its partner rsa private key export in the client , it is very dangerous!)
63 | ```java
64 |
65 | //step 1 create config for ali pay
66 | AliPayAPI.Config config = new AliPayAPI.Config.Builder()
67 | .setRsaPrivate(rsa_private) // rsa private key from partner (pkcs8 format)
68 | .setRsaPublic(rsa_public)//ali rsa public key
69 | .setPartner(partner) //set partner
70 | .setSeller(seller) //set partner seller accout
71 | .create();
72 |
73 | //step 2 create reqeust for ali
74 | AliPayReq aliPayReq = new AliPayReq.Builder()
75 | .with(activity)//Activity instance
76 | .apply(config)// the above custome config
77 | .setOutTradeNo(outTradeNo)//set unique trade no
78 | .setPrice(price)//set price
79 | .setSubject(orderSubject)//set order subject
80 | .setBody(orderBody)//set order detail
81 | .setCallbackUrl(callbackUrl)//set callback for pay reqest
82 | .create()//
83 | .setOnAliPayListener(null);//
84 |
85 | //step 3 send the request for ali pay
86 | PayAPI.getInstance().sendPayRequest(aliPayReq);
87 |
88 | // set the ali pay callback
89 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
90 |
91 | ```
92 |
93 | #### Second way(**Highly Recommend**)
94 |
95 | ```java
96 | //step 1 create raw ali pay order info
97 | String rawAliOrderInfo = new AliPayReq2.AliOrderInfo()
98 | .setPartner(partner) //set partner
99 | .setSeller(seller) //set partner seller accout
100 | .setOutTradeNo(outTradeNo) //set unique trade no
101 | .setSubject(orderSubject) //set order subject
102 | .setBody(orderBody) //set order detail
103 | .setPrice(price) //set price
104 | .setCallbackUrl(callbackUrl) //set callback for pay reqest
105 | .createOrderInfo(); //create ali pay order info
106 |
107 |
108 | //step 2 get the signed ali pay order info
109 | String signAliOrderInfo = getSignAliOrderInfoFromServer(rawAliOrderInfo);
110 |
111 | //step 3 step 3 send the request for ali pay
112 | AliPayReq2 aliPayReq = new AliPayReq2.Builder()
113 | .with(activity)//Activity instance
114 | .setSignedAliPayOrderInfo(signAliOrderInfo)
115 | .setRawAliPayOrderInfo(rawAliOrderInfo)//set the ali pay order info
116 | .setSignedAliPayOrderInfo(signAliOrderInfo) //set the signed ali pay order info
117 | .create()//
118 | .setOnAliPayListener(null);//
119 | PayAPI.getInstance().sendPayRequest(aliPayReq);
120 |
121 |
122 | //set the ali pay callback
123 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
124 |
125 | ```
126 |
127 |
128 | ## ISSUE
129 |
130 | ### About Wechat Pay Callback
131 |
132 |
133 | How to do :
134 |
135 |
136 | 1.new folder named 'wxapi' under your project root package
137 |
138 | 2.and create class WXPayEntryActivity under the wxapi
139 |
140 | 3.implements IWXAPIEventHandler interface
141 |
142 | The sample as follow:
143 |
144 |
145 | ```java
146 |
147 | import android.app.Activity;
148 | import android.app.AlertDialog;
149 | import android.content.Intent;
150 | import android.os.Bundle;
151 | import android.util.Log;
152 | import android.widget.Toast;
153 |
154 | import com.tencent.mm.sdk.constants.ConstantsAPI;
155 | import com.tencent.mm.sdk.modelbase.BaseReq;
156 | import com.tencent.mm.sdk.modelbase.BaseResp;
157 | import com.tencent.mm.sdk.openapi.IWXAPI;
158 | import com.tencent.mm.sdk.openapi.IWXAPIEventHandler;
159 | import com.tencent.mm.sdk.openapi.WXAPIFactory;
160 |
161 | public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler{
162 |
163 | private static final String TAG = "MicroMsg.SDKSample.WXPayEntryActivity";
164 |
165 | private IWXAPI api;
166 |
167 | //TODO need to replace your APP_ID
168 | private String APP_ID = "wx0xxxxxxxxx"; //need to replace your APP_ID
169 |
170 | @Override
171 | public void onCreate(Bundle savedInstanceState) {
172 | super.onCreate(savedInstanceState);
173 | // setContentView(R.layout.pay_result);
174 |
175 | api = WXAPIFactory.createWXAPI(this, APP_ID);
176 | api.handleIntent(getIntent(), this);
177 | }
178 |
179 | @Override
180 | protected void onNewIntent(Intent intent) {
181 | super.onNewIntent(intent);
182 | setIntent(intent);
183 | api.handleIntent(intent, this);
184 | }
185 |
186 | @Override
187 | public void onReq(BaseReq req) {
188 | }
189 |
190 | @Override
191 | public void onResp(BaseResp resp) {
192 | Log.d(TAG, "onPayFinish, errCode = " + resp.errCode);
193 |
194 | if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {
195 | // AlertDialog.Builder builder = new AlertDialog.Builder(this);
196 | // builder.setTitle(R.string.app_tip);
197 | // builder.setMessage(getString(R.string.pay_result_callback_msg, String.valueOf(resp.errCode)));
198 | // builder.show();
199 |
200 | //send the wechat pay broadcast receiver,and you should the define wechat pay broadcast receive
201 |
202 | Intent intent = new Intent();
203 | intent.setAction(WeChatPayReceiver.ACTION_PAY_RESULT);
204 | intent.putExtra("result", resp.errCode);
205 | sendBroadcast(intent);
206 |
207 | finish();
208 | }
209 | }
210 | }
211 |
212 | ```
213 |
214 | ## Proguard
215 |
216 | ```xml
217 |
218 | #pay_library
219 | -dontwarn io.github.mayubao.pay_library.**
220 | -keep class io.github.mayubao.pay_library.** {*;}
221 |
222 | #wechat pay
223 | -dontwarn com.tencent.**
224 | -keep class com.tencent.** {*;}
225 |
226 |
227 | #alipay
228 | -dontwarn com.alipay.**
229 | -keep class com.alipay.** {*;}
230 |
231 | -dontwarn com.ta.utdid2.**
232 | -keep class com.ta.utdid2.** {*;}
233 |
234 | -dontwarn com.ut.device.**
235 | -keep class com.ut.device.** {*;}
236 |
237 | -dontwarn org.json.alipay.**
238 | -keep class corg.json.alipay.** {*;}
239 | ```
240 |
241 |
242 | ## Document
243 |
244 | ### wehcat pay official document
245 | https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5
246 |
247 | ### ali pay official document
248 | https://doc.open.alipay.com/docs/doc.htm?spm=a219a.7629140.0.0.sdGXaH&treeId=204&articleId=105296&docType=1
249 |
250 |
251 | ## Help
252 | If it is helpful to you, could you buy me a cup of coffee?
253 |
254 | ### Wechat
255 | 
256 | ### Ali
257 | 
258 |
259 |
260 | ## Lisence
261 |
262 | Copyright 2017 mayubao
263 |
264 | Licensed under the Apache License, Version 2.0 (the "License");
265 | you may not use this file except in compliance with the License.
266 | You may obtain a copy of the License at
267 |
268 | http://www.apache.org/licenses/LICENSE-2.0
269 |
270 | Unless required by applicable law or agreed to in writing, software
271 | distributed under the License is distributed on an "AS IS" BASIS,
272 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
273 | See the License for the specific language governing permissions and
274 | limitations under the License.
275 |
276 |
277 |
--------------------------------------------------------------------------------
/README_CN.md:
--------------------------------------------------------------------------------
1 | # Android-Pay
2 | 支持微信和支付宝两种主流支付的集成库, 两行代码实现微信支付, 三行代码实现支付宝支付
3 |
4 | ## 引入
5 |
6 | ### gradle
7 | 对应的项目中的build.gradle文件添加依赖:
8 |
9 | ```xml
10 | dependencies {
11 | //添加支付库
12 | compile 'io.github.mayubao:pay_library:1.0.1'
13 | }
14 | ```
15 |
16 | ### maven
17 |
18 | ```xml
19 |
20 | io.github.mayubao
21 | pay_library
22 | 1.0.1
23 | pom
24 |
25 | ```
26 |
27 | ## 使用
28 |
29 | ### 微信支付使用
30 |
31 | ```java
32 | //1.创建微信支付请求
33 | WechatPayReq wechatPayReq = new WechatPayReq.Builder()
34 | .with(this) //activity实例
35 | .setAppId(appid) //微信支付AppID
36 | .setPartnerId(partnerid)//微信支付商户号
37 | .setPrepayId(prepayid)//预支付码
38 | // .setPackageValue(wechatPayReq.get)//"Sign=WXPay"
39 | .setNonceStr(noncestr)
40 | .setTimeStamp(timestamp)//时间戳
41 | .setSign(sign)//签名
42 | .create();
43 | //2.发送微信支付请求
44 | PayAPI.getInstance().sendPayRequest(wechatPayReq);
45 |
46 |
47 | //关于微信支付的回调
48 | //wechatPayReq.setOnWechatPayListener(new OnWechatPayListener);
49 |
50 |
51 | ```
52 |
53 |
54 |
55 | >注意:这里没有金额设置,金额的信息已经包含在预支付码prepayid了。
56 |
57 | ### 支付宝支付使用
58 |
59 | #### 支付宝支付第一种方式(不建议用这种方式,商户私钥暴露在客户端,极其危险,推荐用第二种支付方式)
60 | ```java
61 |
62 | //1.创建支付宝支付配置
63 | AliPayAPI.Config config = new AliPayAPI.Config.Builder()
64 | .setRsaPrivate(rsa_private) //设置私钥 (商户私钥,pkcs8格式)
65 | .setRsaPublic(rsa_public)//设置公钥(// 支付宝公钥)
66 | .setPartner(partner) //设置商户
67 | .setSeller(seller) //设置商户收款账号
68 | .create();
69 |
70 | //2.创建支付宝支付请求
71 | AliPayReq aliPayReq = new AliPayReq.Builder()
72 | .with(activity)//Activity实例
73 | .apply(config)//支付宝支付通用配置
74 | .setOutTradeNo(outTradeNo)//设置唯一订单号
75 | .setPrice(price)//设置订单价格
76 | .setSubject(orderSubject)//设置订单标题
77 | .setBody(orderBody)//设置订单内容 订单详情
78 | .setCallbackUrl(callbackUrl)//设置回调地址
79 | .create()//
80 | .setOnAliPayListener(null);//
81 |
82 | //3.发送支付宝支付请求
83 | PayAPI.getInstance().sendPayRequest(aliPayReq);
84 |
85 | //关于支付宝支付的回调
86 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
87 |
88 | ```
89 |
90 | #### 支付宝支付第二种方式(**强烈推荐**)
91 |
92 | ```java
93 | //1.创建支付宝支付订单的信息
94 | String rawAliOrderInfo = new AliPayReq2.AliOrderInfo()
95 | .setPartner(partner) //商户PID || 签约合作者身份ID
96 | .setSeller(seller) // 商户收款账号 || 签约卖家支付宝账号
97 | .setOutTradeNo(outTradeNo) //设置唯一订单号
98 | .setSubject(orderSubject) //设置订单标题
99 | .setBody(orderBody) //设置订单内容
100 | .setPrice(price) //设置订单价格
101 | .setCallbackUrl(callbackUrl) //设置回调链接
102 | .createOrderInfo(); //创建支付宝支付订单信息
103 |
104 |
105 | //2.签名 支付宝支付订单的信息 ===>>> 商户私钥签名之后的订单信息
106 | //TODO 这里需要从服务器获取用商户私钥签名之后的订单信息
107 | String signAliOrderInfo = getSignAliOrderInfoFromServer(rawAliOrderInfo);
108 |
109 | //3.发送支付宝支付请求
110 | AliPayReq2 aliPayReq = new AliPayReq2.Builder()
111 | .with(activity)//Activity实例
112 | .setRawAliPayOrderInfo(rawAliOrderInfo)//支付宝支付订单信息
113 | .setSignedAliPayOrderInfo(signAliOrderInfo) //设置 商户私钥RSA加密后的支付宝支付订单信息
114 | .create()//
115 | .setOnAliPayListener(null);//
116 | PayAPI.getInstance().sendPayRequest(aliPayReq);
117 |
118 | //关于支付宝支付的回调
119 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
120 | ```
121 |
122 | ## 存在的问题
123 |
124 | ### 微信支付回调的问题
125 |
126 | 之前微信支付是可以支付的,但是存在一个问题,就是微信支付回调的问题,我一直在寻找不同于官方的那样的回调方法,但是还是没有找到,只能想微信支付妥协~如果你想要微信支付的回调,还是可以实现的,只是做法跟微信支付官方文档一样。
127 |
128 |
129 | 解决方案:
130 |
131 | 1.在你的项目的包名下面新建wxapi包,
132 | 2.新建一个WXPayEntryActivity类,
133 | 3.实现IWXAPIEventHandler接口
134 | 4.在你对应的地方注册接收微信支付的广播即可
135 |
136 | 注意:
137 |
138 | 1.下面的是示例代码中的APP_ID需要替换为自己的
139 | 2.下面的是示例代码中的WeChatPayReceiver需要自己定义
140 |
141 | 下面是示例代码:
142 |
143 |
144 | ```java
145 |
146 | import android.app.Activity;
147 | import android.app.AlertDialog;
148 | import android.content.Intent;
149 | import android.os.Bundle;
150 | import android.util.Log;
151 | import android.widget.Toast;
152 |
153 | import com.tencent.mm.sdk.constants.ConstantsAPI;
154 | import com.tencent.mm.sdk.modelbase.BaseReq;
155 | import com.tencent.mm.sdk.modelbase.BaseResp;
156 | import com.tencent.mm.sdk.openapi.IWXAPI;
157 | import com.tencent.mm.sdk.openapi.IWXAPIEventHandler;
158 | import com.tencent.mm.sdk.openapi.WXAPIFactory;
159 |
160 | public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler{
161 |
162 | private static final String TAG = "MicroMsg.SDKSample.WXPayEntryActivity";
163 |
164 | private IWXAPI api;
165 |
166 | //TODO 这里需要替换你的APP_ID
167 | private String APP_ID = "wx0xxxxxxxxx"; //这里需要替换你的APP_ID
168 |
169 | @Override
170 | public void onCreate(Bundle savedInstanceState) {
171 | super.onCreate(savedInstanceState);
172 | // setContentView(R.layout.pay_result);
173 |
174 | api = WXAPIFactory.createWXAPI(this, APP_ID);
175 | api.handleIntent(getIntent(), this);
176 | }
177 |
178 | @Override
179 | protected void onNewIntent(Intent intent) {
180 | super.onNewIntent(intent);
181 | setIntent(intent);
182 | api.handleIntent(intent, this);
183 | }
184 |
185 | @Override
186 | public void onReq(BaseReq req) {
187 | }
188 |
189 | @Override
190 | public void onResp(BaseResp resp) {
191 | Log.d(TAG, "onPayFinish, errCode = " + resp.errCode);
192 |
193 | if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {
194 | // AlertDialog.Builder builder = new AlertDialog.Builder(this);
195 | // builder.setTitle(R.string.app_tip);
196 | // builder.setMessage(getString(R.string.pay_result_callback_msg, String.valueOf(resp.errCode)));
197 | // builder.show();
198 |
199 | //以下是自定义微信支付广播的发送,微信支付广播请自己定义
200 |
201 | Intent intent = new Intent();
202 | intent.setAction(WeChatPayReceiver.ACTION_PAY_RESULT);
203 | intent.putExtra("result", resp.errCode);
204 | sendBroadcast(intent);
205 |
206 | finish();
207 | }
208 | }
209 | }
210 |
211 | ```
212 |
213 | ## 混淆
214 |
215 | ```xml
216 |
217 | #pay_library
218 | -dontwarn io.github.mayubao.pay_library.**
219 | -keep class io.github.mayubao.pay_library.** {*;}
220 |
221 | #wechat pay
222 | -dontwarn com.tencent.**
223 | -keep class com.tencent.** {*;}
224 |
225 |
226 | #alipay
227 | -dontwarn com.alipay.**
228 | -keep class com.alipay.** {*;}
229 |
230 | -dontwarn com.ta.utdid2.**
231 | -keep class com.ta.utdid2.** {*;}
232 |
233 | -dontwarn com.ut.device.**
234 | -keep class com.ut.device.** {*;}
235 |
236 | -dontwarn org.json.alipay.**
237 | -keep class corg.json.alipay.** {*;}
238 | ```
239 |
240 |
241 | ## 文档
242 |
243 | ### 微信支付官方文档 支付流程
244 | https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5
245 |
246 | ### 支付宝支付官方文档 支付流程
247 | https://doc.open.alipay.com/docs/doc.htm?spm=a219a.7629140.0.0.sdGXaH&treeId=204&articleId=105296&docType=1
248 |
249 |
250 |
251 | ## 注意
252 |
253 | ### 微信支付
254 |
255 | - 微信支付必须要在**正式签名**和**正确包名**的应用中才能成功调起。(**重点)
256 |
257 | 即商户在微信开放平台申请开发应用后对应包名和对应签名的应用才能成功调起。
258 | 详情请参考微信支付的开发流程文档。
259 |
260 | https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5
261 |
262 | - 微信支付API没有在客户端显示的设置回调,回调是在Server端设置的。(与支付宝支付的区别,支付宝的回调是在客户端设置的)
263 |
264 | ### 支付宝支付
265 |
266 | - 支付宝支付为了保证交易双方的身份和数据安全, 需要配置双方密钥。
267 |
268 | 详情请参考支付宝支付的密钥处理体系文档。
269 |
270 | https://doc.open.alipay.com/docs/doc.htm?spm=a219a.7629140.0.0.1wPnBT&treeId=204&articleId=106079&docType=1
271 |
272 | ## 打赏
273 | 如果你觉得此项目对你有用,能否赏我一杯咖啡呢?
274 |
275 | ### 微信支付
276 | 
277 | ### 支付宝支付
278 | 
279 |
280 |
281 | ## Lisence
282 |
283 | Copyright 2017 mayubao
284 |
285 | Licensed under the Apache License, Version 2.0 (the "License");
286 | you may not use this file except in compliance with the License.
287 | You may obtain a copy of the License at
288 |
289 | http://www.apache.org/licenses/LICENSE-2.0
290 |
291 | Unless required by applicable law or agreed to in writing, software
292 | distributed under the License is distributed on an "AS IS" BASIS,
293 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
294 | See the License for the specific language governing permissions and
295 | limitations under the License.
296 |
297 |
--------------------------------------------------------------------------------
/README_EN.md:
--------------------------------------------------------------------------------
1 | # Android-Pay
2 | [中文](https://github.com/mayubao/Android-Pay/blob/master/README_CN.md)
3 |
4 | A pay libary for Android, and which support Wechat pay and Ali pay.
5 | And developer can easily use Wechat pay in two line code.
6 | And developer can easily use Ali pay in three line code.
7 |
8 | ## Include
9 |
10 | ### gradle
11 | add these code in your app **build.gradle** as follow:
12 |
13 | ```xml
14 | dependencies {
15 | //add pay libaray
16 | compile 'io.github.mayubao:pay_library:1.0.1'
17 | }
18 | ```
19 |
20 | ### maven
21 |
22 | ```xml
23 |
24 | io.github.mayubao
25 | pay_library
26 | 1.0.1
27 | pom
28 |
29 | ```
30 |
31 | ## Usage
32 |
33 | ### Wechat Pay Usage
34 |
35 | ```java
36 | //1.create request for wechat pay
37 | WechatPayReq wechatPayReq = new WechatPayReq.Builder()
38 | .with(this) //activity instance
39 | .setAppId(appid) //wechat pay AppID
40 | .setPartnerId(partnerid)//wechat pay partner id
41 | .setPrepayId(prepayid)//pre pay id
42 | // .setPackageValue(wechatPayReq.get)//"Sign=WXPay"
43 | .setNonceStr(noncestr)
44 | .setTimeStamp(timestamp)//time stamp
45 | .setSign(sign)//sign
46 | .create();
47 | //2. send the request with wechat pay
48 | PayAPI.getInstance().sendPayRequest(wechatPayReq);
49 |
50 |
51 | //set the callback for wechat pay
52 | //wechatPayReq.setOnWechatPayListener(new OnWechatPayListener);
53 |
54 |
55 | ```
56 |
57 | >Notes:WechatPayReq have no method to set the money, because the money info is include in the parameter 'prepayid'.
58 |
59 | ### Ali Pay Usage
60 |
61 |
62 | #### Ali Pay first way(**Not Recommend**, and its partner rsa private key export in the client , it is very dangerous!)
63 | ```java
64 |
65 | //step 1 create config for ali pay
66 | AliPayAPI.Config config = new AliPayAPI.Config.Builder()
67 | .setRsaPrivate(rsa_private) // rsa private key from partner (pkcs8 format)
68 | .setRsaPublic(rsa_public)//ali rsa public key
69 | .setPartner(partner) //set partner
70 | .setSeller(seller) //set partner seller accout
71 | .create();
72 |
73 | //step 2 create reqeust for ali
74 | AliPayReq aliPayReq = new AliPayReq.Builder()
75 | .with(activity)//Activity instance
76 | .apply(config)// the above custome config
77 | .setOutTradeNo(outTradeNo)//set unique trade no
78 | .setPrice(price)//set price
79 | .setSubject(orderSubject)//set order subject
80 | .setBody(orderBody)//set order detail
81 | .setCallbackUrl(callbackUrl)//set callback for pay reqest
82 | .create()//
83 | .setOnAliPayListener(null);//
84 |
85 | //step 3 send the request for ali pay
86 | PayAPI.getInstance().sendPayRequest(aliPayReq);
87 |
88 | // set the ali pay callback
89 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
90 |
91 | ```
92 |
93 | #### Ali Pay second way(**Highly Recommend**)
94 |
95 | ```java
96 | //step 1 create raw ali pay order info
97 | String rawAliOrderInfo = new AliPayReq2.AliOrderInfo()
98 | .setPartner(partner) //set partner
99 | .setSeller(seller) //set partner seller accout
100 | .setOutTradeNo(outTradeNo) //set unique trade no
101 | .setSubject(orderSubject) //set order subject
102 | .setBody(orderBody) //set order detail
103 | .setPrice(price) //set price
104 | .setCallbackUrl(callbackUrl) //set callback for pay reqest
105 | .createOrderInfo(); //create ali pay order info
106 |
107 |
108 | //step 2 get the signed ali pay order info
109 | String signAliOrderInfo = getSignAliOrderInfoFromServer(rawAliOrderInfo);
110 |
111 | //step 3 step 3 send the request for ali pay
112 | AliPayReq2 aliPayReq = new AliPayReq2.Builder()
113 | .with(activity)//Activity instance
114 | .setSignedAliPayOrderInfo(signAliOrderInfo)
115 | .setRawAliPayOrderInfo(rawAliOrderInfo)//set the ali pay order info
116 | .setSignedAliPayOrderInfo(signAliOrderInfo) //set the signed ali pay order info
117 | .create()//
118 | .setOnAliPayListener(null);//
119 | PayAPI.getInstance().sendPayRequest(aliPayReq);
120 |
121 |
122 | //set the ali pay callback
123 | //aliPayReq.setOnAliPayListener(new OnAliPayListener);
124 |
125 | ```
126 |
127 |
128 | ## Proguard
129 |
130 | ```xml
131 |
132 | #pay_library
133 | -dontwarn io.github.mayubao.pay_library.**
134 | -keep class io.github.mayubao.pay_library.** {*;}
135 |
136 | #wechat pay
137 | -dontwarn com.tencent.**
138 | -keep class com.tencent.** {*;}
139 |
140 |
141 | #alipay
142 | -dontwarn com.alipay.**
143 | -keep class com.alipay.** {*;}
144 |
145 | -dontwarn com.ta.utdid2.**
146 | -keep class com.ta.utdid2.** {*;}
147 |
148 | -dontwarn com.ut.device.**
149 | -keep class com.ut.device.** {*;}
150 |
151 | -dontwarn org.json.alipay.**
152 | -keep class corg.json.alipay.** {*;}
153 | ```
154 |
155 |
156 | ## Document
157 |
158 | ### wehcat pay official document
159 | https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5
160 |
161 | ### ali pay official document
162 | https://doc.open.alipay.com/docs/doc.htm?spm=a219a.7629140.0.0.sdGXaH&treeId=204&articleId=105296&docType=1
163 |
164 |
165 | ## Help
166 | If it is helpful to you, could you buy me a cup of coffee?
167 |
168 | ### Wechat
169 | 
170 | ### Ali
171 | 
172 |
173 |
174 | ## Lisence
175 |
176 | Copyright 2017 mayubao
177 |
178 | Licensed under the Apache License, Version 2.0 (the "License");
179 | you may not use this file except in compliance with the License.
180 | You may obtain a copy of the License at
181 |
182 | http://www.apache.org/licenses/LICENSE-2.0
183 |
184 | Unless required by applicable law or agreed to in writing, software
185 | distributed under the License is distributed on an "AS IS" BASIS,
186 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
187 | See the License for the specific language governing permissions and
188 | limitations under the License.
189 |
190 |
191 |
--------------------------------------------------------------------------------
/app/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.application'
2 |
3 | android {
4 | compileSdkVersion 23
5 | buildToolsVersion "24.0.1"
6 |
7 | defaultConfig {
8 | applicationId "io.github.mayubao.android_pay"
9 | minSdkVersion 14
10 | targetSdkVersion 23
11 | versionCode 1
12 | versionName "1.0"
13 | }
14 | buildTypes {
15 | release {
16 | minifyEnabled false
17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
18 | }
19 | }
20 | }
21 |
22 | dependencies {
23 | compile fileTree(include: ['*.jar'], dir: 'libs')
24 | testCompile 'junit:junit:4.12'
25 | compile 'com.android.support:appcompat-v7:23.4.0'
26 | compile 'com.android.support:design:23.4.0'
27 | compile project(':pay-library')
28 | }
29 |
--------------------------------------------------------------------------------
/app/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in D:\android\adt-bundle-windows-x86_64-20140702\sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/app/src/androidTest/java/io/github/mayubao/android_pay/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
11 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
25 |
29 |
33 |
34 |
35 |
36 |
--------------------------------------------------------------------------------
/app/src/main/java/io/github/mayubao/android_pay/MainActivity.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import android.content.Intent;
4 | import android.os.Bundle;
5 | import android.support.design.widget.FloatingActionButton;
6 | import android.support.design.widget.Snackbar;
7 | import android.support.v7.app.AppCompatActivity;
8 | import android.support.v7.widget.Toolbar;
9 | import android.view.View;
10 | import android.view.Menu;
11 | import android.view.MenuItem;
12 |
13 | public class MainActivity extends AppCompatActivity {
14 |
15 | @Override
16 | protected void onCreate(Bundle savedInstanceState) {
17 | super.onCreate(savedInstanceState);
18 | setContentView(R.layout.activity_main);
19 | Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
20 | setSupportActionBar(toolbar);
21 |
22 | FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
23 | fab.setOnClickListener(new View.OnClickListener() {
24 | @Override
25 | public void onClick(View view) {
26 | Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
27 | .setAction("Action", null).show();
28 | }
29 | });
30 | }
31 |
32 | @Override
33 | public boolean onCreateOptionsMenu(Menu menu) {
34 | // Inflate the menu; this adds items to the action bar if it is present.
35 | getMenuInflater().inflate(R.menu.menu_main, menu);
36 | return true;
37 | }
38 |
39 | @Override
40 | public boolean onOptionsItemSelected(MenuItem item) {
41 | // Handle action bar item clicks here. The action bar will
42 | // automatically handle clicks on the Home/Up button, so long
43 | // as you specify a parent activity in AndroidManifest.xml.
44 | int id = item.getItemId();
45 |
46 | //noinspection SimplifiableIfStatement
47 | if (id == R.id.action_settings) {
48 | gotoUI(TestOnlineActivity.class); //作为库的module的AndroidManifest.xml是会被合并的
49 | return true;
50 | }else if (id == R.id.action_test_wechat_pay) {
51 | gotoUI(PayWechatActivity.class);
52 | return true;
53 | }else if (id == R.id.action_test_ali_pay) {
54 | gotoUI(PayAliPayActivity.class);
55 | return true;
56 | }
57 |
58 | return super.onOptionsItemSelected(item);
59 | }
60 |
61 |
62 | /**
63 | * 跳转到支付测试页面
64 | * @param clazz
65 | */
66 | private void gotoUI(Class clazz){
67 | Intent intent = new Intent(this, clazz);
68 | startActivity(intent);
69 | }
70 |
71 | }
72 |
--------------------------------------------------------------------------------
/app/src/main/java/io/github/mayubao/android_pay/PayAliPayActivity.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import android.os.Bundle;
4 | import android.support.design.widget.FloatingActionButton;
5 | import android.support.design.widget.Snackbar;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.support.v7.widget.Toolbar;
8 | import android.view.View;
9 |
10 | public class PayAliPayActivity extends AppCompatActivity {
11 |
12 | @Override
13 | protected void onCreate(Bundle savedInstanceState) {
14 | super.onCreate(savedInstanceState);
15 | setContentView(R.layout.activity_pay_ali_pay);
16 | Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
17 | setSupportActionBar(toolbar);
18 |
19 | FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
20 | fab.setOnClickListener(new View.OnClickListener() {
21 | @Override
22 | public void onClick(View view) {
23 | Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
24 | .setAction("Action", null).show();
25 | }
26 | });
27 | }
28 |
29 | }
30 |
--------------------------------------------------------------------------------
/app/src/main/java/io/github/mayubao/android_pay/PayWechatActivity.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import android.os.Bundle;
4 | import android.support.design.widget.FloatingActionButton;
5 | import android.support.design.widget.Snackbar;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.support.v7.widget.Toolbar;
8 | import android.view.View;
9 |
10 | public class PayWechatActivity extends AppCompatActivity {
11 |
12 | @Override
13 | protected void onCreate(Bundle savedInstanceState) {
14 | super.onCreate(savedInstanceState);
15 | setContentView(R.layout.activity_pay_wechat);
16 | Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
17 | setSupportActionBar(toolbar);
18 |
19 | FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
20 | fab.setOnClickListener(new View.OnClickListener() {
21 | @Override
22 | public void onClick(View view) {
23 | Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
24 | .setAction("Action", null).show();
25 | }
26 | });
27 | }
28 |
29 | }
30 |
--------------------------------------------------------------------------------
/app/src/main/java/io/github/mayubao/android_pay/TestOnlineActivity.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import android.os.Bundle;
4 | import android.support.design.widget.FloatingActionButton;
5 | import android.support.design.widget.Snackbar;
6 | import android.support.v7.app.AppCompatActivity;
7 | import android.support.v7.widget.Toolbar;
8 | import android.view.View;
9 | import android.webkit.WebChromeClient;
10 | import android.webkit.WebView;
11 | import android.webkit.WebViewClient;
12 |
13 | public class TestOnlineActivity extends AppCompatActivity {
14 |
15 | public static final String DEFAULT_TEST_URL = "http://www.baidu.com";
16 |
17 | @Override
18 | protected void onCreate(Bundle savedInstanceState) {
19 | super.onCreate(savedInstanceState);
20 | setContentView(R.layout.activity_test_online);
21 | Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
22 | setSupportActionBar(toolbar);
23 |
24 | FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
25 | fab.setOnClickListener(new View.OnClickListener() {
26 | @Override
27 | public void onClick(View view) {
28 | Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
29 | .setAction("Action", null).show();
30 | }
31 | });
32 |
33 |
34 | final WebView webView = (WebView) this.findViewById(R.id.webView);
35 |
36 | webView.getSettings().setJavaScriptEnabled(true);
37 | webView.setWebViewClient(new WebViewClient() {
38 | @Override
39 | public boolean shouldOverrideUrlLoading(WebView view, String url) {
40 | webView.loadUrl(url);
41 | return super.shouldOverrideUrlLoading(view, url);
42 | }
43 | });
44 |
45 | webView.setWebChromeClient(new WebChromeClient() {
46 |
47 | });
48 |
49 | webView.loadUrl(DEFAULT_TEST_URL);
50 | }
51 |
52 | }
53 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
21 |
22 |
23 |
24 |
25 |
26 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_pay_ali_pay.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
21 |
22 |
23 |
24 |
25 |
26 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_pay_wechat.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
21 |
22 |
23 |
24 |
25 |
26 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_test_online.xml:
--------------------------------------------------------------------------------
1 |
2 |
9 |
10 |
14 |
15 |
21 |
22 |
23 |
24 |
25 |
26 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/content_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
14 |
15 |
19 |
20 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/content_pay_ali_pay.xml:
--------------------------------------------------------------------------------
1 |
2 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/content_pay_wechat.xml:
--------------------------------------------------------------------------------
1 |
2 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/app/src/main/res/layout/content_test_online.xml:
--------------------------------------------------------------------------------
1 |
2 |
14 |
18 |
19 |
20 |
21 |
22 |
--------------------------------------------------------------------------------
/app/src/main/res/menu/menu_main.xml:
--------------------------------------------------------------------------------
1 |
21 |
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/app/src/main/res/values-v21/styles.xml:
--------------------------------------------------------------------------------
1 | >
2 |
3 |
9 |
10 |
--------------------------------------------------------------------------------
/app/src/main/res/values-w820dp/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 | 64dp
6 |
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #3F51B5
4 | #303F9F
5 | #FF4081
6 |
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values/dimens.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 16dp
4 | 16dp
5 | 16dp
6 |
7 |
--------------------------------------------------------------------------------
/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | Android-Pay
3 | Settings
4 | 微信支付测试
5 | 支付宝支付测试
6 | PayWechatActivity
7 | PayAliPayActivity
8 | TestOnlineActivity
9 |
10 |
--------------------------------------------------------------------------------
/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
10 |
11 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
--------------------------------------------------------------------------------
/app/src/test/java/io/github/mayubao/android_pay/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.android_pay;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 | repositories {
5 | jcenter()
6 | }
7 | dependencies {
8 | classpath 'com.android.tools.build:gradle:1.5.0'
9 |
10 | classpath 'com.novoda:bintray-release:0.3.4'
11 | // NOTE: Do not place your application dependencies here; they belong
12 | // in the individual module build.gradle files
13 | }
14 | }
15 |
16 | allprojects {
17 | repositories {
18 | jcenter()
19 | }
20 |
21 | tasks.withType(Javadoc) {
22 | options.addStringOption('Xdoclint:none', '-quiet')
23 | options.addStringOption('encoding', 'UTF-8')
24 | }
25 | }
26 |
27 | task clean(type: Delete) {
28 | delete rootProject.buildDir
29 | }
30 |
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Wed Oct 21 11:34:03 PDT 2015
2 | distributionBase=GRADLE_USER_HOME
3 | distributionPath=wrapper/dists
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.8-all.zip
7 |
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # Attempt to set APP_HOME
46 | # Resolve links: $0 may be a link
47 | PRG="$0"
48 | # Need this for relative symlinks.
49 | while [ -h "$PRG" ] ; do
50 | ls=`ls -ld "$PRG"`
51 | link=`expr "$ls" : '.*-> \(.*\)$'`
52 | if expr "$link" : '/.*' > /dev/null; then
53 | PRG="$link"
54 | else
55 | PRG=`dirname "$PRG"`"/$link"
56 | fi
57 | done
58 | SAVED="`pwd`"
59 | cd "`dirname \"$PRG\"`/" >/dev/null
60 | APP_HOME="`pwd -P`"
61 | cd "$SAVED" >/dev/null
62 |
63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
64 |
65 | # Determine the Java command to use to start the JVM.
66 | if [ -n "$JAVA_HOME" ] ; then
67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
68 | # IBM's JDK on AIX uses strange locations for the executables
69 | JAVACMD="$JAVA_HOME/jre/sh/java"
70 | else
71 | JAVACMD="$JAVA_HOME/bin/java"
72 | fi
73 | if [ ! -x "$JAVACMD" ] ; then
74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
75 |
76 | Please set the JAVA_HOME variable in your environment to match the
77 | location of your Java installation."
78 | fi
79 | else
80 | JAVACMD="java"
81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
82 |
83 | Please set the JAVA_HOME variable in your environment to match the
84 | location of your Java installation."
85 | fi
86 |
87 | # Increase the maximum file descriptors if we can.
88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
89 | MAX_FD_LIMIT=`ulimit -H -n`
90 | if [ $? -eq 0 ] ; then
91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
92 | MAX_FD="$MAX_FD_LIMIT"
93 | fi
94 | ulimit -n $MAX_FD
95 | if [ $? -ne 0 ] ; then
96 | warn "Could not set maximum file descriptor limit: $MAX_FD"
97 | fi
98 | else
99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
100 | fi
101 | fi
102 |
103 | # For Darwin, add options to specify how the application appears in the dock
104 | if $darwin; then
105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
106 | fi
107 |
108 | # For Cygwin, switch paths to Windows format before running java
109 | if $cygwin ; then
110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
112 | JAVACMD=`cygpath --unix "$JAVACMD"`
113 |
114 | # We build the pattern for arguments to be converted via cygpath
115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
116 | SEP=""
117 | for dir in $ROOTDIRSRAW ; do
118 | ROOTDIRS="$ROOTDIRS$SEP$dir"
119 | SEP="|"
120 | done
121 | OURCYGPATTERN="(^($ROOTDIRS))"
122 | # Add a user-defined pattern to the cygpath arguments
123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
125 | fi
126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
127 | i=0
128 | for arg in "$@" ; do
129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
131 |
132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
134 | else
135 | eval `echo args$i`="\"$arg\""
136 | fi
137 | i=$((i+1))
138 | done
139 | case $i in
140 | (0) set -- ;;
141 | (1) set -- "$args0" ;;
142 | (2) set -- "$args0" "$args1" ;;
143 | (3) set -- "$args0" "$args1" "$args2" ;;
144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
150 | esac
151 | fi
152 |
153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
154 | function splitJvmOpts() {
155 | JVM_OPTS=("$@")
156 | }
157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
159 |
160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
161 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/pay-library/.gitignore:
--------------------------------------------------------------------------------
1 | /build
2 |
--------------------------------------------------------------------------------
/pay-library/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: 'com.android.library'
2 | apply plugin: 'com.novoda.bintray-release'//添加
3 |
4 | android {
5 | compileSdkVersion 23
6 | buildToolsVersion "24.0.1"
7 |
8 | defaultConfig {
9 | minSdkVersion 14
10 | targetSdkVersion 23
11 | versionCode 1
12 | versionName "1.0"
13 | }
14 | buildTypes {
15 | release {
16 | minifyEnabled false
17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
18 | }
19 | }
20 |
21 | lintOptions {
22 | abortOnError false
23 | }
24 |
25 | }
26 |
27 | dependencies {
28 | compile fileTree(include: ['*.jar'], dir: 'libs')
29 | testCompile 'junit:junit:4.12'
30 | compile 'com.android.support:appcompat-v7:23.4.0'
31 | compile files('libs/alipaySDK-20150818.jar')
32 | compile files('libs/libammsdk.jar')
33 | }
34 |
35 |
36 | //添加
37 | publish {
38 | userOrg = 'mayubao'//bintray.com用户名
39 | groupId = 'io.github.mayubao'//jcenter上的路径
40 | artifactId = 'pay_library'//项目名称
41 | publishVersion = '1.0.1'//版本号
42 | desc = 'This is a pay library for android, it is very easy to use to integrated payment'//描述
43 | website = 'https://github.com/mayubao/Android-Pay.git'//网站 默认github上面的
44 | }
45 |
--------------------------------------------------------------------------------
/pay-library/libs/alipaySDK-20150818.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/pay-library/libs/alipaySDK-20150818.jar
--------------------------------------------------------------------------------
/pay-library/libs/libammsdk.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/pay-library/libs/libammsdk.jar
--------------------------------------------------------------------------------
/pay-library/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in D:\android\adt-bundle-windows-x86_64-20140702\sdk/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
--------------------------------------------------------------------------------
/pay-library/src/androidTest/java/io/github/mayubao/pay_library/ApplicationTest.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import android.app.Application;
4 | import android.test.ApplicationTestCase;
5 |
6 | /**
7 | * Testing Fundamentals
8 | */
9 | public class ApplicationTest extends ApplicationTestCase {
10 | public ApplicationTest() {
11 | super(Application.class);
12 | }
13 | }
--------------------------------------------------------------------------------
/pay-library/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
16 |
17 |
23 |
24 |
25 |
26 |
27 |
28 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/AliPayAPI.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | /**
4 | * 支付宝支付API
5 | *
6 | * 使用:
7 | *
8 | * AliPayAPI.getInstance().apply(config).sendPayReq(aliPayReq);
9 | *
10 | * Created by mayubao on 2017/3/5.
11 | * Contact me 345269374@qq.com
12 | */
13 | public class AliPayAPI {
14 |
15 | private Config mConfig;
16 |
17 | /**
18 | * 获取支付宝支付API
19 | */
20 | private static final Object mLock = new Object();
21 | private static AliPayAPI mInstance;
22 |
23 | public static AliPayAPI getInstance(){
24 | if(mInstance == null){
25 | synchronized (mLock){
26 | if(mInstance == null){
27 | mInstance = new AliPayAPI();
28 | }
29 | }
30 | }
31 | return mInstance;
32 | }
33 |
34 |
35 | /**
36 | * 配置支付宝配置
37 | * @param config
38 | * @return
39 | */
40 | public AliPayAPI apply(Config config){
41 | this.mConfig = config;
42 | return this;
43 | }
44 |
45 | /**
46 | * 发送支付宝支付请求
47 | * @param aliPayReq
48 | */
49 | public void sendPayReq(AliPayReq aliPayReq){
50 | aliPayReq.send();
51 | }
52 |
53 |
54 | /**
55 | * 发送支付宝支付请求
56 | * @param aliPayReq2
57 | */
58 | public void sendPayReq(AliPayReq2 aliPayReq2){
59 | aliPayReq2.send();
60 | }
61 |
62 |
63 | /**
64 | * 支付宝支付配置
65 | * @author Administrator
66 | *
67 | */
68 | public static class Config{
69 | //ali pay config
70 | // 商户私钥,pkcs8格式
71 | private String aliRsaPrivate;
72 | // 支付宝公钥
73 | private String aliRsaPublic;
74 | // 商户PID
75 | // 签约合作者身份ID
76 | private String partner;
77 | // 商户收款账号
78 | // 签约卖家支付宝账号
79 | private String seller;
80 |
81 | public String getAliRsaPrivate() {
82 | return aliRsaPrivate;
83 | }
84 |
85 | public void setAliRsaPrivate(String aliRsaPrivate) {
86 | this.aliRsaPrivate = aliRsaPrivate;
87 | }
88 |
89 | public String getAliRsaPublic() {
90 | return aliRsaPublic;
91 | }
92 |
93 | public void setAliRsaPublic(String aliRsaPublic) {
94 | this.aliRsaPublic = aliRsaPublic;
95 | }
96 |
97 | public String getPartner() {
98 | return partner;
99 | }
100 |
101 | public void setPartner(String partner) {
102 | this.partner = partner;
103 | }
104 |
105 | public String getSeller() {
106 | return seller;
107 | }
108 |
109 | public void setSeller(String seller) {
110 | this.seller = seller;
111 | }
112 |
113 | public static class Builder{
114 | //ali pay config
115 | // 商户私钥,pkcs8格式
116 | private String aliRsaPrivate;
117 | // 支付宝公钥
118 | private String aliRsaPublic;
119 | // 商户PID
120 | // 签约合作者身份ID
121 | private String partner;
122 | // 商户收款账号
123 | // 签约卖家支付宝账号
124 | private String seller;
125 |
126 | public Builder() {
127 | super();
128 | }
129 |
130 | public Builder setRsaPrivate(String aliRsaPrivate){
131 | this.aliRsaPrivate = aliRsaPrivate;
132 | return this;
133 | }
134 |
135 | public Builder setRsaPublic(String aliRsaPublic){
136 | this.aliRsaPublic = aliRsaPublic;
137 | return this;
138 | }
139 |
140 | public Builder setPartner(String partner){
141 | this.partner = partner;
142 | return this;
143 | }
144 |
145 | public Builder setSeller(String seller){
146 | this.seller = seller;
147 | return this;
148 | }
149 |
150 | public Config create(){
151 | Config conf = new Config();
152 | conf.aliRsaPrivate = this.aliRsaPrivate;
153 | conf.aliRsaPublic = this.aliRsaPublic;
154 | conf.partner = this.partner;
155 | conf.seller = this.seller;
156 | return conf;
157 | }
158 | }
159 | }
160 | }
161 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/AliPayReq.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import android.app.Activity;
4 | import android.os.Handler;
5 | import android.os.Message;
6 | import android.text.TextUtils;
7 | import android.widget.Toast;
8 |
9 | import com.alipay.sdk.app.PayTask;
10 |
11 | import java.io.UnsupportedEncodingException;
12 | import java.net.URLEncoder;
13 |
14 | import io.github.mayubao.pay_library.alipay.PayResult;
15 | import io.github.mayubao.pay_library.alipay.SignUtils;
16 |
17 | /**
18 | * 支付宝支付请求
19 | *
20 | * Created by mayubao on 2017/3/5.
21 | * Contact me 345269374@qq.com
22 | */
23 | public class AliPayReq {
24 |
25 | /**
26 | * ali pay sdk flag
27 | */
28 | private static final int SDK_PAY_FLAG = 1;
29 | private static final int SDK_CHECK_FLAG = 2;
30 |
31 | private Activity mActivity;
32 |
33 | //支付宝支付的配置
34 | private AliPayAPI.Config mConfig;
35 |
36 | // 商户网站唯一订单号
37 | private String outTradeNo;
38 | // 商品名称
39 | private String subject;
40 | // 商品详情
41 | private String body;
42 | // 商品金额
43 | private String price;
44 | // 服务器异步通知页面路径
45 | private String callbackUrl;
46 |
47 | private Handler mHandler;
48 |
49 | public AliPayReq() {
50 | super();
51 | mHandler = new Handler(){
52 |
53 | @Override
54 | public void handleMessage(Message msg) {
55 | switch (msg.what) {
56 | case SDK_PAY_FLAG: {
57 | PayResult payResult = new PayResult((String) msg.obj);
58 |
59 | // 支付宝返回此次支付结果及加签,建议对支付宝签名信息拿签约时支付宝提供的公钥做验签
60 | String resultInfo = payResult.getResult();
61 |
62 | String resultStatus = payResult.getResultStatus();
63 |
64 | // 判断resultStatus 为“9000”则代表支付成功,具体状态码代表含义可参考接口文档
65 | if (TextUtils.equals(resultStatus, "9000")) {
66 | Toast.makeText(mActivity, "支付成功", Toast.LENGTH_SHORT).show();
67 | if(mOnAliPayListener != null) mOnAliPayListener.onPaySuccess(resultInfo);
68 | } else {
69 | // 判断resultStatus 为非“9000”则代表可能支付失败
70 | // “8000”代表支付结果因为支付渠道原因或者系统原因还在等待支付结果确认,最终交易是否成功以服务端异步通知为准(小概率状态)
71 | if (TextUtils.equals(resultStatus, "8000")) {
72 | Toast.makeText(mActivity, "支付结果确认中", Toast.LENGTH_SHORT).show();
73 | if(mOnAliPayListener != null) mOnAliPayListener.onPayConfirmimg(resultInfo);
74 |
75 | } else {
76 | // 其他值就可以判断为支付失败,包括用户主动取消支付,或者系统返回的错误
77 | Toast.makeText(mActivity, "支付失败", Toast.LENGTH_SHORT).show();
78 | if(mOnAliPayListener != null) mOnAliPayListener.onPayFailure(resultInfo);
79 | }
80 | }
81 | break;
82 | }
83 | case SDK_CHECK_FLAG: {
84 | Toast.makeText(mActivity, "检查结果为:" + msg.obj, Toast.LENGTH_SHORT).show();
85 | if(mOnAliPayListener != null) mOnAliPayListener.onPayCheck(msg.obj.toString());
86 | break;
87 | }
88 | default:
89 | break;
90 | }
91 | }
92 |
93 | };
94 | }
95 |
96 |
97 | /**
98 | * 发送支付宝支付请求
99 | * @param config 支付宝配置
100 | */
101 | public void sendWithConfig(AliPayAPI.Config config) {
102 | this.mConfig = config;
103 | send();
104 | }
105 |
106 |
107 | /**
108 | * 发送支付宝支付请求
109 | */
110 | public void send() {
111 | // 创建订单信息
112 | String orderInfo = getOrderInfo(this.mConfig.getPartner(),
113 | this.mConfig.getSeller(), this.outTradeNo, this.subject, this.body,
114 | this.price, this.callbackUrl);
115 | // 对订单做RSA 签名
116 | String sign = sign(orderInfo);
117 | try {
118 | // 仅需对sign 做URL编码
119 | sign = URLEncoder.encode(sign, "UTF-8");
120 | } catch (UnsupportedEncodingException e) {
121 | e.printStackTrace();
122 | }
123 |
124 | // 完整的符合支付宝参数规范的订单信息
125 | final String payInfo = orderInfo + "&sign=\"" + sign + "\"&"
126 | + getSignType();
127 |
128 | Runnable payRunnable = new Runnable() {
129 |
130 | @Override
131 | public void run() {
132 | // 构造PayTask 对象
133 | PayTask alipay = new PayTask(mActivity);
134 | // 调用支付接口,获取支付结果
135 | String result = alipay.pay(payInfo);
136 |
137 | Message msg = new Message();
138 | msg.what = SDK_PAY_FLAG;
139 | msg.obj = result;
140 | mHandler.sendMessage(msg);
141 | }
142 | };
143 |
144 | // 必须异步调用
145 | Thread payThread = new Thread(payRunnable);
146 | payThread.start();
147 | }
148 |
149 | /**
150 | * 查询终端设备是否存在支付宝认证账户
151 | */
152 | public void check(){
153 | Runnable checkRunnable = new Runnable() {
154 |
155 | @Override
156 | public void run() {
157 | // 构造PayTask 对象
158 | PayTask payTask = new PayTask(mActivity);
159 | // 调用查询接口,获取查询结果
160 | boolean isExist = payTask.checkAccountIfExist();
161 |
162 | Message msg = new Message();
163 | msg.what = SDK_CHECK_FLAG;
164 | msg.obj = isExist;
165 | mHandler.sendMessage(msg);
166 | }
167 | };
168 |
169 | Thread checkThread = new Thread(checkRunnable);
170 | checkThread.start();
171 | }
172 |
173 |
174 | /**
175 | * 创建订单信息
176 | *
177 | * @param partner 签约合作者身份ID
178 | * @param seller 签约卖家支付宝账号
179 | * @param outTradeNo 商户网站唯一订单号
180 | * @param subject 商品名称
181 | * @param body 商品详情
182 | * @param price 商品金额
183 | * @param callbackUrl 服务器异步通知页面路径
184 | * @return
185 | */
186 | public String getOrderInfo(String partner, String seller, String outTradeNo, String subject, String body, String price, String callbackUrl) {
187 |
188 | // 签约合作者身份ID
189 | String orderInfo = "partner=" + "\"" + partner + "\"";
190 |
191 | // 签约卖家支付宝账号
192 | orderInfo += "&seller_id=" + "\"" + seller + "\"";
193 |
194 | // 商户网站唯一订单号
195 | orderInfo += "&out_trade_no=" + "\"" + outTradeNo + "\"";
196 |
197 | // 商品名称
198 | orderInfo += "&subject=" + "\"" + subject + "\"";
199 |
200 | // 商品详情
201 | orderInfo += "&body=" + "\"" + body + "\"";
202 |
203 | // 商品金额
204 | orderInfo += "&total_fee=" + "\"" + price + "\"";
205 |
206 | // 服务器异步通知页面路径
207 | // orderInfo += "¬ify_url=" + "\"" + "http://notify.msp.hk/notify.htm"
208 | // + "\"";
209 | orderInfo += "¬ify_url=" + "\"" + callbackUrl
210 | + "\"";
211 |
212 | // 服务接口名称, 固定值
213 | orderInfo += "&service=\"mobile.securitypay.pay\"";
214 |
215 | // 支付类型, 固定值
216 | orderInfo += "&payment_type=\"1\"";
217 |
218 | // 参数编码, 固定值
219 | orderInfo += "&_input_charset=\"utf-8\"";
220 |
221 | // 设置未付款交易的超时时间
222 | // 默认30分钟,一旦超时,该笔交易就会自动被关闭。
223 | // 取值范围:1m~15d。
224 | // m-分钟,h-小时,d-天,1c-当天(无论交易何时创建,都在0点关闭)。
225 | // 该参数数值不接受小数点,如1.5h,可转换为90m。
226 | orderInfo += "&it_b_pay=\"30m\"";
227 |
228 | // extern_token为经过快登授权获取到的alipay_open_id,带上此参数用户将使用授权的账户进行支付
229 | // orderInfo += "&extern_token=" + "\"" + extern_token + "\"";
230 |
231 | // 支付宝处理完请求后,当前页面跳转到商户指定页面的路径,可空
232 | orderInfo += "&return_url=\"m.alipay.com\"";
233 |
234 | // 调用银行卡支付,需配置此参数,参与签名, 固定值 (需要签约《无线银行卡快捷支付》才能使用)
235 | // orderInfo += "&paymethod=\"expressGateway\"";
236 |
237 | return orderInfo;
238 | }
239 |
240 |
241 | /**
242 | * sign the order info. 对订单信息进行签名
243 | *
244 | * @param content
245 | * 待签名订单信息
246 | */
247 | public String sign(String content) {
248 | return SignUtils.sign(content, this.mConfig.getAliRsaPrivate());
249 | }
250 |
251 | /**
252 | * get the sign type we use. 获取签名方式
253 | *
254 | */
255 | public String getSignType() {
256 | return "sign_type=\"RSA\"";
257 | }
258 |
259 |
260 | public static class Builder{
261 | //上下文
262 | private Activity activity;
263 | //支付宝支付配置
264 | private AliPayAPI.Config config;
265 | // 商户网站唯一订单号
266 | private String outTradeNo;
267 | // 商品名称
268 | private String subject;
269 | // 商品详情
270 | private String body;
271 | // 商品金额
272 | private String price;
273 | // 服务器异步通知页面路径
274 | private String callbackUrl;
275 | public Builder() {
276 | super();
277 | }
278 |
279 | public Builder with(Activity activity){
280 | this.activity = activity;
281 | return this;
282 | }
283 |
284 | public Builder apply(AliPayAPI.Config config){
285 | this.config = config;
286 | return this;
287 | }
288 |
289 | /**
290 | * 设置唯一订单号
291 | * @param outTradeNo
292 | * @return
293 | */
294 | public Builder setOutTradeNo(String outTradeNo){
295 | this.outTradeNo = outTradeNo;
296 | return this;
297 | }
298 |
299 | /**
300 | * 设置订单标题
301 | * @param subject
302 | * @return
303 | */
304 | public Builder setSubject(String subject){
305 | this.subject = subject;
306 | return this;
307 | }
308 |
309 | /**
310 | * 设置订单内容
311 | * @param body
312 | * @return
313 | */
314 | public Builder setBody(String body){
315 | this.body = body;
316 | return this;
317 | }
318 |
319 | /**
320 | * 设置订单价格
321 | * @param price
322 | * @return
323 | */
324 | public Builder setPrice(String price){
325 | this.price = price;
326 | return this;
327 | }
328 |
329 | /**
330 | * 设置回调
331 | * @param callbackUrl
332 | * @return
333 | */
334 | public Builder setCallbackUrl(String callbackUrl){
335 | this.callbackUrl = callbackUrl;
336 | return this;
337 | }
338 |
339 | public AliPayReq create(){
340 | AliPayReq aliPayReq = new AliPayReq();
341 | aliPayReq.mActivity = this.activity;
342 | aliPayReq.mConfig = this.config;
343 | aliPayReq.outTradeNo = this.outTradeNo;
344 | aliPayReq.subject = this.subject;
345 | aliPayReq.body = this.body;
346 | aliPayReq.price = this.price;
347 | aliPayReq.callbackUrl = this.callbackUrl;
348 |
349 | return aliPayReq;
350 | }
351 |
352 | }
353 |
354 |
355 | //支付宝支付监听
356 | private OnAliPayListener mOnAliPayListener;
357 | public AliPayReq setOnAliPayListener(OnAliPayListener onAliPayListener) {
358 | this.mOnAliPayListener = onAliPayListener;
359 | return this;
360 | }
361 |
362 | /**
363 | * 支付宝支付监听
364 | * @author Administrator
365 | *
366 | */
367 | public interface OnAliPayListener{
368 | public void onPaySuccess(String resultInfo);
369 | public void onPayFailure(String resultInfo);
370 | public void onPayConfirmimg(String resultInfo);
371 | public void onPayCheck(String status);
372 | }
373 | }
374 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/AliPayReq2.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import android.app.Activity;
4 | import android.os.Handler;
5 | import android.os.Message;
6 | import android.text.TextUtils;
7 | import android.widget.Toast;
8 |
9 | import com.alipay.sdk.app.PayTask;
10 |
11 | import java.io.UnsupportedEncodingException;
12 | import java.net.URLEncoder;
13 |
14 | import io.github.mayubao.pay_library.alipay.PayResult;
15 |
16 | /**
17 | * 支付宝支付请求2
18 | *
19 | * 安全的的支付宝支付流程,用法
20 | *
21 | * Created by mayubao on 2017/3/5.
22 | * Contact me 345269374@qq.com
23 | */
24 | public class AliPayReq2 {
25 |
26 | /**
27 | * ali pay sdk flag
28 | */
29 | private static final int SDK_PAY_FLAG = 1;
30 | private static final int SDK_CHECK_FLAG = 2;
31 |
32 | private Activity mActivity;
33 |
34 | //未签名的订单信息
35 | private String rawAliPayOrderInfo;
36 | //服务器签名成功的订单信息
37 | private String signedAliPayOrderInfo;
38 |
39 | private Handler mHandler;
40 |
41 | public AliPayReq2() {
42 | super();
43 | mHandler = new Handler(){
44 |
45 | @Override
46 | public void handleMessage(Message msg) {
47 | switch (msg.what) {
48 | case SDK_PAY_FLAG: {
49 | PayResult payResult = new PayResult((String) msg.obj);
50 |
51 | // 支付宝返回此次支付结果及加签,建议对支付宝签名信息拿签约时支付宝提供的公钥做验签
52 | String resultInfo = payResult.getResult();
53 |
54 | String resultStatus = payResult.getResultStatus();
55 |
56 | // 判断resultStatus 为“9000”则代表支付成功,具体状态码代表含义可参考接口文档
57 | if (TextUtils.equals(resultStatus, "9000")) {
58 | Toast.makeText(mActivity, "支付成功", Toast.LENGTH_SHORT).show();
59 | if(mOnAliPayListener != null) mOnAliPayListener.onPaySuccess(resultInfo);
60 | } else {
61 | // 判断resultStatus 为非“9000”则代表可能支付失败
62 | // “8000”代表支付结果因为支付渠道原因或者系统原因还在等待支付结果确认,最终交易是否成功以服务端异步通知为准(小概率状态)
63 | if (TextUtils.equals(resultStatus, "8000")) {
64 | Toast.makeText(mActivity, "支付结果确认中", Toast.LENGTH_SHORT).show();
65 | if(mOnAliPayListener != null) mOnAliPayListener.onPayConfirmimg(resultInfo);
66 |
67 | } else {
68 | // 其他值就可以判断为支付失败,包括用户主动取消支付,或者系统返回的错误
69 | Toast.makeText(mActivity, "支付失败", Toast.LENGTH_SHORT).show();
70 | if(mOnAliPayListener != null) mOnAliPayListener.onPayFailure(resultInfo);
71 | }
72 | }
73 | break;
74 | }
75 | case SDK_CHECK_FLAG: {
76 | Toast.makeText(mActivity, "检查结果为:" + msg.obj, Toast.LENGTH_SHORT).show();
77 | if(mOnAliPayListener != null) mOnAliPayListener.onPayCheck(msg.obj.toString());
78 | break;
79 | }
80 | default:
81 | break;
82 | }
83 | }
84 |
85 | };
86 | }
87 |
88 |
89 | /**
90 | * 发送支付宝支付请求
91 | */
92 | public void send() {
93 | // 创建订单信息
94 | // String orderInfo = getOrderInfo(this.partner,
95 | // this.seller, this.outTradeNo, this.subject, this.body,
96 | // this.price, this.callbackUrl);
97 | String orderInfo = rawAliPayOrderInfo;
98 | // 做RSA签名之后的订单信息
99 | String sign = signedAliPayOrderInfo;
100 | try {
101 | // 仅需对sign 做URL编码
102 | sign = URLEncoder.encode(sign, "UTF-8");
103 | } catch (UnsupportedEncodingException e) {
104 | e.printStackTrace();
105 | }
106 |
107 | // 完整的符合支付宝参数规范的订单信息
108 | final String payInfo = orderInfo + "&sign=\"" + sign + "\"&"
109 | + getSignType();
110 |
111 | Runnable payRunnable = new Runnable() {
112 |
113 | @Override
114 | public void run() {
115 | // 构造PayTask 对象
116 | PayTask alipay = new PayTask(mActivity);
117 | // 调用支付接口,获取支付结果
118 | String result = alipay.pay(payInfo);
119 |
120 | Message msg = new Message();
121 | msg.what = SDK_PAY_FLAG;
122 | msg.obj = result;
123 | mHandler.sendMessage(msg);
124 | }
125 | };
126 |
127 | // 必须异步调用
128 | Thread payThread = new Thread(payRunnable);
129 | payThread.start();
130 | }
131 |
132 | /**
133 | * 查询终端设备是否存在支付宝认证账户
134 | */
135 | public void check(){
136 | Runnable checkRunnable = new Runnable() {
137 |
138 | @Override
139 | public void run() {
140 | // 构造PayTask 对象
141 | PayTask payTask = new PayTask(mActivity);
142 | // 调用查询接口,获取查询结果
143 | boolean isExist = payTask.checkAccountIfExist();
144 |
145 | Message msg = new Message();
146 | msg.what = SDK_CHECK_FLAG;
147 | msg.obj = isExist;
148 | mHandler.sendMessage(msg);
149 | }
150 | };
151 |
152 | Thread checkThread = new Thread(checkRunnable);
153 | checkThread.start();
154 | }
155 |
156 |
157 | /**
158 | * 创建订单信息
159 | *
160 | * @param partner 签约合作者身份ID
161 | * @param seller 签约卖家支付宝账号
162 | * @param outTradeNo 商户网站唯一订单号
163 | * @param subject 商品名称
164 | * @param body 商品详情
165 | * @param price 商品金额
166 | * @param callbackUrl 服务器异步通知页面路径
167 | * @return
168 | */
169 | public static String getOrderInfo(String partner, String seller, String outTradeNo, String subject, String body, String price, String callbackUrl) {
170 |
171 | // 签约合作者身份ID
172 | String orderInfo = "partner=" + "\"" + partner + "\"";
173 |
174 | // 签约卖家支付宝账号
175 | orderInfo += "&seller_id=" + "\"" + seller + "\"";
176 |
177 | // 商户网站唯一订单号
178 | orderInfo += "&out_trade_no=" + "\"" + outTradeNo + "\"";
179 |
180 | // 商品名称
181 | orderInfo += "&subject=" + "\"" + subject + "\"";
182 |
183 | // 商品详情
184 | orderInfo += "&body=" + "\"" + body + "\"";
185 |
186 | // 商品金额
187 | orderInfo += "&total_fee=" + "\"" + price + "\"";
188 |
189 | // 服务器异步通知页面路径
190 | // orderInfo += "¬ify_url=" + "\"" + "http://notify.msp.hk/notify.htm"
191 | // + "\"";
192 | orderInfo += "¬ify_url=" + "\"" + callbackUrl
193 | + "\"";
194 |
195 | // 服务接口名称, 固定值
196 | orderInfo += "&service=\"mobile.securitypay.pay\"";
197 |
198 | // 支付类型, 固定值
199 | orderInfo += "&payment_type=\"1\"";
200 |
201 | // 参数编码, 固定值
202 | orderInfo += "&_input_charset=\"utf-8\"";
203 |
204 | // 设置未付款交易的超时时间
205 | // 默认30分钟,一旦超时,该笔交易就会自动被关闭。
206 | // 取值范围:1m~15d。
207 | // m-分钟,h-小时,d-天,1c-当天(无论交易何时创建,都在0点关闭)。
208 | // 该参数数值不接受小数点,如1.5h,可转换为90m。
209 | orderInfo += "&it_b_pay=\"30m\"";
210 |
211 | // extern_token为经过快登授权获取到的alipay_open_id,带上此参数用户将使用授权的账户进行支付
212 | // orderInfo += "&extern_token=" + "\"" + extern_token + "\"";
213 |
214 | // 支付宝处理完请求后,当前页面跳转到商户指定页面的路径,可空
215 | orderInfo += "&return_url=\"m.alipay.com\"";
216 |
217 | // 调用银行卡支付,需配置此参数,参与签名, 固定值 (需要签约《无线银行卡快捷支付》才能使用)
218 | // orderInfo += "&paymethod=\"expressGateway\"";
219 |
220 | return orderInfo;
221 | }
222 |
223 | /**
224 | * get the sign type we use. 获取签名方式
225 | *
226 | */
227 | public String getSignType() {
228 | return "sign_type=\"RSA\"";
229 | }
230 |
231 |
232 | public static class Builder{
233 | //上下文
234 | private Activity activity;
235 |
236 | //未签名的订单信息
237 | private String rawAliPayOrderInfo;
238 | //服务器签名成功的订单信息
239 | private String signedAliPayOrderInfo;
240 |
241 | public Builder() {
242 | super();
243 | }
244 |
245 | public Builder with(Activity activity){
246 | this.activity = activity;
247 | return this;
248 | }
249 |
250 |
251 | /**
252 | * 设置未签名的订单信息
253 | * @param rawAliPayOrderInfo
254 | * @return
255 | */
256 | public Builder setRawAliPayOrderInfo(String rawAliPayOrderInfo){
257 | this.rawAliPayOrderInfo = rawAliPayOrderInfo;
258 | return this;
259 | }
260 |
261 | /**
262 | * 设置服务器签名成功的订单信息
263 | * @param signedAliPayOrderInfo
264 | * @return
265 | */
266 | public Builder setSignedAliPayOrderInfo(String signedAliPayOrderInfo){
267 | this.signedAliPayOrderInfo = signedAliPayOrderInfo;
268 | return this;
269 | }
270 |
271 | public AliPayReq2 create(){
272 | AliPayReq2 aliPayReq = new AliPayReq2();
273 | aliPayReq.mActivity = this.activity;
274 | aliPayReq.rawAliPayOrderInfo = this.rawAliPayOrderInfo;
275 | aliPayReq.signedAliPayOrderInfo = this.signedAliPayOrderInfo;
276 |
277 | return aliPayReq;
278 | }
279 |
280 | }
281 |
282 |
283 | /**
284 | * 支付宝支付订单信息的信息类
285 | *
286 | * 官方demo是暴露了商户私钥,pkcs8格式的,这是极其不安全的。官方也建议私钥签名订单这一块放到服务器去处理。
287 | * 所以为了避免商户私钥暴露在客户端,订单的加密过程放到服务器去处理
288 | */
289 | public static class AliOrderInfo{
290 | String partner;
291 | String seller;
292 | String outTradeNo;
293 | String subject;
294 | String body;
295 | String price;
296 | String callbackUrl;
297 |
298 | /**
299 | * 设置商户
300 | * @param partner
301 | * @return
302 | */
303 | public AliOrderInfo setPartner(String partner){
304 | this.partner = partner;
305 | return this;
306 | }
307 |
308 | /**
309 | * 设置商户账号
310 | * @param seller
311 | * @return
312 | */
313 | public AliOrderInfo setSeller(String seller){
314 | this.seller = seller;
315 | return this;
316 | }
317 |
318 | /**
319 | * 设置唯一订单号
320 | * @param outTradeNo
321 | * @return
322 | */
323 | public AliOrderInfo setOutTradeNo(String outTradeNo){
324 | this.outTradeNo = outTradeNo;
325 | return this;
326 | }
327 |
328 | /**
329 | * 设置订单标题
330 | * @param subject
331 | * @return
332 | */
333 | public AliOrderInfo setSubject(String subject){
334 | this.subject = subject;
335 | return this;
336 | }
337 |
338 | /**
339 | * 设置订单详情
340 | * @param body
341 | * @return
342 | */
343 | public AliOrderInfo setBody(String body){
344 | this.body = body;
345 | return this;
346 | }
347 |
348 | /**
349 | * 设置价格
350 | * @param price
351 | * @return
352 | */
353 | public AliOrderInfo setPrice(String price){
354 | this.price = price;
355 | return this;
356 | }
357 |
358 | /**
359 | * 设置请求回调
360 | * @param callbackUrl
361 | * @return
362 | */
363 | public AliOrderInfo setCallbackUrl(String callbackUrl){
364 | this.callbackUrl = callbackUrl;
365 | return this;
366 | }
367 |
368 | /**
369 | * 创建订单详情
370 | * @return
371 | */
372 | public String createOrderInfo(){
373 | // (String partner, String seller, String outTradeNo, String subject, String body, String price, String callbackUrl) {
374 | return getOrderInfo(partner, seller, outTradeNo, subject, body, price, callbackUrl);
375 | }
376 | }
377 |
378 |
379 |
380 |
381 | //支付宝支付监听
382 | private OnAliPayListener mOnAliPayListener;
383 | public AliPayReq2 setOnAliPayListener(OnAliPayListener onAliPayListener) {
384 | this.mOnAliPayListener = onAliPayListener;
385 | return this;
386 | }
387 |
388 | /**
389 | * 支付宝支付监听
390 | * @author Administrator
391 | *
392 | */
393 | public interface OnAliPayListener{
394 | public void onPaySuccess(String resultInfo);
395 | public void onPayFailure(String resultInfo);
396 | public void onPayConfirmimg(String resultInfo);
397 | public void onPayCheck(String status);
398 | }
399 | }
400 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/MainActivity.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import android.app.Activity;
4 | import android.support.v7.app.AppCompatActivity;
5 |
6 | /**
7 | * Created by mayubao on 2017/3/5.
8 | * Contact me 345269374@qq.com
9 | */
10 | public class MainActivity extends AppCompatActivity {
11 |
12 | /**
13 | * 微信支付Test
14 | */
15 | public void testWechatPay(){
16 | String appid = "";
17 | String partnerid = "";
18 | String prepayid = "";
19 | String noncestr = "";
20 | String timestamp = "";
21 | String sign = "";
22 | WechatPayReq wechatPayReq = new WechatPayReq.Builder()
23 | .with(this) //activity实例
24 | .setAppId(appid) //微信支付AppID
25 | .setPartnerId(partnerid)//微信支付商户号
26 | .setPrepayId(prepayid)//预支付码
27 | // .setPackageValue(wechatPayReq.get)//"Sign=WXPay"
28 | .setNonceStr(noncestr)
29 | .setTimeStamp(timestamp)//时间戳
30 | .setSign(sign)//签名
31 | .create();
32 |
33 | PayAPI.getInstance().sendPayRequest(wechatPayReq);
34 | // .setOnWechatPayListener(new OnWechatPayListener() {
35 | //
36 | // @Override
37 | // public void onPaySuccess(int errorCode) {
38 | // ToastUtil.show(mContext, "支付成功" + errorCode);
39 | //
40 | // }
41 | //
42 | // @Override
43 | // public void onPayFailure(int errorCode) {
44 | // ToastUtil.show(mContext, "支付失败" + errorCode);
45 | //
46 | // }
47 | // });
48 | // WechatPayAPI.getInstance().sendPayReq(wechatPayReq);
49 |
50 | PayAPI.getInstance().sendPayRequest(wechatPayReq);
51 |
52 | }
53 |
54 |
55 | /**
56 | * 支付宝支付测试
57 | */
58 | public void testAliPay(){
59 | String rsa_private = "";
60 | String rsa_public = "";
61 | String partner = "";
62 | String seller = "";
63 |
64 | Activity activity = this;
65 | String outTradeNo = "";
66 | String price = "";
67 | String orderSubject = "";
68 | String orderBody = "";
69 | String callbackUrl = "";
70 |
71 | AliPayAPI.Config config = new AliPayAPI.Config.Builder()
72 | .setRsaPrivate(rsa_private) //设置私钥
73 | .setRsaPublic(rsa_public)//设置公钥
74 | .setPartner(partner) //设置商户
75 | .setSeller(seller) //设置商户收款账号
76 | .create();
77 |
78 | AliPayReq aliPayReq = new AliPayReq.Builder()
79 | .with(activity)//Activity实例
80 | .apply(config)//支付宝支付通用配置
81 | .setOutTradeNo(outTradeNo)//设置唯一订单号
82 | .setPrice(price)//设置订单价格
83 | .setSubject(orderSubject)//设置订单标题
84 | .setBody(orderBody)//设置订单内容 订单详情
85 | .setCallbackUrl(callbackUrl)//设置回调地址
86 | .create();//
87 | AliPayAPI.getInstance().apply(config).sendPayReq(aliPayReq);
88 |
89 | PayAPI.getInstance().sendPayRequest(aliPayReq);
90 | }
91 |
92 | /**
93 | * 安全的支付宝支付测试
94 | */
95 | public void testAliPaySafely(){
96 | String partner = "";
97 | String seller = "";
98 |
99 | Activity activity = this;
100 | String outTradeNo = "";
101 | String price = "";
102 | String orderSubject = "";
103 | String orderBody = "";
104 | String callbackUrl = "";
105 |
106 |
107 | String rawAliOrderInfo = new AliPayReq2.AliOrderInfo()
108 | .setPartner(partner) //商户PID || 签约合作者身份ID
109 | .setSeller(seller) // 商户收款账号 || 签约卖家支付宝账号
110 | .setOutTradeNo(outTradeNo) //设置唯一订单号
111 | .setSubject(orderSubject) //设置订单标题
112 | .setBody(orderBody) //设置订单内容
113 | .setPrice(price) //设置订单价格
114 | .setCallbackUrl(callbackUrl) //设置回调链接
115 | .createOrderInfo(); //创建支付宝支付订单信息
116 |
117 |
118 | //TODO 这里需要从服务器获取用商户私钥签名之后的订单信息
119 | String signAliOrderInfo = getSignAliOrderInfoFromServer(rawAliOrderInfo);
120 |
121 | AliPayReq2 aliPayReq = new AliPayReq2.Builder()
122 | .with(activity)//Activity实例
123 | .setRawAliPayOrderInfo(rawAliOrderInfo)//set the ali pay order info
124 | .setSignedAliPayOrderInfo(signAliOrderInfo) //set the signed ali pay order info
125 | .create()//
126 | .setOnAliPayListener(null);//
127 | AliPayAPI.getInstance().sendPayReq(aliPayReq);
128 |
129 | PayAPI.getInstance().sendPayRequest(aliPayReq);
130 | }
131 |
132 | /**
133 | * 获取签名后的支付宝订单信息 (用商户私钥RSA加密之后的订单信息)
134 | * @param rawAliOrderInfo
135 | * @return
136 | */
137 | private String getSignAliOrderInfoFromServer(String rawAliOrderInfo) {
138 | return null;
139 | }
140 |
141 | }
142 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/PayAPI.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | /**
4 | * 支付的API
5 | *
6 | * Created by mayubao on 2017/2/22.
7 | * Contact me 345269374@qq.com
8 | */
9 | public class PayAPI {
10 |
11 |
12 | private static final Object mLock = new Object();
13 | private static PayAPI mInstance;
14 |
15 | public static PayAPI getInstance(){
16 | if(mInstance == null){
17 | synchronized (mLock){
18 | if(mInstance == null){
19 | mInstance = new PayAPI();
20 | }
21 | }
22 | }
23 | return mInstance;
24 | }
25 |
26 |
27 | /**
28 | * 支付宝支付请求
29 | * @param aliPayRe
30 | */
31 | public void sendPayRequest(AliPayReq aliPayRe){
32 | AliPayAPI.getInstance().sendPayReq(aliPayRe);
33 | }
34 |
35 | /**
36 | * 支付宝支付请求 - 避免商户私钥暴露在客户端
37 | * @param aliPayRe2
38 | */
39 | public void sendPayRequest(AliPayReq2 aliPayRe2){
40 | AliPayAPI.getInstance().sendPayReq(aliPayRe2);
41 | }
42 |
43 |
44 |
45 | /**
46 | * 微信支付请求
47 | * @param wechatPayReq
48 | */
49 | public void sendPayRequest(WechatPayReq wechatPayReq){
50 | WechatPayAPI.getInstance().sendPayReq(wechatPayReq);
51 | }
52 |
53 | }
54 |
55 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/WechatPayAPI.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | /**
4 | * 微信支付API
5 | *
6 | * 使用:
7 | *
8 | * WechatPayAPI.getInstance().sendPayReq(wechatPayReq);
9 | *
10 | * Created by mayubao on 2017/3/5.
11 | * Contact me 345269374@qq.com
12 | */
13 | public class WechatPayAPI {
14 |
15 | /**
16 | * 获取微信支付API
17 | */
18 | private static final Object mLock = new Object();
19 | private static WechatPayAPI mInstance;
20 |
21 | public static WechatPayAPI getInstance(){
22 | if(mInstance == null){
23 | synchronized (mLock){
24 | if(mInstance == null){
25 | mInstance = new WechatPayAPI();
26 | }
27 | }
28 | }
29 | return mInstance;
30 | }
31 |
32 | /**
33 | * 发送微信支付请求
34 | * @param wechatPayReq
35 | */
36 | public void sendPayReq(WechatPayReq wechatPayReq){
37 | wechatPayReq.send();
38 | }
39 | }
40 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/WechatPayReq.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import android.app.Activity;
4 | import android.util.Log;
5 | import android.widget.Toast;
6 |
7 | import com.tencent.mm.sdk.constants.ConstantsAPI;
8 | import com.tencent.mm.sdk.modelbase.BaseReq;
9 | import com.tencent.mm.sdk.modelbase.BaseResp;
10 | import com.tencent.mm.sdk.modelpay.PayReq;
11 | import com.tencent.mm.sdk.openapi.IWXAPI;
12 | import com.tencent.mm.sdk.openapi.IWXAPIEventHandler;
13 | import com.tencent.mm.sdk.openapi.WXAPIFactory;
14 |
15 | /**
16 | * 微信支付请求
17 | *
18 | * @author Administrator
19 | *
20 | */
21 | public class WechatPayReq implements IWXAPIEventHandler {
22 |
23 | private static final String TAG = WechatPayReq.class.getSimpleName();
24 |
25 | private Activity mActivity;
26 |
27 | //微信支付AppID
28 | private String appId;
29 | //微信支付商户号
30 | private String partnerId;
31 | //预支付码(重要)
32 | private String prepayId;
33 | //"Sign=WXPay"
34 | private String packageValue;
35 | private String nonceStr;
36 | //时间戳
37 | private String timeStamp;
38 | //签名
39 | private String sign;
40 |
41 | //微信支付核心api
42 | IWXAPI mWXApi;
43 |
44 | public WechatPayReq() {
45 | super();
46 | }
47 |
48 |
49 | /**
50 | * 发送微信支付请求
51 | */
52 | public void send() {
53 | mWXApi = WXAPIFactory.createWXAPI(mActivity, null);
54 | mWXApi.handleIntent(mActivity.getIntent(), this);
55 |
56 | mWXApi.registerApp(this.appId);
57 |
58 | PayReq request = new PayReq();
59 |
60 | request.appId = this.appId;
61 | request.partnerId = this.partnerId;
62 | request.prepayId= this.prepayId;
63 | request.packageValue = this.packageValue != null ? this.packageValue : "Sign=WXPay";
64 | request.nonceStr= this.nonceStr;
65 | request.timeStamp= this.timeStamp;
66 | request.sign = this.sign;
67 |
68 | mWXApi.sendReq(request);
69 | }
70 |
71 | public static class Builder{
72 | //上下文
73 | private Activity activity;
74 | //微信支付AppID
75 | private String appId;
76 | //微信支付商户号
77 | private String partnerId;
78 | //预支付码(重要)
79 | private String prepayId;
80 | //"Sign=WXPay"
81 | private String packageValue="Sign=WXPay";
82 | private String nonceStr;
83 | //时间戳
84 | private String timeStamp;
85 | //签名
86 | private String sign;
87 | public Builder() {
88 | super();
89 | }
90 |
91 | public Builder with(Activity activity){
92 | this.activity = activity;
93 | return this;
94 | }
95 |
96 | /**
97 | * 设置微信支付AppID
98 | * @param appId
99 | * @return
100 | */
101 | public Builder setAppId(String appId){
102 | this.appId = appId;
103 | return this;
104 | }
105 |
106 | /**
107 | * 微信支付商户号
108 | * @param partnerId
109 | * @return
110 | */
111 | public Builder setPartnerId(String partnerId){
112 | this.partnerId = partnerId;
113 | return this;
114 | }
115 |
116 | /**
117 | * 设置预支付码(重要)
118 | * @param prepayId
119 | * @return
120 | */
121 | public Builder setPrepayId(String prepayId){
122 | this.prepayId = prepayId;
123 | return this;
124 | }
125 |
126 |
127 | /**
128 | * 设置
129 | * @param packageValue
130 | * @return
131 | */
132 | public Builder setPackageValue(String packageValue){
133 | this.packageValue = packageValue;
134 | return this;
135 | }
136 |
137 |
138 | /**
139 | * 设置
140 | * @param nonceStr
141 | * @return
142 | */
143 | public Builder setNonceStr(String nonceStr){
144 | this.nonceStr = nonceStr;
145 | return this;
146 | }
147 |
148 | /**
149 | * 设置时间戳
150 | * @param timeStamp
151 | * @return
152 | */
153 | public Builder setTimeStamp(String timeStamp){
154 | this.timeStamp = timeStamp;
155 | return this;
156 | }
157 |
158 | /**
159 | * 设置签名
160 | * @param sign
161 | * @return
162 | */
163 | public Builder setSign(String sign){
164 | this.sign = sign;
165 | return this;
166 | }
167 |
168 |
169 |
170 | public WechatPayReq create(){
171 | WechatPayReq wechatPayReq = new WechatPayReq();
172 |
173 | wechatPayReq.mActivity = this.activity;
174 | //微信支付AppID
175 | wechatPayReq.appId = this.appId;
176 | //微信支付商户号
177 | wechatPayReq.partnerId = this.partnerId;
178 | //预支付码(重要)
179 | wechatPayReq.prepayId = this.prepayId;
180 | //"Sign=WXPay"
181 | wechatPayReq.packageValue = this.packageValue;
182 | wechatPayReq.nonceStr = this.nonceStr;
183 | //时间戳
184 | wechatPayReq.timeStamp = this.timeStamp;
185 | //签名
186 | wechatPayReq.sign = this.sign;
187 |
188 | return wechatPayReq;
189 | }
190 |
191 | }
192 |
193 |
194 | //微信支付监听
195 | private OnWechatPayListener mOnWechatPayListener;
196 | public WechatPayReq setOnWechatPayListener(OnWechatPayListener onWechatPayListener) {
197 | this.mOnWechatPayListener = onWechatPayListener;
198 | return this;
199 | }
200 |
201 | /**
202 | * 微信支付监听
203 | * @author Administrator
204 | *
205 | */
206 | public interface OnWechatPayListener{
207 | public void onPaySuccess(int errorCode);
208 | public void onPayFailure(int errorCode);
209 | }
210 |
211 | @Override
212 | public void onReq(BaseReq baseReq) {
213 | Toast.makeText(this.mActivity, "onReq===>>>get baseReq.getType : "+baseReq.getType(), Toast.LENGTH_LONG).show();
214 | Log.d(TAG,"onReq===>>>get baseReq.getType : "+baseReq.getType());
215 | }
216 |
217 |
218 | @Override
219 | public void onResp(BaseResp resp) {
220 | Toast.makeText(this.mActivity, "onResp===>>>get resp.getType : "+ resp.getType(), Toast.LENGTH_LONG).show();
221 |
222 | // 0 成功 展示成功页面
223 | // -1 错误 可能的原因:签名错误、未注册APPID、项目设置APPID不正确、注册的APPID与设置的不匹配、其他异常等。
224 | // -2 用户取消 无需处理。发生场景:用户不支付了,点击取消,返回APP。
225 |
226 | if(resp.getType()== ConstantsAPI.COMMAND_PAY_BY_WX){
227 | Log.d(TAG,"onPayFinish,errCode="+resp.errCode);
228 | if(this.mOnWechatPayListener != null){
229 | if(resp.errCode == BaseResp.ErrCode.ERR_OK){ // 0 成功 展示成功页面
230 | this.mOnWechatPayListener.onPaySuccess(resp.errCode);
231 | }else{// -1 错误 -2 用户取消
232 | this.mOnWechatPayListener.onPayFailure(resp.errCode);
233 | }
234 | }
235 | }
236 | }
237 | }
238 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/alipay/Base64.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library.alipay;
2 |
3 | public final class Base64 {
4 |
5 | private static final int BASELENGTH = 128;
6 | private static final int LOOKUPLENGTH = 64;
7 | private static final int TWENTYFOURBITGROUP = 24;
8 | private static final int EIGHTBIT = 8;
9 | private static final int SIXTEENBIT = 16;
10 | private static final int FOURBYTE = 4;
11 | private static final int SIGN = -128;
12 | private static char PAD = '=';
13 | private static byte[] base64Alphabet = new byte[BASELENGTH];
14 | private static char[] lookUpBase64Alphabet = new char[LOOKUPLENGTH];
15 |
16 | static {
17 | for (int i = 0; i < BASELENGTH; ++i) {
18 | base64Alphabet[i] = -1;
19 | }
20 | for (int i = 'Z'; i >= 'A'; i--) {
21 | base64Alphabet[i] = (byte) (i - 'A');
22 | }
23 | for (int i = 'z'; i >= 'a'; i--) {
24 | base64Alphabet[i] = (byte) (i - 'a' + 26);
25 | }
26 |
27 | for (int i = '9'; i >= '0'; i--) {
28 | base64Alphabet[i] = (byte) (i - '0' + 52);
29 | }
30 |
31 | base64Alphabet['+'] = 62;
32 | base64Alphabet['/'] = 63;
33 |
34 | for (int i = 0; i <= 25; i++) {
35 | lookUpBase64Alphabet[i] = (char) ('A' + i);
36 | }
37 |
38 | for (int i = 26, j = 0; i <= 51; i++, j++) {
39 | lookUpBase64Alphabet[i] = (char) ('a' + j);
40 | }
41 |
42 | for (int i = 52, j = 0; i <= 61; i++, j++) {
43 | lookUpBase64Alphabet[i] = (char) ('0' + j);
44 | }
45 | lookUpBase64Alphabet[62] = (char) '+';
46 | lookUpBase64Alphabet[63] = (char) '/';
47 |
48 | }
49 |
50 | private static boolean isWhiteSpace(char octect) {
51 | return (octect == 0x20 || octect == 0xd || octect == 0xa || octect == 0x9);
52 | }
53 |
54 | private static boolean isPad(char octect) {
55 | return (octect == PAD);
56 | }
57 |
58 | private static boolean isData(char octect) {
59 | return (octect < BASELENGTH && base64Alphabet[octect] != -1);
60 | }
61 |
62 | /**
63 | * Encodes hex octects into Base64
64 | *
65 | * @param binaryData
66 | * Array containing binaryData
67 | * @return Encoded Base64 array
68 | */
69 | public static String encode(byte[] binaryData) {
70 |
71 | if (binaryData == null) {
72 | return null;
73 | }
74 |
75 | int lengthDataBits = binaryData.length * EIGHTBIT;
76 | if (lengthDataBits == 0) {
77 | return "";
78 | }
79 |
80 | int fewerThan24bits = lengthDataBits % TWENTYFOURBITGROUP;
81 | int numberTriplets = lengthDataBits / TWENTYFOURBITGROUP;
82 | int numberQuartet = fewerThan24bits != 0 ? numberTriplets + 1
83 | : numberTriplets;
84 | char encodedData[] = null;
85 |
86 | encodedData = new char[numberQuartet * 4];
87 |
88 | byte k = 0, l = 0, b1 = 0, b2 = 0, b3 = 0;
89 |
90 | int encodedIndex = 0;
91 | int dataIndex = 0;
92 |
93 | for (int i = 0; i < numberTriplets; i++) {
94 | b1 = binaryData[dataIndex++];
95 | b2 = binaryData[dataIndex++];
96 | b3 = binaryData[dataIndex++];
97 |
98 | l = (byte) (b2 & 0x0f);
99 | k = (byte) (b1 & 0x03);
100 |
101 | byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2)
102 | : (byte) ((b1) >> 2 ^ 0xc0);
103 | byte val2 = ((b2 & SIGN) == 0) ? (byte) (b2 >> 4)
104 | : (byte) ((b2) >> 4 ^ 0xf0);
105 | byte val3 = ((b3 & SIGN) == 0) ? (byte) (b3 >> 6)
106 | : (byte) ((b3) >> 6 ^ 0xfc);
107 |
108 | encodedData[encodedIndex++] = lookUpBase64Alphabet[val1];
109 | encodedData[encodedIndex++] = lookUpBase64Alphabet[val2 | (k << 4)];
110 | encodedData[encodedIndex++] = lookUpBase64Alphabet[(l << 2) | val3];
111 | encodedData[encodedIndex++] = lookUpBase64Alphabet[b3 & 0x3f];
112 | }
113 |
114 | // form integral number of 6-bit groups
115 | if (fewerThan24bits == EIGHTBIT) {
116 | b1 = binaryData[dataIndex];
117 | k = (byte) (b1 & 0x03);
118 |
119 | byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2)
120 | : (byte) ((b1) >> 2 ^ 0xc0);
121 | encodedData[encodedIndex++] = lookUpBase64Alphabet[val1];
122 | encodedData[encodedIndex++] = lookUpBase64Alphabet[k << 4];
123 | encodedData[encodedIndex++] = PAD;
124 | encodedData[encodedIndex++] = PAD;
125 | } else if (fewerThan24bits == SIXTEENBIT) {
126 | b1 = binaryData[dataIndex];
127 | b2 = binaryData[dataIndex + 1];
128 | l = (byte) (b2 & 0x0f);
129 | k = (byte) (b1 & 0x03);
130 |
131 | byte val1 = ((b1 & SIGN) == 0) ? (byte) (b1 >> 2)
132 | : (byte) ((b1) >> 2 ^ 0xc0);
133 | byte val2 = ((b2 & SIGN) == 0) ? (byte) (b2 >> 4)
134 | : (byte) ((b2) >> 4 ^ 0xf0);
135 |
136 | encodedData[encodedIndex++] = lookUpBase64Alphabet[val1];
137 | encodedData[encodedIndex++] = lookUpBase64Alphabet[val2 | (k << 4)];
138 | encodedData[encodedIndex++] = lookUpBase64Alphabet[l << 2];
139 | encodedData[encodedIndex++] = PAD;
140 | }
141 |
142 | return new String(encodedData);
143 | }
144 |
145 | /**
146 | * Decodes Base64 data into octects
147 | *
148 | * @param encoded
149 | * string containing Base64 data
150 | * @return Array containind decoded data.
151 | */
152 | public static byte[] decode(String encoded) {
153 |
154 | if (encoded == null) {
155 | return null;
156 | }
157 |
158 | char[] base64Data = encoded.toCharArray();
159 | // remove white spaces
160 | int len = removeWhiteSpace(base64Data);
161 |
162 | if (len % FOURBYTE != 0) {
163 | return null;// should be divisible by four
164 | }
165 |
166 | int numberQuadruple = (len / FOURBYTE);
167 |
168 | if (numberQuadruple == 0) {
169 | return new byte[0];
170 | }
171 |
172 | byte decodedData[] = null;
173 | byte b1 = 0, b2 = 0, b3 = 0, b4 = 0;
174 | char d1 = 0, d2 = 0, d3 = 0, d4 = 0;
175 |
176 | int i = 0;
177 | int encodedIndex = 0;
178 | int dataIndex = 0;
179 | decodedData = new byte[(numberQuadruple) * 3];
180 |
181 | for (; i < numberQuadruple - 1; i++) {
182 |
183 | if (!isData((d1 = base64Data[dataIndex++]))
184 | || !isData((d2 = base64Data[dataIndex++]))
185 | || !isData((d3 = base64Data[dataIndex++]))
186 | || !isData((d4 = base64Data[dataIndex++]))) {
187 | return null;
188 | }// if found "no data" just return null
189 |
190 | b1 = base64Alphabet[d1];
191 | b2 = base64Alphabet[d2];
192 | b3 = base64Alphabet[d3];
193 | b4 = base64Alphabet[d4];
194 |
195 | decodedData[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4);
196 | decodedData[encodedIndex++] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf));
197 | decodedData[encodedIndex++] = (byte) (b3 << 6 | b4);
198 | }
199 |
200 | if (!isData((d1 = base64Data[dataIndex++]))
201 | || !isData((d2 = base64Data[dataIndex++]))) {
202 | return null;// if found "no data" just return null
203 | }
204 |
205 | b1 = base64Alphabet[d1];
206 | b2 = base64Alphabet[d2];
207 |
208 | d3 = base64Data[dataIndex++];
209 | d4 = base64Data[dataIndex++];
210 | if (!isData((d3)) || !isData((d4))) {// Check if they are PAD characters
211 | if (isPad(d3) && isPad(d4)) {
212 | if ((b2 & 0xf) != 0)// last 4 bits should be zero
213 | {
214 | return null;
215 | }
216 | byte[] tmp = new byte[i * 3 + 1];
217 | System.arraycopy(decodedData, 0, tmp, 0, i * 3);
218 | tmp[encodedIndex] = (byte) (b1 << 2 | b2 >> 4);
219 | return tmp;
220 | } else if (!isPad(d3) && isPad(d4)) {
221 | b3 = base64Alphabet[d3];
222 | if ((b3 & 0x3) != 0)// last 2 bits should be zero
223 | {
224 | return null;
225 | }
226 | byte[] tmp = new byte[i * 3 + 2];
227 | System.arraycopy(decodedData, 0, tmp, 0, i * 3);
228 | tmp[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4);
229 | tmp[encodedIndex] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf));
230 | return tmp;
231 | } else {
232 | return null;
233 | }
234 | } else { // No PAD e.g 3cQl
235 | b3 = base64Alphabet[d3];
236 | b4 = base64Alphabet[d4];
237 | decodedData[encodedIndex++] = (byte) (b1 << 2 | b2 >> 4);
238 | decodedData[encodedIndex++] = (byte) (((b2 & 0xf) << 4) | ((b3 >> 2) & 0xf));
239 | decodedData[encodedIndex++] = (byte) (b3 << 6 | b4);
240 |
241 | }
242 |
243 | return decodedData;
244 | }
245 |
246 | /**
247 | * remove WhiteSpace from MIME containing encoded Base64 data.
248 | *
249 | * @param data
250 | * the byte array of base64 data (with WS)
251 | * @return the new length
252 | */
253 | private static int removeWhiteSpace(char[] data) {
254 | if (data == null) {
255 | return 0;
256 | }
257 |
258 | // count characters that's not whitespace
259 | int newSize = 0;
260 | int len = data.length;
261 | for (int i = 0; i < len; i++) {
262 | if (!isWhiteSpace(data[i])) {
263 | data[newSize++] = data[i];
264 | }
265 | }
266 | return newSize;
267 | }
268 | }
269 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/alipay/ExternalFragment.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library.alipay;
2 |
3 | import android.os.Bundle;
4 | import android.support.v4.app.Fragment;
5 | import android.view.LayoutInflater;
6 | import android.view.View;
7 | import android.view.ViewGroup;
8 |
9 | import io.github.mayubao.pay_library.R;
10 |
11 | public class ExternalFragment extends Fragment {
12 |
13 | @Override
14 | public View onCreateView(LayoutInflater inflater, ViewGroup container,
15 | Bundle savedInstanceState) {
16 | return inflater.inflate(R.layout.pay_external, container, false);
17 | }
18 | }
19 |
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/alipay/PayResult.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library.alipay;
import android.text.TextUtils;
public class PayResult {
private String resultStatus;
private String result;
private String memo;
public PayResult(String rawResult) {
if (TextUtils.isEmpty(rawResult))
return;
String[] resultParams = rawResult.split(";");
for (String resultParam : resultParams) {
if (resultParam.startsWith("resultStatus")) {
resultStatus = gatValue(resultParam, "resultStatus");
}
if (resultParam.startsWith("result")) {
result = gatValue(resultParam, "result");
}
if (resultParam.startsWith("memo")) {
memo = gatValue(resultParam, "memo");
}
}
}
@Override
public String toString() {
return "resultStatus={" + resultStatus + "};memo={" + memo
+ "};result={" + result + "}";
}
private String gatValue(String content, String key) {
String prefix = key + "={";
return content.substring(content.indexOf(prefix) + prefix.length(),
content.lastIndexOf("}"));
}
/**
* @return the resultStatus
*/
public String getResultStatus() {
return resultStatus;
}
/**
* @return the memo
*/
public String getMemo() {
return memo;
}
/**
* @return the result
*/
public String getResult() {
return result;
}
}
--------------------------------------------------------------------------------
/pay-library/src/main/java/io/github/mayubao/pay_library/alipay/SignUtils.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library.alipay;
2 |
3 | import java.security.KeyFactory;
4 | import java.security.PrivateKey;
5 | import java.security.spec.PKCS8EncodedKeySpec;
6 |
7 | public class SignUtils {
8 |
9 | private static final String ALGORITHM = "RSA";
10 |
11 | private static final String SIGN_ALGORITHMS = "SHA1WithRSA";
12 |
13 | private static final String DEFAULT_CHARSET = "UTF-8";
14 |
15 | public static String sign(String content, String privateKey) {
16 | try {
17 | PKCS8EncodedKeySpec priPKCS8 = new PKCS8EncodedKeySpec(
18 | Base64.decode(privateKey));
19 | KeyFactory keyf = KeyFactory.getInstance(ALGORITHM);
20 | PrivateKey priKey = keyf.generatePrivate(priPKCS8);
21 |
22 | java.security.Signature signature = java.security.Signature
23 | .getInstance(SIGN_ALGORITHMS);
24 |
25 | signature.initSign(priKey);
26 | signature.update(content.getBytes(DEFAULT_CHARSET));
27 |
28 | byte[] signed = signature.sign();
29 |
30 | return Base64.encode(signed);
31 | } catch (Exception e) {
32 | e.printStackTrace();
33 | }
34 |
35 | return null;
36 | }
37 |
38 | }
39 |
--------------------------------------------------------------------------------
/pay-library/src/main/res/layout/pay_external.xml:
--------------------------------------------------------------------------------
1 |
8 |
9 |
15 |
16 |
23 |
24 |
32 |
33 |
34 |
40 |
41 |
48 |
49 |
56 |
57 |
58 |
64 |
65 |
72 |
73 |
81 |
82 |
83 |
91 |
92 |
100 |
101 |
102 |
--------------------------------------------------------------------------------
/pay-library/src/main/res/layout/pay_main.xml:
--------------------------------------------------------------------------------
1 |
6 |
7 |
14 |
15 |
20 |
21 |
--------------------------------------------------------------------------------
/pay-library/src/main/res/mipmap-hdpi/msp_demo_title.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/pay-library/src/main/res/mipmap-hdpi/msp_demo_title.png
--------------------------------------------------------------------------------
/pay-library/src/main/res/mipmap-hdpi/msp_demo_title_bg.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/pay-library/src/main/res/mipmap-hdpi/msp_demo_title_bg.png
--------------------------------------------------------------------------------
/pay-library/src/main/res/mipmap-hdpi/msp_icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/mayubao/Android-Pay/603233da5d5d78b3f25273b87554e2a06d6d3e61/pay-library/src/main/res/mipmap-hdpi/msp_icon.png
--------------------------------------------------------------------------------
/pay-library/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | pay-library
3 |
4 |
--------------------------------------------------------------------------------
/pay-library/src/test/java/io/github/mayubao/pay_library/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package io.github.mayubao.pay_library;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * To work on unit tests, switch the Test Artifact in the Build Variants view.
9 | */
10 | public class ExampleUnitTest {
11 | @Test
12 | public void addition_isCorrect() throws Exception {
13 | assertEquals(4, 2 + 2);
14 | }
15 | }
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | include ':app', ':pay-library'
2 |
--------------------------------------------------------------------------------