How to create a Compose Multiplatform iOS & Android barcode scanning application using the Scanbot SDK

Ildar July 30, 2024 11 mins read
app store

Developing mobile applications for iOS and Android often involves managing different coding environments and languages. UI frameworks like Compose Multiplatform simplify the development process significantly, especially when combined with a software development kit like our Scanbot Barcode Scanner SDK.

For Android developers, Jetpack Compose has become the standard toolkit for creating user interfaces. With its small but powerful codebase, the UI toolkit has revolutionized how Android apps are developed, making the process less cumbersome and more intuitive. To bring these advantages to more platforms, JetBrains created Compose Multiplatform. This framework enables developers to wield the power of Kotlin to write UI code compatible with both Android and iOS (with support for the latter currently in beta), thereby streamlining development.

SDKs, too, simplify app development. The Scanbot SDK provides a straightforward solution for rapidly integrating fast and reliable barcode scanning into any mobile app. Barcode scanning is an essential feature for countless use cases among various industries, including logistics, healthcare, and self-checkout in retail.

In this article, we’ll walk you through the integration process for iOS and Android using Compose Multiplatform and the Scanbot Barcode Scanner SDK.

💡 You can find the finished example project’s source code in our GitHub repo.

Setting up the Compose Multiplatform project

We will use Compose Multiplatform to create both Android and iOS apps. For this, you need:

Check your environment

Before you start, use the KDoctor tool to ensure that your development environment is configured correctly:

Install KDoctor with Homebrew:

brew install kdoctor

Run KDoctor in your terminal:

kdoctor

Preparing the project

In this tutorial we will use Kotlin Multiplatform Wizard, a new way to setup the KMP project. It uses Kotlin 2.0 at the time of this article.

To get started with Compose Multiplatform using the wizard, open the link and select Android and iOS with the Share UI option selected.

The project was migrated from CocoaPods to a manual XCFramework, which makes the use of third-party dependencies for iOS a bit more complicated. Therefore, we will describe the process in detail. 

After pressing DOWNLOAD and extracting the zip archive, open the project in Android Studio, sync the project, and switch to the Project Files view. You should see the following structure:

Let’s try to run the template app on an Android device (it looks very similar on iOS).

Adding an Android dependency on the Scanbot Barcode Scanner SDK

First off, let’s set up our app on the Android side to work with the Scanbot Barcode Scanner SDK. 

For this, we will follow the steps in our documentation

In settings.gradle.kts, let’s modify the dependencyResolutionManagement section like this:

dependencyResolutionManagement {
    repositories {
        ...
        maven { url = java.net.URI("https://nexus.scanbot.io/nexus/content/repositories/releases/") }
        maven { url = java.net.URI("https://nexus.scanbot.io/nexus/content/repositories/snapshots/") }
        ...
  }
}

Next, let’s modify composeApp/build.gradle.kts by adding our Scanbot SDK dependencies:

kotlin {
 sourceSets {
...
        val androidMain by getting {
            dependencies 
                ...
                implementation("io.scanbot:scanbot-barcode-scanner-sdk:5.2.0")
                implementation("io.scanbot:rtu-ui-v2-barcode:5.2.0")
                ...
            }
        }
...
}

Android Studio will hint that it is recommended to use version catalogs instead of the hardcoded version. Feel free to apply the hint to make the version of the SDK consistent with other dependencies. You can also do this later. 

Adding a Native iOS dependency as XCFramework in Kotlin Multiplatform project

To be able to use our barcode scanner functionalities in iOS apps as well, we need to add a dependency on the Scanbot Barcode Scanner SDK for iOS. We’ll use the XCFramework approach for this, since it’s the default way.

💡 If you use the Cocoapods setup for iOS in your Kotlin Multiplatform project, please follow these steps to add the dependency to our ScanbotBarcodeScannerSDK cocoapod. Then you can proceed directly to the step “Implementing shared Compose Multiplatform UI”.

1. We need to make sure that the project is using the property: 

kotlin.mpp.enableCInteropCommonization=true

Without that, we won’t be able to access the iOS code from the third-party library in the Kotlin layer. 

