Apple Pay - Flutter SDK
This guide shows you how to integrate Nium’s Apple Pay Push Provisioning SDK into a Flutter app. You'll set up the project, configure iOS settings, and implement card provisioning using a platform channel.
Step 1: Create a new Flutter project
- Open Terminal and run:
flutter create apple_push_provisioning_demo
- Navigate into the project directory:
cd apple_push_provisioning_demo
- Open the project in your preferred IDE (such as VS Code or Android Studio).
Step 2: Configure iOS settings
Set up the iOS configuration by updating the Podfile.
- Navigate to the
ios
directory:
cd ios
- If a
Podfile
doesn’t exist, create one:
pod init
- Open the
Podfile
and add:
target 'Runner' do
use_frameworks!
# CocoaPods source repositories
source 'https://github.com/CocoaPods/Specs.git'
source 'https://bitbucket.org/instadevelopers/push-provisioning-pod.git'
# Apple Push Provisioning SDK
pod 'NiumPayPushProvisioning'
pod 'TrustKit'
end
- Save and install dependencies:
pod install
Step 3: Add card provisioning
Use a platform channel to connect your Flutter app with native iOS code and call the SDK.
Set up Flutter method channel
Update lib/main.dart
:
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
home: HomePage(),
);
}
}
class HomePage extends StatefulWidget {
@override
_HomePageState createState() => _HomePageState();
}
class _HomePageState extends State<HomePage> {
static const MethodChannel _channel = MethodChannel('push_provisioning');
Future<void> provisionCard(Map<String, dynamic> parameters) async {
try {
final result = await _channel.invokeMethod('provisionCard', parameters);
debugPrint('Card provisioned: $result');
} on PlatformException catch (e) {
debugPrint('Failed to provision card: ${e.message}');
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text('Apple Push Provisioning SDK')),
body: Center(
child: ElevatedButton(
onPressed: () => provisionCard({
'cardId': 'CardId',
'additionalParam': 'value',
}),
child: Text('Provision Card'),
),
),
);
}
}
Implement the iOS method handler
Update ios/Runner/AppDelegate.swift
:
import UIKit
import Flutter
import NiumPayPushProvisioning
import TrustKit
@UIApplicationMain
@objc class AppDelegate: FlutterAppDelegate {
override func application(
_ application: UIApplication,
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
) -> Bool {
GeneratedPluginRegistrant.register(with: self)
// TrustKit Configuration
let trustKitConfig: [String: Any] = [
kTSKSwizzleNetworkDelegates: false,
kTSKPinnedDomains: [
"nium.com": [
kTSKEnforcePinning: true,
kTSKIncludeSubdomains: true,
kTSKPublicKeyHashes: ["yourHashValue"],
kTSKReportUris: ["<https://overmind.datatheorem.com/trustkit/report>"]
]
]
]
TrustKit.initSharedInstance(withConfiguration: trustKitConfig)
do {
try NiumCardsPushProvisioning.initialize(
clientHashId: "clientHashId",
customerHashId: "customerHashId",
walletHashId: "walletHashId",
apiSecret: "apiSecret",
env: .production
)
print("NiumCardsPushProvisioning SDK initialized successfully")
} catch {
print("Error initializing SDK: \(error)")
}
let controller = window?.rootViewController as! FlutterViewController
let channel = FlutterMethodChannel(name: "push_provisioning", binaryMessenger: controller.binaryMessenger)
channel.setMethodCallHandler { (call: FlutterMethodCall, result: @escaping FlutterResult) in
if call.method == "provisionCard" {
guard let args = call.arguments as? [String: Any], let cardId = args["cardId"] as? String else {
result(FlutterError(code: "INVALID_ARGUMENT", message: "Card ID is required", details: nil))
return
}
let sdk = NiumCardsPushProvisioning.shared
sdk?.addToWallet(viewController: controller, cardHashId: cardId) { sdkResult in
switch sdkResult {
case .success(let message):
print("Success: \(message)")
case .failure(let error):
print("Error: \(error)")
}
}
}
}
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
}
}
Step 4: Build and test
- Run your app on an iOS device:
flutter run
- In the app, tap Provision Card to start the provisioning flow. If successful, the card is added to the Apple Wallet.
Troubleshooting
Authentication errors with Bitbucket
If you encounter access errors when pulling the SDK from Bitbucket, use a Bitbucket username and App Password.
Use this format in your Podfile
:
source 'https://<username>:<app_password>@bitbucket.org/instadevelopers/push-provisioning-pod.git'
CocoaPods issues
Clear the CocoaPods cache if you face dependency problems:
rm -rf ~/.cocoapods/repos/bitbucket-instadevelopers-push-provisioning-pod
pod cache clean --all
pod repo update
pod install
If you continue to run into any issues, please contact your Nium account manager or support@nium.com.