如何在 Swift 中实现 FlutterPlugin 的方法处理程序?

Ren*_*ato 5 swift flutter

我正在尝试使用 Flutter 编写移动应用程序。不幸的是,我需要实现一个 Flutter 插件才能调用一些原生 API。

Objective-C 对我来说太陌生了,所以我真的更喜欢使用 Swift(我也不太了解,但它与 Kotlin 足够接近,我觉得它很熟悉)来创建插件。

我得到了 Flutter 的插件框架设置,但现在我正在尝试实现方法处理函数:

public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult)
Run Code Online (Sandbox Code Playgroud)

我尝试使用的本机函数具有以下签名(使用 gomobile 生成):

FOUNDATION_EXPORT MobileapiDatabase* MobileapiReadDatabase(
  NSString* filePath, NSString* password, NSError** error);
Run Code Online (Sandbox Code Playgroud)

我不明白几件事:

  1. 如何使用result对象返回错误?
  2. 如何报告方法未执行?
  3. 如何使用本机函数的error对象?

我只能找到 Objective-C 示例,而没有通过谷歌搜索找到的单个 Swift 示例(我发现的唯一示例没有处理错误或其他任何内容)。

这是我到目前为止所得到的:

public func handle(_ call: FlutterMethodCall, 
    result: @escaping FlutterResult) {
  switch call.method {
  case "getDb":
    let args = call.arguments as! [String]
    if args.count == 2 {
        let error = NSError() // ERROR in next line
        MobileapiReadDatabase(args[0], args[1], error)
        // how to check if there was an error?
    }
    // TODO report wrong arguments
  default:
    // TODO report method not implemented
  }
}
Run Code Online (Sandbox Code Playgroud)

真的很感激任何帮助。

Ren*_*ato 10

通过阅读一些调用 Objective-C 的 Swift 代码示例并检查 Objective-C 源代码,我自己设法弄明白了。

这是我如何实施它:

import Flutter
import UIKit
import Mobileapi

public class SwiftGohashMobilePlugin: NSObject, FlutterPlugin {
    public static func register(with registrar: FlutterPluginRegistrar) {
        let channel = FlutterMethodChannel(name: "gohash_mobile", binaryMessenger: registrar.messenger())
        let instance = SwiftGohashMobilePlugin()
        registrar.addMethodCallDelegate(instance, channel: channel)
    }

    public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
        switch call.method {
        case "getDb":
            if let args = call.arguments as? [String] {
                if args.count == 2 {
                    var error : NSError?
                    let db = MobileapiReadDatabase(args[0], args[1], &error)
                    if let errorMessage = error?.userInfo.description {
                        result(FlutterError.init(code: "NATIVE_ERR",
                                                 message: "Error: " + errorMessage,
                                                 details: nil))
                    } else {
                        // SUCCESS!!
                        result(db!)
                    }
                } else {
                    result(FlutterError.init(code: "BAD_ARGS",
                                             message: "Wrong arg count (getDb expects 2 args): " + args.count.description,
                                             details: nil))
                }
            } else {
                result(FlutterError.init(code: "BAD_ARGS",
                                         message: "Wrong argument types",
                                         details: nil))
            }
        default:
            result(FlutterMethodNotImplemented)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

欢迎提出改进建议!

  • Flutter 是否会通过该名称自动调用函数“handle”? (2认同)
  • 是的,请参阅 https://api.flutter.dev/objcdoc/Protocols/FlutterPlugin.html#/c:objc(pl)FlutterPlugin(im)handleMethodCall:result: (2认同)