2. We need to define the .def file with the following path: 

composeApp/src/nativeInterop/cinterop/ScanbotBarcodeScannerSDK.def

… and with the following contents:

language = Objective-C
modules = ScanbotBarcodeScannerSDK 
modules = ScanbotBarcodeScannerSDK 
package = ScanbotBarcodeScannerSDK

3. We need to download the Scanbot Barcode Scanner SDK for the iOS XCFramework (you can find a link to the latest version here). For this tutorial, we’ll use version 5.2.1.

Extract the zip file somewhere into the project folder, for example in scanbotsdk/ScanbotBarcodeScannerSDK.xcframework. We only need the xcframework file itself.

4. Compile and link the frameworks for the corresponding architectures in composeApp/build.gradle.kts

Now add the paths for the .def and frameworks. To do that, we need to add the following snippet into the kotlin { ... } part of the configuration:

iosArm64 {
    val path = "$rootDir/scanbotsdk/ScanbotBarcodeScannerSDK.xcframework/ios-arm64"
    compilations.getByName("main") {
        val ScanbotBarcodeScannerSDK by cinterops.creating {
            defFile("src/nativeInterop/cinterop/ScanbotBarcodeScannerSDK.def")
            compilerOpts("-F$path", "-framework", "ScanbotBarcodeScannerSDK", "-rpath", path)
            extraOpts += listOf("-compiler-option", "-fmodules")
        }
    }
    binaries.all {
        linkerOpts("-framework", "ScanbotBarcodeScannerSDK", "-F$path")
    }
}

listOf(
    iosX64(),
    iosSimulatorArm64()
).forEach {
    val path =
        "$rootDir/scanbotsdk/ScanbotBarcodeScannerSDK.xcframework/ios-arm64_x86_64-simulator"
    it.compilations.getByName("main") {
        val ScanbotBarcodeScannerSDK by cinterops.creating {
            defFile("src/nativeInterop/cinterop/ScanbotBarcodeScannerSDK.def")
            compilerOpts("-F$path", "-framework", "ScanbotBarcodeScannerSDK", "-rpath", path)
            extraOpts += listOf("-compiler-option", "-fmodules")
        }
    }
    it.binaries.all {
        linkerOpts("-framework", "ScanbotBarcodeScannerSDK", "-F$path")
    }
}

You can put it next to the following lines:

listOf(
    iosX64(),
    iosArm64(),
    iosSimulatorArm64()
).forEach { iosTarget ->
    iosTarget.binaries.framework {
        baseName = "ComposeApp"
        isStatic = true
    }
}

Now sync the project and you will be able to access the iOS dependency file from the Kotlin classes.

However, the final app also needs to bundle the XCFramework. For that we need to open the project in Xcode and add the framework manually. Just drag the iosApp folder into Xcode.

Press Add other and select the XCFramework file we put into the scanbotsdk folder. Make sure it says Embed & Sign when it appears in the list:

Finally, our iOS dependency setup is done. 

Let’s try to run our app on an iOS device. Before we continue, we have to make sure the signing properties are in order:

Implementing shared Compose Multiplatform UI 

It’s pretty cool that we can create a shared UI for Android and iOS using Compose, even though iOS support is still in beta. Let’s get right to it!

We start by adding a button and a text element in the template code. The text element is where we will put the scanning results.

The template provides the following code in the App class (composeApp/src/commonMain/kotlin/App.kt):

@Composable
@Preview
fun App() {
    MaterialTheme {
        var showContent by remember { mutableStateOf(false) }
        Column(Modifier.fillMaxWidth(), horizontalAlignment = Alignment.CenterHorizontally) {
            Button(onClick = { showContent = !showContent }) {
                Text("Click me!")
            }
            AnimatedVisibility(showContent) {
                val greeting = remember { Greeting().greet() }
                Column(Modifier.fillMaxWidth(), horizontalAlignment = Alignment.CenterHorizontally) {
                    Image(painterResource(Res.drawable.compose_multiplatform), null)
                    Text("Compose: $greeting")
                }
            }
        }
    }
}

