A Unity sample project demonstrating a clean, maintainable integration of AppLovin MAX for showing ads:
- Multiple Banner, Interstitial, and Rewarded placements
- ScriptableObject profile to configure everything in the Inspector (no code edits)
- Tiny, single-purpose controllers (per ad type) with safe callback unsubscribe
- A button helper that triggers ads by label
Test the app on your own (APK Build 1.2.2)
Watch the app in action (Video Recording)
The project implements all listener and delegate methods for each ad type — even if only for logging — so you can clearly see the lifecycle of ads and how to hook into them.
This project and documentation can be extended upon request. New features like Zenject and Reactive demonstration, automated android version or similar useful implementations can be added.
- Banner – Toggle on/off, positioned via settings.
- Interstitial – Fullscreen ad between content.
- Rewarded A & Rewarded B – Two separate reward scenarios for testing.
All relevant MAX callbacks are implemented and logged:
- SDK
- OnSdkInitializedEvent
 
- Banner
- OnAdLoadedEvent
- OnAdLoadFailedEvent
- OnAdClickedEvent
- OnAdExpandedEvent
- OnAdCollapsedEvent
- OnAdRevenuePaidEvent
 
- Interstitial
- OnAdLoadedEvent
- OnAdLoadFailedEvent
- OnAdDisplayedEvent
- OnAdDisplayFailedEvent
- OnAdClickedEvent
- OnAdHiddenEvent
- OnAdRevenuePaidEvent
 
- Rewarded
- OnAdLoadedEvent
- OnAdLoadFailedEvent
- OnAdDisplayedEvent
- OnAdDisplayFailedEvent
- OnAdClickedEvent
- OnAdHiddenEvent
- OnAdReceivedRewardEvent
- OnAdRevenuePaidEvent
 
- 
Create a Profile Right-click in Project: Create → Ads → MAX Ads Profile 
- 
Add items to: - 
Banners: set label, adUnitId, anchor, background, startHidden 
- 
Interstitials: set label, adUnitId, preloadOnStart 
- 
Rewardeds: set label, adUnitId, preloadOnStart, optional rewardKey (e.g., coins, revive) and fallbackAmount 
 The profile validates duplicates and empty fields in OnValidate (warnings in Console). 
- 
- 
Drop a Bootstrap Prefab in Your First Scene - Create an empty GameObject called MaxAdsBootstrap
 
- 
Add: - 
MaxSdkBootstrap (SDK init; optional verbose logging and test GAID) 
- 
MaxAds (assign your MaxAdsProfile) 
 Both components are designed to live across scenes (via DontDestroyOnLoad [Singleton] in scripts). 
- 
- 
Add Buttons (Optional) Add a Button and attach MaxAdsButton - Set Action (e.g., ShowRewarded), Label (must match a profile label), and (optionally) drag a reference to your MaxAds (or let it auto-find).
- This script is helpful for you to check how to call the desired ads in an event driven way.
 
That’s it—press Play and use the buttons. Logs will show lifecycle events.
- 
Banners MaxAds.ShowBanner("Home_Banner_1"); MaxAds.HideBanner("Home_Banner_1"); MaxAds.ToggleBanner("Home_Banner_1");
- 
Interstitials MaxAds.ShowInterstitial("LevelEnd_Interstitial"); MaxAds.PreloadInterstitial("LevelEnd_Interstitial");
- 
Rewardeds MaxAds.ShowRewarded("Revive_Rewarded"); MaxAds.PreloadRewarded("Revive_Rewarded");
### Reward Routing (Optional, Built-In)
MaxAds listens to each RewardedAds controller’s OnRewardGranted event and calls a tiny router:
If rewardKey == "coins" → ResourceA.ReceiveReward(fallbackAmount)
If rewardKey == "revive" → ResourceB.ReceiveReward(fallbackAmount)
- 
BannerAds Creates banner with AdViewConfiguration (anchor / background), safe unsubscribe of callbacks - Show,
- Hide,
- Toggle,
 
- 
InterstitialAds - Initialize,
- Preload,
- Show
 
Exponential backoff retry on load failures (1→64s)
Re-preloads after Hidden / DisplayFailed
- RewardedAds
Same lifecycle as Interstitials
- Raises OnRewardGranted(label) on OnAdReceivedReward
All controllers use named handlers for MAX callbacks and unsubscribe in OnDestroy, preventing event buildup after script reloads.
- 
- MaxSdkBootstrapruns at startup.
 
- 
- MaxAdscreates and initializes controllers for- Banner,
- Interstitial,
- Rewarded A, and Rewarded B.
 
- Exposes simple methods:
- ToggleBanner(),
- ShowInterstitial(),
- ShowRewardedA(),
- ShowRewardedB().
 
 
- 
Controllers - Each ad type has its own controller:
- Subscribes to all MAX events and logs them.
- Includes retry logic for failed loads (Interstitial & Rewarded).
- Banner appearance and placement is configurable via MaxAdProfile.
 
- 
Rewards - ResourcesController listens to OnAdReceivedRewardEvent.
- Matches ad unit IDs to ResourceA or ResourceB.
- Grants rewards (with fallback amounts if reward.Amountis 0).
- ResourceLabel updates UI automatically when resource values change.
 
- ResourcesController listens to 
- 
UI Binding - AdButtonsBinder connects Unity UI Buttons to ad service methods once MaxAdsServiceis ready.
- MediationButton opens the Mediation Debugger.
 
- AdButtonsBinder connects Unity UI Buttons to ad service methods once 
- Mediation Debugger – Use the “Open Mediation Debugger” button to inspect network setup.
- Verbose Logging – Toggle in MaxInitializerto see detailed logs.
- Event Logs – All ad lifecycle events are logged with [Banner],[Interstitial],[Rewarded]tags.
There is no guard to protect the current Ad Service implementation from pausing/unfocusing the app. There may occur unwanted issues accordingly.
This repo includes automated versioning and continuous integration setup, so commit on main branch triggers version bumping without the need for manual actions.
- Automated Versioning – Uses semantic-release to bump version numbers, generate changelogs, and tag releases based on commit history.
- Why it matters – Ensures that demo builds shared with clients are always reproducible, versioned, and match the code in the repo.
This sample is for demonstration purposes.
Logo / Branding Notice:
The application icon and any sample branding used in this demo do not belong to me. They are included solely to improve UX and presentation for demonstration purposes. All trademarks, service marks, and logos remain the property of their respective owners. No affiliation or endorsement is implied. If you are a rights holder and would like any asset removed, please contact me and I will take it down immediately.
Developer: Alp Kurt, Berlin, Germany
If you have questions about this demo or the implementation details, feel free to reach out.
