Unable to return value via Platform Channel when using Swift Callback Handlers

877 Views Asked by At

I am working with Flutter and Swift. I'm trying to authorize a credit card using the AuthrizeNet SDK, which unfortunately, does not natively support Flutter. To get around this, I created a Platform Channel in my iOS AppDelegate, which I successfully managed receive data in from Flutter. My question now is how can I return that data from inside the callback handler of the authorizing function (from the AuthorizeNet SDK)? When trying to call the Flutter result function, the Swift compiler throws this error: Escaping closure captures non-escaping parameter 'result'. Here's my code:

handler!.getTokenWithRequest(request, successHandler: { (inResponse:AcceptSDKTokenResponse) -> () in
                let paymentResponseModel: PaymentResponseModel = PaymentResponseModel.init(
                    token: inResponse.getOpaqueData().getDataValue(),
                    resultCode: inResponse.getMessages().getResultCode(),
                    tokenResultDescription: inResponse.getOpaqueData().getDataDescriptor(),
                    messageCode: inResponse.getMessages().getMessages()[0].getCode(),
                    messageText: inResponse.getMessages().getMessages()[0].getText()
                )
                result(String(data: try! JSONEncoder().encode(paymentResponseModel), encoding: String.Encoding.utf8))
            }) { (inError:AcceptSDKErrorResponse) in
                let paymentResponseModel: PaymentModelErrorResponse = PaymentModelErrorResponse.init(
                    code: inError.getMessages().getResultCode(),
                    message: inError.getMessages().getMessages()[0].getCode(),
                    details: inError.getMessages().getMessages()[0].getText()
                )
                result(String(data: try! JSONEncoder().encode(paymentResponseModel), encoding: String.Encoding.utf8))
            }

I've tried different way of calling the result function outside of the getTokenWithRequest function, but I couldn't get it to work. I'm fairly new to Swift programming, so I apologize if this question is vague or has a simple solution. Thank you!

1

There are 1 best solutions below

6
On BEST ANSWER

The problem is that the successHandler will be called in the future (as a callback), but result must be called immediately and you must return from the method of your code snippet immediately. So, how do you get the success/error response back? You have to make a call from native to Dart when the the token response becomes available. Then, at the Dart end, you'll have to hang about until that response call is made when you can update your app state, and the UI.

It becomes a bi-directional method channel like this:

---token request--->
<---result----------

<---token response--
----result--------->

Neither of the results contains much information, but could be used to convey errors. For example, if the first result comes back false, something went wrong and the Dart end shouldn't expect the token response - it won't happen.