Instead of opening an image after the button is clicked, we will open the corresponding Ready-to-Use UI Component from the Scanbot SDK for Android and iOS respectively. Later, we’ll also need to define a Composable expect function BarcodeScannerNativeView on the native side.

We save the text of the last scanned barcode in a variable and define a callback for it for each of the Native views. Since the SDK must be initialized first, we’ll do just that and extract a variable for the license key. 

In short, we’ll replace the existing class with the following:

@Composable
@Preview
fun App() {
    MaterialTheme {
        var showContent by remember { mutableStateOf(false) }
        var text by remember { mutableStateOf("") }

        /*
        * TODO: Add the Scanbot Barcode SDK license key here.
        * Please note: The Scanbot Barcode SDK will run without a license key for one minute per session!
        * After the trial period is over all Scanbot SDK functions as well as the UI components will stop working.
        * You can get an unrestricted "no-strings-attached" 30 day trial license key for free.
        * Please submit the trial license form (https://scanbot.io/sdk/trial.html) on our website by using
        * the app identifier.
        */
        val licenseKey = ""
        initializeScanbot(licenseKey)

        Column(Modifier.fillMaxWidth(), horizontalAlignment = Alignment.CenterHorizontally) {
            Button(onClick = { showContent = !showContent }) {
                Text("Start barcode scanner")
            }
            AnimatedVisibility(showContent) {
                val greeting = remember { Greeting().greet() }
                Column(Modifier.fillMaxWidth(), horizontalAlignment = Alignment.CenterHorizontally) {
                    Text(text)
                    Text("Compose: $greeting")
                    BarcodeScannerNativeView(object : OnBarcodeScanned {
                        override fun onBarcodeScanned(barcode: String) {
                            text = barcode
                        }
                    })
                }
            }
        }
    }
}

@Composable
expect fun BarcodeScannerNativeView(onBarcodeScanned: OnBarcodeScanned)

@Composable
expect fun initializeScanbot(licenseKey: String)

interface OnBarcodeScanned {
    fun onBarcodeScanned(barcode: String)
}

Now that we have some expect functions, we need to adapt the corresponding platform-specific parts. 

You can already create the androidMain and iosMain implementations by resolving a complaint from Android Studio about our expect functions.

Android integration

Let’s start with the Android-specific part.

First of all, as our Barcode Scanner feature will use the device’s camera, we need to add the Camera permission in our Android manifest (composeApp/src/androidMain/AndroidManifest.xml):

<uses-feature
    android:name="android.hardware.camera"
    android:required="false" />
<uses-permission android:name="android.permission.CAMERA" />

Initializing the SDK

We need to initialize the Scanbot Barcode Scanner SDK for Android before using it. Normally, we recommend doing this in the Application subclass in your Native Android part. 

However, for simplicity’s sake, we’ll initialize it with LaunchedEffect in this tutorial. To call the ScanbotBarcodeScannerSDKInitializer, we also need to get the Application object, which we can request from the LocalContext.current.

💡 Note: Without a license key, the Scanbot SDK will only work for 60 seconds. You can request a free trial license key on our website.

We need to implement the expect function defined ealier with the actual function. We’ll do it here: composeApp/src/androidMain/kotlin/App.android.kt:

@Composable
actual fun initializeScanbot(licenseKey: String) {
    val current = LocalContext.current
    LaunchedEffect(true) {
        val application = current.applicationContext as Application
        ScanbotBarcodeScannerSDKInitializer()
            .license(application, licenseKey)
            .initialize(application)
    }
}

Implement calling the RTU UI Barcode Scanner screen

Now we follow the steps in the Ready-to-Use UI section.

With our recently introduced RTU UI v.2.0, we’ve added support for integrating the screen as a Jetpack Compose Sub-Component. This makes implementing the BarcodeScannerNativeView for Android very straightforward (we can do it in the same file):

