AMDevIT.Admob.Wrapper
0.0.1.1-alpha
dotnet add package AMDevIT.Admob.Wrapper --version 0.0.1.1-alpha
NuGet\Install-Package AMDevIT.Admob.Wrapper -Version 0.0.1.1-alpha
<PackageReference Include="AMDevIT.Admob.Wrapper" Version="0.0.1.1-alpha" />
<PackageVersion Include="AMDevIT.Admob.Wrapper" Version="0.0.1.1-alpha" />
<PackageReference Include="AMDevIT.Admob.Wrapper" />
paket add AMDevIT.Admob.Wrapper --version 0.0.1.1-alpha
#r "nuget: AMDevIT.Admob.Wrapper, 0.0.1.1-alpha"
#:package AMDevIT.Admob.Wrapper@0.0.1.1-alpha
#addin nuget:?package=AMDevIT.Admob.Wrapper&version=0.0.1.1-alpha&prerelease
#tool nuget:?package=AMDevIT.Admob.Wrapper&version=0.0.1.1-alpha&prerelease
AMDev.IT AdMob Wrapper
A modern, lightweight AdMob wrapper for .NET 10 Android, .NET 10 iOS, and .NET MAUI, designed to solve the lack of working AdMob bindings in the current .NET ecosystem.
Why this exists: As of 2025, there are no fully functional, up-to-date AdMob bindings for .NET MAUI. The official
Xamarin.GooglePlayServices.Adsbinding has broken classes (InterstitialAdLoadCallback.OnAdLoadedand others), and the iOS binding repository has been archived with no replacement. This library wraps the native Android and iOS SDKs in clean, bindable Kotlin and Swift layers and exposes them to .NET with both callback andasync/awaitAPIs.
Packages
Requirements
- .NET 10
- Android API 33+ (Android 13)
- iOS 15.0+
Xamarin.GooglePlayServices.Ads125.0.0.1
Installation
Android project
<PackageReference Include="AMDevIT.Admob.Wrapper.Android" Version="1.0.0" />
iOS project
<PackageReference Include="AMDevIT.Admob.Wrapper.iOS" Version="1.0.0" />
Android or iOS project with async/await support
<PackageReference Include="AMDevIT.Admob.Wrapper" Version="1.0.0" />
MAUI project with async/await support and XAML controls
<PackageReference Include="AMDevIT.Admob.Wrapper" Version="1.0.0" />
<PackageReference Include="AMDevIT.Admob.Wrapper.MAUICross" Version="1.0.0" />
AndroidManifest.xml
Add your AdMob App ID inside the <application> tag:
<application ...>
<meta-data
android:name="com.google.android.gms.ads.APPLICATION_ID"
android:value="ca-app-pub-XXXXXXXXXXXXXXXX~YYYYYYYYYY" />
</application>
Info.plist (iOS)
Add your AdMob App ID:
<key>GADApplicationIdentifier</key>
<string>ca-app-pub-XXXXXXXXXXXXXXXX~YYYYYYYYYY</string>
For testing, use the official Google test App ID:
ca-app-pub-3940256099942544~3347511713
Ad formats supported
| Format | Android class | iOS class |
|---|---|---|
| Banner | BannerAdWrapper |
BannerAdWrapper |
| Interstitial | InterstitialAdWrapper |
InterstitialAdWrapper |
| Rewarded | RewardedAdWrapper |
RewardedAdWrapper |
| App Open | AppOpenAdWrapper |
AppOpenAdWrapper |
Usage — Android
Initialization
Callback style
AdMobManager.Instance.Initialize(this, new MyInitListener());
private class MyInitListener : Java.Lang.Object, IOnInitializedListener
{
public void OnInitialized()
{
// SDK ready, load ads
}
public void OnInitializationFailed(string error)
{
Console.WriteLine($"AdMob init failed: {error}");
}
}
Async style
await AdMobManager.Instance.InitializeAsync(this);
Banner Ad
Callback style
var bannerWrapper = new BannerAdWrapper(this);
var adView = bannerWrapper.Load(
adUnitId: "ca-app-pub-3940256099942544/6300978111",
loadListener: new MyBannerLoadListener()
);
bannerContainer.AddView(adView);
private class MyBannerLoadListener : Java.Lang.Object, IOnAdLoadedListener
{
public void OnAdLoaded() => Console.WriteLine("Banner loaded");
public void OnAdFailedToLoad(int errorCode, string errorMessage) =>
Console.WriteLine($"Banner failed: [{errorCode}] {errorMessage}");
}
Async style
var bannerWrapper = new BannerAdWrapper(this);
var adView = await bannerWrapper.LoadAsync("ca-app-pub-3940256099942544/6300978111");
bannerContainer.AddView(adView);
Interstitial Ad
Callback style
var interstitialWrapper = new InterstitialAdWrapper(this);
interstitialWrapper.Load(
adUnitId: "ca-app-pub-3940256099942544/1033173712",
loadListener: new MyLoadListener(),
eventListener: new MyEventListener()
);
if (interstitialWrapper.IsLoaded())
interstitialWrapper.Show(this);
private class MyLoadListener : Java.Lang.Object, IOnAdLoadedListener
{
public void OnAdLoaded() => Console.WriteLine("Interstitial loaded");
public void OnAdFailedToLoad(int errorCode, string errorMessage) =>
Console.WriteLine($"Interstitial failed: [{errorCode}] {errorMessage}");
}
private class MyEventListener : Java.Lang.Object, IOnAdEventListener
{
public void OnAdShown() => Console.WriteLine("Interstitial shown");
public void OnAdDismissed() => Console.WriteLine("Interstitial dismissed");
public void OnAdClicked() => Console.WriteLine("Interstitial clicked");
public void OnAdImpression() => Console.WriteLine("Interstitial impression");
public void OnAdFailedToShow(int errorCode, string errorMessage) =>
Console.WriteLine($"Interstitial show failed: [{errorCode}] {errorMessage}");
}
Async style
var interstitialWrapper = new InterstitialAdWrapper(this);
await interstitialWrapper.LoadAsync("ca-app-pub-3940256099942544/1033173712");
interstitialWrapper.Show(this);
Note: Interstitial ads are one-shot. Once dismissed, you need to call
Loadagain before showing. This is by design — it gives you full control over which Ad Unit ID to use on the next load.
Rewarded Ad
Callback style
var rewardedWrapper = new RewardedAdWrapper(this);
rewardedWrapper.Load(
adUnitId: "ca-app-pub-3940256099942544/5224354917",
loadListener: new MyLoadListener()
);
if (rewardedWrapper.IsLoaded())
rewardedWrapper.Show(this, new MyRewardListener());
private class MyRewardListener : Java.Lang.Object, IOnRewardEarnedListener
{
public void OnRewardEarned(string type, int amount) =>
Console.WriteLine($"Reward earned: {amount} {type}");
}
Async style
var rewardedWrapper = new RewardedAdWrapper(this);
await rewardedWrapper.LoadAsync("ca-app-pub-3940256099942544/5224354917");
var (type, amount) = await rewardedWrapper.ShowAsync(this);
Console.WriteLine($"Reward earned: {amount} {type}");
App Open Ad
Callback style
var appOpenWrapper = new AppOpenAdWrapper(this);
appOpenWrapper.Load(
adUnitId: "ca-app-pub-3940256099942544/9257395921",
loadListener: new MyLoadListener(),
eventListener: new MyEventListener()
);
if (appOpenWrapper.IsLoaded() && !appOpenWrapper.IsShowing())
appOpenWrapper.Show(this);
Async style
var appOpenWrapper = new AppOpenAdWrapper(this);
await appOpenWrapper.LoadAsync("ca-app-pub-3940256099942544/9257395921");
if (!appOpenWrapper.IsShowing())
appOpenWrapper.Show(this);
Usage — iOS
Initialization
Callback style
AdMobManager.Instance.InitializeWithViewController(this, new MyInitListener());
private class MyInitListener : NSObject, IOnInitializedListener
{
public void OnInitialized()
{
// SDK ready, load ads
}
public void OnInitializationFailedWithError(string error)
{
Console.WriteLine($"AdMob init failed: {error}");
}
}
Async style
await AdMobManager.Instance.InitializeAsync(this);
Banner Ad
Callback style
var bannerWrapper = new BannerAdWrapper();
var adView = bannerWrapper.LoadWithAdUnitId(
adUnitId: "ca-app-pub-3940256099942544/6300978111",
viewController: this,
loadListener: new MyBannerLoadListener(),
eventListener: null
);
bannerContainer.AddSubview(adView);
private class MyBannerLoadListener : NSObject, IOnAdLoadedListener
{
public void OnAdLoaded() => Console.WriteLine("Banner loaded");
public void OnAdFailedToLoadWithErrorCode(nint errorCode, string errorMessage) =>
Console.WriteLine($"Banner failed: [{errorCode}] {errorMessage}");
}
Async style
var bannerWrapper = new BannerAdWrapper();
var adView = await bannerWrapper.LoadAsync("ca-app-pub-3940256099942544/6300978111", this);
bannerContainer.AddSubview(adView);
Interstitial Ad
Callback style
var interstitialWrapper = new InterstitialAdWrapper();
interstitialWrapper.LoadWithAdUnitId(
adUnitId: "ca-app-pub-3940256099942544/1033173712",
loadListener: new MyLoadListener(),
eventListener: new MyEventListener()
);
if (interstitialWrapper.IsLoaded)
interstitialWrapper.ShowWithViewController(this);
Async style
var interstitialWrapper = new InterstitialAdWrapper();
await interstitialWrapper.LoadAsync("ca-app-pub-3940256099942544/1033173712");
interstitialWrapper.ShowWithViewController(this);
Rewarded Ad
Callback style
var rewardedWrapper = new RewardedAdWrapper();
rewardedWrapper.LoadWithAdUnitId(
adUnitId: "ca-app-pub-3940256099942544/5224354917",
loadListener: new MyLoadListener(),
eventListener: null
);
if (rewardedWrapper.IsLoaded)
rewardedWrapper.ShowWithViewController(this, new MyRewardListener());
Async style
var rewardedWrapper = new RewardedAdWrapper();
await rewardedWrapper.LoadAsync("ca-app-pub-3940256099942544/5224354917");
var (type, amount) = await rewardedWrapper.ShowAsync(this);
Console.WriteLine($"Reward earned: {amount} {type}");
App Open Ad
Callback style
var appOpenWrapper = new AppOpenAdWrapper();
appOpenWrapper.LoadWithAdUnitId(
adUnitId: "ca-app-pub-3940256099942544/9257395921",
loadListener: new MyLoadListener(),
eventListener: null
);
if (appOpenWrapper.IsLoaded && !appOpenWrapper.IsShowing)
appOpenWrapper.ShowWithViewController(this);
Async style
var appOpenWrapper = new AppOpenAdWrapper();
await appOpenWrapper.LoadAsync("ca-app-pub-3940256099942544/9257395921");
if (!appOpenWrapper.IsShowing)
appOpenWrapper.ShowWithViewController(this);
Usage — MAUI (AMDevIT.Admob.Wrapper.MAUICross)
Setup
Register the handler in MauiProgram.cs:
builder.UseAdMobWrapper();
Banner Ad in XAML
<?xml version="1.0" encoding="utf-8" ?>
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:admob="clr-namespace:AMDevIT.Admob.Wrapper.MAUICross;assembly=AMDevIT.Admob.Wrapper.MAUICross"
x:Class="YourApp.MainPage">
<Grid RowDefinitions="*, Auto">
<Label Grid.Row="0" Text="Hello MAUI!" />
<admob:BannerAd Grid.Row="1"
AdUnitId="ca-app-pub-3940256099942544/6300978111"
AdSize="Adaptive"
AdLoaded="OnBannerLoaded"
AdFailed="OnBannerFailed" />
</Grid>
</ContentPage>
private void OnBannerLoaded(object sender, EventArgs e)
{
Console.WriteLine("Banner loaded");
}
private void OnBannerFailed(object sender, BannerAdFailedEventArgs e)
{
Console.WriteLine($"Banner failed: [{e.ErrorCode}] {e.ErrorMessage}");
}
Banner Ad sizes
| Value | Description |
|---|---|
Adaptive |
Adapts to the container width (default) |
Banner |
Standard 320x50 |
LargeBanner |
320x100 |
MediumRectangle |
300x250 |
FullBanner |
468x60 |
Leaderboard |
728x90 |
Error handling
All async methods throw AdException on failure:
try
{
await AdMobManager.Instance.InitializeAsync(this);
var adView = await bannerWrapper.LoadAsync(adUnitId);
bannerContainer.AddView(adView);
}
catch (AdException ex)
{
Console.WriteLine($"AdMob error [{ex.ErrorCode}]: {ex.Message}");
}
Test Ad Unit IDs
Use these IDs during development. Never use real Ad Unit IDs on a device you own.
| Format | Test Ad Unit ID |
|---|---|
| App Open | ca-app-pub-3940256099942544/9257395921 |
| Banner | ca-app-pub-3940256099942544/6300978111 |
| Interstitial | ca-app-pub-3940256099942544/1033173712 |
| Rewarded | ca-app-pub-3940256099942544/5224354917 |
| Rewarded Interstitial | ca-app-pub-3940256099942544/5354046379 |
| Native | ca-app-pub-3940256099942544/2247696110 |
Project structure
AMDevIT.Admob.Wrapper/
├── native/
│ ├── android/ # Kotlin source (Android Studio)
│ │ └── admob-wrapper/
│ │ ├── AdMobManager.kt
│ │ └── ads/
│ │ ├── BannerAdWrapper.kt
│ │ ├── InterstitialAdWrapper.kt
│ │ ├── RewardedAdWrapper.kt
│ │ └── AppOpenAdWrapper.kt
│ └── ios/ # Swift source (Xcode)
│ ├── build_xcframework.sh
│ └── AdMobWrapper/
│ ├── AdMobManager.swift
│ └── Ads/
│ ├── BannerAdWrapper.swift
│ ├── InterstitialAdWrapper.swift
│ ├── RewardedAdWrapper.swift
│ └── AppOpenAdWrapper.swift
└── dotnet/
├── AMDevIT.Admob.Wrapper.Android/ # .NET binding project (Android)
├── AMDevIT.Admob.Wrapper.iOSNative/ # .NET binding project (iOS)
├── AMDevIT.Admob.Wrapper/ # Multi-platform wrapper + async extensions
├── AMDevIT.Admob.Wrapper.MAUICross/ # MAUI controls and handlers (optional)
├── AMDevIT.Admob.Wrapper.DroidTestApp/ # Android test app
└── AMDevIT.Admob.Wrapper.iOSTestApp/ # iOS test app
Notes about building
Android
The native Android SDK is built as an AAR using Gradle. The binding project references both debug and release flavors — the correct one is selected automatically based on the build configuration. When making changes to the native code, recompile the AAR from Android Studio and update the files in AMDevIT.Admob.Wrapper.Android/Jars/.
iOS
The native iOS SDK is built as an xcframework using Xcode. A build script is provided at native/ios/build_xcframework.sh to automate the process. When making changes to the Swift code, run the script from the native/ios/ directory:
./build_xcframework.sh
Then copy the generated build/AdMobWrapper.xcframework to AMDevIT.Admob.Wrapper.iOSNative/native/.
Contributing
Contributions are welcome. Please open an issue before submitting a pull request for significant changes.
When updating the native Android SDK version:
- Update
playServicesAdsVersioninlibs.versions.toml - Recompile the AAR from Android Studio
- Replace the AARs in
AMDevIT.Admob.Wrapper.Android/Jars/ - Update the
Xamarin.GooglePlayServices.AdsNuGet version accordingly - Bump the package version and publish
When updating the native iOS SDK version:
- Update the SPM dependency version in Xcode
- Run
./build_xcframework.shfromnative/ios/ - Replace the xcframework in
AMDevIT.Admob.Wrapper.iOSNative/native/ - Bump the package version and publish
License
Apache 2.0 License — see LICENSE for details.
This library is not affiliated with or endorsed by Google. AdMob is a trademark of Google LLC.
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net10.0-android36.0 is compatible. net10.0-ios26.0 is compatible. |
-
net10.0-android36.0
- AMDevIT.Admob.Wrapper.Droid (>= 0.0.1-alpha)
-
net10.0-ios26.0
- AMDevIT.Admob.Wrapper.iOSNative (>= 0.0.1.1-alpha)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 0.0.1.1-alpha | 63 | 4/23/2026 |
| 0.0.1-alpha | 71 | 4/21/2026 |
First version that includes Android bindings.