Getting started with Native Library Interop
This article covers how to get started with Native Library Interop using Maui.NativeLibraryInterop to simplify the setup.
These instructions outline the basic steps, key decision points, and guiding examples for creating bindings via Native Library Interop. For further guidance on specific API and implementation details, please refer to documentation for the native SDKs and libraries of interest.
Prerequisites
Install prerequisites:
- .NET 8.0 SDK
- .NET MAUI workload (via Visual Studio or CLI
dotnet workload install maui
) - Android SDK
- Android Studio
- Objective-Sharpie (used to manually generate the C# APIs)
- Xamarin.iOS (required for Objective-Sharpie to work)
- Visual Studio or Visual Studio Code
- Xcode
- Xcode Command Line Tools (
xcode-select --install
)
Note
It's possible to install the Android SDK and/or the Xcode Command Line Tools in a standalone manner. However, installation of the Xcode Command Line Tools is typically handled via Xcode. Likewise, Android SDK installation is also typically handled via Android Studio and/or the .NET MAUI VS Code Extension as-per the .NET MAUI Getting Started documentation.
Create a new binding
The easiest way to get started with creating a new binding is by cloning the template in the Maui.NativeLibraryInterop repo and making modifications from there. To better understand the full scope of how Maui.NativeLibraryInterop is currently set up, read more in the overview documentation.
Set up the .NET binding libraries
The template includes starter .NET for Android and .NET for iOS binding libraries.
Update the binding libraries to reflect the target platforms and .NET version as needed in your .NET app.
Note
For example: If you aim to create only an iOS binding using .NET 9, you can:
- Delete the Android binding library at template/android/NewBinding.Android.Binding, and
- Update the target framework in template/macios/NewBinding.MaciOS.Binding/NewBinding.MaciOS.Binding.csproj to be set to
net9.0-ios
.
Set up the native wrapper projects and libraries
The template also includes starter Android Studio projects and Xcode projects.
Update the native projects to reflect the target platforms and versions as needed in your .NET app, and include the native libraries of interest with the following steps.
Setup: iOS & Mac Catalyst
The Xcode project is located at template/macios/native/NewBinding.
Update the Xcode project to reflect the target platforms and versions as supported in your .NET app. In the Xcode project, click on the top-level framework, and in Targets > General:
- Add/remove any Support Destinations as needed.
- Adjust the iOS version as needed.
Bring in the native library for iOS and/or MacCatalyst into your Xcode project, through whatever method works best for your library and your needs (e.g., CocoaPods, Swift Package Manager).
Setup: Android
The Android Studio project is located at template/android/native.
Update the Android Studio project to reflect the target versions supported in your .NET app.
- Navigate to the build.gradle.kts (:app) file
- Update the
compileSdk
version as needed
Bring in the Android native library through gradle
- Add the package dependency in the dependencies block of the build.gradle.kts (:app) file.
- Add the repository to the
dependencyResolutionManagement
repositories
block in the settings.gradle.kts file. - Sync project with gradle files (via the button in the upper right corner of Android Studio).
Create the API interface
Create the API interface between your native projects and your .NET binding projects with the following steps.
API Definition: iOS & Mac Catalyst
On the native side, make updates in template/macios/native/NewBinding/NewBinding/DotnetNewBinding.swift:
- Add an import statement to import the native library you just added.
- Write the API definitions for the native library APIs of interest.
- Ensure the Xcode project builds successfully and you are satisfied with the APIs.
Back on the .NET side, we are now ready to interop with the native library:
- Run
dotnet build
from template/macios/NewBinding.MaciOS.Binding to test everything is plugged in correctly and good to go. - Use objective sharpie to generate the C# bindings for your Swift API updates:
- Navigate to template/macios/NewBinding.MaciOS.Binding/bin/Debug/net9.0-ios/NewBinding.MaciOS.Binding.resources/NewBindingiOS.xcframework/ios-arm64/NewBinding.framework in your MaciOS binding projects output folder.
- Run
sharpie xcode -sdks
to get a list of valid target SDK values for the bind command. Select the value that aligns with the platform and version you are targeting to use with the next command, for exampleiphoneos18.0
. - Run
sharpie bind
against the header files in the xcframework created by the binding project:sharpie bind --output=sharpie-out --namespace=NewBindingMaciOS --sdk=iphoneos18.0 --scope=Headers Headers/NewBinding-Swift.h
- Update the contents of template/macios/NewBinding.MaciOS.Binding/ApiDefinition.cs by replacing it with the contents of template/macios/NewBinding.MaciOS.Binding/bin/Debug/net9.0-ios/NewBinding.MaciOS.Binding.resources/NewBindingiOS.xcframework/ios-arm64/NewBinding.framework/sharpie-out/ApiDefinitions.cs and tweaking as desired (e.g. naming).
- Run
dotnet build
from template/macios/NewBinding.MaciOS.Binding again.
See also the objective-sharpie documentation to learn more about this tool.
API Definition: Android
On the native side, make updates in template/android/native/newbinding/src/main/java/com/example/newbinding/DotnetNewBinding.java:
- Add an import statement to import the native library you just added.
- Write the API definitions for the native library APIs of interest.
- Ensure the Android Studio project builds successfully and you are satisfied with the APIs.
Back on the .NET side, we are now ready to interop with the native library:
- Run
dotnet build
from template/android/NewBinding.Android.Binding to test everything is plugged in correctly and good to go. (Note: This step will require that you have JDK 17 installed) - Reference any Android binding dependencies by adding an
@(AndroidMavenLibrary)
item to template/sample/MauiSample.csproj for each maven dependency being bound in your native Android project. This will enable Java dependency verification for your project and cause subsequent builds to produce build warnings or errors for missing dependencies. You can address these warnings/errors by adding@(AndroidMavenLibrary)
or@(PackageReference)
elements as suggested to satisfy the java dependency chain for the native library you are binding. (Note: The gradle/maven dependencies often need to be explicitly referenced, as they are not automatically bundled into your library.)
<ItemGroup Condition="$(TargetFramework.Contains('android'))">
<AndroidMavenLibrary Include="{DependencyGroupId}:{DependencyName}" Version="{DependencyVersion}" Bind="false" />
</ItemGroup>
See also the AndroidMavenLibrary reference and Java dependency verification documentation for more information about this process.
Note
You can rename the placeholder DotnetNewBinding
class to better reflect the native library being wrapped. For more examples and tips for writing the API definitions, read more in the section below: Modify an existing binding.
Consume the APIs in your .NET app
The template includes a .NET MAUI sample app at template/sample/MauiSample, which references the .NET binding projects and makes the native libraries immediately ready to use!
If you are interested in using your own .NET MAUI, .NET for Android, .NET for iOS, and/or .NET for Mac Catalyst apps, however, you may do so by modifying your .NET app project files to reference the binding libraries:
<!-- Reference to MaciOS Binding project -->
<ItemGroup Condition="$(TargetFramework.Contains('ios')) Or $(TargetFramework.Contains('maccatalyst'))">
<ProjectReference Include="..\..\macios\NewBinding.MaciOS.Binding\NewBinding.MaciOS.Binding.csproj" />
</ItemGroup>
<!-- Reference to Android Binding project -->
<ItemGroup Condition="$(TargetFramework.Contains('android'))">
<ProjectReference Include="..\..\android\NewBinding.Android.Binding\NewBinding.Android.Binding.csproj" />
</ItemGroup>
Modify an existing binding
If the existing API surface doesn't expose the functionality you need in your own project, it's time to make your own modifications!
iOS & Mac Catalyst
Inside the Xcode project, you will find one or more Swift files which define the public API surface for the binding. For example, the register
method for Firebase Messaging is defined as:
@objc(MauiFIRMessaging)
public class MauiFIRMessaging : NSObject {
@objc(register:completion:)
public static func register(apnsToken: NSData, completion: @escaping (String?, NSError?) -> Void) {
let data = Data(referencing: apnsToken);
Messaging.messaging().apnsToken = data
Messaging.messaging().token(completion: { fid, error in
completion(fid, error as NSError?)
})
}
// ...
}
Note
Native wrapper API types which will be used by the .NET Binding must be declared as public
and need to be annoted with @objc(NameOfType)
and methods also need to be public
, and can also benefit from similar annotations @objc(methodName:parameter1:)
where the name and parameters are specified which help influence the binding which objective sharpie will generate.
You can see in this method that the public API surface only uses types which .NET for iOS is already aware of: NSData
, String
, NSError
and a callback.
In the Firebase.MaciOS.Binding
project, the ApiDefinitions.cs file contains the binding definition for this native wrapper API:
using System;
using Foundation;
namespace Firebase
{
// @interface MauiFIRMessaging : NSObject
[BaseType (typeof(NSObject))]
interface MauiFIRMessaging
{
[Static]
[Export ("register:completion:")]
[Async]
void Register (NSData apnsToken, Action<string?, NSError?> completion);
// ...
}
Let's say you want to add a method for unregistering. The Swift code would look something like this:
@objc(unregister:)
public static func unregister(completion: @escaping (NSError?) -> Void) {
// need delegate to watch for fcmToken updates
Messaging.messaging().deleteToken(completion: { error in
completion(error as NSError?)
})
}
The other half will be to update the ApiDefinitions.cs file in the binding project to expose this new method. There are two ways you can go about this:
- You can manually add the required code
- After building the binding project, you can run the objective sharpie tool to generate an ApiDefinitions.cs file. You can try to find the relevant changes from this file and copy them over manually, or try copying over the whole file and looking at the diff to find the part you need.
In this case, the changes to ApiDefinitions.cs would be:
[Static]
[Export("unregister:")]
[Async]
void UnRegister(Action completion);
Once you've made these changes, you can rebuild the Binding project, and the new API will be ready to use from your .NET MAUI project.
Note
Binding projects for Mac/iOS are not using source generators, and so the project system and intellisense may not know about the new API's until you've rebuilt the binding project, and reloaded the solution so that the project reference picks up the newer assembly. Your app project should still compile regardless of intellisense errors.
Android
Inside the Android Studio project you will find a module directory which contains a java file definining the public API surface for the binding. For example, the initialize
method for Facebook is defined as below:
package com.microsoft.mauifacebook;
import android.app.Activity;
import android.app.Application;
import android.os.Bundle;
import android.util.Log;
import com.facebook.LoggingBehavior;
import com.facebook.appevents.AppEventsLogger;
public class FacebookSdk {
static AppEventsLogger _logger;
public static void initialize(Activity activity, Boolean isDebug) {
Application application = activity.getApplication();
if (isDebug) {
com.facebook.FacebookSdk.setIsDebugEnabled(true);
}
com.facebook.FacebookSdk.addLoggingBehavior(LoggingBehavior.APP_EVENTS);
AppEventsLogger.activateApp(application);
_logger = AppEventsLogger.newLogger(activity);
}
// ...
}
You can see in this method that the public API surface only uses types which .NET for Android is already aware of: Activity
and Boolean
.
In the Facebook.Android.Binding project, the Transforms/Metadata.xml file contains only some xml to describe how to map the java package name (com.microsoft.mauifacebook
) to a more C# friendly namespace (Facebook
). Generally android bindings are more 'automatic' than Mac/iOS at this point, and you rarely should need to make changes to these transform files.
<metadata>
<attr path="/api/package[@name='com.microsoft.mauifacebook']" name="managedName">Facebook</attr>
</metadata>
Let's say you want to add a method for logging an event. The java code would look something like this:
public static void logEvent(String eventName) {
_logger.logEvent(eventName);
}
From this simple change, binding project requires no updates to the Transforms/Metadata.xml or other files. You can simply rebuild the Binding project, and the new API will be ready to use from your .NET MAUI project.
Note
Binding projects for Android are not using source generators, and so the project system and intellisense may not know about the new API's until you've rebuilt the binding project, and reloaded the solution so that the project reference picks up the newer assembly. Your app project should still compile regardless of intellisense errors.
.NET MAUI Community Toolkit