@Composable
actual fun BarcodeScannerNativeView(onBarcodeScanned: OnBarcodeScanned) {
    val configuration = BarcodeScannerConfiguration()
    CompositionLocalProvider(
        LocalScanbotNativeConfiguration provides BarcodeNativeConfiguration(
            enableContinuousScanning = true
        )
    ) {
        // This view is coming from the import io.scanbot.sdk.ui_v2.barcode.BarcodeScannerView:
        BarcodeScannerView(
            configuration = configuration,
            onBarcodeScanned = {
                onBarcodeScanned.onBarcodeScanned(it.items.first().text)
            },
            onBarcodeScannerClosed = {
            }
        )
    }
}

Now the Android part is ready! With just a few lines of code, we’re now ready to run the application on our Android device:

It scans the barcode very quickly. Perfect! 

iOS integration

Before we go on, let’s also add the Camera permission, just as we did for Android. For the iOS project, you add it to the Info.plist file:

With our RTU UI v.2.0 and the recent updates to Compose Multiplatform, integrating the iOS view controller became much easier. Now we can do it entirely in Kotlin using Android Studio. 

Let’s start with implementation of the BarcodeScannerViewController wrapper around the Barcode Scanner RTU UI view contoller. 

For that we just wrap it in a UIViewController and attach in the viewDidLoad function in the file composeApp/src/iosMain/kotlin/BarcodeScannerViewController.kt.

@OptIn(ExperimentalForeignApi::class)
class BarcodeScannerViewController(val onBarcodeScanned: OnBarcodeScanned) : UIViewController(nibName = null, bundle = null) {

    @OptIn(ExperimentalForeignApi::class)
    private lateinit var scannerViewController: SBSDKUI2BarcodeScannerViewController

    override fun viewDidLoad() {
        super.viewDidLoad()

        val configuration = SBSDKUI2BarcodeScannerConfiguration()
        scannerViewController =
            SBSDKUI2BarcodeScannerViewController.createNewWithConfiguration(
                configuration = configuration,
                handler = object : (SBSDKUI2BarcodeScannerViewController?, Boolean, NSError?, SBSDKUI2BarcodeScannerResult?) -> Unit {
                    override fun invoke(
                        p1: SBSDKUI2BarcodeScannerViewController?,
                        p2: Boolean,
                        p3: NSError?,
                        result: SBSDKUI2BarcodeScannerResult?
                    ) {
                        if (result?.items()?.isNotEmpty() == true) {
                            val firstBarcode = result.items().first() as SBSDKUI2BarcodeItem
                            onBarcodeScanned.onBarcodeScanned(firstBarcode.text())
                        }
                    }
                }
            )

        this.sbsdk_attachViewController(scannerViewController, this.view)
    }
}

Finally, we need to call these from composeApp/src/iosMain/kotlin/App.ios.kt.

We’ll also do as we did with Android regarding the initialization in LaunchedEffect.

@OptIn(ExperimentalForeignApi::class)
@Composable
actual fun initializeScanbot(licenseKey: String) {
    LaunchedEffect(true) {
        if (licenseKey.isNotEmpty()) {
            Scanbot.setLicense(licenseKey)
        }
        Scanbot.initialize()
    }
}

The call to the view contoller wrapper is straightforward:

@OptIn(ExperimentalForeignApi::class)
@Composable
actual fun BarcodeScannerNativeView(onBarcodeScanned: OnBarcodeScanned) {
    UIKitViewController(
        factory = { BarcodeScannerViewController(onBarcodeScanned) },
        modifier = Modifier.fillMaxSize(),
    )
}

That’s it! Our native iOS view controller is used in the Compose Multiplatform app’s view hierarchy.

Building may take some time. But after a short wait, we can see our Compose Multiplatform app with barcode scanning functionalities running on our iOS device as well!

Concluding remarks

We hope you enjoyed this tutorial! We’re looking forward to seeing the apps you build with Compose Multiplatform and the Scanbot Barcode Scanner SDK!

If you have questions about this tutorial or ran into any issues, we’re happy to help! Just shoot us an email via tutorial-support@scanbot.io.

Want to get started integrating the Scanbot SDK into your app? We offer a free 7-day trial license – no strings attached. If you have any questions, don’t hesitate to get in touch. And keep an eye on this blog for more tutorials!