Skip to content

React Native 包使用微信分享、登录、收藏、支付等功能。

Notifications You must be signed in to change notification settings

EatherToo/react-native-wechat-lib

Repository files navigation

react-native-wechat-libs

安装: 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 表示苹果账号的团队 IDcom.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];

一、接入指南

  • 安卓
    1. 在项目的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:+'
          ......
      }
    2. 新建消息接收类

      1. 微信支付:在项目的根package下新建一个子packagewxapi,然后新建一个名为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);
              }
      
          }
      }
    3. AndroidManifest.xml中加入对应的activity

      1. 微信支付

        <activity
                    android:name=".wxapi.WXPayEntryActivity"
                    android:exported="true"
                    android:launchMode="singleTop"
                    android:theme="@android:style/Theme.NoDisplay" />
    4. 微信登录认证/跳转小程序

        <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" 需要替换为你自己的包名 -->
  • iOS
    1. cdios 目录下,运行pod install

    2. 事件通知

      1. AppDelegate.m文件中添加依赖库

        #import <WXApi.h>
        #import <RNWechat/RNWechat.h>
      2. 添加微信消息接收事件的方法: 根据实际需要(按注释说明配置即可)定义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);
}

三、使用示例

  1. 注册、判断是否安装、支持微信示例
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);
    });
  }

About

React Native 包使用微信分享、登录、收藏、支付等功能。

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •