安装: yarn add react-native-wechat-libs
此项目改写自 https://github.com/uiwjs/react-native-wechat
iOS: 微信授权登录 Universal Link(通用链接)
Universal Link(通用链接)是苹果在 iOS9 推出的,一种能够方便的通过传统 HTTPS 链接来启动 APP 的功能,可以使用相同的网址打开网址和 APP。
看起来就是一条普通的 https 链接,当然是我们在该链接域名根目录配置过的一个链接,也可以在该链接中放置对应的H5页面。当用户的点击该链接,只要手机中安装了支持该链接的 APP 就会直接进入到 APP 中。如果没有安装APP则会跳转到 Safari 浏览器中,展示 H5 页面。对用户来说则是一个无缝跳转的过程。
创建一个名为 apple-app-site-association
的文件,如下:
{
"applinks": {
"details": [
{
"appID": "968DSZ49MT.com.uiwjs.react.example.wechat",
"paths": ["/react-native-wechat/*"]
}
]
}
}
说明: 字段 appID 中的 968DSZ49MT
表示苹果账号的团队 ID
,com.uiwjs.react.example.wechat
表示项目的 BundleID
。
<Application Identifier Prefix>.<Bundle Identifier>
上传该文件到你的域名所对应的根目录
或xxx目录
下,apple-app-site-association
文件不需要扩展名。
注意: 苹果提供了一个网页来验证我们编写的这个 apple-app-site-association 是否合法有效。
https://<fully qualified domain>/.well-known/apple-app-site-association
根目录
https://uiwjs.github.io/apple-app-site-association
xxx目录
https://uiwjs.github.io/react-native-wechat/apple-app-site-association
打开 Associated Domains
开关,将 Universal Links
域名加到配置上,如果 URL
地址是 https://uiwjs.github.io/apple-app-site-association,那么,
Associated Domains
中填写 applinks: uiwjs.github.io
。
登录苹果开发者后台,在设置证书的页面找到 Identifiers
里,在对应的 BundleId
下勾选 Associated Domains
iOS: -canOpenURL: failed for URL: "weixin://".
-canOpenURL: failed for URL: "weixin://" - error: "The operation couldn’t be completed. (OSStatus error -10814.)"
设置 URL Schemes 并列为白名单,在 ios/<应用名称>/Info.plist
中添加
<key>CFBundleURLTypes</key>
<array>
<dict>
<key>CFBundleURLName</key>
<string>weixin</string>
<key>CFBundleURLSchemes</key>
<array>
<string>wx500b695a47bd364b</string>
</array>
</dict>
</array>
<key>LSApplicationQueriesSchemes</key>
<array>
<string>weixin</string>
<string>weixinULAPI</string>
</array>
iOS: RCTBridge required dispatch_sync to load RCTDevLoadingView.
错误内容: RCTBridge required dispatch_sync to load RCTDevLoadingView. This may lead to deadlocks
错误解决方案:可以通过下面代码可以解决,事实上我通过关闭 debug 浏览器页面就没有错误消息了。错误原因可能是你打开了 debug 浏览器,但是你模拟器并没有开启 debug 模式。
+ #if RCT_DEV
+ #import <React/RCTDevLoadingView.h>
+ #endif
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
#ifdef FB_SONARKIT_ENABLED
InitializeFlipper(application);
#endif
RCTBridge *bridge = [[RCTBridge alloc] initWithDelegate:self launchOptions:launchOptions];
+ #if RCT_DEV
+ [bridge moduleForClass:[RCTDevLoadingView class]];
+ #endif
RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:bridge moduleName:@"example" initialProperties:nil];
-
-
在项目的
android/app
目录下找到build.gradle
文件,并在dependencies
加入下面的依赖dependencies { ...... // 微信SDK // Android Studio环境下:已改用gradle形式,发布到jcenter,请开发者使用gradle来编译、更新微信SDK。 // 在build.gradle文件中,添加如下依赖即可: implementation 'com.tencent.mm.opensdk:wechat-sdk-android-without-mta:+' ...... }
-
新建消息接收类
-
微信支付:在项目的根
package
下新建一个子package
叫wxapi
,然后新建一个名为WXPayEntryActivity
的类package xxx.xxx.wxapi; // 根package的名称可以在AndroidManifest.xml的manifest中找到 import android.app.Activity; import android.os.Bundle; import android.util.Log; import com.tencent.mm.opensdk.modelbase.BaseReq; import com.tencent.mm.opensdk.modelbase.BaseResp; import com.tencent.mm.opensdk.openapi.IWXAPI; import com.tencent.mm.opensdk.openapi.IWXAPIEventHandler; import com.tencent.mm.opensdk.openapi.WXAPIFactory; import net.sourceforge.simcpux.RNWechatModule; public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler { private IWXAPI api = null; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); api = WXAPIFactory.createWXAPI(this, "微信的appId"); api.handleIntent(getIntent(), this); } @Override protected void onStart() { super.onStart(); finish(); } @Override public void onReq(BaseReq baseReq) { } @Override public void onResp(BaseResp baseResp) { Log.d("WXPAY", "onPayFinish, errCode = " + baseResp.errCode); RNWechatModule.sendReqPromise.resolve(baseResp.errCode); } }
2、微信登录认证/跳转小程序:同上,在
wxapi
包下再新建一个名为WXEntryActivity
的类package xx.xx.wxapi; import android.app.Activity; import android.os.Bundle; import android.util.Log; import com.facebook.react.bridge.Arguments; import com.facebook.react.bridge.WritableMap; import com.tencent.mm.opensdk.modelbase.BaseReq; import com.tencent.mm.opensdk.modelbase.BaseResp; import com.tencent.mm.opensdk.modelmsg.SendAuth; import com.tencent.mm.opensdk.openapi.IWXAPI; import com.tencent.mm.opensdk.openapi.IWXAPIEventHandler; import com.tencent.mm.opensdk.openapi.WXAPIFactory; import net.sourceforge.simcpux.RNWechatModule; public class WXEntryActivity extends Activity implements IWXAPIEventHandler { private IWXAPI api = null; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); api = WXAPIFactory.createWXAPI(this, "微信的appId"); api.handleIntent(getIntent(), this); } @Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); setIntent(intent); api.handleIntent(intent, this); } @Override protected void onStart() { super.onStart(); finish(); } @Override // 微信请求应用的响应结果回调 public void onReq(BaseReq baseReq) { } @Override // 应用请求微信的响应结果回调 public void onResp(BaseResp baseResp) { // 调用小程序回调 if (baseResp.getType() == ConstantsAPI.COMMAND_LAUNCH_WX_MINIPROGRAM) { WXLaunchMiniProgram.Resp launchMiniProResp = (WXLaunchMiniProgram.Resp) baseResp; String extraData =launchMiniProResp.extMsg; //对应小程序组件 <button open-type="launchApp"> 中的 app-parameter 属性 RNWechatModule.sendReqPromise.resolve(extraData); } else if (baseResp.getType() == ConstantsAPI.COMMAND_SENDAUTH) { // 微信登录认证回调 Log.d("WXLogin", "onLoginFinish, errCode = " + baseResp.errCode); SendAuth.Resp resp = (SendAuth.Resp) baseResp; WritableMap params = Arguments.createMap(); params.putInt("errCode", baseResp.errCode); params.putString("code", resp.code); params.putString("state", resp.state); RNWechatModule.sendReqPromise.resolve(params); } } }
-
-
在
AndroidManifest.xml
中加入对应的activity
-
微信支付
<activity android:name=".wxapi.WXPayEntryActivity" android:exported="true" android:launchMode="singleTop" android:theme="@android:style/Theme.NoDisplay" />
-
-
微信登录认证/跳转小程序
<activity android:name=".wxapi.WXEntryActivity" android:label="应用名" android:theme="@android:style/Theme.Translucent.NoTitleBar" android:exported="true" android:taskAffinity="xx.xx.wxapi" android:launchMode="singleTask"> <!-- android:taskAffinity="xx.xx.wxapi" 需要替换为你自己的包名 -->
-
-
-
先
cd
到ios
目录下,运行pod install
-
事件通知
-
在
AppDelegate.m
文件中添加依赖库#import <WXApi.h> #import <RNWechat/RNWechat.h>
-
添加微信消息接收事件的方法: 根据实际需要(按注释说明配置即可)定义onResp内类别判断
//ios9以后 - (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary*)options { return [WXApi handleOpenURL:url delegate:self]; } //ios9以后的方法 - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation { return [WXApi handleOpenURL:url delegate:self]; } //ios9以前 - (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url { return [WXApi handleOpenURL:url delegate:self]; } #pragma mark - wx callback - (void) onReq:(BaseReq*)req { // TODO Something } - (void)onResp:(BaseResp *)resp { // 登录认证的类型 if([resp isKindOfClass:[SendAuthResp class]]) { // 需要注意的是 resolver、rejecter 登录认证、支付是不同的方法,不能混用 RCTPromiseResolveBlock resolver = [RNWechat getSendLoginResolverStatic]; RCTPromiseRejectBlock rejecter = [RNWechat getSendLoginRejecterStatic]; // 登录认证返回结果 NSString *loginResult = [NSString stringWithFormat:@"errcode:%d", resp.errCode]; //登录认证返回结果,实际支付结果需要去微信服务器端查询 switch (resp.errCode) { case 0: loginResult = @"登录认证结果:成功!"; break; case -1: loginResult = @"登录认证结果:失败!"; break; case -2: loginResult = @"用户已经退出登录认证!"; break; default: loginResult = [NSString stringWithFormat:@"登录认证结果:失败!retcode = %d, retstr = %@", resp.errCode,resp.errStr]; break; } NSLog(@"微信登录:%@", loginResult); SendAuthResp *sendResp = (SendAuthResp *)resp; NSDictionary *params = @{ @"errCode": [NSString stringWithFormat:@"%d", sendResp.errCode], @"code": [NSString stringWithFormat:@"%@", sendResp.code], @"state": [NSString stringWithFormat:@"%@", sendResp.state], }; resolver(params); NSLog(@"WeChatSDK---------: %@", loginResult); } //判断是否是微信支付回调类型 (注意是PayResp 而不是PayReq) if ([resp isKindOfClass:[PayResp class]]) { RCTPromiseResolveBlock resolver = [RNWechat getSendPayResolverStatic]; RCTPromiseRejectBlock rejecter = [RNWechat getSendPayRejecterStatic]; //启动微信支付的response NSString *payResoult = [NSString stringWithFormat:@"errcode:%d", resp.errCode]; if([resp isKindOfClass:[PayResp class]]){ //支付返回结果,实际支付结果需要去微信服务器端查询 switch (resp.errCode) { case 0: payResoult = @"支付结果:成功!"; break; case -1: payResoult = @"支付结果:失败!"; break; case -2: payResoult = @"用户已经退出支付!"; break; default: payResoult = [NSString stringWithFormat:@"支付结果:失败!retcode = %d, retstr = %@", resp.errCode,resp.errStr]; break; } resolver(@(resp.errCode)); } else { rejecter(@"-10404", @"失败", nil); } NSLog(@"WeChatSDK: %@", payResoult); } // 拉起小程序 if ([resp isKindOfClass:[WXLaunchMiniProgramResp class]]) { RCTPromiseResolveBlock resolver = [RNWechat getSendMiniProResolverStatic]; RCTPromiseRejectBlock rejecter = [RNWechat getSendMiniProRejecterStatic]; WXLaunchMiniProgramResp *miniProResp = (WXLaunchMiniProgramResp *)resp; NSString *extMsg = miniProResp.extMsg; resolver(extMsg); // 对应JsApi navigateBackApplication中的extraData字段数据 } } - (BOOL)application:(UIApplication *)application continueUserActivity:(nonnull NSUserActivity *)userActivity restorationHandler:(nonnull void (^)(NSArray<id<UIUserActivityRestoring>> * _Nullable))restorationHandler{ return [WXApi handleOpenUniversalLink:userActivity delegate:self]; }
-
-
/**
* 向微信注册应用
* 必须先注册应用,在 Android 后面的调用才会起作用
* @param appid 通过微信开放平台,[获取appid](https://open.weixin.qq.com/)
* @param universalLink 参数在 iOS 中有效,Universal Link(通用链接)是苹果在 iOS9 推出的,一种能够方便的通过传统 HTTPS 链接来启动 APP 的功能,可以使用相同的网址打开网址和 APP。
*/
export function registerApp(appid: string, universalLink: string): Promise<any>;
/**
* 检查微信是否已被用户安装
* 微信已安装返回 `true`,未安装返回 `false`。
*/
export function isWXAppInstalled(): Promise<boolean>;
/**
* 判断当前微信的版本是否支持 OpenApi
* 支持返回 `true`,不支持返回 `false`
*/
export function isWXAppSupportApi(): Promise<boolean>;
/**
* 打开微信,成功返回 `true`,不支持返回 失败返回 `false`
*/
export function openWXApp(): Promise<boolean>;
/**
* 获取当前微信SDK的版本号
*/
export function getApiVersion(): Promise<string>;
export type RequestOption = {
appId: string;
partnerId: string;
prepayId: string;
nonceStr: string;
timestamp: string;
packageValue: string;
sign: string;
extData?: string;
}
/**
* 发送请求支付请求
*/
export function sendPayRequest(requestOption: RequestOption) : Promise<any>;
/**
* 发送登录认证请求
* @param state 唯一标识码
*/
export function sendLoginRequest(requestOption: {state: string}) : Promise<any>;
/**
* 跳转小程序
*/
static openMiniProgram(requestOption) {
return NativeModules.RNWechat.openMiniProgram(requestOption);
}
- 注册、判断是否安装、支持微信示例
import Wechat from 'react-native-wechat-libs';
const wechatInit = () => {
await Wechat.registerApp(
'wx7363cc9581927cb3',
'https://www.52dian.com/app',
).then(res => {
console.log(res, 'registerApp');
});
const _isInstall = await Wechat.isWXAppInstalled();
const _isWXAppSupportApi = await Wechat.isWXAppSupportApi();
}
2、 微信登录示例
import Wechat from 'react-native-wechat-libs';
const WXLogin = () => {
let state = new Date().getTime(); // 这里state取值可根据项目实际设置,此处为简单示例
Wechat.sendLoginRequest({state: state}).then((res: any) => {
console.log('&**********微信登录请求', res);
if (res.errCode === 0) { // 用户同意
if (res.state === state) { // state是原样返回的数据,检测正常请求而非外部攻击
// do
}
} else {
// res.errCode === -4 用户拒绝
// res.errCode === -2 用户取消
}
});
};
2、 跳转小程序
import Wechat from 'eather-react-native-wechat';
const goWXMiniProgram = () => {
Wechat.openMiniProgram({
// 小程序id
userName: 'XXXXXX',
// 小程序页面路径,不填默认首页
path: 'pages/XXX/XXX',
// 小程序版本 0: 正式版, 1:开发版, 2:体验版
miniProgramType: 0,
}).then(res => {
// 这里需要注意的是,这里的回调指的是小程序代码中下方按钮传递的app-parameter值,如果没有或者不需要这个按钮,就没有回调。。。。。
// <button class="back_app" open-type="launchApp" app-parameter="wechat" binderror="launchAppError">打开app</button>
console.log('前往微信小程序', res);
});
}