Last Updated: 17 March 2025
Flutter integration with Web SDK
Overview
The integration consists of using web_flutter (the Flutter equivalent of an iframe) to display a HTML form, which uses the Access Checkout Web SDK to interface with the Flutter layer.
Integration
- Follow one of our guides (e.g. create a session to pay with a card) to integrate the Access Checkout Web SDK into a HTML page.
- Add the webview_flutter package to your application by following the webview_flutter installation steps.
- Your iOS folder
Podfile
must be set up to automatically source an additional dependency calledwebview_flutter_wkwebview
.- if you have a
Podfile
, make sure that it contains the same code as in the webview_flutter example application Podfile - if you don't have a
Podfile
under theios
folder, you must generate one and it should be correctly set up by default. Runflutter run
and select an iOS device as the destination or, runflutter build ios
and follow the instructions
- if you have a
- Import the
webview_flutter
package in your stateful widget usingimport 'package:webview_flutter/webview_flutter.dart';
. - Create an instance of
WebViewController
in your widget's state and set theJavaScript
mode asunrestricted
:
class _WebViewState extends State<AccessCheckoutWebWidget> {
late WebViewController controller;
// ...
@override
void initState() {
super.initState();
// ...
controller = WebViewController()
..setJavaScriptMode(JavaScriptMode.unrestricted)
..setNavigationDelegate(
NavigationDelegate(
onProgress: (int progress) {},
onPageStarted: (String url) {},
onPageFinished: (String url) {},
onHttpError: (HttpResponseError error) {
print(error.response);
},
onWebResourceError: (WebResourceError error) {
print(error.description);
},
),
);
}
}
- Add a JavaScript channel with a message listener to the
WebViewController
. The JavaScript channel intercepts the messages that is sent to Flutter from the JavaScript layer:
controller = WebViewController()
..addJavaScriptChannel(
'flutterWebView',
onMessageReceived: (dynamic message) {
setState(() {
sessionToken = message.message;
});
},
)
// ...
- In
WebViewController
, load the URL of the HTML page that contains your card form. At this stage, you should be able to see your card form displayed in your Flutter application:
controller = WebViewController()
// ...
..loadRequest(
Uri.parse("..."),
);
- Use the Flutter 'postMessage' function from the JavaScript channel in your JavaScript code to communicate from the JavaScript layer to the Flutter layer. It is available in the JavaScript context as a variable that has the same name as the name used when defining the JavaScript channel in the Flutter code:
Worldpay.checkout.init(
{
...
},
function (error, checkout) {
// ...
form.addEventListener("submit", function (event) {
event.preventDefault();
checkout.generateSessionState(function (error, sessionState) {
// the session is sent to the Flutter layer using the Flutter postMessage mechanism
flutterJSChannel.postMessage(sessionState)
});
});
// ...
}
);
Full integration example
import 'package:flutter/material.dart';
import 'package:webview_flutter/webview_flutter.dart';
class AccessCheckoutWebWidget extends StatefulWidget {
const AccessCheckoutWebWidget({super.key});
@override
State<AccessCheckoutWebWidget> createState() => _WebViewState();
}
class _WebViewState extends State<AccessCheckoutWebWidget> {
late WebViewController controller;
String sessionToken = "";
@override
void initState() {
super.initState();
checkoutId = widget.checkoutId;
iframeBaseUrl = widget.iframeBaseUrl;
controller = WebViewController()
..setJavaScriptMode(JavaScriptMode.unrestricted)
..addJavaScriptChannel(
'flutterJSChannel',
onMessageReceived: (dynamic message) {
setState(() {
sessionToken = message.message;
});
},
)
..setNavigationDelegate(
NavigationDelegate(
onProgress: (int progress) {},
onPageStarted: (String url) {},
onPageFinished: (String url) {},
onHttpError: (HttpResponseError error) {
print(error.response);
},
onWebResourceError: (WebResourceError error) {
print(error.description);
},
),
)
..loadRequest(
Uri.parse("<replace-with-your-url-to-form.html>"),
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
body: Padding(
padding: const EdgeInsets.all(0),
child: Column(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
SizedBox(
width: 400,
height: 450,
child: WebViewWidget(controller: controller)
),
if (sessionToken != "") Text(sessionToken),
],
)));
}
}
Tips for Android emulators
Tips for Android emulators
On Android emulator devices, localhost
points to the emulated device. If you want the iframe to load a page hosted in your local environment you must use the 10.0.2.2
IP address. More information is available on the official Android Emulator networking page.
The use of the https
protocol has been enforced since Android 6.0. As this can be impractical for local development, you can configure an exception to allow plain text http on 10.0.2.2
. Note that this is for development only, and that you must always use https in Production.
- Create a
network_config.xml
in theres/xml
folder with the following content:
<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
<domain-config cleartextTrafficPermitted="true">
<domain includeSubdomains="true">10.0.2.2</domain>
</domain-config>
</network-security-config>
- Configure your Android app to use this file as network security configuration by adding the
android:networkSecurityConfig="@xml/network_config"
attribute to the<application>
node of yourAndroidManifest.xml
file.