如何将 React Native Promise 连接到 Swift

Posted

技术标签:

【中文标题】如何将 React Native Promise 连接到 Swift【英文标题】:How to bridge React Native Promise to Swift 【发布时间】:2017-03-03 11:19:17 【问题描述】:

各位软件爱好者,

我目前正在开发一个 React 本机项目,我需要为其添加一些用 swift 编写的逻辑。我可以通过桥接到 Objective C 和 Swift 来触发一个基本的 swift 函数。

当我尝试使用 Promise 做某事时,就会出现问题。我描述这一点的页面在 Promises 的 Objective C 部分以及桥接到 Swift 上都很清楚,但对于 swift 的承诺则不然:https://facebook.github.io/react-native/docs/native-modules-ios.html

这就是我所拥有的:

Project-Bridging-Header.h

#import <React/RCTBridgeModule.h>

MyLoginBridge.m

#import <Foundation/Foundation.h>
#import <React/RCTBridgeModule.h>

@interface RCT_EXTERN_REMAP_MODULE(MyCustomLoginJSName, MyLoginModule, NSObject)

RCT_EXTERN_REMAP_METHOD(loginWithEmail,
                    resolver:(RCTPromiseResolveBlock)resolve
                    rejecter:(RCTPromiseRejectBlock)reject)

RCT_EXTERN_METHOD(testMethod)

@end

MyLoginModule.swift

import Foundation

@objc(TripleASDKModule)
class TripleASDKModule: NSObject 

  @objc
  func loginWithEmail(resolver resolve: RCTPromiseResolveBlock,  rejecter reject: RCTPromiseRejectBlock) -> Void 
    resolve("This method is troublesome")
  

  @objc func testMethod() -> Void 
    print("This Does appear")
  

当我触发 testMethod 时,打印显示在 Xcode 中,以便执行 swift 代码。但是当我调用 loginWithEmail 方法时,我得到了臭名昭著的红色 React Native 错误屏幕:

Exception 'resolver:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject is not a recognized Objective-C method.' was thrown while invoking loginWithEmail on target MyCustomLoginJSName with params (
    30,
    31
)

为了完整起见,javascript方面:

const loginWithEmail = () => NativeModules.TripleA.loginWithEmail()
    .then(result => console.log(result));

我尝试了几乎所有 RCT_EXTERN_REMAP_METHOD 的变体以及我能找到的类似的东西,无论有没有重新映射重复名称等。 因此,如果这个问题听起来很熟悉,或者您可以引导我朝着正确的方向前进,请这样做,我们将不胜感激。

【问题讨论】:

【参考方案1】:

取自Got "is not a recognized Objective-C method" when bridging Swift to React-Native的答案;它不起作用的事实是因为第一个参数标签的不同。

要使其与您的初始代码一起使用,您应该将 Swift 的第一个参数编写为没有名称,如下所示:

@objc
func loginWithEmail(_ resolve: RCTPromiseResolveBlock, rejecter reject: RCTPromiseRejectBlock) -> Void 
    //   the trick  ^
    resolve("This method is no longer troublesome")

【讨论】:

当我这样做时,我得到error: use of undeclared type RCTPromiseResolveBlock'error: use of undeclared type 'RCTPromiseRejectBlock' - 有什么想法吗? 是否在桥接头中导入桥接模块?例如。 #import &lt;React/RCTBridgeModule.h&gt; 也有这个问题,我已经在网桥中导入了:( 我说的不是 react native 桥,而是ProjectName-Bridging-Header.h 文件。 原来 Xcode 中的桥接头设置不正确...感谢您的帮助!【参考方案2】:

添加以便显示完整的解决方案

.m - 注意 resolve 参数不是“命名的”

RCT_EXTERN_METHOD(loginWithEmail: (RCTPromiseResolveBlock)resolve
                                  rejecter:(RCTPromiseRejectBlock)reject)

.swift - 与@Koen 相同

@objc func loginWithEmail(_ resolve: @escaping RCTPromiseResolveBlock,
                            rejecter reject: @escaping RCTPromiseRejectBlock ) -> Void 

【讨论】:

感谢添加 mate - 获得与 Swift 签名匹配的外部定义很重要【参考方案3】:

我最终想出了一个解决方案。我相信它并不完全像 React 本地人想要的那样,但它确实有效。所以我可以继续,也许其他人用我的解决方案走上了正确的轨道。不过,请按应有的方式发布。

所以我决定先从 Objective-C 的方式开始。所以我为我的模块创建了一个 .h 文件。

MyLoginBridge.h

#import <React/RCTBridgeModule.h>

@interface MyLoginBridge : NSObject <RCTBridgeModule>
@end

然后修改 .m 文件

#import "MyLoginBridge.h"
#import "MyProject-Swift.h" // Include the swift header manually

@implementation MyLoginBridge

RCT_EXPORT_MODULE(MyCustomLoginJSName);

RCT_EXPORT_METHOD(loginWithEmail:(RCTPromiseResolveBlock)resolve   rejecter:(RCTPromiseRejectBlock)reject)

    // Manually init the module and call swift function
    MyLoginModule* module = [[MyLoginModule alloc] init];
    [module loginWithEmailWithResolver:resolve rejecter:reject];


@end

swift 文件和桥接头保持不变。这行得通。

【讨论】:

以上是关于如何将 React Native Promise 连接到 Swift的主要内容,如果未能解决你的问题,请参考以下文章

React Native 中的 Promise

React Native 中的意外 Promise

React Native AsyncStorage:从 Promise 对象访问值

react-native async 函数返回 promise 但不返回我的 json 数据?

如何在reducer中处理异步promise

React Native 网络请求封装:使用Promise封装fetch请求