# Superwall Documentation # External Paywall Creation The legacy editor is deprecated. Please visit the docs covering our new [editor](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-overview). The following is only necessary if you're looking to build your own paywall website outside of Superwall. We strongly recommend using the paywall editor or templates instead. Data Tags [#data-tags] Data tags are specific attributes attached to elements, like text, links, or buttons, on the paywall webpage.\ These are recognised by a special script called [Paywall.js](/docs/advanced-paywall-creation#paywalljs) that interprets user feedback and transmits information to and from the SDK. These are built in to the templates we've created and you don't need to worry about them unless you're building your own paywall webpage. When you configure your paywall webpage on the Superwall dashboard, the dashboard instantly recognizes the data tags on your website. From there you can edit the text that appears in the tagged elements. There are seven different types of data tag: | Name | Value | Purpose | | ----------------------- | ----------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | data-pw-var | The reference name of the element, e.g. "Title". | This indicates to Superwall that the element should be remotely configurable. The Superwall dashboard will create an editable text field for this element. The reference name of that text field will be equal to the value you provide. | | data-pw-close | Add anything as a value. It's ignored. | Closes the paywall. | | data-pw-restore | Add anything as a value. It's ignored. | For restoring purchases, if you bought on a different device. | | data-pw-purchase | Either primary, secondary, or tertiary. | This relates to your primary, secondary, or tertiary product set up in the Superwall dashboard respectively. When the tagged element is tapped, it tells the SDK to initialize a purchase of the specified product. | | data-pw-open-url | A valid url. E.g.` .` | Normal href links do not work in a paywall. However, attaching this tag to an element opens the provided url. | | data-pw-open-deep-link | A valid deep link url. E.g. fb://profile/33138223345. | Opens a deeplink from a button on a paywall. | | data-pw-custom | A name for a custom action | When the tagged element is tapped, the SDK calls its delegate method handleCustomPaywallAction(withName:). The value you provide is passed to the function, from which you can perform custom logic. Check out [Custom Paywall Buttons](/docs/sdk/guides/advanced/custom-paywall-actions) for more. | | data-pw-skip-inner-html | true | This prevents an element's text from being edited. It tells Paywall.js to keep looking into the element's children for more data-pw-var tags. This is useful when you want to hide a whole section or edit the style of a container, rather than edit its text. | Here are a few examples of what data tags look like in the webpage HTML: ```html HTML

``` Paywall.js [#paywalljs] Paywall.js is a javascript utility that recognizes data tags and acts as the interface between the Superwall client SDK and the webpage presented to the user. It does a few fancy tricks to make the HTML feel native, but it's main purpose is to interpret user feedback and transmit information to and from the SDK. In addition, when you configure a paywall on the Superwall dashboard, the dashboard instantly recognizes the data tags you’ve provided. From there, you can edit the text of all tagged items. To get your webpage to be operable with Superwall, you need to add the Paywall.js script to the header of your site. We recommend using [Webflow](https://webflow.com). To add custom scripts to a Webflow webpage you need to have a paid Webflow account. However, our [clonable Webflow project](https://webflow.com/website/45-Paywall-Elements?ref=showcase-search\&searchValue=superwall) already includes this script. Therefore, we recommend using the cloned project as the basis of your paywalls. In this project, we've also removed the Webflow watermark that is attached to all free Webflow webpages. If you'd like to add this script yourself, open the **Pages panel**, select the **cog wheel** next to the page name, then scroll down to the **Custom Code** section. In the **Inside head tag** section, add the following code: ```html HTML ``` Do not remove async Paywall.js is available to Superwall customers subject to the terms of the subscription agreement between Superwall and the customer, and is not available as an open-source project. Game Controller [#game-controller] Paywall.js now supports game controller input so users can make purchases while maintaining the native game controller experience. Basic Setup [#basic-setup] To attach an on-screen button to a game controller button, simply add the `data-pw-gc-button` tag to the element. The value of this tag must match the [unmappedLocalizedName](https://developer.apple.com/documentation/gamecontroller/gccontrollerelement/3681941-unmappedlocalizedname) of the game controller button in iOS. For example, the following button would purchase the primary product if a user pressed the "A" button on their game controller. ```html HTML
Purchase
``` Styling [#styling] To allow users to style buttons, we will add and remove classes during the button press event. When the game controller input becomes depressed, we add `pw-gc-press`. When the button is let up, we will remove that class. Many users will use these classes to slightly shirk the button on press and restore it on press end. ```css CSS .pw-gc-press { opacity: 0.9; transition: all .2s ease-in-out; transform: scale(0.93); } ``` Advanced [#advanced] Some users may want to take other actions based on the game controller input so we will automatically forward all game controller events to a function on the window if it is defined. ```typescript TypeScript type DirectionalInput = { controller_element: string directional: true x: number y: number } type NonDirectionalInput = { controller_element: string directional: false value: number } export type Input = | DirectionalInput | NonDirectionalInput window.onGameControllerInput = (data: Input) => { // Do anything you want with the input! } ``` # Changelog 2.7.14 [#2714] Fixes [#fixes] * Fixes test mode products not being loaded properly in PW * Improve exponential backoff retry for Play Service unavailable * Fix Custom Info date serialization issues 2.7.13 [#2713] Fixes [#fixes-1] * Fix `device.appVersionPadded` and `device.sdkVersionPadded` emitting non-ASCII digits on devices whose default locale uses a non-Latin numbering system (e.g. `ar-EG`, `fa-IR`, `bn-BD`), which caused audience-rule version comparisons to misbucket affected users. * Ensures timeout applies to HttpUrlConnection for enrichment and subscription API's * Remove unnecessary sync access causing ANR lock in React Native 2.7.12 [#2712] Enhancements [#enhancements] * Add customer info to paywall info and tracked events * Add stripe/paddle intro offer eligibility Fixes [#fixes-2] * Fix dismiss animation for bottom sheets and modals on newer Samsung devices 2.7.11 [#2711] Enhancements [#enhancements-1] * Improved startup performance and reduced blockage of caller threads * Add `PaywallOptions.preloadDeviceOverrides` - this allows you to override `shouldPreload` for different device tiers, i.e. disable it for low-end devices. For more details, see KDoc. Fixes [#fixes-3] * Reduce CookieManager ANR potential * Fix concurrency issues with double onFinished calls being invoked 2.7.10 [#2710] Potentially impactful changes [#potentially-impactful-changes] * Billing errors on register are now wrapped in `SubscriptionStatusTimeout` errors * This is used to clarify that the status is timing out due to billing errors. If you are depending on the `NoPaywallView` to distinguish between these, ensure you are checking for the proper status. Enhancements [#enhancements-2] * Adds onboarding analytics * Adds prioritized preloading support * Improved error handling Fixes [#fixes-4] * Prevents paywalls from dismissing on return from deep link * Fixes deadlock in `SerialTaskManager` * Fix issues with bottom sheet on certain Samsung devices 2.7.9 [#279] Fixes [#fixes-5] * Fix review dialog closing paywall 2.7.8 [#278] Fixes [#fixes-6] * Fix serialization issue with R8 and dates * Fix scope cancellation for test mode 2.7.7 [#277] Enhancements [#enhancements-3] * Adds support for local resource loading in paywalls via `Superwall.instance.localResources` Fixes [#fixes-7] * Fix issues with stripe period types failing to deserialize 2.7.6 [#276] Fixes [#fixes-8] * Fix concurrency issue with early paywall displays and product loading * Improve edge case handling in billing * Improve paywall timeout cases and failAt stamping * Fix issue with param templating in re-presentation * Fix race issue with test mode 2.7.5 [#275] Enhancements [#enhancements-4] * Add appstack integration attribute identifier Fixes [#fixes-9] * Ensure test mode does not interfere with expo * Ensure isActive is properly returned and not calculated via expiration date * Fix potential memory leak when webview crashes * Ensure O(n) cleanup doesn't run multiple times 2.7.4 [#274] Enhancements [#enhancements-5] * Adds support for "Test Mode", which allows you to simulate in-app purchases without involving Google Play. Test Mode can be enabled through the Superwall dashboard by marking specific users as test store users, or activates automatically when a an application ID mismatch is detected or behavior is set to `ALWAYS`. When active, a configuration modal lets you select starting entitlements and override free trial availability. Purchases are simulated with a UI that lets users complete, abandon, or fail transactions, with all purchase events firing normally for end-to-end paywall testing. * Improved logging in web entitlement and entitlement redeeming for easier debugging * Adds active entitlements to subscription status change event * Ensures purchases are always queried with connected clients and retried in background * Adds `expirationDate`, `subscriptionGroupId` and `offerId` to `StoreTransactionType` Fixes [#fixes-10] * Improve rounding to match editor's pricing consistently 2.7.3 [#273] Enhancements [#enhancements-6] * Adds state and params to `PaywallInfo` for `PaywallClose` events 2.7.2 [#272] Fixes [#fixes-11] * Fixes issue with `enableExperimentalDeviceVariables` option causing subscription status sync to fail 2.7.1 [#271] Enhancements [#enhancements-7] * Adds improved constructors for `SuperwallOptions`,`PaywallOptions` and `PaywallPresentationHandler`, allowing a DSL like usage,i.e. ` PaywallPresentationHandler { onPresent { ... } }` * Adds haptic feedback action support Fixes [#fixes-12] * Ensures poster is always visible on video preview * Fix bug with loading not dismissing post purchase if user remains in paywall * Fix warning about old edge-to-edge API usages * Fix shimmerview theme warnings and memory leak 2.7.0 [#270] Enhancements [#enhancements-8] * Enables paywall post-purchase action execution instead of dismissing * Enables triggering custom callback requests from paywall * Adds a new method to PaywallPresentationHandler called onCustomCallback that allows user to handle custom callback requests * Adds retrieving of paywall state inside paywall info * Adds support for new one time purchases with purchase options and offers * Update Superscript to version 1.0.13, find more in the [Superscript changelog](https://github.com/superwall/superscript/releases/tag/1.0.13) Deprecations [#deprecations] * Deprecated `paywallWebviewLoad_timeout` - this event was causing confusion due to its naming, leading to it being deprecated Fixes [#fixes-13] * Fixes late initialization authorization issue for Stripe checkouts * Improves how Shimmer duration is measured * Fixes wrong redemption type being displayed due to integration attributes 2.6.8 [#268] Enhancements [#enhancements-9] * Adds microphone permission Fixes [#fixes-14] * Fixes error when redeeming external purchases 2.6.7 [#267] Enhancements [#enhancements-10] * Adds permission granting and callbacks to/from paywalls * Adds `PaywallPreloadStart` and `PaywallPreloadComplete` events Fixes [#fixes-15] * Fix handling of deep links when paywall is detached * Enables permission granting from paywall and callbacks * Fix crash when handling drawer style paywalls with 100% height 2.6.6 [#266] Enhancements [#enhancements-11] * Add dynamic notification support and scheduling enabling deeper personalization of notifications * Provides a `CustomerInfo` class API, allowing you to observe the customer's purchases and subscription lifecycle via `Superwall.instance.customerInfo` flow * Provides a new delegate method to observe customer info changes - `fun customerInfoDidChange(from: CustomerInfo, to: CustomerInfo)` * Provides a `CustomerInfoDidChange` event to track customer info changes * Overrides `Collection.plus`and `.toSet` methods to ensure our merging methods are used. * Provides a `userAttributesDidChange(newAttributes: Map)` method in Superwall Delegate to track external (i.e. paywall) attribute changesg * Allows triggering a `transaction_abandon` offer on a `paywall_decline` offer and vice-versa, whereas previously it would trigger a presentation error. * Add a `Superwall.teardown` method and `Superwall.instance.refreshConfiguration()` for development with hot-reload based frameworks ⚠️ Warning ⚠️ [#️-warning-️] If you are using a Purchase Controller and web2app or app2web purchases, you will have to update your purchase controller to listen to `Superwall.instance.customerInfo` which will provide you with the relevant web entitlements and call `setSubscriptionStatus` accordingly. 2.6.5 [#265] Dependencies [#dependencies] * Reverts `androidx.lifecycle:lifecycle-runtime-ktx` to 2.8.4 to ensure old Compose BOM compatiblity Enhancements [#enhancements-12] * Improves error messaging in play store errors Fixes [#fixes-16] * Fixes edge case bug with wrong entitlement being matched in cases where product ID's match and base plans differentiate by suffix only * Fixes issue with composable paywall state updates not firing in onAttach 2.6.4 [#264] Enhancements [#enhancements-13] * Improves error and timeout handling * Hardens paywall recreation in case of render process crash 2.6.3 [#263] Enhancements [#enhancements-14] * Adds `productIdentifier` to RedemptionResult's `PaywallInfo` object Fixes [#fixes-17] * Fixes nested scrolling issue in Modal webviews * Removes node removal for `com.google.android.gms.permission.AD_ID` from Manifest * Ensures remote entitlements in the background refresh without feature flags 2.6.2 [#262] Enhancements [#enhancements-15] * Adds `Superwall.instance.consume(purchaseToken)` method to help easily consume in-app purchases Fixes [#fixes-18] * Fixes issue with deeplink params not being handled properly in some cases * Fixes issue with Drawer and Modal displays on Android 14 Samsung devices * Fixes selection issue with some OTP, ensures after consuming the Status is synced 2.6.1 [#261] Enhancements [#enhancements-16] * Enables Stripe and Paddle checkout via in-app payment sheets * Improves product handling and redemption for Stripe and Paddle Fixes [#fixes-19] * Fixes issue with Google's Play Billing library auto-reconnection 2.6.0 ⚠️ [Deprecated] [#260-️-deprecated] Notes [#notes] * This version is deprecated due to discovery of an issue in Play Billing library which could cause runtime issues * Please use version 2.6.1 2.5.8 [#258] Fixes [#fixes-20] * Fix lifetime purchase entitlements not being discovered in some cases on purchase * Fix potential ANR issues where some animations would end up looping over on main thread * Fix webview client not behaving properly when using a resetted paywall 2.6.0-alpha [#260-alpha] * Add app2web support, allowing users to purchase Stripe or Paddle products without leaving your app * Add `PaymentSheet` purchase type enabling quick bottom sheet purchases * Add support for Android app links purchase redeeming 2.5.7 [#257] Fixes [#fixes-21] * Fix `demandScore` and `demandTier` getting removed from some events * Fixes paywall navigation resetting after backgrounding * Removes webview flags which can cause off-screen render issues 2.5.6 [#256] Enhancements [#enhancements-17] * Add support for rerouting back button if enabled in paywall settings * Handled by `SuperwallOptions.PaywallOptions.onBackPressed`, which enables you to consume the back press or let the SDK consume it * Add support for redeeming web entitlements with Paddle Fixes [#fixes-22] * Fix binary .so file regression to ensure 16kb page size compatibility * Fix potential issue with paywall not dismissing due to paywall\_decline concurrency issue 2.5.5 [#255] Enhancements [#enhancements-18] * Expose `signature` on `StoreTransaction` Fixes [#fixes-23] * ⚠️ Important - in the recent versions we have added usage of Google AppSetId and AdID to enable automatic attribution with Google's ad networks. * Due to issues with Google's detection of the usage, they have been removed temporarily and will be added back once the issue is resolved. * As an alternative, you can set those attribution identifiers using `AttributionProvider.GOOGLE_APP_SET | AttributionProvider.GOOGLE_ADS` 2.5.4 [#254] Enhancements [#enhancements-19] * `PaywallComposable` now reapplies theme on system change and `PaywallView` now exposes an `onThemeChanged()` method * `DeepLink` event now exposes query params 2.5.3 [#253] Enhancements [#enhancements-20] * Adds ability to specify a custom height and corner radius for the drawer presentation style. * Adds ability to display a `Popup` presentation style * Adds ability to request reviews from paywall actions, including: * Value `device.reviewRequestCount` that returns total request counts * Method `device.reviewRequestsInHour|Day|Week|Year|reviewRequestsSinceInstall` computed methods for granular targeting * Adds `Superwall.instance.setIntegrationAttributes` method enabling you to set integration identifiers for the users from different platforms (Adjust, Mixpanel, Meta, etc.) * Adds `Superwall.instance.showAlert` method enabling you to easily show an alert over the current paywall * Adds `PaywallOptions.timeoutAfter` to easily control timeout of paywalls when not using fallback loading * Adds `Superwall.instance.setIntegrationAttributes` and `Superwall.instance.integrationAttributes` to set integration identifiers (i.e. Mixpanel, Appsflyer, etc) * Adds user attributes to `TransactionComplete` and `PaywallOpen` events * Adds device ID to device attributes Fixes [#fixes-24] * Fixes memory leak issues in `PaywallBuilder` would keep the paywall view alive in some cases * Fixes issue where some events would be missing properties 2.5.1 [#251] Enhancements [#enhancements-21] * Improves subscribed user experience in cases with slow configuration 2.5.0 [#250] Enhancements [#enhancements-22] * Updates Google Play billing library to v8. Unfortunately, Google has broken backwards compatibility with previous versions, so if you're using the standalone library too ensure it is compatible with v8. * Adds kotlin version to device variables Fixes [#fixes-25] * Reduces noisy logging when product is missing an offer * Ensures that getting experimental properties works for consumable products 2.4.1 [#241] Fixes [#fixes-26] * Google Play Billing integration with newer libraries (such as RC 9.\*) * Reduces noisy logging when product is missing an offer * Ensures that getting experimental properties works for consumable products 2.4.0 [#240] Enhancements [#enhancements-23] * Increases superscript version to 1.0.2 with improved type and null safety 2.3.3 [#233] Fixes [#fixes-27] * Ensure properties are always properly serialized 2.3.2 [#232] Enhancements [#enhancements-24] * Adds new properties to count placement occurrences in specific time: `placementsInHours`, `placementsInDay`, `placementsInWeek`, `placementsInMonth`, `placementsSinceInstall` Fixes [#fixes-28] * Fixes an issue where a redemption could succeed but throw an error * Fixes an issue with receipt manager when there is no purchases and experimental properties are enabled 2.3.1 [#231] Fixes [#fixes-29] * Fixes an issue where entitlements would not be reset on time * Ensures `redeem` is only done on initial config not on config refresh Enhancements [#enhancements-25] * Adds `overrideProductsByName` property to allow globally overriding products on paywalls. This property accepts a map of product names to product identifiers (strings). Local overrides provided via `PaywallOverrides` take precedence over global overrides. * Adds `ProductOverride` sealed class to provide flexible product override handling with support for both product IDs and `StoreProduct` objects. 2.3.0 [#230] Enhancements [#enhancements-26] * Deprecated `Superwall.instance.handleDeepLink` in favor of static `Superwall.handleDeepLink` to ensure links received before `configure` completion are handled properly * Adds `externalAcountId`, provided to Google Play billing upon purchase as a SHA256 of the userId or the userId itself if `passIdentifiersToPlayStore` option is provided. * Adds a `SuperwallOption` named `enableExperimentalDeviceVariables`. When set to true, this enables additional device-level variables: `latestSubscriptionPeriodType`, `latestSubscriptionState`, and `latestSubscriptionWillAutoRenew`. These properties provide information about the most recent Google Play subscription on the device and can be used in audience filters. Note that due to their experimental nature, they are subject to change in future updates. * Update `com.android.billingclient` to version 7.1.1 to align with Google's latest requirements Fixes [#fixes-30] * Fixes issues with paywall destruction when activity performs a hot reload (i.e. during update) * Fixes issue where the feature block would be triggered on non-gated paywalls when the app is minimised 2.2.3 [#223] Fixes [#fixes-31] * Fix potential issue with device enrichment and attributes synchronisation causing a lock 2.2.2 [#222] Enhancements [#enhancements-27] * Adds `demandScore` and `demandTier` to device attributes using an off-device advanced machine learning model. A user is assigned these based on a variety of factors to determine whether they're more or less likely to convert and can be used within audience filters. * Adds `deviceTier` to the device attributes using an on-device scoring system to place the device in a tier based on it's capabilities. This can be used in audience filters to display different paywalls based on the user's device capabilities. The value can be one of the following values `ultraLow`, `low`, `mid`, `high`, `ultra_high`, `unknown`. NOTE: This property is still experimental Fix [#fix] * Fixes potential issue in `web2app` with subscription being overriden by the polling due to extra API calls 2.2.0 [#220] Enhancements [#enhancements-28] * Updates binaries to work on 16kb page sizes 2.1.2 [#212] Fixes [#fixes-32] * Fix issue with deep link referrer throwing a DeadObjectException 2.1.1 [#211] Enhancements [#enhancements-29] * Add optimisticLoading paywall option that hides the shimmer when HTML is loaded * Prevent stopping the paywall handler listening with onDismiss when reason is None * Improve PaywallBuilder API for non-kotlin and non-coroutine users * Expose `deviceAttributes()` function to retrieve session device attributes 2.1.0 [#210] Enhancements [#enhancements-30] * Updates kotlin version to 2.0.21 * Updates `compileSDK` to 35 * Adds web checkout and redemption support * Adds SuperwallDelegate methods `willRedeemLink` and `didRedeemLink` Fixes [#fixes-33] * Removes lock while reading cache that could cause ANR on the main thread * Fixes issue where experiment and variant ID would be missing due to concurrency issues 2.1.0-beta.1 [#210-beta1] Enhancements [#enhancements-31] * Updates kotlin version to 2.0.21 * Updates `compileSDK` to 35 * Adds web checkout and redemption support * Adds SuperwallDelegate methods `willRedeemLink` and `didRedeemLink` 2.0.8 [#208] Fixes [#fixes-34] * Fixes the serialization issue with Kotlin 2.0 2.0.7 [#207] Enhancements [#enhancements-32] * Improves how errors are handled when loading, improving the UX and reloading in real failure cases * Added `device.subscriptionStatus` to the device object Fixes [#fixes-35] * Fixes an issue where users of kotlin 2.0 would experience a `NoAudienceMatch` when evaluating rules 2.0.6 [#206] Fixes [#fixes-36] * Fix potential crash while setting render priority 2.0.5 [#205] Fixes [#fixes-37] * Fix issue with `original_transaction_id` missing when using a `PurchaseController` 2.0.4 [#204] Enhancements [#enhancements-33] * Provide overloads for Java interop * Provide utility functions for Java interop with `PurchaseController` 2.0.3 [#203] Enhancements [#enhancements-34] * Renames `SuperwallPlacement` back to `SuperwallEvent` 2.0.2 [#202] Fixes [#fixes-38] * Fixes issue with `NoAudienceMatch` appearing on some devices and issues with certain campaign rules 2.0.1 [#201] Enhancements [#enhancements-35] * Changes back to `handleSuperwallEvent` naming with a deprecation notice and a typealias for previous methods Fixes [#fixes-39] * Removes extra failure logging when displaying alerts * Finds nearest activity instead of relying just on Context in `PaywallComposable` * Improves cleanup in `PaywallComposable` 2.0.0 [#200] Our 2.0.0 release brings some major and minor changes to both our API's and core features. For more information, please look at our [migration docs](https://superwall.com/docs/migrating-to-v2-android) Enhancements [#enhancements-36] * Adds `PaywallBuilder` class as an alternative to existing `getPaywallView` method. This provides a cleaner API and an ability to change purchase loading bar and shimmer view. * Ensure safety of static webview calls that are known to fail randomly due to Webview's internal issues * Adds `purchase` method to `Superwall` you can use to purchase products without having to resort on paywalls. To purchase a product you can pass it in one of the following objects: * Google Play's `ProductDetails` * Superwall's `StoreProduct` object * Or a string containing the product identifier, i.e. `Superwall.instance.purchase("product_id:base_plan:offer")` * Adds `restorePurchases` method to `Superwall` you can use to handle restoring purchases * Adds `getProducts` method to `Superwall` you can use to retrieve a list of `ProductDetails` given the product ID, i.e. i.e. `Superwall.instance.purchase("product_id:base_plan:offer")` * Adds support for observing purchases done outside of Superwall paywalls. You can now observe purchases done outside of Superwall paywalls by setting the `shouldObservePurchases` option to true and either: * Manually by calling `Superwall.instance.observe(PurchasingObserverState)` or utility methods `Superwall.instance.observePurchaseStart/observePurchaseError/observePurchaseResult` * Automatically by replacing `launchBillingFlow` with `launchBillingFlowWithSuperwall`. This will automatically observe purchases done outside of Superwall paywalls. * Adds consumer proguard rules to enable consumer minification * `Superwall.instance` now provides blocking or callback based version of multiple calls, suffixed with `*Sync` * Improves preloading performance and reduces impact on the main thread * Reduces minSDK to 22 Breaking Changes [#breaking-changes] * `SuperwallPaywallActivity` and `PaywallView` have been moved into `com.superwall.sdk.paywall.view` package from `com.superwall.sdk.paywall.vc` package. * Removes `PaywallComposable` and Jetpack Compose support from the main SDK artifact in favor of `Superwall-Compose` module for Jetpack Compose support: * You can find it at `com.superwall.sdk:superwall-compose:2.0.0-alpha` * Usage remains the same as before, but now you need to include the `superwall-compose` module in your project. * Removed methods previously marked as Deprecated * `SubscriptionStatus.Active` now takes in a set of `Entitlements`, while `Inactive` and `Active` have been turned into objects. * `Superwall.instance.register` now uses `placement` instead of `event` as the argument name * `preloadPaywalls` now uses `placementNames` instead of `eventNames` as the argument name * `PaywallPresentationHandler.onDismiss` now has two arguments, `PaywallInfo` and `PaywallResult` * `PaywallComposable` now uses `placement` argument instead of `event` * `TriggerResult.NoRuleMatch` and `TriggerResult.EventNotFound` have been renamed to `TriggerResult.NoAudienceMatch` and `TriggerResult.PlacementNotFound` * `PresentationResult.NoRuleMatch` and `PresentationResult.EventNotFound` have been renamed to `PresentationResult.NoAudienceMatch` and `PresentationResult.PlacementNotFound` * `SuperwallEvent` has been renamed to `SuperwallPlacement`, belonging properties with `eventName` have been renamed to `placementName` * `SuperwallEventInfo` has been renamed to `SuperwallPlacementInfo` * `ComputedPropertyRequest.eventName` has been renamed to `ComputedPropertyRequest.placementName` * `Superwall.instance.events` has been renamed to `Superwall.instance.placements` * `LogScope.events` has been renamed to `LogScope.placements` * `PaywallPresentationRequestStatusReason.EventNotFound` has been renamed to `PaywallPresentationRequestStatusReason.PlacementNotFound` * `PaywallSkippedReason.EventNotFound` has been renamed to `PaywallSkippedReason.PlacementNotFound` * `SuperwallDelegate.handleSuperwallEvent` method has been renamed to `SuperwallDelegate.handleSuperwallPlacement` * Removed `PurchaseResult.Restored` 2.0.0-beta.5 [#200-beta5] Breaking changes [#breaking-changes-1] * `Superwall.instance.register` now uses `placement` instead of `event` as the argument name * `preloadPaywalls` now uses `placementNames` instead of `eventNames` as the argument name * Superwall's `PaywallPresentationHandler.onDismiss` now has two arguments, `PaywallInfo` and `PaywallResult` * `PaywallComposable` now uses `placement` argument instead of `event` * Remove `PurchaseResult.Restored` 2.0.0-beta.4 [#200-beta4] Breaking changes [#breaking-changes-2] * `SuperwallEvents.entitlementStatusDidChange` has been renamed to `SuperwallEvents.subscriptionStatusDidChange` * `SuperwallDelegate.entitlementStatusDidChange` has been renamed to `SuperwallEvents.entitlementStatusDidChange` * `TriggerResult.NoRuleMatch` and `TriggerResult.EventNotFound` have been renamed to `TriggerResult.NoAudienceMatch` and `TriggerResult.PlacementNotFound` * `PresentationResult.NoRuleMatch` and `PresentationResult.EventNotFound` have been renamed to `PresentationResult.NoAudienceMatch` and `PresentationResult.PlacementNotFound` 2.0.0-beta.3 [#200-beta3] Breaking changes [#breaking-changes-3] * `SuperwallEvent` has been renamed to `SuperwallPlacement`, belonging properties with `eventName` have been renamed to `placementName` * `SuperwallEventInfo` has been renamed to `SuperwallPlacementInfo` * `ComputedPropertyRequest.eventName` has been renamed to `ComputedPropertyRequest.placementName` * `Superwall.instance.events` has been renamed to `Superwall.instance.placements` * `LogScope.events` has been renamed to `LogScope.placements` * `PaywallPresentationRequestStatusReason.EventNotFound` has been renamed to `PaywallPresentationRequestStatusReason.PlacementNotFound` * `PaywallSkippedReason.EventNotFound` has been renamed to `PaywallSkippedReason.PlacementNotFound` * `SuperwallDelegate.handleSuperwallEvent` method has been renamed to `SuperwallDelegate.handleSuperwallPlacement` 2.0.0-beta.2 [#200-beta2] Breaking Changes [#breaking-changes-4] * API Changes: * Migration of `setEntitlementStatus` to `setSubscriptionStatus` * Exposing `Superwall.instance.entitlementsStatus` * Migration of `SuperwallDelegate.entitlementStatusDidChange` to `SuperwallDelegate.subscriptionStatusDidChange` 2.0.0-beta.1 [#200-beta1] Enhancements [#enhancements-37] * Add `PaywallBuilder` class as an alternative to existing `getPaywallView` method. This provides a cleaner API and an ability to change purchase loading bar and shimmer view. * Add callback versions of new 2.0 methods * Ensure safety of static webview calls that are known to fail randomly due to Webview's internal issues 2.0.0-Alpha.1 [#200-alpha1] Breaking Changes [#breaking-changes-5] * `SuperwallPaywallActivity` and `PaywallView` have been moved into `com.superwall.sdk.paywall.view` package from `com.superwall.sdk.paywall.vc` package. * Removes `PaywallComposable` and Jetpack Compose support from the main SDK artifact in favor of `Superwall-Compose` module for Jetpack Compose support: * You can find it at `com.superwall.sdk:superwall-compose:2.0.0-alpha` * Usage remains the same as before, but now you need to include the `superwall-compose` module in your project. * Removed methods previously marked as Deprecated * Removes `SubscriptionStatus`, together with belonging update methods and `subscriptionStatusDidChange` callback. * These are replaced with `EntitlementStatus` and `entitlementStatusDidChange` callback. You can find more details on this migration in our docs. Enhancements [#enhancements-38] * Adds `purchase` method to `Superwall` you can use to purchase products without having to resort on paywalls. To purchase a product you can pass it in one of the following objects: * Google Play's `ProductDetails` * Superwall's `StoreProduct` object * Or a string containing the product identifier, i.e. `Superwall.instance.purchase("product_id:base_plan:offer")` * Adds `restorePurchases` method to `Superwall` you can use to handle restoring purchases * Adds `getProducts` method to `Superwall` you can use to retrieve a list of `ProductDetails` given the product ID, i.e. i.e. `Superwall.instance.purchase("product_id:base_plan:offer")` * Adds support for observing purchases done outside of Superwall paywalls. You can now observe purchases done outside of Superwall paywalls by setting the `shouldObservePurchases` option to true and either: * Manually by calling `Superwall.instance.observe(PurchasingObserverState)` or utility methods `Superwall.instance.observePurchaseStart/observePurchaseError/observePurchaseResult` * Automatically by replacing `launchBillingFlow` with `launchBillingFlowWithSuperwall`. This will automatically observe purchases done outside of Superwall paywalls. * Adds consumer proguard rules to enable consumer minification * Reduces minSDK to 22 * Adds `purchaseToken` to purchase events * `Superwall.instance` now provides blocking or callback based version of multiple calls, suffixed with `*Sync` * Improves preloading performance and reduces impact on the main thread 1.5.5 [#155] Fixes [#fixes-40] * Fixes potential distribution issues for variant selection in edge cases * Fixes potential memory leaks of paywall calling activity 1.5.4 [#154] Fixes [#fixes-41] * Fixes issue when a paywall would dismiss with `ForNextPaywall` but next paywall could not be shown due to triggers or limits. Now it resolves into the proper dismiss status. 1.5.3 [#153] Enhancements [#enhancements-39] * Add `purchaseToken` to TransactionComplete 1.5.2 [#152] Fixes [#fixes-42] * Fix chromium crashes caused by race conditions in webview's implementation 1.5.1 [#151] Enhancements [#enhancements-40] * Updates superscript dependencies to reduce minSDK version Fixes [#fixes-43] * Adds consumer proguard rules to avoid minifying JNA classes during minification 1.5.0 [#150] Enhancements [#enhancements-41] * Adds `shimmerView_start` and `shimmerView_complete` events to track the loading of the shimmer animation. * Makes `hasFreeTrial` match iOS SDK behavior by returning `true` for both free trials and non-free introductory offers * Adds `Superwall.instance.events` - A SharedFlow instance emitting all Superwall events as `SuperwallEventInfo`. This can be used as an alternative to a delegate for listening to events. * Adds a new shimmer animation * Adds support for SuperScript expression evaluator Fixes [#fixes-44] * Fixes concurrency issues with subscriptions triggered in Cordova apps 1.5.0-beta.2 [#150-beta2] Enhancements [#enhancements-42] * Adds `shimmerView_start` and `shimmerView_complete` events to track the loading of the shimmer animation. * Makes `hasFreeTrial` match iOS SDK behavior by returning `true` for both free trials and non-free introductory offers 1.5.0-beta.1 [#150-beta1] Enhancements [#enhancements-43] * Adds `Superwall.instance.events` - A SharedFlow instance emitting all Superwall events as `SuperwallEventInfo`. This can be used as an alternative to a delegate for listening to events. * Adds a new shimmer animation * Adds support for SuperScript expression evaluator 1.4.1 [#141] Enhancements [#enhancements-44] * Adds `appVersionPadded` attribute Fixes [#fixes-45] * Fixes issue where `PaywallPresentationHandler.onError` would be skipped in case of `BillingError`s 1.4.0 [#140] Enhancements [#enhancements-45] * Improves paywall loading and preloading performance * Reduces impact of preloading on render performance * Updates methods to return `kotlin.Result` instead of relying on throwing exceptions * This introduces some minor breaking changes: * `configure` completion block now provides a `Result` that can be used to check for success or failure * `handleDeepLink` now returns a `Result` * `getAssignments` now returns a `Result>` * `confirmAllAssignments` now returns a `Result>` * `getPresentationResult` now returns a `Result` * `getPaywallComponents` now returns a `Result` * Removes Gson dependency * Adds `isScrollEnabled` flag to enable remote control of Paywall scrollability * Adds `PaywallResourceLoadFail` event to enable tracking of failed resources in Paywall * Improves bottom navigation bar color handling Fixes [#fixes-46] * Fixes issue where paywalls without fallback would fail to load and missing resource would cause a failure event * Fixes issue with `trialPeriodDays` rounding to the higher value instead of lower, i.e. where `P4W2D` would return 28 days instead of 30, it now returns 30. * Fixes issue with system navigation bar not respecting paywall color * Fixes issues with cursor allocation in Room transaction * Improves handling of chromium render process crash 1.4.0-beta.3 [#140-beta3] * Fixes issue where paywalls without fallback would fail to load and missing resource would cause a failure event 1.4.0-beta.2 [#140-beta2] Enhancements [#enhancements-46] * Removes Gson dependency * Adds `isScrollEnabled` flag to enable remote controll of Paywall scrollability Fixes [#fixes-47] * Fixes issue with `trialPeriodDays` rounding to the higher value instead of lower, i.e. where `P4W2D` would return 28 days instead of 30, it now returns 30. * Fixes issue with system navigation bar not respecting paywall color * Reduces impact of preloading on render performance * Fixes issues with cursor allocation in Room transaction 1.4.0-beta.1 [#140-beta1] * Updates methods to return `kotlin.Result` instead of relying on throwing exceptions * This introduces some minor breaking changes: * `configure` completion block now provides a `Result` that can be used to check for success or failure * `handleDeepLink` now returns a `Result` * `getAssignments` now returns a `Result>` * `confirmAllAssignments` now returns a `Result>` * `getPresentationResult` now returns a `Result` * `getPaywallComponents` now returns a `Result` 1.3.1 [#131] Fixes [#fixes-48] * Fixes issue when destroying activities during sleep would cause a background crash * Fixes issue when using Superwall with some SDK's would cause a crash (i.e. Smartlook SDK) 1.3.0 [#130] Enhancements [#enhancements-47] * The existing `getPaywall` method has been deprecated and renamed to `getPaywallOrThrow`. The new `getPaywall` method now returns a `kotlin.Result` instead of throwing an exception. * Adds a new option to `SuperwallOptions` - `passIdentifiersToPlayStore` which allows you to pass the user's identifiers (from `Superwall.instance.identify(userId: String, ...)`) to the Play Store when making a purchase. Note: When passing in identifiers to use with the play store, please make sure to follow their \[guidelines]\([https://developer.android.com/reference/com/android/billingclient/api/BillingFlowParams.Builder#setObfuscatedAccountId(java.lang.String)](https://developer.android.com/reference/com/android/billingclient/api/BillingFlowParams.Builder#setObfuscatedAccountId\(java.lang.String\)). * Adds `Superwall.instance.confirmAllAssignments()`, which confirms assignments for all placements and returns an array of all confirmed experiment assignments. Note that the assignments may be different when a placement is registered due to changes in user, placement, or device parameters used in audience filters. Fixes [#fixes-49] * Fixes issues with Paywall sometimes not displaying when returning from background * Fixes issue with SDK crashing when WebView is not available * Fixes issue with `SuperwallPaywallActivity` NPE * Update visibility of internal `getPaywall` methods to `internal` to prevent misuse 1.2.9 [#129] Fixes [#fixes-50] * Fixes issues with `MODAL` presentation style and scrolling containers * Fixes issues with `FULLSCREEN` presentation style rendering behind navigation 1.2.8 [#128] Fixes [#fixes-51] * Fixes issues with Paywall presentation styles not being properly passed 1.2.7 [#127] Enhancements [#enhancements-48] * Exposes current configuration status via `Superwall.instance.configurationStatus` Fixes [#fixes-52] * Fixes issues with Paywall previews not loading 1.2.6 [#126] Fixes [#fixes-53] * Fixes issue where the paywall would not show in some cases when using `minutes_since` * Fixes issue with wrong URL being logged when a paywall fails to load 1.2.5 [#125] Enhancements [#enhancements-49] * Adds a `Modifier` to `PaywallComposable` to allow for more control * Adds a `PaywallView.setup(...)` method to allow for easy setup when using `PaywallView` directly * Adds support for `MODAL` presentation style Fixes [#fixes-54] * Fixes issue with displaying `PaywallComposable` * Resolves issue where users would get `UninitializedPropertyAccessException` when calling `Superwall.instance` 1.2.4 [#124] Enhancements [#enhancements-50] * For users who are not able to upgrade their AGP or Gradle versions, we have added a new artifact `superwall-android-agp-7` which keeps compatibility. Enhancements [#enhancements-51] * Fixes issue with decoding custom placements from paywalls. 1.2.3 [#123] Enhancements [#enhancements-52] * Expose `placementName`, `paywallInfo` and `params` on a `custom_placement` event 1.2.2 [#122] Enhancements [#enhancements-53] * Adds support for multiple paywall URLs, in case one CDN provider fails. * `ActivityEncapsulatable` now uses a WeakReference instead of a reference * SW-2900: Adds Superwall.instance.localeIdentifier as a convenience variable that you can use to dynamically update the locale used for evaluating rules and getting localized paywalls. * SW-2919: Adds a `custom_placement` event that you can attach to any element in the paywall with a dictionary of parameters. When the element is tapped, the event will be tracked. The name of the placement can be used to trigger a paywall and its params used in audience filters. * Adds support for bottom sheet presentation style (DRAWER), no animation style and default animation. * Adds `build_id` and `cache_key` to `PaywallInfo`. * SW-2917: Tracks a `config_attributes` event after calling `Superwall.configure`, which contains info about the configuration of the SDK. This gets tracked whenever you set the delegate. * Adds in device attributes tracking after setting the interface style override. * To comply with new Google Play Billing requirements we now avoid setting empty `offerToken` for one-time purchases 1.2.1 [#121] Enhancements [#enhancements-54] * Adds the ability for the SDK to refresh the Superwall configuration every session start, subject to a feature flag. * Tracks a `config_refresh` Superwall event when the configuration is refreshed. * SW-2890: Adds `capabilities` to device attributes. This is a comma-separated list of capabilities the SDK has that you can target in audience filters. This release adds the `paywall_event_receiver` capability. This indicates that the paywall can receive transaction from the SDK. * SW-2902: Adds `abandoned_product_id` to a `transaction_abandon` event to use in audience filters. You can use this to show a paywall if a user abandons the transaction for a specific product. 1.2.0 [#120] Enhancements [#enhancements-55] * Adds DSL methods for configuring the SDK. You can now use a configuration block: ```kotlin fun Application.configureSuperwall( apiKey: String, configure: SuperwallBuilder.() -> Unit, ) ``` This allows you to configure the SDK in a more idiomatic way: ```kotlin configureSuperwall(CONSTANT_API_KEY){ options { logging { level = LogLevel.debug } paywalls { shouldPreload = false } } } ``` Deprecations [#deprecations-1] This release includes multiple deprecations that will be removed in upcoming versions. Most are internal and will not affect the public API, those that will are marked as such and a simple migration path is provided. The notable ones in the public API are as follows: * Deprecated `DebugViewControllerActivity` in favor of `DebugViewActivity` * Deprecated `PaywallViewController` in favor of `PaywallView` * Deprecated belonging methods: * `viewWillAppear` in favor of `beforeViewCreated` * `viewDidAppear` in favor of `onViewCreated` * `viewWillDisappear` in favor of `beforeOnDestroy` * `viewDidDisappear` in favor of `destroyed` * `presentAlert` in favor of `showAlert` * Deprecated `PaywallViewControllerDelegate` in favor of `PaywallViewCallback` * Deprecated belonging methods: * `didFinish` in favor of `onFinished` * Deprecated `PaywallViewControllerEventDelegate` in favor of `PaywallViewEventCallback` * Users might also note deprecation of `PaywallWebEvent.OpenedUrlInSafari` in favor of `PaywallWebEvent.OpenedUrlInChrome` * `didFinish` in favor of `onFinished` * In `Superwall`, the following methods were deprecated: * `Superwall.paywallViewController` in favor of `Superwall.paywallView` * `Superwall.eventDidOccur` argument `paywallViewController` in favor of `paywallView` * `Superwall.dismiss` in favor of \`Superwall.presentPaywallView * `Superwall.presentPaywallViewController` in favor of `Superwall.presentPaywallView` * Deprecated `Paywallmanager.getPaywallViewController` in favor of `PaywallManager.getPaywallView` * Deprecated `DebugManager.viewController` in favor of `DebugManager.view` * Deprecated `DebugViewController` in favor of `DebugView` * Deprecated `LogScope.debugViewController` in favor of `LogScope.debugView` * Deprecated `PaywallPresentationRequestStatus.NoPaywallViewController` in favor of `NoPaywallView` 1.1.9 [#119] Deprecations [#deprecations-2] * Deprecated configuration method `Superwall.configure(applicationContext: Context, ...)` in favor of `Superwall.configure(applicationContext: Application, ...)` to enforce type safety. The rest of the method signature remains the same. Fixes [#fixes-55] * SW-2878: and it's related leaks. The `PaywallViewController` was not being properly detached when activity was stopped, causing memory leaks. * SW-2872: Fixes issue where `deviceAttributes` event and fetching would not await for IP geo to complete. * Fixes issues on tablet devices where the paywall would close after rotation/configuration change. 1.1.8 [#118] Enhancements [#enhancements-56] * SW-2859: Adds error message to `paywallWebviewLoad_fail`. * SW-2866: Logs error when trying to purchase a product that has failed to load. * SW-2869: Add `Reset` event to track when `Superwall.instance.reset` is called. * SW-2867: Prevents Geo api from being called when app is in the background * SW-2431: Improves coroutine scope usages & threading limits * Toolchain and dependency updates Fixes [#fixes-56] * SW-2863: Fixed a `NullPointerException` some users on Android 12 & 13 would experience when calling `configure`. 1.1.7 [#117] Enhancements [#enhancements-57] * SW-2805: Exposes a `presentation` property on the `PaywallInfo` object. This contains information about the presentation of the paywall. * SW-2855: Adds `restore_start`, `restore_complete`, and `restore_fail` events. Fixes [#fixes-57] * SW-2854: Fixed issue where abandoning the transaction by pressing back would prevent the user from restarting the transaction. 1.1.6 [#116] Enhancements [#enhancements-58] * SW-2833: Adds support for dark mode paywall background color. * Adds ability to target devices based on their IP address location. Use `device.ipRegion`, `device.ipRegionCode`, `device.ipCountry`, `device.ipCity`, `device.ipContinent`, or `device.ipTimezone`. * Adds `event_name` to the event params for use with audience filters. Fixes [#fixes-58] * Fixes issue with products whose labels weren't primary/secondary/tertiary. 1.1.5 [#115] Fixes [#fixes-59] * Fixes thread safety crash when multiple threads attempted to initialize the `JavaScriptSandbox` internally. 1.1.3 [#113] Enhancements [#enhancements-59] * Tracks an `identity_alias` event whenever identify is called to alias Superwall's anonymous ID with a developer provided id. * Adds `setInterfaceStyle(interfaceStyle:)` which can be used to override the system interface style. * Adds `device.interfaceStyleMode` to the device template, which can be `automatic` or `manual` if overriding the interface style. Fixes [#fixes-60] * Uses `JavascriptSandbox` when available for filter expression evaluation on a background thread instead of running code on the main thread in a webview. * Fixes crash where the loading spinner inside the `PaywallViewController` was being updated outside the main thread. 1.1.2 [#112] Enhancements [#enhancements-60] * Updates build.gradle configuration which means we can now upload the SDK to maven central. You no longer need to specify our custom repo in your build.gradle to get our SDK and therefore installation should be easier. Fixes [#fixes-61] * Fixes `ConcurrentModificationException` crash that sometimes happened when identifying a user. * Fixes crash on purchasing a free trial when using `getPaywall`. 1.1.1 [#111] Fixes [#fixes-62] * Fixes an issue loading products with offers. 1.1.0 [#110] Enhancements [#enhancements-61] * SW-2768: Adds `device.regionCode` and `device.preferredRegionCode`, which returns the `regionCode` of the locale. For example, if a locale is `en_GB`, the `regionCode` will be `GB`. You can use this in the filters of your campaign. * Adds support for unlimited products in a paywall. * SW-2785: Adds internal feature flag to disable verbose events like `paywallResponseLoad_start`. Fixes [#fixes-63] * SW-2732: User attributes weren't being sent on app open until identify was called. Now they are sent every time there's a new session. * SW-2733: Fixes issue where the spinner would still show on a paywall if a user had previously purchased on it. * SW-2744: Fixes issue where using the back button to dismiss a paywall presented via `getPaywall` would call `didFinish` in the `PaywallViewControllerDelegate` with the incorrect values. * Fixes issue where an invalid paywall background color would prevent the paywall from opening. If this happens, it will now default to white. * SW-2748: Exposes `viewWillAppear`, `viewDidAppear`, `viewWillDisappear` and `viewDidDisappear` methods of `PaywallViewController` which you must call when using `getPaywall`. * Stops `Superwall.configure` from being called multiple times. * `getPresentationResult` now confirms assignments for holdouts. * Gracefully handles unknown local notification types if new ones are added in the future. * SW-2761: Fixes issue where "other" responses in paywall surveys weren't showing in the dashboard. 1.0.2 [#102] Fixes [#fixes-64] * Prevents a paywall from opening in a separate activity inside a task manager when using `taskAffinity` within your app. 1.0.1 [#101] Fixes [#fixes-65] * Fixes serialization of `feature_gating` in `SuperwallEvents`. * Changes the product loading so that if preloading is enabled, it makes one API request to get all products available in paywalls. This results in fewer API requests. Also, it adds retry logic on failure. If billing isn't available on device, the `onError` handler will be called. 1.0.0 [#100] Breaking Changes [#breaking-changes-6] * Changes the import path for the `LogScope`, and `LogLevel`. Fixes [#fixes-66] * Fixes rare thread-safety crash when sending events back to Superwall's servers. * Calls the `onError` presentation handler block when there's no activity to present a paywall on. * Fixes issue where the wrong product may be presented to purchase if a free trial had already been used and you were letting Superwall handle purchases. * Fixes `IllegalStateException` on Samsung devices when letting Superwall handle purchases. * Keeps the text zoom of paywalls to 100% rather than respecting the accessibility settings text zoom, which caused unexpected UI issues. * Fixes rare `UninitializedPropertyAccessException` crash caused by a threading issue. * Fixes crash when the user has disabled the Android System WebView. 1.0.0-alpha.45 [#100-alpha45] Fixes [#fixes-67] * Fixes issue where the `paywallProductsLoad_fail` event wasn't correctly being logged. This is a "soft fail", meaning that even though it gets logged, your paywall will still show. The error message with the event has been updated to include all product subscription IDs that are failing to be retrieved. 1.0.0-alpha.44 [#100-alpha44] Fixes [#fixes-68] * Fixes rare issue where paywall preloading was preventing paywalls from showing. 1.0.0-alpha.43 [#100-alpha43] Enhancements [#enhancements-62] * Adds `handleLog` to the `SuperwallDelegate`. 1.0.0-alpha.42 [#100-alpha42] Fixes [#fixes-69] * Makes sure client apps use our proguard file. 1.0.0-alpha.41 [#100-alpha41] Fixes [#fixes-70] * Removes need for `SCHEDULED_EXACT_ALARM` permission in manifest. 1.0.0-alpha.40 [#100-alpha40] Fixes [#fixes-71] * Fixes issue presenting paywalls to users who had their device language set to Russian, Polish or Czech. 1.0.0-alpha.39 [#100-alpha39] Fixes [#fixes-72] * Adds missing `presentationSourceType` to `PaywallInfo`. * Fixes issue where the status bar color was always dark regardless of paywall color. * Adds `TypeToken` to proguard rules to prevent r8 from 'optimizing' our code and causing a crash. 1.0.0-alpha.38 [#100-alpha38] Enhancements [#enhancements-63] * SW-2682: Adds `Superwall.instance.latestPaywallInfo`, which you can use to get the `PaywallInfo` of the most recently presented view controller. * SW-2683: Adds `Superwall.instance.isLoggedIn`, which you can use to check if the user is logged in. Fixes [#fixes-73] * Removes use of `USE_EXACT_ALARM` permission that was getting apps rejected. * Fixes issue with scheduling notifications. The paywall wasn't waiting to schedule notifications before dismissal so the permissions wasn't always showing. 1.0.0-alpha.37 [#100-alpha37] Enhancements [#enhancements-64] * SW-2684: Adds error logging if the `currentActivity` is `null` when trying to present a paywall. Fixes [#fixes-74] * Fixes bug where paywalls might not present on first app install. * Fixes bug where all `paywallResponseLoad_` events were being counted as `paywallResponseLoad_start`. * Adds ProGuard rule to prevent `DefaultLifecycleObserver` from being removed. 1.0.0-alpha.36 [#100-alpha36] Enhancements [#enhancements-65] * Adds `X-Subscription-Status` header to all requests. * Caches the last `subscriptionStatus`. * Adds `subscriptionStatus_didChange` event that is fired whenever the subscription status changes. * Calls the delegate method `subscriptionStatusDidChange` whenever the subscription status changes. * SW-2676: Adds a completion block to the `configure` method. Fixes [#fixes-75] * Fixes issue where the main thread was blocked when accessing some properties internally. * SW-2679: Fixes issue where the `subscription_start` event was being fired even if a non-recurring product was purchased. 1.0.0-alpha.35 [#100-alpha35] Fixes [#fixes-76] * Fixes issue where `transaction_complete` events were being rejected by the server. 1.0.0-alpha.34 [#100-alpha34] Breaking Changes [#breaking-changes-7] * Changes `Superwall.instance.getUserAttributes()` to `Superwall.instance.userAttributes`. * `SuperwallOptions.logging.logLevel` is now non-optional. Set it to `LogLevel.none` to prevent logs from being printed to the console. Enhancements [#enhancements-66] * SW-2663: Adds `preloadAllPaywalls()` and `preloadPaywalls(eventNames:)` to be able to manually preload paywalls. * SW-2665: Adds `Superwall.instance.userId` so you can access the current user's id. * SW-2668: Adds `preferredLocale` and `preferredLanguageLocale` to the device attributes for use in rules. * Adds `Superwall.instance.logLevel` as a convenience variable to set and get the log level. Fixes [#fixes-77] * SW-2664: Fixes race condition between resetting and presenting paywalls. 1.0.0-alpha.33 [#100-alpha33] Fixes [#fixes-78] * Fixes issue where a user would be asked to enable notifications even if there weren't any attached to the paywall. 1.0.0-alpha.32 [#100-alpha32] Enhancements [#enhancements-67] * SW-2214: Adds ability to use free trial notifications with a paywall. * Adds `cancelAllScheduledNotifications()` to cancel any scheduled free trial notifications. * SW-2640: Adds `computedPropertyRequests` to `PaywallInfo`. * SW-2641: Makes `closeReason` in `PaywallInfo` non-optional. Fixes [#fixes-79] * Fixes issue where thrown exceptions weren't always being caught. 1.0.0-alpha.31 [#100-alpha31] Enhancements [#enhancements-68] * SW-2638: Adds `Restored` to `PurchaseResult`. * SW-2644: Adds `RestoreType` to `SuperwallEvent.TransactionRestore`. * SW-2643: Makes `storePayment` non-optional for a `StoreTransaction`. * SW-2642: Adds `productIdentifier` to `StorePayment`. Fixes [#fixes-80] * SW-2635: Fixes crash that sometimes occurred if an app was trying to get Superwall's paywall configuration in the background. 1.0.0-alpha.30 [#100-alpha30] Enhancements [#enhancements-69] * SW-2154: The SDK now includes a paywall debugger, meaning you can scan the QR code of any paywall in the editor to preview it on device. You can change its localization, view product attributes, and view the paywall with or without a trial. Fixes [#fixes-81] * More bug fixes relating to the loading of products. 1.0.0-alpha.29 [#100-alpha29] Fixes [#fixes-82] * SW-2631: Fixes issue where paywalls weren't showing if the products within them had a base plan or offer ID set. 1.0.0-alpha.28 [#100-alpha28] Fixes [#fixes-83] * SW-2615: Fixes crash on Android versions \< 8 when accessing the Android 8+ only class Period. * SW-2616: Fixes crash where the `PaywallViewController` was sometimes being added to a new parent view before being removed from it's existing parent view. 1.0.0-alpha.27 [#100-alpha27] Breaking Changes [#breaking-changes-8] * \#SW-2218: Changes the `PurchaseController` purchase function to `purchase(activity:productDetails:basePlanId:offerId:)`. This adds support for purchasing offers and base plans. Enhancements [#enhancements-70] * \#SW-2600: Backport device attributes * Adds support for localized paywalls. * Paywalls are only preloaded if their associated rules can match. * Adds status bar to full screen paywalls. Fixes [#fixes-84] * Fixes issue where holdouts were still matching even if the limit set for their corresponding rules were exceeded. * \#SW-2584: Fixes issue where prices with non-terminating decimals were causing products to fail to load. 1.0.0-alpha.26 [#100-alpha26] Fixes [#fixes-85] * Additional fixes to make Google billing more robust. * Fixes an issue that causes `transaction_complete` events to fail. 1.0.0-alpha.25 [#100-alpha25] Fixes [#fixes-86] * Fixes [Google Billing Crash on Samsung devices](https://community.revenuecat.com/sdks-51/how-to-fix-crash-too-many-bind-requests-999-for-service-intent-inappbillingservice-3317). 1.0.0-alpha.24 [#100-alpha24] Fixes [#fixes-87] * Fixes an issue that could cause "n/a" to be displayed on a paywall in place of the proper subscription period string. 1.0.0-alpha.23 [#100-alpha23] Fixes [#fixes-88] * Fixes an issue where calling `identify` right after `configure` would hang b/c network requests need to access the user id to add to headers. * Fixes a potential crash when loading data from disk 1.0.0-alpha.22 [#100-alpha22] Fixes [#fixes-89] * Fixes threading issue 1.0.0-alpha.21 [#100-alpha21] Fixes [#fixes-90] * Changes Activity to AppCompatActivity 1.0.0-alpha.20 [#100-alpha20] Enhancements [#enhancements-71] Fixes [#fixes-91] * Fixes `app_open` race condition * Prevents calling purchase twice * Disables interactivity during purchase 1.0.0-alpha.19 [#100-alpha19] Fixes [#fixes-92] * Fixes `app_launch` event not triggering 1.0.0-alpha.18 [#100-alpha18] Enhancements [#enhancements-72] * Adds the ability to provide an `ActivityProvider` when configuring the SDK. This is an interface containing the function `getCurrentActivity()`, which Superwall will use to present paywalls from. You would typically conform an `ActivityLifecycleTracker` to this interface. Fixes [#fixes-93] * Fixes a crash when storing a `List` to user attributes and if that List or a Map had a null value. 1.0.0-alpha.17 [#100-alpha17] Enhancements [#enhancements-73] * Adds automatic purchase controller * Improves memory handling for webviews * Hides the loading indicator on a paywall if transactionBackgroundView is set to NONE 1.0.0-alpha.14 [#100-alpha14] Enhancements [#enhancements-74] * Adds `trigger_session_id` to Superwall Events. * Resets the scroll position of the paywall on close. Fixes [#fixes-94] * Fixes issue where an invalid currency code on a device would crash the app when trying to retrieve products. 1.0.0-alpha.13 [#100-alpha13] Fixes [#fixes-95] * Fixes concurrency issues when setting and retrieving values like the appUserId and seed. 1.0.0-alpha.11 [#100-alpha11] Fixes [#fixes-96] * Can now use both non-recurring products and subscription products in paywalls. * Fixes a crash issue that was caused by a lazy variable being accessed before it was initialized. # Cohorting in 3rd Party Tools :::android ```kotlin Kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when(eventInfo.event) { is SuperwallEvent.TriggerFire -> { MyAnalyticsService.shared.setUserAttributes( mapOf( "sw_experiment_${eventInfo.params.get("experiment_id").toString()}" to true, "sw_variant_${eventInfo.params.get("variant_id").toString()}" to true ) ) } else -> {} } } ``` ::: :::flutter ```dart Flutter @override void handleSuperwallEvent(SuperwallEventInfo eventInfo) async { final experimentId = eventInfo.params?['experiment_id']; final variantId = eventInfo.params?['variant_id']; switch (eventInfo.event.type) { case EventType.triggerFire: MyAnalyticsService.shared.setUserAttributes({ "sw_experiment_$experimentId": true, "sw_variant_$variantId": true }); break; default: break; } } ``` ::: :::expo ```typescript React Native handleSuperwallEvent(eventInfo: SuperwallEventInfo) { const experimentId = eventInfo.params?['experiment_id'] const variantId = eventInfo.params?['variant_id'] if (!experimentId || !variantId) { return } switch (eventInfo.event.type) { case EventType.triggerFire: MyAnalyticsService.shared.setUserAttributes({ `sw_experiment_${experimentId}`: true, `sw_variant_${variantId}`: true }); break; default: break; } } ``` ::: Once you've set this up, you can easily ask for all users who have an attribute `sw_experiment_1234` and breakdown by both variants to see how users in a Superwall experiment behave in other areas of your app. # Custom Paywall Analytics You can create customized analytics tracking for any paywall event by using custom placements. With them, you can get callbacks for actions such as interacting with an element on a paywall sent to your [Superwall delegate](/docs/sdk/guides/using-superwall-delegate). This can be useful for tracking how users interact with your paywall and how that affects their behavior in other areas of your app. For example, in the paywall below, perhaps you're interested in tracking when people switch the plan from "Standard" and "Pro": You could create a custom placement [tap behavior](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-styling-elements#tap-behaviors) which fires when a segment is tapped: Then, you can listen for this placement and forward it to your analytics service: ```swift Swift extension SuperwallService: SuperwallDelegate { func handleSuperwallEvent(withInfo eventInfo: SuperwallEventInfo) { switch eventInfo.event { case let .customPlacement(name, params, paywallInfo): // Prints out didTapPro or didTapStandard print("\(name) - \(params) - \(paywallInfo)") MyAnalyticsService.shared.send(event: name, params: params) default: print("Default event: \(eventInfo.event.description)") } } } ``` For a walkthrough example, check out this [video on YouTube](https://youtu.be/4rM1rGRqDL0). # 3rd Party Analytics Hooking up Superwall events to 3rd party tools [#hooking-up-superwall-events-to-3rd-party-tools] SuperwallKit automatically tracks some internal events. You can [view the list of events here](/docs/sdk/guides/3rd-party-analytics/tracking-analytics). We encourage you to also track them in your own analytics by implementing the [Superwall delegate](/docs/sdk/guides/using-superwall-delegate). Using the `handleSuperwallEvent(withInfo:)` function, you can forward events to your analytics service: :::android ```kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { println("analytics event: ${eventInfo.event.rawName}") MyAnalytics.shared.track(eventInfo.event.rawName, eventInfo.params) } ``` :::
You might also want to set user attribute to allow for [Cohorting in 3rd Party Tools](/docs/sdk/guides/3rd-party-analytics/cohorting-in-3rd-party-tools). Alternatively, if you want typed versions of all these events with associated values, you can access them via `eventInfo.event`: :::android ```kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when(eventInfo.event) { is SuperwallPlacement.AppClose -> TODO() is SuperwallPlacement.AppInstall -> TODO() is SuperwallPlacement.AppLaunch -> TODO() is SuperwallPlacement.AppOpen -> TODO() is SuperwallPlacement.DeepLink -> TODO() is SuperwallPlacement.FirstSeen -> TODO() is SuperwallPlacement.FreeTrialStart -> TODO() is SuperwallPlacement.NonRecurringProductPurchase -> TODO() is SuperwallPlacement.PaywallClose -> TODO() is SuperwallPlacement.PaywallDecline -> TODO() is SuperwallPlacement.PaywallOpen -> TODO() is SuperwallPlacement.PaywallPresentationRequest -> TODO() is SuperwallPlacement.PaywallProductsLoadComplete -> TODO() is SuperwallPlacement.PaywallProductsLoadFail -> TODO() is SuperwallPlacement.PaywallProductsLoadStart -> TODO() is SuperwallPlacement.PaywallResponseLoadComplete -> TODO() is SuperwallPlacement.PaywallResponseLoadFail -> TODO() is SuperwallPlacement.PaywallResponseLoadNotFound -> TODO() is SuperwallPlacement.PaywallResponseLoadStart -> TODO() is SuperwallPlacement.PaywallWebviewLoadComplete -> TODO() is SuperwallPlacement.PaywallWebviewLoadFail -> TODO() is SuperwallPlacement.PaywallWebviewLoadStart -> TODO() is SuperwallPlacement.PaywallWebviewLoadTimeout -> TODO() is SuperwallPlacement.SessionStart -> TODO() is SuperwallPlacement.SubscriptionStart -> TODO() is SuperwallPlacement.SubscriptionStatusDidChange -> TODO() is SuperwallPlacement.SurveyClose -> TODO() is SuperwallPlacement.SurveyResponse -> TODO() is SuperwallPlacement.TransactionAbandon -> TODO() is SuperwallPlacement.TransactionComplete -> TODO() is SuperwallPlacement.TransactionFail -> TODO() is SuperwallPlacement.TransactionRestore -> TODO() is SuperwallPlacement.TransactionStart -> TODO() is SuperwallPlacement.TransactionTimeout -> TODO() is SuperwallPlacement.TriggerFire -> TODO() is SuperwallPlacement.UserAttributes -> TODO() } } ``` ::: Wanting to use events to see which product was purchased on a paywall? Check out this [doc](/docs/sdk/guides/advanced/viewing-purchased-products). # Superwall Events We encourage you to track them in your own analytics as described in [3rd Party Analytics](..). The following Superwall events can be used as placements to present paywalls: * `app_install` * `app_launch` * `deepLink_open` * `session_start` * `paywall_decline` * `transaction_fail` * `transaction_abandon` * `survey_response` For more info about how to use these, check out [how to add them using a Placement](/docs/dashboard/dashboard-campaigns/campaigns-placements#adding-a-placement). The full list of events is as follows: | **Event Name** | **Action** | **Parameters** | | ---------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | | `adServicesTokenRequestComplete` | When the AdServices token request finishes. | `["token": String]` | | `adServicesTokenRequestFail` | When the AdServices token request fails. | `["error": Error]` | | `adServicesTokenRequestStart` | When the AdServices token request starts. | None | | `app_close` | Anytime the app leaves the foreground. | Same as `app_install` | | `app_install` | When the SDK is configured for the first time. | `["is_superwall": true, "app_session_id": String, "using_purchase_controller": Bool]` | | `app_launch` | When the app is launched from a cold start. | Same as `app_install` | | `app_open` | Anytime the app enters the foreground. | Same as `app_install` | | `configAttributes` | When the attributes affecting Superwall's configuration are set or changed. | None | | `configFail` | When the Superwall configuration fails to be retrieved. | None | | `configRefresh` | When the Superwall configuration is refreshed. | None | | `confirmAllAssignments` | When all experiment assignments are confirmed. | None | | `customPlacement` | When the user taps on an element in the paywall that has a `custom_placement` action. | `["name": String, "params": [String: Any], "paywallInfo": PaywallInfo]` | | [`deepLink_open`](/docs/dashboard/dashboard-campaigns/campaigns-standard-placements#using-the-deeplink-open-event) | When a user opens the app via a deep link. | `["url": String, "path": String", "pathExtension": String, "lastPathComponent": String, "host": String, "query": String, "fragment": String]` + any query parameters in the deep link URL | | `device_attributes` | When device attributes are sent to the backend every session. | Includes `app_session_id`, `app_version`, `os_version`, `device_model`, `device_locale`, and various hardware/software details. | | `first_seen` | When the user is first seen in the app, regardless of login status. | Same as `app_install` | | `freeTrial_start` | When a user completes a transaction for a subscription product with an introductory offer. | Same as `subscription_start` | | `identityAlias` | When the user's identity aliases after calling `identify`. | None | | `nonRecurringProduct_purchase` | When the user purchases a non-recurring product. | Same as `subscription_start` | | `paywall_close` | When a paywall is closed (either manually or after a transaction succeeds). | \[“paywall\_webview\_load\_complete\_time”: String?, “paywall\_url”: String, “paywall\_response\_load\_start\_time”: String?, “paywall\_products\_load\_fail\_time”: String?, “secondary\_product\_id”: String, “feature\_gating”: Int, “paywall\_response\_load\_complete\_time”: String?, “is\_free\_trial\_available”: Bool, “is\_superwall”: true, “presented\_by”: String, “paywall\_name”: String, “paywall\_response\_load\_duration”: String?, “paywall\_identifier”: String, “paywall\_webview\_load\_start\_time”: String?, “paywall\_products\_load\_complete\_time”: String?, “paywall\_product\_ids”: String, “tertiary\_product\_id”: String, “paywall\_id”: String, “app\_session\_id”: String, “paywall\_products\_load\_start\_time”: String?, “primary\_product\_id”: String, “survey\_attached”: Bool, “survey\_presentation”: String?] | | [`paywall_decline`](/docs/dashboard/dashboard-campaigns/campaigns-standard-placements#using-the-paywall-decline-event) | When a user manually dismisses a paywall. | Same as `paywall_close` | | `paywall_open` | When a paywall is opened. | Same as `paywall_close` | | `paywallPresentationRequest` | When something happened during the paywall presentation, whether a success or failure. | `[“source_event_name”: String, “status”: String, “is_superwall”: true, “app_session_id”: String, “pipeline_type”: String, “status_reason”: String]` | | `paywallProductsLoad_complete` | When the request to load a paywall's products completes. | Same as `paywallResponseLoad_start` | | `paywallProductsLoad_fail` | When the request to load a paywall's products fails. | Same as `paywallResponseLoad_start` | | `paywallProductsLoad_retry` | When the request to load a paywall's products fails and is being retried. | `["triggeredPlacementName": String?, "paywallInfo": PaywallInfo, "attempt": Int]` | | `paywallProductsLoad_start` | When the request to load a paywall's products starts. | Same as `paywallResponseLoad_start` | | `paywallResponseLoad_complete` | When a paywall request to Superwall's servers completes. | Same as `paywallResponseLoad_start` | | `paywallResponseLoad_fail` | When a paywall request to Superwall's servers fails. | Same as `paywallResponseLoad_start` | | `paywallResponseLoad_notFound` | When a paywall request returns a 404 error. | Same as `paywallResponseLoad_start` | | `paywallResponseLoad_start` | When a paywall request to Superwall's servers has started. | Same as `app_install` + `["is_triggered_from_event": Bool]` | | `paywallWebviewLoad_complete` | When a paywall's webpage completes loading. | Same as `paywall_close` | | `paywallWebviewLoad_fail` | When a paywall's webpage fails to load. | Same as `paywall_close` | | `paywallWebviewLoad_fallback` | When a paywall's webpage fails and loads a fallback version. | Same as `paywall_close` | | `paywallWebviewLoad_start` | When a paywall's webpage begins to load. | Same as `paywall_close` | | `paywallWebviewLoad_processTerminated` | When the paywall's web view content process terminates. | Same as `paywall_close` | | `reset` | When `Superwall.reset()` is called. | None | | `restoreComplete` | When a restore completes successfully. | None | | `restoreFail` | When a restore fails. | `["message": String]` | | `restoreStart` | When a restore is initiated. | None | | `session_start` | When the app is opened after at least 60 minutes since last `app_close`. | Same as `app_install` | | `shimmerViewComplete` | When the shimmer view stops showing. | None | | `shimmerViewStart` | When the shimmer view starts showing. | None | | `subscription_start` | When a user completes a transaction for a subscription product without an introductory offer. | \[“product\_period\_days”: String, “product\_price”: String, “presentation\_source\_type”: String?, “paywall\_response\_load\_complete\_time”: String?, “product\_language\_code”: String, “product\_trial\_period\_monthly\_price”: String, “paywall\_products\_load\_duration”: String?, “product\_currency\_symbol”: String, “is\_superwall”: true, “app\_session\_id”: String, “product\_period\_months”: String, “presented\_by\_event\_id”: String?, “product\_id”: String, “trigger\_session\_id”: String, “paywall\_webview\_load\_complete\_time”: String?, “paywall\_response\_load\_start\_time”: String?, “product\_raw\_trial\_period\_price”: String, “feature\_gating”: Int, “paywall\_id”: String, “product\_trial\_period\_daily\_price”: String, “product\_period\_years”: String, “presented\_by”: String, “product\_period”: String, “paywall\_url”: String, “paywall\_name”: String, “paywall\_identifier”: String, “paywall\_products\_load\_start\_time”: String?, “product\_trial\_period\_months”: String, “product\_currency\_code”: String, “product\_period\_weeks”: String, “product\_periodly”: String, “product\_trial\_period\_text”: String, “paywall\_webview\_load\_start\_time”: String?, “paywall\_products\_load\_complete\_time”: String?, “primary\_product\_id”: String, “product\_trial\_period\_yearly\_price”: String, “paywalljs\_version”: String?, “product\_trial\_period\_years”: String, “tertiary\_product\_id”: String, “paywall\_products\_load\_fail\_time”: String?, “product\_trial\_period\_end\_date”: String, “product\_weekly\_price”: String, “variant\_id”: String, “presented\_by\_event\_timestamp”: String?, “paywall\_response\_load\_duration”: String?, “secondary\_product\_id”: String, “product\_trial\_period\_days”: String, “product\_monthly\_price”: String, “paywall\_product\_ids”: String, “product\_locale”: String, “product\_daily\_price”: String, “product\_raw\_price”: String, “product\_yearly\_price”: String, “product\_trial\_period\_price”: String, “product\_localized\_period”: String, “product\_identifier”: String, “experiment\_id”: String, “is\_free\_trial\_available”: Bool, “product\_trial\_period\_weeks”: String, “paywall\_webview\_load\_duration”: String?, “product\_period\_alt”: String, “product\_trial\_period\_weekly\_price”: String, “presented\_by\_event\_name”: String?] | | `subscriptionStatus_didChange` | When a user's subscription status changes. | `["is_superwall": true, "app_session_id": String, "subscription_status": String]` | | `surveyClose` | When the user chooses to close a survey instead of responding. | None | | [`survey_response`](/docs/dashboard/dashboard-campaigns/campaigns-standard-placements#using-the-survey-response-event) | When a user responds to a paywall survey. | `["survey_selected_option_title": String, "survey_custom_response": String, "survey_id": String, "survey_assignment_key": String, "survey_selected_option_id": String]` | | `touches_began` | When the user touches the app's UIWindow for the first time (if tracked by a campaign). | Same as `app_install` | | `transaction_abandon` | When the user cancels a transaction. | Same as `subscription_start` | | `transaction_complete` | When the user completes checkout and any product is purchased. | Same as subscription\_start + \[“web\_order\_line\_item\_id”: String, “app\_bundle\_id”: String, “config\_request\_id”: String, “state”: String, “subscription\_group\_id”: String, “is\_upgraded”: String, “expiration\_date”: String, “trigger\_session\_id”: String, “original\_transaction\_identifier”: String, “id”: String, “transaction\_date”: String, “is\_superwall”: true, “store\_transaction\_id”: String, “original\_transaction\_date”: String, “app\_session\_id”: String] | | `transaction_fail` | When the payment sheet fails to complete a transaction (ignores user cancellation). | Same as `subscription_start` + `["message": String]` | | `transaction_restore` | When the user successfully restores their purchases. | Same as `subscription_start` | | `transaction_start` | When the payment sheet is displayed to the user. | Same as `subscription_start` | | `transaction_timeout` | When the transaction takes longer than 5 seconds to display the payment sheet. | `["paywallInfo": PaywallInfo]` | | `trigger_fire` | When a registered placement triggers a paywall. | `[“trigger_name”: String, “trigger_session_id”: String, “variant_id”: String?, “experiment_id”: String?, “paywall_identifier”: String?, “result”: String, “unmatched_rule_”: “”]. unmatched_rule_ indicates why a rule (with a specfiic experiment id) didn’t match. It will only exist if the result is no_rule_match. Its outcome will either be OCCURRENCE, referring to the limit applied to a rule, or EXPRESSION.` | | `user_attributes` | When the user attributes are set. | `[“aliasId”: String, “seed”: Int, “app_session_id”: String, “applicationInstalledAt”: String, “is_superwall”: true, “application_installed_at”: String] + provided attributes` | # Advanced Purchasing Using a `PurchaseController` is only recommended for **advanced** use cases. By default, Superwall handles all subscription-related logic and purchasing operations for you out of the box. By default, Superwall handles basic subscription-related logic for you: 1. **Purchasing**: When the user initiates a checkout on a paywall. 2. **Restoring**: When the user restores previously purchased products. 3. **Subscription Status**: When the user's subscription status changes to active or expired (by checking the local receipt). However, if you want more control, you can pass in a `PurchaseController` when configuring the SDK via `configure(apiKey:purchaseController:options:)` and manually set `Superwall.shared.subscriptionStatus` to take over this responsibility. On iOS, starting in `4.15.0`, a `PurchaseController` is also how you handle custom products attached to Superwall paywalls. Those products are not purchased with StoreKit, so your controller must route them through your own billing system. See [Custom Store Products](/docs/ios/guides/custom-store-products) for the full iOS setup. Step 1: Creating a `PurchaseController` [#step-1-creating-a-purchasecontroller] A `PurchaseController` handles purchasing and restoring via protocol methods that you implement. :::android ```kotlin Kotlin // MyPurchaseController.kt class MyPurchaseController(val context: Context): PurchaseController { // 1 override suspend fun purchase( activity: Activity, productDetails: ProductDetails, basePlanId: String?, offerId: String? ): PurchaseResult { // TODO // ---- // Purchase via GoogleBilling, RevenueCat, Qonversion or however // you like and return a valid PurchaseResult return PurchaseResult.Purchased() } // 2 override suspend fun restorePurchases(): RestorationResult { // TODO // ---- // Restore purchases and return true if successful. return RestorationResult.Success() } } ``` ::: Here’s what each method is responsible for: 1. Purchasing a given product. In here, enter your code that you use to purchase a product. Then, return the result of the purchase as a `PurchaseResult`. For Flutter, this is separated into purchasing from the App Store and Google Play. This is an enum that contains the following cases, all of which must be handled: 1. `.cancelled`: The purchase was cancelled. 2. `.purchased`: The product was purchased. 3. `.pending`: The purchase is pending/deferred and requires action from the developer. 4. `.failed(Error)`: The purchase failed for a reason other than the user cancelling or the payment pending. 2. Restoring purchases. Here, you restore purchases and return a `RestorationResult` indicating whether the restoration was successful or not. If it was, return `.restore`, or `failed` along with the error reason. Step 2: Configuring the SDK With Your `PurchaseController` [#step-2-configuring-the-sdk-with-your-purchasecontroller] Pass your purchase controller to the `configure(apiKey:purchaseController:options:)` method: :::android ```kotlin Kotlin // MainApplication.kt class MainApplication : android.app.Application(), SuperwallDelegate { override fun onCreate() { super.onCreate() Superwall.configure(this, "MY_API_KEY", MyPurchaseController(this)) // OR using the DSL configureSuperwall("MY_API_KEY") { purchaseController = MyPurchaseController(this@MainApplication) } } } ``` ::: Step 3: Keeping `subscriptionStatus` Up-To-Date [#step-3-keeping-subscriptionstatus-up-to-date] You **must** set `Superwall.shared.subscriptionStatus` every time the user's subscription status changes, otherwise the SDK won't know who to show a paywall to. This is an enum that has three possible cases: 1. **`.unknown`**: This is the default value. In this state, paywalls will not show and their presentation will be ***automatically delayed*** until `subscriptionStatus` changes to a different value. 2. **`.active(let entitlements)`**: Indicates that the user has an active entitlement. Paywalls will not show in this state unless you remotely set the paywall to ignore subscription status. A user can have one or more active entitlement. 3. **`.inactive`**: Indicates that the user doesn't have an active entitlement. Paywalls can show in this state. Here's how you might do this: :::android ```kotlin Kotlin // When a subscription is purchased, restored, validated, expired, etc... myService.subscriptionStatusDidChange { if (it.hasActiveSubscription) { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Active(entitlements)) } else { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive(entitlements)) } } ``` :::
`subscriptionStatus` is cached between app launches Listening for subscription status changes [#listening-for-subscription-status-changes] If you need a simple way to observe when a user's subscription status changes, on iOS you can use the `Publisher` for it. Here's an example: :::android ```kotlin Kotlin Superwall.instance.subscriptionStatus.collect { status: SubscriptionStatus -> // React to changes } ``` ::: You can do similar tasks with the `SuperwallDelegate`, such as [viewing which product was purchased from a paywall](/docs/sdk/guides/3rd-party-analytics#using-events-to-see-purchased-products). Product Overrides [#product-overrides] Product overrides allow you to dynamically substitute products on paywalls without modifying the paywall design in the Superwall dashboard. When using a `PurchaseController`, you may want to override specific products shown on your paywalls. This is useful for: * A/B testing different subscription tiers * Showing region-specific products * Dynamically changing products based on user segments * Testing promotional pricing without modifying paywalls **How Product Overrides Work:** 1. Product names (e.g., "primary", "secondary") must match exactly as defined in the Superwall dashboard's Paywall Editor 2. The SDK substitutes the original product IDs with your override IDs before fetching from the App Store 3. The paywall maintains its visual design while showing the substituted products 4. Your `PurchaseController` will receive the overridden products when `purchase(product:)` is called Product overrides only affect the products shown on paywalls. They don't change your subscription logic or entitlement validation. # Custom callbacks Available from Android SDK 2.7.0. Overview [#overview] Custom callbacks let a paywall request arbitrary actions from your app and receive results that determine which branch (`onSuccess` / `onFailure`) executes inside the paywall. Common use cases include validating user input, fetching data, or running business logic that lives outside the paywall. How it works [#how-it-works] 1. In the paywall editor, attach a **Custom callback** action to an element (button, form submit, etc.) and give it a name (e.g. `validate_email`). 2. When the user triggers that element the SDK calls your `onCustomCallback` handler with a `CustomCallback` object. 3. Your handler runs whatever logic is needed and returns a `CustomCallbackResult` — either `.success()` or `.failure()` — with optional data. 4. The paywall receives the result and executes the matching `onSuccess` or `onFailure` branch. Setting up the handler [#setting-up-the-handler] Register the handler on a `PaywallPresentationHandler` before calling `register`: ```kotlin val handler = PaywallPresentationHandler() handler.onCustomCallback { callback -> when (callback.name) { "validate_email" -> { val email = callback.variables?.get("email") as? String if (isValidEmail(email)) { CustomCallbackResult.success(mapOf("validated" to true)) } else { CustomCallbackResult.failure(mapOf("error" to "Invalid email")) } } else -> CustomCallbackResult.failure() } } Superwall.instance.register(placement = "campaign_trigger", handler = handler) { // Feature launched } ``` CustomCallback [#customcallback] The `CustomCallback` data class is passed to your handler: CustomCallbackResult [#customcallbackresult] Return one of the following from your handler to signal the outcome: ```kotlin // Success — the paywall's onSuccess branch runs CustomCallbackResult.success(data = mapOf("key" to "value")) // Failure — the paywall's onFailure branch runs CustomCallbackResult.failure(data = mapOf("error" to "Something went wrong")) ``` Both `success()` and `failure()` accept an optional `data` map whose values are sent back to the paywall and accessible as `callbacks..data.`. Callback behavior [#callback-behavior] When configuring the custom callback action in the paywall editor you can choose between two behaviors: * **Blocking** — the paywall waits for your handler to return before continuing the tap-action chain. Use this when the next step depends on the result (e.g. form validation). * **Non-blocking** — the paywall continues immediately. The `onSuccess` / `onFailure` handlers still fire when the result arrives, but subsequent actions in the chain do not wait. Accessing returned data in the paywall [#accessing-returned-data-in-the-paywall] Inside the paywall you can reference the returned data using the pattern `callbacks..data.`. For example, if the callback named `validate_email` returns `mapOf("validated" to true)`, the paywall can access `callbacks.validate_email.data.validated`. # Custom Paywall Actions For example, adding a custom action called `help_center` to a button in your paywall gives you the opportunity to present a help center whenever that button is pressed. To set this up, implement `handleCustomPaywallAction(withName:)` in your `SuperwallDelegate`: :::android ```kotlin override fun handleCustomPaywallAction(name: String) { if (name == "help_center") { HelpCenterManager.present() } } ``` :::
Remember to set `Superwall.shared.delegate`! For implementation details, see the [Superwall Delegate](/docs/sdk/guides/using-superwall-delegate) guide. # Purchasing Products Outside of a Paywall If you're using Superwall for revenue tracking, but want a hand with making purchases in your implementation, you can use our `purchase` methods: :::android ```kotlin Android // Purchase product with first available baseplan Superwall.instance.purchase("my_product_id") // Purchase product with base plan and cheapest/free offer Superwall.instance.purchase("my_product_id:base_plan:sw-auto") // Purchase product with a specified offer Superwall.instance.purchase("my_product_id:base_plan:offer") // Purchase product with no offer Superwall.instance.purchase("my_product_id:base_plan:sw-none") ``` ::: For iOS, the `purchase()` method supports StoreKit 1, 2 and Superwall's abstraction over a product, `StoreProduct`. You can fetch the products you've added to Superwall via the `products(for:)` method. Similarly, in Android, you can fetch a product using a product identifier — and the first base plan will be selected: :::android ```kotlin Android val products = Superwall.instance.getProducts("id1", "id2") ``` ::: If you already have your own product fetching code, simply pass the product representation to these methods. For example, in StoreKit 1 — an `SKProduct` instance, in StoreKit 2, `Product`, etc. Each `purchase()` implementation returns a `PurchaseResult`, which informs you of the transaction's resolution: * `.cancelled`: The purchase was cancelled. * `.purchased`: The product was purchased. * `.pending`: The purchase is pending/deferred and requires action from the developer. * `.failed(Error)`: The purchase failed for a reason other than the user cancelling or the payment pending. # Game Controller Support :::android First set the `SuperwallOption` `isGameControllerEnabled` to `true`: ```kotlin Superwall.instance.options.isGameControllerEnabled = true ``` Then Superwall will automatically listen for gamepad events and forward them to your paywall! ::: :::flutter First set the `SuperwallOption` `isGameControllerEnabled` to `true`: ```dart Superwall.instance.options.isGameControllerEnabled = true ``` Then Superwall will automatically listen for gamepad events and forward them to your paywall! ::: :::expo First set the `SuperwallOption` `isGameControllerEnabled` to `true`: ```typescript Superwall.instance.options.isGameControllerEnabled = true ``` Then Superwall will automatically listen for gamepad events and forward them to your paywall! ::: # Observer Mode If you wish to make purchases outside of Superwall's SDK and paywalls, you can use **observer mode** to report purchases that will appear in the Superwall dashboard, such as transactions: This is useful if you are using Superwall solely for revenue tracking, and you're making purchases using frameworks like StoreKit or Google Play Billing Library directly. Observer mode will also properly link user identifiers to transactions. To enable observer mode, set it using `SuperwallOptions` when configuring the SDK: :::android ```kotlin Android (Automatic) val options = SuperwallOptions() options.shouldObservePurchases = true Superwall.configure(this, "your_api_key", options = options) // In your purchase methods, replace Google's billing code with Superwall's proxy val productDetailsParams = SuperwallBillingFlowParams.ProductDetailsParams .newBuilder() .setOfferToken("test_offer_token") .setProductDetails(mockProductDetails) .build() val params = SuperwallBillingFlowParams .newBuilder() .setIsOfferPersonalized(true) .setObfuscatedAccountId(...) .setObfuscatedProfileId(...) .setProductDetailsParamsList(listOf(productDetailsParams)) .build() billingClient.launchBillingFlowWithSuperwall() ``` ```kotlin Android (Manual) val options = SuperwallOptions() options.shouldObservePurchases = true Superwall.configure(this, "your_api_key", options = options) // In your purchase methods, begin by tracking the purchase start Superwall.instance.observePurchaseStart(productDetails) // On succesful purchase Superwall.instance.observePurchaseResult(billingResult, purchases) // On purchase error Superwall.instance.observePurchaseError(productDetails, error) ``` ::: There are a few things to keep in mind when using observer mode: 1. On iOS, if you're using StoreKit 2, then Superwall solely reports transaction completions. If you're using StoreKit 1, then Superwall will report transaction starts, abandons, and completions. 2. When using observer mode, you can't make purchases using our SDK — such as `Superwall.shared.purchase(aProduct)`. For more on setting up revenue tracking, check out this [doc](/docs/dashboard/dashboard-settings/overview-settings-revenue-tracking). # Retrieving and Presenting a Paywall Yourself If you want complete control over the paywall presentation process, you can use `getPaywall(forPlacement:params:paywallOverrides:delegate:)`. This returns the `UIViewController` subclass `PaywallViewController`, which you can then present however you like. Or, you can use a SwiftUI `View` via `PaywallView`. The following is code is how you'd mimic [register](/docs/sdk/quickstart/feature-gating): ```swift Swift final class MyViewController: UIViewController { private func presentPaywall() async { do { // 1 let paywallVc = try await Superwall.shared.getPaywall( forPlacement: "campaign_trigger", delegate: self ) self.present(paywallVc, animated: true) } catch let skippedReason as PaywallSkippedReason { // 2 switch skippedReason { case .holdout, .noAudienceMatch, .placementNotFound: break } } catch { // 3 print(error) } } private func launchFeature() { // Insert code to launch a feature that's behind your paywall. } } // 4 extension MyViewController: PaywallViewControllerDelegate { func paywall( _ paywall: PaywallViewController, didFinishWith result: PaywallResult, shouldDismiss: Bool ) { if shouldDismiss { paywall.dismiss(animated: true) } switch result { case .purchased, .restored: launchFeature() case .declined: let closeReason = paywall.info.closeReason let featureGating = paywall.info.featureGatingBehavior if closeReason != .forNextPaywall && featureGating == .nonGated { launchFeature() } } } } ``` ```swift Objective-C @interface MyViewController : UIViewController - (void)presentPaywall; @end @interface MyViewController () @end @implementation MyViewController - (void)presentPaywall { // 1 [[Superwall sharedInstance] getPaywallForEvent:@"campaign_trigger" params:nil paywallOverrides:nil delegate:self completion:^(SWKGetPaywallResult * _Nonnull result) { if (result.paywall != nil) { [self presentViewController:result.paywall animated:YES completion:nil]; } else if (result.skippedReason != SWKPaywallSkippedReasonNone) { switch (result.skippedReason) { // 2 case SWKPaywallSkippedReasonHoldout: case SWKPaywallSkippedReasonUserIsSubscribed: case SWKPaywallSkippedReasonEventNotFound: case SWKPaywallSkippedReasonNoRuleMatch: case SWKPaywallSkippedReasonNone: break; }; } else if (result.error) { // 3 NSLog(@"%@", result.error); } }]; } -(void)launchFeature { // Insert code to launch a feature that's behind your paywall. } // 4 - (void)paywall:(SWKPaywallViewController *)paywall didFinishWithResult:(enum SWKPaywallResult)result shouldDismiss:(BOOL)shouldDismiss { if (shouldDismiss) { [paywall dismissViewControllerAnimated:true completion:nil]; } SWKPaywallCloseReason closeReason; SWKFeatureGatingBehavior featureGating; switch (result) { case SWKPaywallResultPurchased: case SWKPaywallResultRestored: [self launchFeature]; break; case SWKPaywallResultDeclined: closeReason = paywall.info.closeReason; featureGating = paywall.info.featureGatingBehavior; if (closeReason != SWKPaywallCloseReasonForNextPaywall && featureGating == SWKFeatureGatingBehaviorNonGated) { [self launchFeature]; } break; } } @end ``` ```swift SwiftUI import SuperwallKit struct MyAwesomeApp: App { @State var store: AppStore = .init() init() { Superwall.configure(apiKey: "MyAPIKey") } var body: some Scene { WindowGroup { ContentView() .fullScreenCover(isPresented: $store.showPaywall) { // You can just use 'placement' at a minimum. The 'feature' // Closure fires if they convert PaywallView(placement: "a_placement", onSkippedView: { skip in switch skip { case .userIsSubscribed, .holdout(_), .noRuleMatch, .eventNotFound: MySkipView() } }, onErrorView: { error in MyErrorView() }, feature: { // User is subscribed as a result of the paywall purchase // Or they already were (which would happen in `onSkippedView`) }) } } } } ``` ```kotlin Kotlin // This is an example of how to use `getPaywall` to use a composable` import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.material3.CircularProgressIndicator import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.viewinterop.AndroidView import com.superwall.sdk.Superwall import com.superwall.sdk.paywall.presentation.get_paywall.getPaywall import com.superwall.sdk.paywall.presentation.internal.request.PaywallOverrides import com.superwall.sdk.paywall.vc.PaywallView import com.superwall.sdk.paywall.vc.delegate.PaywallViewCallback @Composable fun PaywallComposable( event: String, params: Map? = null, paywallOverrides: PaywallOverrides? = null, callback: PaywallViewCallback, errorComposable: @Composable ((Throwable) -> Unit) = { error: Throwable -> // Default error composable Text(text = "No paywall to display") }, loadingComposable: @Composable (() -> Unit) = { // Default loading composable Box(modifier = Modifier.fillMaxSize()) { Column( modifier = Modifier.align(Alignment.Center), verticalArrangement = Arrangement.Center, horizontalAlignment = Alignment.CenterHorizontally ) { CircularProgressIndicator() } } } ) { val viewState = remember { mutableStateOf(null) } val errorState = remember { mutableStateOf(null) } val context = LocalContext.current LaunchedEffect(Unit) { PaywallBuilder(event) .params(params) .overrides(paywallOverrides) .delegate(delegate) .activity(context as Activity) .build() .fold(onSuccess = { viewState.value = it }, onFailure = { errorState.value = it }) } when { viewState.value != null -> { viewState.value?.let { viewToRender -> DisposableEffect(viewToRender) { viewToRender.onViewCreated() onDispose { viewToRender.beforeOnDestroy() viewToRender.encapsulatingActivity = null CoroutineScope(Dispatchers.Main).launch { viewToRender.destroyed() } } } AndroidView( factory = { context -> viewToRender } ) } } errorState.value != null -> { errorComposable(errorState.value!!) } else -> { loadingComposable() } } } ``` This does the following: 1. Gets the paywall view controller. 2. Handles the cases where the paywall was skipped. 3. Catches any presentation errors. 4. Implements the delegate. This is called when the user is finished with the paywall. First, it checks `shouldDismiss`. If this is true then is dismissed the paywall from view before launching any features. This may depend on the `result` depending on how you first presented your view. Then, it switches over the `result`. If the result is `purchased` or `restored` the feature can be launched. However, if the result is `declined`, it checks that the the `featureGating` property of `paywall.info` is `nonGated` and that the `closeReason` isn't `.forNextPaywall`. Best practices [#best-practices] 1. **Make sure to prevent a paywall from being accessed after a purchase has occurred**. If a user purchases from a paywall, it is your responsibility to make sure that the user can't access that paywall again. For example, if after successful purchase you decide to push a new view on to the navigation stack, you should make sure that the user can't go back to access the paywall. 2. **Make sure the paywall view controller deallocates before presenting it elsewhere**. If you have a paywall view controller presented somewhere and you try to present the same view controller elsewhere, you will get a crash. For example, you may have a paywall in a tab bar controller, and then you also try to present it modally. We plan on improving this, but currently it's your responsibility to ensure this doesn't happen. # Request permissions from paywalls Overview [#overview] Use the **Request permission** action in the paywall editor when you want to gate features behind Android permissions without bouncing users back to native screens. When the user taps the element, the SDK: * Presents the corresponding Android system dialog. * Emits analytics events (`permission_requested`, `permission_granted`, `permission_denied`). * Sends the result back to the paywall so you can branch the UI (for example, swap a checklist item for a success state). Add the action in the editor [#add-the-action-in-the-editor] 1. Open your paywall, select the button (or any element) that should prompt the permission, and set its action to **Request permission**. 2. Choose the permission you want to request. You can wire multiple buttons if you need to prime several permissions in a single flow. 3. Republish the paywall. No extra SDK configuration is required beyond having the proper `AndroidManifest.xml` entries. Declare the permissions in `AndroidManifest.xml` [#declare-the-permissions-in-androidmanifestxml] | Editor option | `permission_type` sent from the paywall | Required manifest entries | Notes | | --------------------- | --------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------ | ---------------------------------------------------------------------------------------------------- | | Notifications | `notification` | `` (API 33+) | Devices below Android 13 do not require a runtime permission; the SDK reports `granted` immediately. | | Location (Foreground) | `location` | `` | Also covers coarse location because FINE implies COARSE. | | Location (Background) | `background_location` | Foreground entry above **and** `` (API 29+) | The SDK first ensures foreground access, then escalates to background. | | Photos / Images | `read_images` | `` (API 33+) or `READ_EXTERNAL_STORAGE` for older OS versions | Automatically picks the right permission at runtime. | | Videos | `read_video` | `` (API 33+) or `READ_EXTERNAL_STORAGE` pre-33 | | | Contacts | `contacts` | `` | | | Camera | `camera` | `` | | | Microphone | `microphone` | `` | Added in 2.6.8. | If a manifest entry is missing—or the permission is unsupported on the current OS level—the SDK responds with an `unsupported` status so you can show fallback copy. Analytics and delegate callbacks [#analytics-and-delegate-callbacks] Forward the new events through `SuperwallDelegate.handleSuperwallEvent` to keep your analytics platform and feature flags in sync: ```kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when (val event = eventInfo.event) { is SuperwallEvent.PermissionRequested -> { analytics.track("permission_requested", mapOf( "permission" to event.permissionName, "paywall_id" to event.paywallIdentifier )) } is SuperwallEvent.PermissionGranted -> { FeatureFlags.unlock(event.permissionName) } is SuperwallEvent.PermissionDenied -> { Alerts.showPermissionDeclinedSheet(event.permissionName) } else -> Unit } } ``` You can also log the newer [`customerInfoDidChange`](/docs/android/sdk-reference/SuperwallDelegate#customerinfodidchangefrom-customerinfo-to-customerinfo) callback if the permission subsequently unlocks new paywalls that grant entitlements. Status values returned to the paywall [#status-values-returned-to-the-paywall] The paywall receives a `permission_result` web event with: * `granted` – The system dialog reported success (or no dialog was needed). * `denied` – The user denied the request or previously denied it. * `unsupported` – The platform or manifest doesn't allow the requested permission. Use Liquid or custom Javascript inside the paywall to branch on these statuses—for example, replace a “Grant notification access” button with a checkmark when the result equals `granted`. Troubleshooting [#troubleshooting] * Seeing `unsupported`? Double-check the manifest entries above and confirm the permission exists on the device's API level (for example, notification permissions only apply on Android 13+). * Nothing happens when you tap the button? Ensure the action is set to **Request permission** in the released paywall version. * Want to provide next steps after a denial? Listen for `PermissionDenied` in your delegate to deep-link users into Settings or show educational copy. # Device Tier Targeting The `deviceTier` variable allows you to create targeted audiences based on device performance capabilities. This helps optimize paywall experiences by showing resource-appropriate content to different device types. You can reference this in campaign filters, dynamic values, or in paywall text via the `device.deviceTier` variable. Device tier targeting is available starting in Android SDK version `2.2.3`. Make sure you're using this version or later to access this feature. How device tier works [#how-device-tier-works] Device tier classification is based on several hardware factors: * CPU performance * Available RAM * 4K/2K codec support * Display quality This automatic classification helps you deliver paywalls that perform well across the full spectrum of Android devices. Matching device ranges [#matching-device-ranges] When creating device tier filters, you can use `contains` or `equals` for narrower matching: * **`contains`** - Broader matching that includes partial matches. For example, `deviceTier contains high` matches both `high` and `ultra_high` devices. * **`equals`** - Exact matching for precise targeting. For example, `deviceTier equals high` matches only `high` tier devices, not `ultra_high`. Use `contains` when you want to target a range of similar device capabilities, and `equals` when you need precise control over which specific tier to target. Device tier values [#device-tier-values] The `device.deviceTier` attribute returns one of these values: * **`ultraLow`** - Entry-level devices with limited resources. * **`low`** - Budget devices with basic performance capabilities. * **`mid`** - Mid-range devices with moderate performance. * **`high`** - Premium devices with strong performance. * **`ultra_high`** - Flagship devices with top-tier specifications. * **`unknown`** - Device tier couldn't be determined. Creating device tier audiences [#creating-device-tier-audiences] To target users by device tier, create an audience using the `device.deviceTier` attribute: 1. Navigate to **Campaigns** in your dashboard. 2. Click on the campaign you want to target. 3. Edit, or create, a new audience. 4. Add a filter where `device.deviceTier` contains your target tier(s). 5. Save your audience. You can target multiple tiers in a single audience. For example, use `deviceTier contains LOW` to target both `ultraLow` and `low` tier devices. Optimizing for lower-end devices [#optimizing-for-lower-end-devices] Create lightweight paywalls for devices that may struggle with resource-intensive content: ``` device.deviceTier contains ultraLow OR device.deviceTier contains low ``` Show these users paywalls with: * Static images instead of videos. * Compressed media files. * Simplified animations. # Using the Presentation Handler You can provide a `PaywallPresentationHandler` to `register`, whose functions provide status updates for a paywall: * `onDismiss`: Called when the paywall is dismissed. Accepts a `PaywallInfo` object containing info about the dismissed paywall, and there is a `PaywallResult` informing you of any transaction. * `onPresent`: Called when the paywall did present. Accepts a `PaywallInfo` object containing info about the presented paywall. * `onError`: Called when an error occurred when trying to present a paywall. Accepts an `Error` indicating why the paywall could not present. * `onSkip`: Called when a paywall is skipped. Accepts a `PaywallSkippedReason` enum indicating why the paywall was skipped. * `onCustomCallback` *(Android 2.7.0+)*: Called when the paywall requests a custom callback. Accepts a `CustomCallback` containing the callback name and optional variables, and returns a `CustomCallbackResult` indicating success or failure with optional data to pass back to the paywall. ```swift Swift let handler = PaywallPresentationHandler() handler.onDismiss { paywallInfo, result in print("The paywall dismissed. PaywallInfo: \(paywallInfo). Result: \(result)") } handler.onPresent { paywallInfo in print("The paywall presented. PaywallInfo:", paywallInfo) } handler.onError { error in print("The paywall presentation failed with error \(error)") } handler.onSkip { reason in switch reason { case .holdout(let experiment): print("Paywall not shown because user is in a holdout group in Experiment: \(experiment.id)") case .noAudienceMatch: print("Paywall not shown because user doesn't match any audiences.") case .placementNotFound: print("Paywall not shown because this placement isn't part of a campaign.") } } Superwall.shared.register(placement: "campaign_trigger", handler: handler) { // Feature launched } ``` ```swift Objective-C SWKPaywallPresentationHandler *handler = [[SWKPaywallPresentationHandler alloc] init]; [handler onDismiss:^(SWKPaywallInfo * _Nonnull paywallInfo, enum SWKPaywallResult result, SWKStoreProduct * _Nullable product) { NSLog(@"The paywall presented. PaywallInfo: %@ - result: %ld", paywallInfo, (long)result); }]; [handler onPresent:^(SWKPaywallInfo * _Nonnull paywallInfo) { NSLog(@"The paywall presented. PaywallInfo: %@", paywallInfo); }]; [handler onError:^(NSError * _Nonnull error) { NSLog(@"The paywall presentation failed with error %@", error); }]; [handler onSkip:^(enum SWKPaywallSkippedReason reason) { switch (reason) { case SWKPaywallSkippedReasonUserIsSubscribed: NSLog(@"Paywall not shown because user is subscribed."); break; case SWKPaywallSkippedReasonHoldout: NSLog(@"Paywall not shown because user is in a holdout group."); break; case SWKPaywallSkippedReasonNoAudienceMatch: NSLog(@"Paywall not shown because user doesn't match any audiences."); break; case SWKPaywallSkippedReasonPlacementNotFound: NSLog(@"Paywall not shown because this placement isn't part of a campaign."); break; case SWKPaywallSkippedReasonNone: // The paywall wasn't skipped. break; } }]; [[Superwall sharedInstance] registerWithPlacement:@"campaign_trigger" params:nil handler:handler feature:^{ // Feature launched. }]; ``` ```kotlin Kotlin val handler = PaywallPresentationHandler() handler.onDismiss { paywallInfo, result -> println("The paywall dismissed. PaywallInfo: ${it}") } handler.onPresent { println("The paywall presented. PaywallInfo: ${it}") } handler.onError { println("The paywall errored. Error: ${it}") } handler.onSkip { when (it) { is PaywallSkippedReason.PlacementNotFound -> { println("The paywall was skipped because the placement was not found.") } is PaywallSkippedReason.Holdout -> { println("The paywall was skipped because the user is in a holdout group.") } is PaywallSkippedReason.NoAudienceMatch -> { println("The paywall was skipped because no audience matched.") } } } Superwall.instance.register(placement = "campaign_trigger", handler = handler) { // Feature launched } ``` ```dart Flutter PaywallPresentationHandler handler = PaywallPresentationHandler(); handler.onPresent((paywallInfo) async { String name = await paywallInfo.name; print("Handler (onPresent): $name"); }); handler.onDismiss((paywallInfo, paywallResult) async { String name = await paywallInfo.name; print("Handler (onDismiss): $name"); }); handler.onError((error) { print("Handler (onError): ${error}"); }); handler.onSkip((skipReason) async { String description = await skipReason.description; if (skipReason is PaywallSkippedReasonHoldout) { print("Handler (onSkip): $description"); final experiment = await skipReason.experiment; final experimentId = await experiment.id; print("Holdout with experiment: ${experimentId}"); } else if (skipReason is PaywallSkippedReasonNoAudienceMatch) { print("Handler (onSkip): $description"); } else if (skipReason is PaywallSkippedReasonPlacementNotFound) { print("Handler (onSkip): $description"); } else { print("Handler (onSkip): Unknown skip reason"); } }); Superwall.shared.registerPlacement("campaign_trigger", handler: handler, feature: () { // Feature launched }); ``` ```typescript React Native const handler = new PaywallPresentationHandler() handler.onPresent((paywallInfo) => { const name = paywallInfo.name console.log(`Handler (onPresent): ${name}`) }) handler.onDismiss((paywallInfo, paywallResult) => { const name = paywallInfo.name console.log(`Handler (onDismiss): ${name}`) }) handler.onError((error) => { console.log(`Handler (onError): ${error}`) }) handler.onSkip((skipReason) => { const description = skipReason.description if (skipReason instanceof PaywallSkippedReasonHoldout) { console.log(`Handler (onSkip): ${description}`) const experiment = skipReason.experiment const experimentId = experiment.id console.log(`Holdout with experiment: ${experimentId}`) } else if (skipReason instanceof PaywallSkippedReasonNoAudienceMatch) { console.log(`Handler (onSkip): ${description}`) } else if (skipReason instanceof PaywallSkippedReasonPlacementNotFound) { console.log(`Handler (onSkip): ${description}`) } else { console.log(`Handler (onSkip): Unknown skip reason`) } }) Superwall.shared.register({ placement: 'campaign_trigger', handler: handler, feature: () => { // Feature launched } }); ``` Wanting to see which product was just purchased from a paywall? Use `onDismiss` and the `result` parameter. Or, you can use the [SuperwallDelegate](/docs/sdk/guides/3rd-party-analytics#using-events-to-see-purchased-products). # Viewing Purchased Products When a paywall is presenting and a user converts, you can view the purchased products in several different ways. Use the `PaywallPresentationHandler` [#use-the-paywallpresentationhandler] Arguably the easiest of the options — simply pass in a presentation handler and check out the product within the `onDismiss` block. ```swift Swift let handler = PaywallPresentationHandler() handler.onDismiss { _, result in switch result { case .declined: print("No purchased occurred.") case .purchased(let product): print("Purchased \(product.productIdentifier)") case .restored: print("Restored purchases.") } } Superwall.shared.register(placement: "caffeineLogged", handler: handler) { logCaffeine() } ``` ```swift Objective-C SWKPaywallPresentationHandler *handler = [SWKPaywallPresentationHandler new]; [handler onDismiss:^(SWKPaywallInfo * _Nonnull info, enum SWKPaywallResult result, SWKStoreProduct * _Nullable product) { switch (result) { case SWKPaywallResultPurchased: NSLog(@"Purchased %@", product.productIdentifier); default: NSLog(@"Unhandled event."); } }]; [[Superwall sharedInstance] registerWithPlacement:@"caffeineLogged" params:@{} handler:handler feature:^{ [self logCaffeine]; }]; ``` ```kotlin Android val handler = PaywallPresentationHandler() handler.onDismiss { _, paywallResult -> when (paywallResult) { is PaywallResult.Purchased -> { // The user made a purchase! val purchasedProductId = paywallResult.productId println("User purchased product: $purchasedProductId") // ... do something with the purchased product ID ... } is PaywallResult.Declined -> { // The user declined to make a purchase. println("User declined to make a purchase.") // ... handle the declined case ... } is PaywallResult.Restored -> { // The user restored a purchase. println("User restored a purchase.") // ... handle the restored case ... } } } Superwall.instance.register(placement = "caffeineLogged", handler = handler) { logCaffeine() } ``` ```dart Flutter PaywallPresentationHandler handler = PaywallPresentationHandler(); handler.onDismiss((paywallInfo, paywallResult) async { String name = await paywallInfo.name; print("Handler (onDismiss): $name"); switch (paywallResult) { case PurchasedPaywallResult(productId: var id): // The user made a purchase! print('User purchased product: $id'); // ... do something with the purchased product ID ... break; case DeclinedPaywallResult(): // The user declined to make a purchase. print('User declined the paywall.'); // ... handle the declined case ... break; case RestoredPaywallResult(): // The user restored a purchase. print('User restored a previous purchase.'); // ... handle the restored case ... break; } }); Superwall.shared.registerPlacement( "caffeineLogged", handler: handler, feature: () { logCaffeine(); }); ``` ```typescript React Native import * as React from "react" import Superwall from "../../src" import { PaywallPresentationHandler, PaywallInfo } from "../../src" import type { PaywallResult } from "../../src/public/PaywallResult" const Home = () => { const navigation = useNavigation() const presentationHandler: PaywallPresentationHandler = { onDismiss: (handler: (info: PaywallInfo, result: PaywallResult) => void) => { handler = (info, result) => { console.log("Paywall dismissed with info:", info, "and result:", result) if (result.type === "purchased") { console.log("Product purchased with ID:", result.productId) } } }, onPresent: (handler: (info: PaywallInfo) => void) => { handler = (info) => { console.log("Paywall presented with info:", info) // Add logic for when the paywall is presented } }, onError: (handler: (error: string) => void) => { handler = (error) => { console.error("Error presenting paywall:", error) // Handle any errors that occur during presentation } }, onSkip: () => { console.log("Paywall presentation skipped") // Handle the case where the paywall presentation is skipped }, } const nonGated = () => { Superwall.shared.register({ placement: "non_gated", handler: presentationHandler, feature: () => { navigation.navigate("caffeineLogged", { value: "Go for caffeine logging", }) }); } return // Your view code here } ``` Use `SuperwallDelegate` [#use-superwalldelegate] Next, the [SuperwallDelegate](/docs/sdk/guides/using-superwall-delegate) offers up much more information, and can inform you of virtually any Superwall event that occurred: ```swift Swift class SWDelegate: SuperwallDelegate { func handleSuperwallEvent(withInfo eventInfo: SuperwallEventInfo) { switch eventInfo.event { case .transactionComplete(_, let product, _, _): print("Transaction complete: product: \(product.productIdentifier)") case .subscriptionStart(let product, _): print("Subscription start: product: \(product.productIdentifier)") case .freeTrialStart(let product, _): print("Free trial start: product: \(product.productIdentifier)") case .transactionRestore(_, _): print("Transaction restored") case .nonRecurringProductPurchase(let product, _): print("Consumable product purchased: \(product.id)") default: print("Unhandled event.") } } } @main struct Caffeine_PalApp: App { @State private var swDelegate: SWDelegate = .init() init() { Superwall.configure(apiKey: "my_api_key") Superwall.shared.delegate = swDelegate } var body: some Scene { WindowGroup { ContentView() } } } ``` ```swift Objective-C // SWDelegate.h... #import @import SuperwallKit; NS_ASSUME_NONNULL_BEGIN @interface SWDelegate : NSObject @end NS_ASSUME_NONNULL_END // SWDelegate.m... @implementation SWDelegate - (void)handleSuperwallEventWithInfo:(SWKSuperwallEventInfo *)eventInfo { switch(eventInfo.event) { case SWKSuperwallEventTransactionComplete: NSLog(@"Transaction complete: %@", eventInfo.params[@"primary_product_id"]); } } // In AppDelegate.m... #import "AppDelegate.h" #import "SWDelegate.h" @import SuperwallKit; @interface AppDelegate () @property (strong, nonatomic) SWDelegate *delegate; @end @implementation AppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // Override point for customization after application launch. self.delegate = [SWDelegate new]; [Superwall configureWithApiKey:@"my_api_key"]; [Superwall sharedInstance].delegate = self.delegate; return YES; } ``` ```kotlin Android class SWDelegate : SuperwallDelegate { override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when (eventInfo.event) { is SuperwallPlacement.TransactionComplete -> { val transaction = (eventInfo.event as SuperwallPlacement.TransactionComplete).transaction val product = (eventInfo.event as SuperwallPlacement.TransactionComplete).product val paywallInfo = (eventInfo.event as SuperwallPlacement.TransactionComplete).paywallInfo println("Transaction Complete: $transaction, Product: $product, Paywall Info: $paywallInfo") } else -> { // Handle other cases } } } } class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure(this, "my_api_key") Superwall.instance.delegate = SWDelegate() } } ``` ```dart Flutter import 'dart:io'; import 'package:flutter/material.dart'; import 'package:superwallkit_flutter/superwallkit_flutter.dart'; class _MyAppState extends State implements SuperwallDelegate { final logging = Logging(); @override void initState() { super.initState(); configureSuperwall(useRevenueCat); } Future configureSuperwall(bool useRevenueCat) async { try { final apiKey = Platform.isIOS ? 'ios_api_project_key' : 'android_api_project_key'; final logging = Logging(); logging.level = LogLevel.warn; logging.scopes = {LogScope.all}; final options = SuperwallOptions(); options.paywalls.shouldPreload = false; options.logging = logging; Superwall.configure(apiKey, purchaseController: null, options: options, completion: () { logging.info('Executing Superwall configure completion block'); }); Superwall.shared.setDelegate(this); } catch (e) { // Handle any errors that occur during configuration logging.error('Failed to configure Superwall:', e); } } @override Future handleSuperwallEvent(SuperwallEventInfo eventInfo) async { switch (eventInfo.event.type) { case PlacementType.transactionComplete: final product = eventInfo.params?['product']; logging.info('Transaction complete event received with product: $product'); // Add any additional logic you need to handle the transaction complete event break; // Handle other events if necessary default: logging.info('Unhandled event type: ${eventInfo.event.type}'); break; } } } ``` ```typescript React Native import { PaywallInfo, SubscriptionStatus, SuperwallDelegate, SuperwallPlacementInfo, PlacementType, } from '../../src'; export class MySuperwallDelegate extends SuperwallDelegate { handleSuperwallPlacement(placementInfo: SuperwallPlacementInfo) { console.log('Handling Superwall placement:', placementInfo); switch (placementInfo.placement.type) { case PlacementType.transactionComplete: const product = placementInfo.params?.["product"]; if (product) { console.log(`Product: ${product}`); } else { console.log("Product not found in params."); } break; default: break; } } } export default function App() { const delegate = new MySuperwallDelegate(); React.useEffect(() => { const setupSuperwall = async () => { const apiKey = Platform.OS === 'ios' ? 'ios_api_project_key' : 'android_api_project_key'; Superwall.configure({ apiKey: apiKey, }); Superwall.shared.setDelegate(delegate); }; } } ``` Use a purchase controller [#use-a-purchase-controller] If you are controlling the purchasing pipeline yourself via a [purchase controller](/docs/sdk/guides/advanced-configuration), then naturally the purchased product is available: ```swift Swift final class MyPurchaseController: PurchaseController { func purchase(product: StoreProduct) async -> PurchaseResult { print("Kicking off purchase of \(product.productIdentifier)") do { let result = try await MyPurchaseLogic.purchase(product: product) return .purchased // .cancelled, .pending, .failed(Error) } catch { return .failed(error) } } // 2 func restorePurchases() async -> RestorationResult { print("Restoring purchases") return .restored // false } } @main struct Caffeine_PalApp: App { private let pc: MyPurchaseController = .init() init() { Superwall.configure(apiKey: "my_api_key", purchaseController: pc) } var body: some Scene { WindowGroup { ContentView() } } } ``` ```swift Objective-C // In MyPurchaseController.h... #import @import SuperwallKit; @import StoreKit; NS_ASSUME_NONNULL_BEGIN @interface MyPurchaseController : NSObject + (instancetype)sharedInstance; @end NS_ASSUME_NONNULL_END // In MyPurchaseController.m... #import "MyPurchaseController.h" @implementation MyPurchaseController + (instancetype)sharedInstance { static MyPurchaseController *sharedInstance = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ sharedInstance = [MyPurchaseController new]; }); return sharedInstance; } - (void)purchaseWithProduct:(SWKStoreProduct * _Nonnull)product completion:(void (^ _Nonnull)(enum SWKPurchaseResult, NSError * _Nullable))completion { NSLog(@"Kicking off purchase of %@", product.productIdentifier); // Do purchase logic here completion(SWKPurchaseResultPurchased, nil); } - (void)restorePurchasesWithCompletion:(void (^ _Nonnull)(enum SWKRestorationResult, NSError * _Nullable))completion { // Do restore logic here completion(SWKRestorationResultRestored, nil); } @end // In AppDelegate.m... #import "AppDelegate.h" #import "MyPurchaseController.h" @import SuperwallKit; @interface AppDelegate () @end @implementation AppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // Override point for customization after application launch. [Superwall configureWithApiKey:@"my_api_key" purchaseController:[MyPurchaseController sharedInstance] options:nil completion:^{ }]; return YES; } ``` ```kotlin Android class MyPurchaseController(val context: Context): PurchaseController { override suspend fun purchase( activity: Activity, productDetails: ProductDetails, basePlanId: String?, offerId: String? ): PurchaseResult { println("Kicking off purchase of $basePlanId") return PurchaseResult.Purchased() } override suspend fun restorePurchases(): RestorationResult { TODO("Not yet implemented") } } class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure(this, "my_api_key", purchaseController = MyPurchaseController(this)) } } ``` ```dart Flutter class MyPurchaseController extends PurchaseController { // 1 @override Future purchaseFromAppStore(String productId) async { print('Attempting to purchase product with ID: $productId'); // Do purchase logic return PurchaseResult.purchased; } @override Future purchaseFromGooglePlay( String productId, String? basePlanId, String? offerId ) async { print('Attempting to purchase product with ID: $productId and basePlanId: $basePlanId'); // Do purchase logic return PurchaseResult.purchased; } @override Future restorePurchases() async { // Do resture logic } } ``` ```typescript React Native export class MyPurchaseController extends PurchaseController { // 1 async purchaseFromAppStore(productId: string): Promise { console.log("Kicking off purchase of ", productId) // Purchase logic return await this._purchaseStoreProduct(storeProduct) } async purchaseFromGooglePlay( productId: string, basePlanId?: string, offerId?: string ): Promise { console.log("Kicking off purchase of ", productId, " base plan ID", basePlanId) // Purchase logic return await this._purchaseStoreProduct(storeProduct) } // 2 async restorePurchases(): Promise { // TODO // ---- // Restore purchases and return true if successful. } } ``` SwiftUI - Use `PaywallView` [#swiftui---use-paywallview] The `PaywallView` allows you to show a paywall by sending it a placement. It also has a dismiss handler where the purchased product will be vended: ```swift @main struct Caffeine_PalApp: App { @State private var presentPaywall: Bool = false init() { Superwall.configure(apiKey: "my_api_key") } var body: some Scene { WindowGroup { Button("Log") { presentPaywall.toggle() } .sheet(isPresented: $presentPaywall) { PaywallView(placement: "caffeineLogged", params: nil, paywallOverrides: nil) { info, result in switch result { case .declined: print("No purchased occurred.") case .purchased(let product): print("Purchased \(product.productIdentifier)") case .restored: print("Restored purchases.") } } feature: { print("Converted") presentPaywall.toggle() } } } } } ``` # Advanced Configuration Logging [#logging] Logging is enabled by default in the SDK and is controlled by two properties: `level` and `scopes`. `level` determines the minimum log level to print to the console. There are five types of log level: 1. **debug**: Prints all logs from the SDK to the console. Useful for debugging your app if something isn't working as expected. 2. **info**: Prints errors, warnings, and useful information from the SDK to the console. 3. **warn**: Prints errors and warnings from the SDK to the console. 4. **error**: Only prints errors from the SDK to the console. 5. **none**: Turns off all logs. The SDK defaults to `info`. `scopes` defines the scope of logs to print to the console. For example, you might only care about logs relating to `paywallPresentation` and `paywallTransactions`. This defaults to `.all`. Check out [LogScope](https://sdk.superwall.me/documentation/superwallkit/logscope) for all possible cases. You set these properties like this: :::android ```kotlin val options = SuperwallOptions() options.logging.level = LogLevel.warn options.logging.scopes = EnumSet.of(LogScope.paywallPresentation, LogScope.paywallTransactions) Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or you can set: Superwall.instance.logLevel = LogLevel.warn // Or use the configuration DSL configureSuperwall("MY_API_KEY") { options { logging { level = LogLevel.warn scopes = EnumSet.of(LogScope.paywallPresentation, LogScope.paywallTransactions) } } } ``` ::: Preloading Paywalls [#preloading-paywalls] Paywalls are preloaded by default when the app is launched from a cold start. The paywalls that are preloaded are determined by the list of placements that result in a paywall for the user when [registered](/docs/sdk/quickstart/feature-gating). Preloading is smart, only preloading paywalls that belong to audiences that could be matched. Paywalls are cached by default, which means after they load once, they don't need to be reloaded from the network unless you make a change to them on the dashboard. However, if you have a lot of paywalls, preloading may increase network usage of your app on first load of the paywalls and result in slower loading times overall. To make an onboarding or first-launch paywall load before the rest of your campaigns, prioritize the campaign from the dashboard with [Priority Placements](/docs/dashboard/dashboard-campaigns/campaigns-placements-prioritized). Use the SDK methods below when you need to disable automatic preloading or manually preload specific placements. You can turn off preloading by setting `shouldPreload` to `false`: :::android ```kotlin val options = SuperwallOptions() options.paywalls.shouldPreload = false Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { shouldPreload = false } } } ``` ::: Then, if you'd like to preload paywalls for specific placements you can use `preloadPaywalls(forPlacements:)`: :::android ```kotlin val eventNames = setOf("campaign_trigger") Superwall.instance.preloadPaywalls(eventNames) ``` ::: If you'd like to preload all paywalls you can use `preloadAllPaywalls()`: :::android ```kotlin Superwall.instance.preloadAllPaywalls() ``` ::: Note: These methods will not reload any paywalls that have already been preloaded. External Data Collection [#external-data-collection] By default, Superwall sends all registered events and properties back to the Superwall servers. However, if you have privacy concerns, you can stop this by setting `isExternalDataCollectionEnabled` to `false`: :::android ```kotlin val options = SuperwallOptions() options.isExternalDataCollectionEnabled = false Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { isExternalDataCollectionEnabled = false } } ``` ::: Disabling this will not affect your ability to create triggers based on properties. Automatically Dismissing the Paywall [#automatically-dismissing-the-paywall] By default, Superwall automatically dismisses the paywall when a product is purchased or restored. You can disable this by setting `automaticallyDismiss` to `false`: :::android ```kotlin val options = SuperwallOptions() options.paywalls.automaticallyDismiss = false Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { automaticallyDismiss = false } } } ``` ::: To manually dismiss the paywall , call `Superwall.shared.dismiss()`. Custom Restore Failure Message [#custom-restore-failure-message] You can set the title, message and close button title for the alert that appears after a restoration failure: :::android ```kotlin val options = SuperwallOptions() options.paywalls.restoreFailed.title = "My Title" options.paywalls.restoreFailed.message = "My message" options.paywalls.restoreFailed.closeButtonTitle = "Close" Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { restoreFailed.title = "My Title" restoreFailed.message = "My message" restoreFailed.closeButtonTitle = "Close" } } } ``` ::: Haptic Feedback [#haptic-feedback] On iOS, the paywall uses haptic feedback by default after a user purchases or restores a product, opens a URL from the paywall, or closes the paywall. To disable this, set the `isHapticFeedbackEnabled` `PaywallOption` to false: Note: Android does not use haptic feedback. Transaction Background View [#transaction-background-view] During a transaction, we add a `UIActivityIndicator` behind the view to indicate a loading status. However, you can remove this by setting the `transactionBackgroundView` to `nil`: :::android ```kotlin val options = SuperwallOptions() options.paywalls.transactionBackgroundView = null Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { transactionBackgroundView = null } } } ``` ::: Purchase Failure Alert [#purchase-failure-alert] When a purchase fails, we automatically present an alert with the error message. If you'd like to show your own alert after failure, set the `shouldShowPurchaseFailureAlert` `PaywallOption` to `false`: :::android ```kotlin val options = SuperwallOptions() options.paywalls.shouldShowPurchaseFailureAlert = false Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { shouldShowPurchaseFailureAlert = false } } } ``` ::: Web Purchase Confirmation Alert [#web-purchase-confirmation-alert] When a user completes a purchase via web checkout (app2web flow), you can control whether to show a confirmation alert. By default, this is set to `false` to prevent duplicate alerts. Set `shouldShowWebPurchaseConfirmationAlert` to `true` if you want to show the native confirmation alert: :::android ```kotlin val options = SuperwallOptions() options.paywalls.shouldShowWebPurchaseConfirmationAlert = true Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { paywalls { shouldShowWebPurchaseConfirmationAlert = true } } } ``` ::: Locale Identifier [#locale-identifier] When evaluating rules, the device locale identifier is set to `autoupdatingCurrent`. However, you can override this if you want to test a specific locale: :::android ```kotlin val options = SuperwallOptions() options.localeIdentifier = "en_GB" Superwall.configure( this, apiKey = "MY_API_KEY", options = options ) // Or using the configuration DSL configureSuperwall("MY_API_KEY") { options { localeIdentifier = "en_GB" } } ``` ::: For a list of locales that are available on iOS, take a look at [this list](https://gist.github.com/jacobbubu/1836273). You can also preview your paywall in different locales using [In-App Previews](/docs/sdk/quickstart/in-app-paywall-previews). Game Controller [#game-controller] If you're using a game controller, you can enable this in `SuperwallOptions` too. Check out our [Game Controller Support](/docs/sdk/guides/advanced/game-controller-support) article. Take a look at [SuperwallOptions](https://sdk.superwall.me/documentation/superwallkit/superwalloptions) in our SDK reference for more info. # Consumable Products Use consumable products when a purchase should grant a quantity that can be used up, such as credits, coins, boosts, or tokens. This guide assumes purchases are made from Superwall paywalls and that you are not using a `PurchaseController`. Consumable products are one-time purchases that users can buy repeatedly, such as credits, tokens, boosts, or packs. Non-consumable products are also one-time purchases, but they grant permanent access, such as a lifetime unlock. Superwall uses entitlements to decide whether a user has ongoing access. Because consumables are meant to be used up, they should usually not grant entitlements. Your app should listen for the purchase, grant the consumable benefit in your own system, and then consume the Google Play purchase token so the item can be purchased again. Dashboard Setup [#dashboard-setup] 1. Create the product as an in-app product in Google Play Console. 2. Add the product in Superwall from **Products**. 3. Use the Google Play product ID. 4. Set **Period** to **None (Lifetime / Consumable)**. 5. Leave **Entitlements** empty. 6. Add the product to any paywall that should sell it. Do not attach an entitlement to a consumable unless the purchase should also unlock ongoing access. If a consumable has no entitlement, buying it does not make the user's subscription status active. Consume The Purchase Token [#consume-the-purchase-token] Google Play in-app products must be consumed after you grant the benefit. If you do not consume the purchase token, the user may not be able to buy that same consumable again. Use the `purchaseToken` from the `TransactionComplete` event, grant the benefit, then call `Superwall.instance.consume(purchaseToken)`. `Superwall.instance.consume(purchaseToken)` is available in Android SDK 2.6.2 and later. ```kotlin Kotlin import androidx.lifecycle.lifecycleScope import com.superwall.sdk.Superwall import com.superwall.sdk.analytics.superwall.SuperwallEvent import com.superwall.sdk.analytics.superwall.SuperwallEventInfo import com.superwall.sdk.delegate.SuperwallDelegate import kotlinx.coroutines.launch class MainActivity : AppCompatActivity(), SuperwallDelegate { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) Superwall.instance.delegate = this } override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { val event = eventInfo.event if (event !is SuperwallEvent.TransactionComplete) { return } if (event.product.productIdentifier != "coins_100") { return } val purchaseToken = event.transaction?.purchaseToken ?: return lifecycleScope.launch { ConsumablesService.grantCoins( count = 100, productId = event.product.productIdentifier, purchaseToken = purchaseToken, ) Superwall.instance.consume(purchaseToken) .onFailure { error -> // Retry consumption after confirming the benefit was granted. println("Failed to consume purchase: ${error.message}") } } } } ``` ```java Java import com.superwall.sdk.Superwall; import com.superwall.sdk.analytics.superwall.SuperwallEvent; import com.superwall.sdk.analytics.superwall.SuperwallEventInfo; import com.superwall.sdk.delegate.SuperwallDelegateJava; import com.superwall.sdk.store.abstractions.transactions.StoreTransactionType; import kotlin.Unit; public class MainActivity extends AppCompatActivity implements SuperwallDelegateJava { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Superwall.getInstance().setJavaDelegate(this); } @Override public void handleSuperwallEvent(SuperwallEventInfo eventInfo) { if (!(eventInfo.getEvent() instanceof SuperwallEvent.TransactionComplete)) { return; } SuperwallEvent.TransactionComplete event = (SuperwallEvent.TransactionComplete) eventInfo.getEvent(); if (!event.getProduct().getProductIdentifier().equals("coins_100")) { return; } StoreTransactionType transaction = event.getTransaction(); if (transaction == null) { return; } String purchaseToken = transaction.getPurchaseToken(); ConsumablesService.grantCoins(100, event.getProduct().getProductIdentifier(), purchaseToken); Superwall.getInstance().consume(purchaseToken, result -> { // Check the Result in your app and retry if consumption fails. return Unit.INSTANCE; }); } } ``` Grant the benefit before consuming the token. If your grant call fails, leave the purchase unconsumed and retry after your app confirms the user received the benefit. Read Purchase History [#read-purchase-history] Consumable and non-consumable purchases appear in `customerInfo.nonSubscriptions`. Use `isConsumable` to distinguish consumables from lifetime purchases. ```kotlin Kotlin val customerInfo = Superwall.instance.getCustomerInfo() val consumables = customerInfo.nonSubscriptions.filter { it.isConsumable } for (purchase in consumables) { println("Consumable purchased: ${purchase.productId}") } ``` ```java Java CustomerInfo customerInfo = Superwall.getInstance().getCustomerInfo(); for (NonSubscriptionTransaction purchase : customerInfo.getNonSubscriptions()) { if (purchase.isConsumable()) { System.out.println("Consumable purchased: " + purchase.getProductId()); } } ``` # Experimental Flags Support for Experimental Flags in Android is not yet available. # Handling Deep Links When your app receives a deep link, you might be tempted to write a switch statement that maps each URL to a specific placement and calls `register`. This works, but it means every time you add a new link or change which paywall shows, you have to ship an app update. A better approach is to pass the URL to `handleDeepLink` and let Superwall's [`deepLink_open`](/docs/dashboard/dashboard-campaigns/campaigns-standard-placements#deeplink_open) standard placement handle the rest. The SDK extracts the URL's path, query parameters, and other components, then fires `deepLink_open` as a placement. You write campaign rules on the dashboard to decide which paywall to show, which means there is no app update required. The problem [#the-problem] Here's a common pattern where deep link routing is hardcoded in the app: :::android ```kotlin fun handleUrl(url: Uri) { val placement = when (url.path) { "/promo" -> "promoPlacement" "/onboarding" -> "onboardingPlacement" "/upgrade" -> "upgradePlacement" "/special-offer" -> "specialOfferPlacement" else -> null } placement?.let { Superwall.instance.register(placement = it) } } ``` ::: Every new URL path means a code change, a build, and an app store review. If you want to change which paywall shows for `/promo`, that's another update too. The solution: `handleDeepLink` + campaign rules [#the-solution-handledeeplink--campaign-rules] Instead, pass the URL to `handleDeepLink`. The SDK fires the `deepLink_open` standard placement with all of the URL's components as parameters. Then, on the Superwall dashboard, you create campaign rules that match on those parameters to decide what to show. :::android ```kotlin fun handleUrl(url: Uri) { Superwall.instance.handleDeepLink(url) } ``` ::: That's it on the app side. The routing logic lives on the dashboard. Setting up campaign rules [#setting-up-campaign-rules] Once `handleDeepLink` is wired up, the `deepLink_open` placement fires every time a deep link arrives. The URL's path, host, query parameters, and other components are available as parameters you can match against in your campaign's audience filters. On the Superwall dashboard, create a new [campaign](/docs/dashboard/dashboard-campaigns/campaigns) — for example, "Deep Link Paywalls". In your campaign, [add a placement](/docs/dashboard/dashboard-campaigns/campaigns-placements#adding-a-placement) and select `deepLink_open` from the standard placements list. Edit the default audience and add filters that match the URL components you care about. For example, if your deep link is `myapp://promo?offer=summer`: * Set `params.path` **is** `promo` to match the path. * Set `params.offer` **is** `summer` to match the query parameter. See [`deepLink_open` parameters](/docs/dashboard/dashboard-campaigns/campaigns-standard-placements#deeplink_open) for the full list of available fields. Click **Paywalls** at the top of the campaign and choose which paywall to present when the filters match. Now when a user opens `myapp://promo?offer=summer`, the SDK fires `deepLink_open`, the campaign rule matches, and the paywall shows. That's all without touching your app code. To add a new deep link path or change which paywall it shows, just update the campaign on the dashboard. Multiple deep link routes [#multiple-deep-link-routes] You can handle several deep link patterns from a single campaign by adding multiple audiences, each with its own filters and paywalls. For example: | Deep link | Filter | Paywall | | ----------------------------- | -------------------------------------------------------- | --------------- | | `myapp://promo?offer=summer` | `params.path` is `promo` AND `params.offer` is `summer` | Summer Sale | | `myapp://promo?offer=newyear` | `params.path` is `promo` AND `params.offer` is `newyear` | New Year Offer | | `myapp://upgrade` | `params.path` is `upgrade` | Upgrade Paywall | Each audience evaluates independently. When you need to add a new route, create a new audience on the dashboard — no app update needed. Prerequisites [#prerequisites] To use `handleDeepLink`, your app needs deep link handling set up first. If you haven't done that yet, follow the setup guide: :::android * [Deep link setup](/docs/sdk/quickstart/in-app-paywall-previews) ::: Related deep link guides [#related-deep-link-guides] :::android * [Deep Link Setup](/docs/sdk/quickstart/in-app-paywall-previews) — Configure URL schemes and wire `handleDeepLink` into your app so Superwall can respond to incoming links. ::: # Local Resources Local resources let your paywalls load bundled assets directly from the device instead of fetching them over the network. This is useful for hero images, onboarding videos, and other media that should appear immediately even when the connection is slow. :::android Local resources require **Android SDK v2.7.7+**. ::: Registering local resources [#registering-local-resources] Choose a stable resource ID for each asset you want to serve locally. That same ID is what you'll select in the [paywall editor](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-local-resources) when configuring image or video components. :::android On Android, local resources are configured on `Superwall.instance.localResources` after calling [`configure()`](/docs/sdk/sdk-reference/configure), and before presenting paywalls that depend on those assets. ```kotlin Kotlin import android.app.Application import android.net.Uri import com.superwall.sdk.Superwall import com.superwall.sdk.paywall.view.webview.PaywallResource import java.io.File class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure( application = this, apiKey = "pk_your_api_key", ) Superwall.instance.localResources = mapOf( "hero-image" to PaywallResource.FromResources(R.drawable.hero), "onboarding-video" to PaywallResource.FromUri( Uri.fromFile(File(filesDir, "welcome.mp4")) ), "background-animation" to PaywallResource.FromResources(R.raw.paywall_bg) ) } } ``` Set `localResources` before presenting paywalls that depend on those assets. Updating the map later only affects paywalls loaded after the change. ::: Supported source types [#supported-source-types] :::android Android supports two local resource source types: | Type | Use for | | -------------------------------------- | --------------------------------------------------------------------------------- | | `PaywallResource.FromResources(resId)` | Assets packaged in `res/drawable`, `res/raw`, and other Android resource folders | | `PaywallResource.FromUri(uri)` | Files addressed by a `Uri`, such as files in app storage or content provider URLs | ::: Choosing resource IDs [#choosing-resource-ids] Resource IDs are the contract between your app and the paywall editor. A few guidelines: * Use stable, descriptive names like `"hero-image"` and `"onboarding-video"`. * Keep the casing consistent. `"Hero-Image"` and `"hero-image"` are different IDs. * If you rename an ID, update any paywalls that reference it. Referencing local resources in a paywall [#referencing-local-resources-in-a-paywall] In the paywall editor, set a local resource on an image or video component and select the resource ID you registered in the SDK. You can still provide a remote URL as a fallback. Under the hood, paywalls load these resources through `swlocal://` URLs. For example: ```html ``` If the SDK cannot resolve a local resource, the paywall can fall back to the remote URL configured in the editor. Debugging [#debugging] If a resource ID does not appear in the editor or fails to load: * Make sure the app is running a compatible SDK version. * Confirm the resource ID in your paywall exactly matches the key you registered in the SDK. * Open a paywall on a test device after configuring local resources so the editor can discover recently used IDs. * Keep a remote fallback URL on critical media so older builds still render correctly. Related [#related] * [iOS `localResources`](/docs/sdk/sdk-reference/localResources): SDK reference for the iOS property. * [Android `localResources`](/docs/sdk/sdk-reference/localResources): SDK reference for the Android property. * [Paywall Editor: Local Resources](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-local-resources): How to assign local resource IDs in the dashboard. # Migrating from v1 to v2 - Android Migration steps [#migration-steps] 1\. Update code references [#1-update-code-references] 1.1 Rename references from `event` to `placement` [#11-rename-references-from-event-to-placement] In some most cases, the updates are simple and consist of renaming `events` to `placements` where necessary, for others, you'll need to run through this list to manually update your code. | Before | After | | ------------------------------------ | ---------------------------------------- | | fun register(event:) | fun register(placement:) | | fun preloadPaywalls(forEvents:) | fun preloadPaywalls(forPlacements:) | | fun getPaywall(forEvent:) | fun getPaywall(forPlacement:) | | fun getPresentationResult(forEvent:) | fun getPresentationResult(forPlacement:) | | TriggerResult.EventNotFound | TriggerResult.PlacementNotFound | | TriggerResult.NoRuleMatch | TriggerResult.NoAudienceMatch | 1.2 If using Compose and Paywall Composable [#12-if-using-compose-and-paywall-composable] The `PaywallComposable` has been removed from the main Superwall SDK and moved into an optional `superwall-compose` library. To use it, besides including the main superwall library, you'll now need to also include a `superwall-compose` artifact ```groovy gradle implementation "com.superwall.sdk:superwall-compose:2.6.5" ``` 2\. Replacing getPaywall with PaywallBuilder [#2-replacing-getpaywall-with-paywallbuilder] The default method for retrieving a paywall to display it yourself is changing to use the Builder pattern, allowing you more flexibility in both retrieving and displaying paywalls. The builder allows for improved customisation of your paywall experience, allowing you to pass in both custom Shimmer and Loading views by implementing proper interfaces. Usage example: ```kotlin Android val paywallView = PaywallBuilder("placement_name") .params(mapOf("key" to "value")) .overrides(PaywallOverrides()) .delegate(mySuperwallDelegate) .shimmerView(MyShimmerView(context)) .purchaseLoadingView(MyPurchaseLoadingView(context)) .activity(activity) .build() ``` 3\. Getting the purchased product [#3-getting-the-purchased-product] The `onDismiss` block of the `PaywallPresentationHandler` now accepts both a `PaywallInfo` object and a `PaywallResult` object. This allows you to easily access the purchased product from the result when the paywall dismisses. 4\. Entitlements [#4-entitlements] The `subscriptionStatus` has been changed to accept a set of `Entitlement` objects. This allows you to give access to entitlements based on products purchased. For example, in your app you might have Bronze, Silver, and Gold subscription tiers, i.e. entitlements, which entitle a user to access a certain set of features within your app. Every subscription product must be associated with one or more entitlements, which is controlled via the dashboard. Superwall will already have associated all your products with a default entitlement. If you don't use more than one entitlement tier within your app and you only use subscription products, you don't need to do anything extra. However, if you use one-time purchases or multiple entitlements, you should review your products and their entitlements. In general, consumables should not be associated with an entitlement, whereas non-consumables should be. Check your products [here](https://superwall.com/applications/\:app/products/v2). If you're using a `PurchaseController`, you'll need to set the `entitlements` with the `subscriptionStatus`: | Before | After | | ----------------------------------------------------------------- | ------------------------------------------------------------------------------- | | Superwall.shared.setSubscriptionStatus(SubscriptionStatus.ACTIVE) | Superwall.shared.setSubscriptionStatus(SubscriptionStatus.Active(entitlements)) | You can get the `ProductDetails` and their associated entitlements from Superwall by calling the method `products(for:)`. Here is an example of how you'd sync your subscription status with Superwall using these methods: ```kotlin With Play Billing suspend fun syncSubscriptionStatus() { // We await for configuration to be set so our entitlements are available Superwall.instance.configurationStateListener.first { it is ConfigurationStatus.Configured } // Query purchases from your BillingClient val subscriptionPurchases = queryPurchasesOfType(BillingClient.ProductType.SUBS) val inAppPurchases = queryPurchasesOfType(BillingClient.ProductType.INAPP) val allPurchases = subscriptionPurchases + inAppPurchases val hasActivePurchaseOrSubscription = allPurchases.any { it.purchaseState == Purchase.PurchaseState.PURCHASED } val status: SubscriptionStatus = if (hasActivePurchaseOrSubscription) { subscriptionPurchases .flatMap { // Extract the productId it.products }.toSet() // Ensure uniqueness .flatMap { // Receive entitlements val res = entitlementsInfo.byProductId(it) res }.toSet() .let { entitlements -> if (entitlements.isNotEmpty()) { SubscriptionStatus.Active(entitlements) } else { SubscriptionStatus.Inactive } } } else { SubscriptionStatus.Inactive } Superwall.instance.setSubscriptionStatus(status) } ``` ```kotlin with RevenueCat fun syncSubscriptionStatus() { Purchases.sharedInstance.getCustomerInfoWith { if (hasAnyActiveEntitlements(it)) { setSubscriptionStatus( SubscriptionStatus.Active( it.entitlements.active .map { Entitlement(it.key, Entitlement.Type.SERVICE_LEVEL) }.toSet(), ), ) } else { setSubscriptionStatus(SubscriptionStatus.Inactive) } } } ``` You can listen to the flowable property `Superwall.instance.subscriptionStatus` to be notified when the subscriptionStatus changes. Or you can use the `SuperwallDelegate` method `subscriptionStatusDidChange(from:to:)`, which replaces `subscriptionStatusDidChange(to:)`. 5\. Paywall Presentation Condition [#5-paywall-presentation-condition] In the Paywall Editor you can choose whether to always present a paywall or ask the SDK to check the user subscription before presenting a paywall. For users on v2 of the SDK, this is replaced with a check on the entitlements within the audience filter. As you migrate your users from v1 to v2 of the SDK, you'll need to make sure you set both the entitlements check and the paywall presentation condition in the paywall editor. 6\. Check out the full change log [#6-check-out-the-full-change-log] You can view this on [our GitHub page](https://github.com/superwall/Superwall-Android/blob/develop/CHANGELOG.md). 7\. Check out our updated example apps [#7-check-out-our-updated-example-apps] All of our [example apps](https://github.com/superwall/Superwall-Android/tree/develop/example) have been updated to use the latest SDK. We now only have two apps: Basic and Advanced. Basic shows you the basic integration of Superwall without needing a purchase controller or multiple entitlements. Advanced has multiple flavors, showing you how to use entitlements within your app as well as optionally using a purchase controller with Play Billing or RevenueCat. # Test Mode Test mode lets you simulate in-app purchases without involving Google Play Billing or any external purchase controller. When active, all purchases are faked and product data is retrieved from the Superwall dashboard. This makes it easy to test your entire paywall flow end-to-end, including purchase, restore, and entitlement changes, without needing a Google Play sandbox account or test tracks. How it works [#how-it-works] When test mode is active: * **Product data comes from the dashboard** instead of Google Play, so you don't need products set up in Google Play Console. * **Purchases are simulated.** Instead of the Google Play purchase flow, a test mode drawer appears letting you choose to complete, abandon, or fail the transaction. Purchase events fire normally, so your analytics and delegate callbacks work as expected. * **Restores are simulated.** A restore drawer lets you pick which entitlements to restore and in what state. * **A configuration modal** appears on launch showing why test mode activated, your User ID, purchase controller status, device and user attributes, free trial override, and starting entitlements. You can use this to configure the test session before interacting with your paywalls. Activating test mode [#activating-test-mode] There are two ways to activate test mode: 1\. From the dashboard [#1-from-the-dashboard] Mark specific users as **test store users** in the Superwall dashboard. When the SDK detects that the current user's ID matches a test store user from your config, test mode activates automatically. This is the most common approach. 2\. From the SDK [#2-from-the-sdk] Set `testModeBehavior` on `SuperwallOptions` before calling `configure`: ```kotlin val options = SuperwallOptions().apply { testModeBehavior = TestModeBehavior.ALWAYS } Superwall.configure(this, "your-api-key", options = options) ``` The available behaviors are: | Behavior | Description | | ----------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | `AUTOMATIC` | **(Default)** Activates when the current user is marked as a test store user in the dashboard, or when the app's application ID doesn't match the one configured in the dashboard. Never activates when JUnit or Espresso is detected on the classpath. | | `WHEN_ENABLED_FOR_USER` | Activates only when the current user is marked as a test store user in the dashboard. Ignores application ID mismatches. | | `ALWAYS` | Always activates test mode, regardless of dashboard configuration. Useful during local development. | | `NEVER` | Never activates test mode, regardless of configuration. | The configuration modal [#the-configuration-modal] When test mode activates, a modal appears before you interact with any paywalls. It displays: * **Reason:** Why test mode activated (e.g., application ID mismatch, matched test store user, or always-on via options). * **User ID:** Your current user ID, with a link to view the user in the dashboard. * **Purchase Controller:** Whether you've provided a custom purchase controller. * **Device Attributes:** Tap to view all device-level attributes the SDK is tracking. * **User Attributes:** Tap to view all user-level attributes. * **Free Trial Override:** Override free trial availability for all products. Choose **Use Default** (respects the product's actual trial status), **Force Available**, or **Force Unavailable**. * **Starting Entitlements:** If you have entitlements configured, you can set each one's state before dismissing the modal. Available states include **Subscribed**, **In Grace Period**, **Billing Retry**, **Expired**, **Revoked**, and **Inactive**. For active states you can also set the offer type (**None**, **Trial**, or **Promotional**). This lets you test how your paywalls behave for users with different entitlement states. Tap **Continue** to dismiss the modal and begin testing. Your selections persist across sessions. Simulating purchases [#simulating-purchases] When you tap a purchase button on a paywall while test mode is active, a drawer appears instead of the Google Play purchase flow. The drawer shows the product details (identifier, price, period, free trial availability, and entitlements) along with these options: * **Confirm Purchase** (or **Start Free Trial** if the product has a free trial): Simulates a successful purchase. The product's entitlements are activated and your subscription status updates accordingly. * **Abandon:** Closes the drawer without completing the purchase. * **Fail:** Simulates a purchase failure. Standard Superwall events (`transaction_start`, `transaction_complete`, `transaction_abandon`, `transaction_fail`, etc.) fire as they normally would, so you can verify your analytics and delegate callbacks. Simulating restores [#simulating-restores] When a restore is triggered while test mode is active, a drawer appears letting you select which entitlements to restore and in what state. This is useful for testing how your app handles different restore scenarios. When to use test mode vs. Google Play testing [#when-to-use-test-mode-vs-google-play-testing] | | Test mode | Google Play testing | | ---------------- | -------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------- | | **Setup** | No Google Play Console setup needed | Requires products in Google Play Console and license testers configured | | **Products** | Pulled from the Superwall dashboard | Must exist in Google Play Console | | **Transactions** | Simulated via UI drawer | Real Google Play Billing transactions in a test environment | | **Best for** | End-to-end paywall flow testing, verifying entitlement gating, testing without Google Play Console setup | Testing real billing behavior, receipt validation, subscription lifecycle | Test mode is ideal for quickly validating your paywall presentation, purchase flows, and entitlement gating without any Google Play setup. For testing actual Google Play Billing behavior, use Google Play's license testing and test tracks instead. # Using RevenueCat Not using RevenueCat? No problem! Superwall works out of the box without any additional SDKs. You only need to use a `PurchaseController` if you want end-to-end control of the purchasing pipeline. The recommended way to use RevenueCat with Superwall is by putting it in observer mode. You can integrate RevenueCat with Superwall using several approaches: 1. [**Using a purchase controller:**](#using-a-purchase-controller) Use this route if you want to maintain control over purchasing logic and code. 2. [**Using PurchasesAreCompletedBy:**](#using-purchasesarecompletedby) Here, you don't use a purchase controller and you tell RevenueCat that purchases are completed by your app using StoreKit. In this mode, RevenueCat will observe the purchases that the Superwall SDK makes. For more info [see here](https://www.revenuecat.com/docs/migrating-to-revenuecat/sdk-or-not/finishing-transactions). Using a purchase controller [#using-a-purchase-controller] 1\. Create a `PurchaseController` [#1-create-a-purchasecontroller] Create a new file called `RCPurchaseController`, then copy and paste the following: :::android ```kotlin package com.superwall.superapp import android.app.Activity import android.content.Context import com.android.billingclient.api.ProductDetails import com.revenuecat.purchases.CustomerInfo import com.revenuecat.purchases.LogLevel import com.revenuecat.purchases.ProductType import com.revenuecat.purchases.PurchaseParams import com.revenuecat.purchases.Purchases import com.revenuecat.purchases.PurchasesConfiguration import com.revenuecat.purchases.PurchasesError import com.revenuecat.purchases.PurchasesErrorCode import com.revenuecat.purchases.getCustomerInfoWith import com.revenuecat.purchases.interfaces.GetStoreProductsCallback import com.revenuecat.purchases.interfaces.PurchaseCallback import com.revenuecat.purchases.interfaces.ReceiveCustomerInfoCallback import com.revenuecat.purchases.interfaces.UpdatedCustomerInfoListener import com.revenuecat.purchases.models.StoreProduct import com.revenuecat.purchases.models.StoreTransaction import com.revenuecat.purchases.models.SubscriptionOption import com.revenuecat.purchases.models.googleProduct import com.revenuecat.purchases.purchaseWith import com.superwall.sdk.Superwall import com.superwall.sdk.delegate.PurchaseResult import com.superwall.sdk.delegate.RestorationResult import com.superwall.sdk.delegate.subscription_controller.PurchaseController import com.superwall.sdk.models.entitlements.Entitlement import com.superwall.sdk.models.entitlements.SubscriptionStatus import kotlinx.coroutines.CompletableDeferred suspend fun Purchases.awaitProducts(productIds: List): List { val deferred = CompletableDeferred>() getProducts( productIds, object : GetStoreProductsCallback { override fun onReceived(storeProducts: List) { deferred.complete(storeProducts) } override fun onError(error: PurchasesError) { deferred.completeExceptionally(Exception(error.message)) } }, ) return deferred.await() } interface PurchaseCompletion { var storeTransaction: StoreTransaction var customerInfo: CustomerInfo } // Create a custom exception class that wraps PurchasesError private class PurchasesException( val purchasesError: PurchasesError, ) : Exception(purchasesError.toString()) suspend fun Purchases.awaitPurchase( activity: Activity, storeProduct: StoreProduct, ): PurchaseCompletion { val deferred = CompletableDeferred() purchase( PurchaseParams.Builder(activity, storeProduct).build(), object : PurchaseCallback { override fun onCompleted( storeTransaction: StoreTransaction, customerInfo: CustomerInfo, ) { deferred.complete( object : PurchaseCompletion { override var storeTransaction: StoreTransaction = storeTransaction override var customerInfo: CustomerInfo = customerInfo }, ) } override fun onError( error: PurchasesError, p1: Boolean, ) { deferred.completeExceptionally(PurchasesException(error)) } }, ) return deferred.await() } suspend fun Purchases.awaitRestoration(): CustomerInfo { val deferred = CompletableDeferred() restorePurchases( object : ReceiveCustomerInfoCallback { override fun onReceived(purchaserInfo: CustomerInfo) { deferred.complete(purchaserInfo) } override fun onError(error: PurchasesError) { deferred.completeExceptionally(error as Throwable) } }, ) return deferred.await() } class RevenueCatPurchaseController( val context: Context, ) : PurchaseController, UpdatedCustomerInfoListener { init { Purchases.logLevel = LogLevel.DEBUG Purchases.configure( PurchasesConfiguration .Builder( context, "android_rc_key", ).build(), ) // Make sure we get the updates Purchases.sharedInstance.updatedCustomerInfoListener = this } fun syncSubscriptionStatus() { // Refetch the customer info on load Purchases.sharedInstance.getCustomerInfoWith { if (hasAnyActiveEntitlements(it)) { setSubscriptionStatus( SubscriptionStatus.Active( it.entitlements.active .map { Entitlement(it.key, Entitlement.Type.SERVICE_LEVEL) }.toSet(), ), ) } else { setSubscriptionStatus(SubscriptionStatus.Inactive) } } } /** * Callback for rc customer updated info */ override fun onReceived(customerInfo: CustomerInfo) { if (hasAnyActiveEntitlements(customerInfo)) { setSubscriptionStatus( SubscriptionStatus.Active( customerInfo.entitlements.active .map { Entitlement(it.key, Entitlement.Type.SERVICE_LEVEL) }.toSet(), ), ) } else { setSubscriptionStatus(SubscriptionStatus.Inactive) } } /** * Initiate a purchase */ override suspend fun purchase( activity: Activity, productDetails: ProductDetails, basePlanId: String?, offerId: String?, ): PurchaseResult { // Find products matching productId from RevenueCat val products = Purchases.sharedInstance.awaitProducts(listOf(productDetails.productId)) // Choose the product which matches the given base plan. // If no base plan set, select first product or fail. val product = products.firstOrNull { it.googleProduct?.basePlanId == basePlanId } ?: products.firstOrNull() ?: return PurchaseResult.Failed("Product not found") return when (product.type) { ProductType.SUBS, ProductType.UNKNOWN -> handleSubscription( activity, product, basePlanId, offerId, ) ProductType.INAPP -> handleInAppPurchase(activity, product) } } private fun buildSubscriptionOptionId( basePlanId: String?, offerId: String?, ): String = buildString { basePlanId?.let { append("$it") } offerId?.let { append(":$it") } } private suspend fun handleSubscription( activity: Activity, storeProduct: StoreProduct, basePlanId: String?, offerId: String?, ): PurchaseResult { storeProduct.subscriptionOptions?.let { subscriptionOptions -> // If subscription option exists, concatenate base + offer ID. val subscriptionOptionId = buildSubscriptionOptionId(basePlanId, offerId) // Find first subscription option that matches the subscription option ID or default // to letting revenuecat choose. val subscriptionOption = subscriptionOptions.firstOrNull { it.id == subscriptionOptionId } ?: subscriptionOptions.defaultOffer // Purchase subscription option, otherwise fail. if (subscriptionOption != null) { return purchaseSubscription(activity, subscriptionOption) } } return PurchaseResult.Failed("Valid subscription option not found for product.") } private suspend fun purchaseSubscription( activity: Activity, subscriptionOption: SubscriptionOption, ): PurchaseResult { val deferred = CompletableDeferred() Purchases.sharedInstance.purchaseWith( PurchaseParams.Builder(activity, subscriptionOption).build(), onError = { error, userCancelled -> deferred.complete( if (userCancelled) { PurchaseResult.Cancelled() } else { PurchaseResult.Failed( error.message, ) }, ) }, onSuccess = { _, _ -> deferred.complete(PurchaseResult.Purchased()) }, ) return deferred.await() } private suspend fun handleInAppPurchase( activity: Activity, storeProduct: StoreProduct, ): PurchaseResult = try { Purchases.sharedInstance.awaitPurchase(activity, storeProduct) PurchaseResult.Purchased() } catch (e: PurchasesException) { when (e.purchasesError.code) { PurchasesErrorCode.PurchaseCancelledError -> PurchaseResult.Cancelled() else -> PurchaseResult.Failed( e.message ?: "Purchase failed due to an unknown error", ) } } /** * Restore purchases */ override suspend fun restorePurchases(): RestorationResult { try { if (hasAnyActiveEntitlements(Purchases.sharedInstance.awaitRestoration())) { return RestorationResult.Restored() } else { return RestorationResult.Failed(Exception("No active entitlements")) } } catch (e: Throwable) { return RestorationResult.Failed(e) } } /** * Check if the customer has any active entitlements */ private fun hasAnyActiveEntitlements(customerInfo: CustomerInfo): Boolean { val entitlements = customerInfo.entitlements.active.values .map { it.identifier } return entitlements.isNotEmpty() } private fun setSubscriptionStatus(subscriptionStatus: SubscriptionStatus) { if (Superwall.initialized) { Superwall.instance.setSubscriptionStatus(subscriptionStatus) } } } ``` ::: As discussed in [Purchases and Subscription Status](/docs/sdk/guides/advanced-configuration), this `CustomPurchaseControllerProvider` is responsible for handling the subscription-related logic using the modern hooks-based approach. 2\. Configure Superwall (Continued) [#2-configure-superwall-continued] The example above shows the complete setup. The `CustomPurchaseControllerProvider` wraps your `SuperwallProvider` and handles all purchase and restore logic through RevenueCat. For more advanced implementations, see the [example app](https://github.com/superwall/expo-superwall/tree/main/example). **Legacy Approach**: If you're migrating from the old SDK or need the class-based purchase controller, you can use `expo-superwall/compat`. However, we recommend using the modern `CustomPurchaseControllerProvider` approach shown above. Removed Legacy Code Section [#removed-legacy-code-section] The following section contains the legacy class-based approach. Skip to the next section for the modern configuration. As discussed in [Purchases and Subscription Status](/docs/sdk/guides/advanced-configuration), this `PurchaseController` is responsible for handling the subscription-related logic. Take a few moments to look through the code to understand how it does this. 2\. Configure Superwall [#2-configure-superwall] Initialize an instance of `RCPurchaseController` and pass it in to `Superwall.configure(apiKey:purchaseController)`: :::android ```kotlin val purchaseController = RCPurchaseController(this) Superwall.configure( this, "MY_API_KEY", purchaseController ) // Make sure we sync the subscription status // on first load and whenever it changes purchaseController.syncSubscriptionStatus() ``` ::: 3\. Sync the subscription status [#3-sync-the-subscription-status] Then, call `purchaseController.syncSubscriptionStatus()` to keep Superwall's subscription status up to date with RevenueCat. That's it! Check out our sample app for working examples: :::android * [Android](https://github.com/superwall/Superwall-Android/tree/develop/example/app/src/revenuecat) ::: Using PurchasesAreCompletedBy [#using-purchasesarecompletedby] If you're using RevenueCat's [PurchasesAreCompletedBy](https://www.revenuecat.com/docs/migrating-to-revenuecat/sdk-or-not/finishing-transactions), you don't need to create a purchase controller. Register your placements, present a paywall — and Superwall will take care of completing any purchase the user starts. However, there are a few things to note if you use this setup: 1. Here, you aren't using RevenueCat's [entitlements](https://www.revenuecat.com/docs/getting-started/entitlements#entitlements) as a source of truth. If your app is multiplatform, you'll need to consider how to link up pro features or purchased products for users. 2. If you require custom logic when purchases occur, then you'll want to add a purchase controller. In that case, Superwall handles purchasing flows and RevenueCat will still observe transactions to power their analytics and charts. 3. Be sure that user identifiers are set the same way across Superwall and RevenueCat. For more information on observer mode, visit [RevenueCat's docs](https://www.revenuecat.com/docs/migrating-to-revenuecat/sdk-or-not/finishing-transactions). # Using the Superwall Delegate Use Superwall's delegate to extend our SDK's functionality across several surface areas by assigning to the `delegate` property: :::android ```kotlin class SWDelegate : SuperwallDelegate { // Implement delegate methods here } // When configuring the SDK... Superwall.instance.delegate = SWDelegate() ``` ::: Some common use cases for using the Superwall delegate include: * **Custom actions:** [Respond to custom tap actions from a paywall.](/docs/sdk/guides/advanced/custom-paywall-actions#custom-paywall-actions) * **Respond to purchases:** [See which product was purchased from the presented paywall.](/docs/sdk/guides/advanced/viewing-purchased-products) * **Analytics:** [Forward events from Superwall to your own analytics.](/docs/sdk/guides/3rd-party-analytics) Below are some commonly used implementations when using the delegate. Superwall Events [#superwall-events] Most of what occurs in Superwall can be viewed using the delegate method to respond to events: :::android ```kotlin class SWDelegate : SuperwallDelegate { override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { // Handle any relevant events here... when (eventInfo.event) { is SuperwallPlacement.TransactionComplete -> { val transaction = (eventInfo.event as SuperwallPlacement.TransactionComplete).transaction val product = (eventInfo.event as SuperwallPlacement.TransactionComplete).product val paywallInfo = (eventInfo.event as SuperwallPlacement.TransactionComplete).paywallInfo println("Transaction Complete: $transaction, Product: $product, Paywall Info: $paywallInfo") } else -> { // Handle other cases } } } } ``` ::: Paywall Custom Actions [#paywall-custom-actions] Using the [custom tap action](/docs/sdk/guides/advanced/custom-paywall-actions#custom-paywall-actions), you can respond to any arbitrary event from a paywall: :::android ```kotlin class SWDelegate : SuperwallDelegate { override fun handleCustomPaywallAction(withName: String) { if (withName == "caffeineLogged") { println("Custom paywall action: $withName") } } } ``` ::: Subscription status changes [#subscription-status-changes] You can be informed of subscription status changes using the delegate. If you need to set or handle the status on your own, use a [purchase controller](/docs/sdk/guides/advanced-configuration) — this function is only for informational, tracking or similar purposes: :::android ```kotlin class SWDelegate : SuperwallDelegate { override fun subscriptionStatusDidChange(from: SubscriptionStatus, to: SubscriptionStatus) { println("Subscription status changed from $from to $to") } } ``` ::: Paywall events [#paywall-events] The delegate also has callbacks for several paywall events, such dismissing, presenting, and more. Here's an example: :::android ```kotlin class SWDelegate : SuperwallDelegate { override fun didPresentPaywall(withInfo: PaywallInfo) { println("Paywall presented: $withInfo") } } ``` ::: # Vibe Coding Overview [#overview] We've built a few tools to help you Vibe Code using the knowledge of the Superwall Docs, access your Superwall account, and more right in your favorite AI tools: * [Superwall Skill (Recommended)](#superwall-skill-recommended): Gives AI agents live docs, API access, and step-by-step SDK integration guides. If you're unsure which tool to use, pick this one. * [Superwall MCP](#superwall-mcp): Expose your Superwall account (projects, paywalls, campaigns) to work with AI tools. And right here in the Superwall Docs: * [Superwall AI](#superwall-ai) * [Docs Links](#docs-links) * [LLMs.txt](#llmstxt) Superwall Skill (Recommended) [#superwall-skill-recommended] The [Superwall Skill](/docs/dashboard/guides/superwall-skill) is the best way to give AI coding agents full context on Superwall. It bundles live documentation, API access, dashboard links, and guided SDK integration flows for every platform, all in one install. If you're unsure which tool to use, pick this one. ```bash npx skills add superwall/skills ``` Once installed, your agent can look up any Superwall doc on demand, call the API to inspect your projects and applications, and walk through a complete SDK integration step by step. It supports iOS, Android, Flutter, and Expo out of the box with platform-specific quickstart skills. If you're only going to set up one tool, this is the one to use. See the full [Superwall Skill guide](/docs/dashboard/guides/superwall-skill) for details. Superwall MCP [#superwall-mcp] The Superwall MCP connects AI tools to your **Superwall account**, letting agents create and manage projects, paywalls, campaigns, products, entitlements, and webhooks directly. Instead of switching to the dashboard, your AI assistant can set everything up for you. If you also want live docs access and guided SDK integration help, use the [Superwall Skill](/docs/dashboard/guides/superwall-skill). The MCP is focused on account and resource management. See the full [Superwall MCP guide](/docs/dashboard/guides/superwall-mcp) for installation, a step-by-step quick setup, and the complete tool reference. Superwall AI [#superwall-ai] Superwall AI is available in the bottom right 💬 and is a great place to start if you have a question or issue. Docs Links [#docs-links] At the top of each page of the Superwall Docs (including this one!): * **Copy Markdown**: to copy the page in Markdown format. Also in the **Open** dropdown menu, you can access these options: * **View as Markdown**: to view the page in Markdown format * **Open in ChatGPT**, **Open in Claude**: to open the page in the respective AI tool and add the page as context for your conversation LLMs.txt [#llmstxt] The Superwall Docs website has `llms.txt` and `llms-full.txt` files, in total and for each SDK, that you can use to add context to your LLMs. `llms.txt` is a summary of the docs with links to each page. `llms-full.txt` is the full text of all of the docs. | SDK | Summary | Full Text | | ------------------------- | --------------------------------------------------------------------------- | ------------------------------------------------------------------------------------- | | All | [`llms.txt`](https://superwall.com/docs/llms.txt) | [`llms-full.txt`](https://superwall.com/docs/llms-full.txt) | | Dashboard | [`llms-dashboard.txt`](https://superwall.com/docs/llms-dashboard.txt) | [`llms-full-dashboard.txt`](https://superwall.com/docs/llms-full-dashboard.txt) | | iOS | [`llms-ios.txt`](https://superwall.com/docs/llms-ios.txt) | [`llms-full-ios.txt`](https://superwall.com/docs/llms-full-ios.txt) | | Android | [`llms-android.txt`](https://superwall.com/docs/llms-android.txt) | [`llms-full-android.txt`](https://superwall.com/docs/llms-full-android.txt) | | Flutter | [`llms-flutter.txt`](https://superwall.com/docs/llms-flutter.txt) | [`llms-full-flutter.txt`](https://superwall.com/docs/llms-full-flutter.txt) | | Expo | [`llms-expo.txt`](https://superwall.com/docs/llms-expo.txt) | [`llms-full-expo.txt`](https://superwall.com/docs/llms-full-expo.txt) | | React Native (Deprecated) | [`llms-react-native.txt`](https://superwall.com/docs/llms-react-native.txt) | [`llms-full-react-native.txt`](https://superwall.com/docs/llms-full-react-native.txt) | | Integrations | [`llms-integrations.txt`](https://superwall.com/docs/llms-integrations.txt) | [`llms-full-integrations.txt`](https://superwall.com/docs/llms-full-integrations.txt) | | Web Checkout | [`llms-web-checkout.txt`](https://superwall.com/docs/llms-web-checkout.txt) | [`llms-full-web-checkout.txt`](https://superwall.com/docs/llms-full-web-checkout.txt) | To minimize token use, we recommend using the files specific to your SDK. # Web Checkout Dashboard Setup [#dashboard-setup] 1. [Set up Web Checkout in the dashboard](/docs/web-checkout) 2. [Add web products to your paywall](/docs/web-checkout/web-checkout-direct-stripe-checkout) SDK Setup [#sdk-setup] :::android 1. [Set up deep links](/docs/sdk/quickstart/in-app-paywall-previews) 2. [Handle Post-Checkout redirecting](/docs/sdk/guides/web-checkout/post-checkout-redirecting) 3. **Only if you're using RevenueCat:** [Using RevenueCat](/docs/sdk/guides/web-checkout/using-revenuecat) 4. **Only if you're using your own PurchaseController:** [Redeeming In-App](/docs/sdk/guides/web-checkout/linking-membership-to-iOS-app) ::: Testing [#testing] 1. [Testing Purchases](/docs/web-checkout/web-checkout-testing-purchases) 2. [Managing Memberships](/docs/web-checkout/web-checkout-managing-memberships) Troubleshooting [#troubleshooting] If a user has issues accessing their purchase in your app after paying via web checkout, direct them to your plan management page to retrieve their redemption link or manage billing: For example: `http://yourapp.superwall.app/manage` FAQ [#faq] [Web Checkout FAQ](/docs/web-checkout/web-checkout-faq) # Redeeming In-App After purchasing from a web paywall, the user will be redirected to your app by a deep link to redeem their purchase on device. Please follow our [Post-Checkout Redirecting](/docs/android/guides/web-checkout/post-checkout-redirecting) guide to handle this user experience. If you're using Superwall to handle purchases, then you don't need to do anything here. If you're using your own `PurchaseController`, you will need to update the subscription status with the redeemed web entitlements. If you're using RevenueCat, you should follow our [Using RevenueCat](/docs/android/guides/web-checkout/using-revenuecat) guide. Using a PurchaseController [#using-a-purchasecontroller] If you're using Google Play Billing in your PurchaseController, you'll need to merge the web entitlements with the device entitlements before setting the subscription status. Here's an example of how you might do this: ```kotlin import com.android.billingclient.api.BillingClient import com.android.billingclient.api.Purchase import com.android.billingclient.api.QueryPurchasesParams import com.superwall.sdk.Superwall import com.superwall.sdk.models.entitlements.SubscriptionStatus import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext suspend fun syncSubscriptionStatus(billingClient: BillingClient) { withContext(Dispatchers.IO) { val productIds = mutableSetOf() // Get the device entitlements from Google Play Billing val params = QueryPurchasesParams.newBuilder() .setProductType(BillingClient.ProductType.SUBS) .build() val purchasesResult = billingClient.queryPurchasesAsync(params) // Collect purchased product IDs purchasesResult.purchasesList.forEach { purchase -> if (purchase.purchaseState == Purchase.PurchaseState.PURCHASED) { purchase.products.forEach { productId -> productIds.add(productId) } } } // Get products from Superwall and extract their entitlements val storeProducts = Superwall.instance.getProducts(productIds) val deviceEntitlements = storeProducts.flatMap { it.entitlements }.toSet() // Get the web entitlements from Superwall val webEntitlements = Superwall.instance.entitlements.web // Merge the two sets of entitlements val allEntitlements = deviceEntitlements + webEntitlements // Update subscription status on the main thread withContext(Dispatchers.Main) { if (allEntitlements.isNotEmpty()) { Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(allEntitlements) ) } else { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) } } } } ``` In addition to syncing the subscription status when purchasing and restoring, you'll need to sync it whenever `didRedeemLink(result:)` is called: ```kotlin import com.superwall.sdk.delegate.SuperwallDelegate import com.superwall.sdk.models.redemption.RedemptionResult import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch class SWDelegate(private val billingClient: BillingClient) : SuperwallDelegate { private val coroutineScope = CoroutineScope(Dispatchers.Main) override fun didRedeemLink(result: RedemptionResult) { coroutineScope.launch { syncSubscriptionStatus(billingClient) } } } ``` Refreshing of web entitlements [#refreshing-of-web-entitlements] If you aren't using a Purchase Controller, the SDK will refresh the web entitlements every 24 hours. Redeeming while a paywall is open [#redeeming-while-a-paywall-is-open] If a redeem event occurs when a paywall is open, the SDK will track that as a restore event and the paywall will close. # Post-Checkout Redirecting After a user completes a web purchase, Superwall needs to redirect them back to your app. You can configure this behavior in two ways: Post-Purchase Behavior Modes [#post-purchase-behavior-modes] You can configure how users are redirected after checkout in your [Application Settings](/docs/web-checkout/web-checkout-configuring-stripe-keys-and-settings#post-purchase-behavior): Redeem Mode (Default) [#redeem-mode-default] Superwall manages the entire redemption experience: * Users are automatically deep linked to your app with a redemption code * Fallback to App Store/Play Store if the app isn't installed * Redemption emails are sent automatically * The SDK handles redemption via delegate methods (detailed below) This is the recommended mode for most apps. Redirect Mode [#redirect-mode] Redirect users to your own custom URL with purchase information: * **When to use**: You want to show a custom success page, perform additional actions before redemption, or have your own deep linking infrastructure * **What you receive**: Purchase data is passed as query parameters to your URL **Query Parameters Included**: * `app_user_id` - The user's identifier from your app * `email` - User's email address * `stripe_subscription_id` - The Stripe subscription ID, or the Stripe Checkout session ID for one-time purchases * Any custom placement parameters you set **Example**: ``` https://yourapp.com/success? app_user_id=user_123& email=user@example.com& stripe_subscription_id=sub_1234567890& campaign_id=summer_sale ``` You'll need to implement your own logic to handle the redirect and deep link users into your app. *** Setting Up Deep Links [#setting-up-deep-links] Whether checkout starts from a web link or from a paywall that opens an external browser, the Superwall SDK relies on deep links to redirect back to your app. Prerequisites [#prerequisites] 1. [Configuring Stripe Keys and Settings](/docs/web-checkout/web-checkout-configuring-stripe-keys-and-settings) 2. [Deep Links](/docs/android/quickstart/in-app-paywall-previews) If you're not using Superwall to handle purchases, then you'll need to follow extra steps to redeem the web purchase in your app. * [Using RevenueCat](/docs/android/guides/web-checkout/using-revenuecat) * [Using a PurchaseController](/docs/android/guides/web-checkout/linking-membership-to-iOS-app#using-a-purchasecontroller) *** Handling Redemption (Redeem Mode) [#handling-redemption-redeem-mode] When using Redeem mode (the default), handle the user experience when they're redirected back to your app using `SuperwallDelegate` methods: willRedeemLink [#willredeemlink] When your app opens via the deep link, we will call the delegate method `willRedeemLink()` before making a network call to redeem the code. At this point, you might wish to display a loading indicator in your app so the user knows that the purchase is being redeemed. ```kotlin class SWDelegate : SuperwallDelegate { override fun willRedeemLink() { // Show a loading indicator to the user showToast("Activating your purchase...") } } ``` You can manually dismiss the paywall at this point if needed, but note that the paywall will be dismissed automatically when the `didRedeemLink` method is called. didRedeemLink [#didredeemlink] After receiving a response from the network, we will call `didRedeemLink(result:)` with the result of redeeming the code. This result can be one of the following: * `RedemptionResult.Success`: The redemption succeeded and contains information about the redeemed code. * `RedemptionResult.Error`: An error occurred while redeeming. You can check the error message via the error parameter. * `RedemptionResult.ExpiredCode`: The code expired and contains information about whether a redemption email has been resent and an optional obfuscated email address. * `RedemptionResult.InvalidCode`: The code that was redeemed was invalid. * `RedemptionResult.ExpiredSubscription`: The subscription that the code redeemed has expired. On network failure, the SDK will retry up to 6 times before returning an `Error` `RedemptionResult` in `didRedeemLink(result:)`. Here, you should remove any loading UI you added in `willRedeemLink` and show a message to the user based on the result. If a paywall is presented, it will be dismissed automatically. ```kotlin class SWDelegate : SuperwallDelegate { override fun didRedeemLink(result: RedemptionResult) { when (result) { is RedemptionResult.ExpiredCode -> { showToast("Expired Link") Log.d("Superwall", "Code expired: ${result.code}, ${result.expiredInfo}") } is RedemptionResult.Error -> { showToast(result.error.message) Log.d("Superwall", "Error: ${result.code}, ${result.error}") } is RedemptionResult.ExpiredSubscription -> { showToast("Expired Subscription") Log.d("Superwall", "Expired subscription: ${result.code}, ${result.redemptionInfo}") } is RedemptionResult.InvalidCode -> { showToast("Invalid Link") Log.d("Superwall", "Invalid code: ${result.code}") } is RedemptionResult.Success -> { val email = result.redemptionInfo.purchaserInfo.email val productIdentifier = result.redemptionInfo.paywallInfo?.productIdentifier if (email != null) { Superwall.instance.setUserAttributes(mapOf("email" to email)) showToast("Welcome, $email!") } else { showToast("Welcome!") } // Access the product identifier if available (2.6.3+) productIdentifier?.let { Log.d("Superwall", "Redeemed product: $it") } } } } } ``` # Using RevenueCat After purchasing from a web paywall, the user will be redirected to your app by a deep link to redeem their purchase on device. Please follow our [Post-Checkout Redirecting](/docs/android/guides/web-checkout/post-checkout-redirecting) guide to handle this user experience. If you're using Superwall to handle purchases, then you don't need to do anything here. You only need to use a `PurchaseController` if you want end-to-end control of the purchasing pipeline. The recommended way to use RevenueCat with Superwall is by putting it in observer mode. If you're using your own `PurchaseController`, you should follow our [Redeeming In-App](/docs/android/guides/web-checkout/linking-membership-to-iOS-app) guide. Using a PurchaseController with RevenueCat [#using-a-purchasecontroller-with-revenuecat] If you're using RevenueCat, you'll need to follow [steps 1 to 4 in their guide](https://www.revenuecat.com/docs/web/integrations/stripe) to set up Stripe with RevenueCat. Then, you'll need to associate the RevenueCat customer with the Stripe subscription IDs returned from redeeming the code. You can do this by extracting the ids from the `RedemptionResult` and sending them to RevenueCat's API by using the `didRedeemLink(result:)` delegate method: This flow is for Stripe subscriptions. Stripe one-time purchases can return Stripe Checkout session IDs through the same legacy `stripeSubscriptionIds` field, but those IDs are not Stripe subscription IDs. Handle one-time purchases with Superwall entitlements or your own backend instead of sending those IDs to RevenueCat's Stripe subscription endpoint. ```kotlin import com.revenuecat.purchases.Purchases import com.superwall.sdk.Superwall import com.superwall.sdk.delegate.SuperwallDelegate import com.superwall.sdk.models.redemption.RedemptionResult import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext import okhttp3.MediaType.Companion.toMediaType import okhttp3.OkHttpClient import okhttp3.Request import okhttp3.RequestBody.Companion.toRequestBody import org.json.JSONObject import java.io.IOException class SWDelegate : SuperwallDelegate { private val client = OkHttpClient() private val coroutineScope = CoroutineScope(Dispatchers.IO) // The user tapped on a deep link to redeem a code override fun willRedeemLink() { Log.d("Superwall", "[!] willRedeemLink") // Optionally show a loading indicator here } // Superwall received a redemption result and validated the purchase with Stripe. override fun didRedeemLink(result: RedemptionResult) { Log.d("Superwall", "[!] didRedeemLink: $result") // Send Stripe IDs to RevenueCat to link purchases to the customer // Get a list of subscription ids tied to the customer. val stripeSubscriptionIds = when (result) { is RedemptionResult.Success -> result.stripeSubscriptionIds else -> null } ?: return val revenueCatStripePublicAPIKey = "strp....." // replace with your RevenueCat Stripe Public API Key val appUserId = Purchases.sharedInstance.appUserID // In the background using coroutines... coroutineScope.launch { // For each subscription id, link it to the user in RevenueCat stripeSubscriptionIds.forEach { stripeSubscriptionId -> try { val json = JSONObject().apply { put("app_user_id", appUserId) put("fetch_token", stripeSubscriptionId) } val requestBody = json.toString() .toRequestBody("application/json".toMediaType()) val request = Request.Builder() .url("https://api.revenuecat.com/v1/receipts") .post(requestBody) .addHeader("Content-Type", "application/json") .addHeader("Accept", "application/json") .addHeader("X-Platform", "stripe") .addHeader("Authorization", "Bearer $revenueCatStripePublicAPIKey") .build() client.newCall(request).execute().use { response -> val responseBody = response.body?.string().orEmpty() if (!response.isSuccessful) { throw IOException("RevenueCat responded with ${response.code}: $responseBody") } Log.d("Superwall", "[!] Success: linked $stripeSubscriptionId to user $appUserId: $responseBody") } } catch (e: Exception) { Log.e("Superwall", "[!] Error: unable to link $stripeSubscriptionId to user $appUserId", e) } } /// After all network calls complete, invalidate the cache Purchases.sharedInstance.getCustomerInfo( onSuccess = { customerInfo -> /// If you're using RevenueCat's `UpdatedCustomerInfoListener`, or keeping Superwall Entitlements in sync /// via RevenueCat's listener methods, you don't need to do anything here. Those methods will be /// called automatically when this call fetches the most up to date customer info, ignoring any local caches. /// Otherwise, if you're manually calling `Purchases.sharedInstance.getCustomerInfo` to keep Superwall's entitlements /// in sync, you should use the newly updated customer info here to do so. /// You could always access web entitlements here as well /// val webEntitlements = Superwall.instance.entitlements.web }, onError = { error -> Log.e("Superwall", "Error getting customer info", error) } ) // After all network calls complete, update UI on the main thread withContext(Dispatchers.Main) { // Perform UI updates on the main thread, like letting the user know their subscription was redeemed } } } } ``` The example surfaces non-200 responses and network exceptions so you can add retries, user messaging, or monitoring. Customize the error handling to fit your production logging and UX. If you call `logIn` from RevenueCat's SDK, then you need to call the logic you've implemented inside `didRedeemLink(result:)` again. For example, that means if `logIn` was invoked from RevenueCat, you'd either abstract out this logic above into a function to call again, or simply call this function directly. The web entitlements will be returned along with other existing entitlements in the `CustomerInfo` object accessible via RevenueCat's SDK. If you're logging in and out of RevenueCat, make sure to resend the Stripe subscription IDs to RevenueCat's endpoint after logging in. # Welcome Quick Links [#quick-links] Get up and running with the Superwall Android SDK Most common features and use cases Reference the Superwall Android SDK Guides for specific use cases Example app for the Superwall Android SDK Guides for troubleshooting common issues Feedback [#feedback] We are always improving our SDKs and documentation! If you have feedback on any of our docs, please leave a rating and message at the bottom of the page. If you have any issues with the SDK, please [open an issue on GitHub](https://github.com/superwall/superwall-android/issues). # Configure the SDK As soon as your app launches, you need to configure the SDK with your **Public API Key**. You'll retrieve this from the Superwall settings page. Sign Up & Grab Keys [#sign-up--grab-keys] If you haven't already, [sign up for a free account](https://superwall.com/sign-up) on Superwall. Then, when you're through to the Dashboard, click **Settings** from the panel on the left, click **Keys** and copy your **Public API Key**: Initialize Superwall in your app [#initialize-superwall-in-your-app] Begin by editing your main Application entrypoint. Depending on the platform this could be `AppDelegate.swift` or `SceneDelegate.swift` for iOS, `MainApplication.kt` for Android, `main.dart` in Flutter, or `App.tsx` for React Native: :::android ```kotlin Kotlin // MainApplication.kt class MainApplication : android.app.Application(), SuperwallDelegate { override fun onCreate() { super.onCreate() // Setup Superwall.configure(this, "MY_API_KEY") // OR using the DSL configureSuperwall("MY_API_KEY") { purchaseController = MyPurchaseController(this@MainApplication) } } } ``` ::: This configures a shared instance of `Superwall`, the primary class for interacting with the SDK's API. Make sure to replace `MY_API_KEY` with your public API key that you just retrieved. By default, Superwall handles basic subscription-related logic for you. However, if you’d like greater control over this process (e.g. if you’re using RevenueCat), you’ll want to pass in a `PurchaseController` to your configuration call and manually set the `subscriptionStatus`. You can also pass in `SuperwallOptions` to customize the appearance and behavior of the SDK. See [Purchases and Subscription Status](/docs/sdk/guides/advanced-configuration) for more. You've now configured Superwall! :::android For further help, check out our [Android example apps](https://github.com/superwall/Superwall-Android/tree/master/Examples) for working examples of implementing the Superwall SDK. ::: # Presenting Paywalls This allows you to register a [placement](/docs/dashboard/dashboard-campaigns/campaigns-placements) to access a feature that may or may not be paywalled later in time. It also allows you to choose whether the user can access the feature even if they don't make a purchase. Here's an example. With Superwall [#with-superwall] :::android ```kotlin Kotlin fun pressedWorkoutButton() { // remotely decide if a paywall is shown and if // navigation.startWorkout() is a paid-only feature Superwall.instance.register("StartWorkout") { navigation.startWorkout() } } ``` ::: Without Superwall [#without-superwall] :::android ```kotlin Kotlin fun pressedWorkoutButton() { if (user.hasActiveSubscription) { navigation.startWorkout() } else { navigation.presentPaywall { result -> if (result) { navigation.startWorkout() } else { // user didn't pay, developer decides what to do } } } } ``` ::: How registering placements presents paywalls [#how-registering-placements-presents-paywalls] You can configure `"StartWorkout"` to present a paywall by [creating a campaign, adding the placement, and adding a paywall to an audience](/docs/dashboard/dashboard-campaigns/campaigns) in the dashboard. 1. The SDK retrieves your campaign settings from the dashboard on app launch. 2. When a placement is called that belongs to a campaign, audiences are evaluated ***on device*** and the user enters an experiment — this means there's no delay between registering a placement and presenting a paywall. 3. If it's the first time a user is entering an experiment, a paywall is decided for the user based on the percentages you set in the dashboard 4. Once a user is assigned a paywall for an audience, they will continue to see that paywall until you remove the paywall from the audience or reset assignments to the paywall. 5. After the paywall is closed, the Superwall SDK looks at the *Feature Gating* value associated with your paywall, configurable from the paywall editor under General > Feature Gating (more on this below) 1. If the paywall is set to ***Non Gated***, the `feature:` closure on `register(placement: ...)` gets called when the paywall is dismissed (whether they paid or not) 2. If the paywall is set to ***Gated***, the `feature:` closure on `register(placement: ...)` gets called only if the user is already paying or if they begin paying. 6. If no paywall is configured, the feature gets executed immediately without any additional network calls. Given the low cost nature of how register works, we strongly recommend registering **all core functionality** in order to remotely configure which features you want to gate – **without an app update**. :::android ```kotlin Kotlin // on the welcome screen fun pressedSignUp() { Superwall.instance.register("SignUp") { navigation.beginOnboarding() } } // in another view controller fun pressedWorkoutButton() { Superwall.instance.register("StartWorkout") { navigation.startWorkout() } } ``` ::: Automatically Registered Placements [#automatically-registered-placements] The SDK [automatically registers](/docs/sdk/guides/3rd-party-analytics/tracking-analytics) some internal placements which can be used to present paywalls: Register. Everything. [#register-everything] To provide your team with ultimate flexibility, we recommend registering *all* of your analytics events, even if you don't pass feature blocks through. This way you can retroactively add a paywall almost anywhere – **without an app update**! If you're already set up with an analytics provider, you'll typically have an `Analytics.swift` singleton (or similar) to disperse all your events from. Here's how that file might look: Getting a presentation result [#getting-a-presentation-result] Use `getPresentationResult(forPlacement:params:)` when you need to ask the SDK what would happen when registering a placement — without actually showing a paywall. Superwall evaluates the placement and its audience filters then returns a `PresentationResult`. You can use this to adapt your app's behavior based on the outcome (such as showing a lock icon next to a pro feature if they aren't subscribed). In short, this lets you peek at the outcome first and decide how your app should respond: # Handling Deep Links 1. Previewing paywalls on your device before going live. 2. Deep linking to specific [campaigns](/docs/dashboard/dashboard-campaigns/campaigns). :::android 3) Web Checkout [Post-Checkout Redirecting](/docs/sdk/guides/web-checkout/post-checkout-redirecting) ::: Setup [#setup] :::android The way to deep link into your app is URL Schemes. ::: Adding a Custom URL Scheme [#adding-a-custom-url-scheme] :::android Add the following to your `AndroidManifest.xml` file: ```xml ``` This configuration allows your app to open in response to a deep link with the format `exampleapp://` from your `MainActivity` class. ::: Handling Deep Links [#handling-deep-links] :::android In your `MainActivity` (or the activity specified in your intent-filter), add the following Kotlin code to handle deep links: ```kotlin Kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) // Respond to deep links respondToDeepLinks() } private fun respondToDeepLinks() { intent?.data?.let { uri -> Superwall.instance.handleDeepLink(uri) } } } ``` ::: ::::android Handling App Links [#handling-app-links] Adding app links [#adding-app-links] Android App links enable seamless integration between the Web checkout and your app, enabling users to redeem the purchase automatically. To allow Android app links to open your app, you will need to follow these steps: 1\. Add your app's fingerprint and schema to Stripe settings [#1-add-your-apps-fingerprint-and-schema-to-stripe-settings] To verify that the request to open the app is legitimate, Android requires your app's keystore SHA256 fingerprint, with at least one for your development keystore and one for your release keystore. You can obtain these fingerprints in the following way: Development fingerprints [#development-fingerprints] If you're using Android studio or have Android components installed, you can obtain your debug key by running the following command in your terminal: `keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android` And then copying the outputted SHA256 fingerprint. Release fingerprints [#release-fingerprints] To obtain the release fingerprints, you'll need your own keystore file (the one you use to sign the final application package before publishing). You can do this by running the following command in your terminal: `keytool -list -v -keystore -alias ` And then copying the outputted SHA256 fingerprint. Adding the fingerprints to the project [#adding-the-fingerprints-to-the-project] To add the fingerprints to Superwall, open the Settings tab of your Superwall Stripe application. Under the `Android Configuration` title, you should see three fields: * Package schema - this allows us to know which schema your app uses to open and parse deep links * Package name - your app's package name, i.e. `com.mydomain.myapp` * App fingerprints - One or more of your app's fingerprints, comma separated Once added, click the `Update Configuration` button which will ensure the application asset links are properly generated for Google to verify. 2\. Add the schema to your app's Android Manifest [#2-add-the-schema-to-your-apps-android-manifest] For this, you'll need to copy the domain from your Superwall Stripe settings. Then, open your `AndroidManifest.xml` and inside the `` tag declaring your deep link handling activity, add the following, replacing the domain with the one from the settings: ```xml ``` 3\. Handle incoming deep links using Superwall SDK [#3-handle-incoming-deep-links-using-superwall-sdk] In the same activity as in step #2, you'll need to pass deeplinks along to Superwall SDK. You can do this by overriding your Activity's `onCreate` and `onNewIntent` methods and passing along the intent data to Superwall using `Superwall.handleDeepLinks()` method. The method returns a `kotlin.Result` indicating if the deep link will be handled by Superwall SDK. ```kotlin override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) // your onCreate code intent?.data?.let { uri -> Superwall.handleDeepLink(uri) } } override fun onNewIntent(intent: Intent, caller: ComponentCaller) { super.onNewIntent(intent, caller) // ... Your onNewIntent code intent?.data?.let { uri -> Superwall.handleDeepLink(uri) } } ``` 4\. Handling links while testing [#4-handling-links-while-testing] If you are running your app using Android Studio, you need to be aware that it won't automatically allow verified links to be opened using the app, but you will need to enable it in settings yourself. This is not the case when installing from Play Store, and all the links will be handled automatically. To do that, once you install the app on your device, open: `Settings > Apps > Your Application Name > Open By Default` Under there, the `Open supported links` should be enabled. Tap `Add Links` button and selected the available links. Testing & more details [#testing--more-details] For details regarding testing and setup, you can refer to [Android's guide for verifying app links](https://developer.android.com/training/app-links/verify-android-applinks). Note - Superwall generates the assetlinks.json for you. To check the file, you can use the subdomain from your Superwall stripe configuation: `https://my-app.superwall.app/.well-known/assetlinks.json` :::android ::: :::: Previewing Paywalls [#previewing-paywalls] Next, build and run your app on your phone. Then, head to the Superwall Dashboard. Click on **Settings** from the Dashboard panel on the left, then select **General**: With the **General** tab selected, type your custom URL scheme, without slashes, into the **Apple Custom URL Scheme** field: Next, open your paywall from the dashboard and click **Preview**. You'll see a QR code appear in a pop-up:
On your device, scan this QR code. You can do this via Apple's Camera app. This will take you to a paywall viewer within your app, where you can preview all your paywalls in different configurations. Using Deep Links to Present Paywalls [#using-deep-links-to-present-paywalls] Deep links can also be used as a placement in a campaign to present paywalls. Simply add `deepLink_open` as an placement, and the URL parameters of the deep link can be used as parameters! You can also use custom placements for this purpose. [Read this doc](/docs/dashboard/guides/presenting-paywalls-from-one-another) for examples of both. Related deep link guides [#related-deep-link-guides] :::android * [Handling Deep Links](/docs/sdk/guides/handling-deep-links) — Use `handleDeepLink` with the `deepLink_open` standard placement and dashboard campaign rules to present paywalls from deep links, without hardcoding routing logic in your app. ::: # Install the SDK Overview [#overview] To see the latest release, [check out the repository](https://github.com/superwall/Superwall-Android) Install via Gradle [#install-via-gradle] [Gradle](https://developer.android.com/build/releases/gradle-plugin) is the preferred way to install Superwall for Android. In your `build.gradle` or `build.gradle.kts` add the latest Superwall SDK. You can find the [latest release here](https://github.com/superwall/Superwall-Android/releases). ```groovy build.gradle implementation "com.superwall.sdk:superwall-android:2.7.14" ``` ```kotlin build.gradle.kts implementation("com.superwall.sdk:superwall-android:2.7.14") ``` ```toml libs.version.toml [libraries] superwall-android = { group = "com.superwall.sdk", name = "superwall-android", version = "2.7.14" } // And in your build.gradle.kts dependencies { implementation(libs.superwall.android) } ``` Make sure to run **Sync Now** to force Android Studio to update.
Go to your `AndroidManifest.xml` and add the following permissions: ```xml AndroidManifest.xml ``` Then add our Activity to your `AndroidManifest.xml`: ```xml AndroidManifest.xml ``` Set your app's theme in the `android:theme` section. When choosing a device or emulator to run on make sure that it has the Play Store app and that you are signed in to your Google account on the Play Store. **And you're done!** Now you're ready to configure the SDK 👇 # Setting User Attributes By setting user attributes, you can display information about the user on the paywall. You can also define [audiences](/docs/dashboard/dashboard-campaigns/campaigns-audience) in a campaign to determine which paywall to show to a user, based on their user attributes. If a paywall uses the **Set user attributes** action, the merged attributes are sent back to your app via `SuperwallDelegate.userAttributesDidChange(newAttributes:)`. You do this by passing a `[String: Any?]` dictionary of attributes to `Superwall.shared.setUserAttributes(_:)`: :::android ```kotlin Kotlin val attributes = mapOf( "name" to user.name, "apnsToken" to user.apnsTokenString, "email" to user.email, "username" to user.username, "profilePic" to user.profilePicUrl, "stripe_customer_id" to user.stripeCustomerId // Optional: For Stripe checkout prefilling ) Superwall.instance.setUserAttributes(attributes) // (merges existing attributes) ``` ::: Usage [#usage] This is a merge operation, such that if the existing user attributes dictionary already has a value for a given property, the old value is overwritten. Other existing properties will not be affected. To unset/delete a value, you can pass `nil` for the value. You can reference user attributes in [audience filters](/docs/dashboard/dashboard-campaigns/campaigns-audience) to help decide when to display your paywall. When you configure your paywall, you can also reference the user attributes in its text variables. For more information on how to that, see [Configuring a Paywall](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-overview). # Tracking Subscription State Superwall tracks the subscription state of a user for you. However, there are times in your app where you need to know if a user is on a paid plan or not. For example, you might want to conditionally show certain UI elements or enable premium features based on their subscription status. Using subscriptionStatus [#using-subscriptionstatus] The easiest way to track subscription status in Android is by accessing the `subscriptionStatus` StateFlow: ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) // Get current status val status = Superwall.instance.subscriptionStatus.value when (status) { is SubscriptionStatus.Active -> { Log.d("Superwall", "User has active entitlements: ${status.entitlements}") showPremiumContent() } is SubscriptionStatus.Inactive -> { Log.d("Superwall", "User is on free plan") showFreeContent() } is SubscriptionStatus.Unknown -> { Log.d("Superwall", "Subscription status unknown") showLoadingState() } } } } ``` The `SubscriptionStatus` sealed class has three possible states: * `SubscriptionStatus.Unknown` - Status is not yet determined * `SubscriptionStatus.Active(Set)` - User has active entitlements (set of entitlement identifiers) * `SubscriptionStatus.Inactive` - User has no active entitlements Observing subscription status changes [#observing-subscription-status-changes] You can observe real-time subscription status changes using Kotlin's StateFlow: ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) lifecycleScope.launch { Superwall.instance.subscriptionStatus.collect { status -> when (status) { is SubscriptionStatus.Active -> { Log.d("Superwall", "User upgraded to pro!") updateUiForPremiumUser() } is SubscriptionStatus.Inactive -> { Log.d("Superwall", "User is on free plan") updateUiForFreeUser() } is SubscriptionStatus.Unknown -> { Log.d("Superwall", "Loading subscription status...") showLoadingState() } } } } } } ``` Using with Jetpack Compose [#using-with-jetpack-compose] If you're using Jetpack Compose, you can observe subscription status reactively: ```kotlin @Composable fun ContentScreen() { val subscriptionStatus by Superwall.instance.subscriptionStatus .collectAsState() Column { when (subscriptionStatus) { is SubscriptionStatus.Active -> { val entitlements = (subscriptionStatus as SubscriptionStatus.Active).entitlements Text("Premium user with: ${entitlements.joinToString()}") PremiumContent() } is SubscriptionStatus.Inactive -> { Text("Free user") FreeContent() } is SubscriptionStatus.Unknown -> { Text("Loading...") LoadingIndicator() } } } } ``` Reading detailed purchase history (2.6.6+) [#reading-detailed-purchase-history-266] When you need more context than `SubscriptionStatus` provides (for example, to show the full transaction history or mix web redemptions with Google Play receipts), subscribe to `Superwall.instance.customerInfo`. The flow emits a `CustomerInfo` object that merges device, web, and external purchase controller data. ```kotlin class BillingDashboardFragment : Fragment() { override fun onViewCreated(view: View, savedInstanceState: Bundle?) { viewLifecycleOwner.lifecycleScope.launch { Superwall.instance.customerInfo.collect { info -> val subscriptions = info.subscriptions.map { it.productId to it.expiresDate } val nonSubscriptions = info.nonSubscriptions.map { it.productId to it.purchaseDate } val entitlementIds = info.entitlements.filter { it.isActive }.map { it.id } renderCustomerInfo( activeProducts = info.activeSubscriptionProductIds, entitlements = entitlementIds, subscriptions = subscriptions, oneTimePurchases = nonSubscriptions ) } } } } ``` Need the latest value immediately (for example, during cold start)? Call `Superwall.instance.getCustomerInfo()` to synchronously read the most recent snapshot before collecting the flow: ```kotlin val cached = Superwall.instance.getCustomerInfo() renderCustomerInfo( activeProducts = cached.activeSubscriptionProductIds, entitlements = cached.entitlements.filter { it.isActive }.map { it.id }, subscriptions = cached.subscriptions.map { it.productId to it.purchaseDate }, oneTimePurchases = cached.nonSubscriptions.map { it.productId to it.purchaseDate } ) ``` After you start collecting, you can also watch for [`SuperwallDelegate.customerInfoDidChange(from:to:)`](/docs/android/sdk-reference/SuperwallDelegate#customerinfodidchangefrom-customerinfo-to-customerinfo) to run analytics or sync other systems whenever purchases change. Checking for specific entitlements [#checking-for-specific-entitlements] If your app has multiple subscription tiers (e.g., Bronze, Silver, Gold), you can check for specific entitlements: ```kotlin val status = Superwall.instance.subscriptionStatus.value when (status) { is SubscriptionStatus.Active -> { if (status.entitlements.contains("gold")) { // Show gold-tier features showGoldFeatures() } else if (status.entitlements.contains("silver")) { // Show silver-tier features showSilverFeatures() } } else -> showFreeFeatures() } ``` Setting subscription status [#setting-subscription-status] When using Superwall with a custom purchase controller or third-party billing service, you need to manually update the subscription status. Here's how to sync with RevenueCat: ```kotlin class RevenueCatPurchaseController : PurchaseController { override suspend fun purchase( activity: Activity, product: StoreProduct ): PurchaseResult { return try { val result = Purchases.sharedInstance.purchase(activity, product.sku) // Update Superwall subscription status based on RevenueCat result if (result.isSuccessful) { val entitlements = result.customerInfo.entitlements.active.keys Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(entitlements) ) PurchaseResult.Purchased } else { PurchaseResult.Failed(Exception("Purchase failed")) } } catch (e: Exception) { PurchaseResult.Failed(e) } } override suspend fun restorePurchases(): RestorationResult { return try { val customerInfo = Purchases.sharedInstance.restorePurchases() val activeEntitlements = customerInfo.entitlements.active.keys if (activeEntitlements.isNotEmpty()) { Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(activeEntitlements) ) } else { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) } RestorationResult.Restored } catch (e: Exception) { RestorationResult.Failed(e) } } } ``` You can also listen for subscription changes from your billing service: ```kotlin class SubscriptionManager { fun syncSubscriptionStatus() { Purchases.sharedInstance.getCustomerInfoWith { customerInfo -> val activeEntitlements = customerInfo.entitlements.active.keys if (activeEntitlements.isNotEmpty()) { Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(activeEntitlements) ) } else { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) } } } } ``` Using SuperwallDelegate [#using-superwalldelegate] You can also listen for subscription status changes using the `SuperwallDelegate`: ```kotlin class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure( applicationContext = this, apiKey = "YOUR_API_KEY", options = SuperwallOptions().apply { delegate = object : SuperwallDelegate() { override fun subscriptionStatusDidChange( from: SubscriptionStatus, to: SubscriptionStatus ) { when (to) { is SubscriptionStatus.Active -> { Log.d("Superwall", "User is now premium") } is SubscriptionStatus.Inactive -> { Log.d("Superwall", "User is now free") } is SubscriptionStatus.Unknown -> { Log.d("Superwall", "Status unknown") } } } } } ) } } ``` Superwall checks subscription status for you [#superwall-checks-subscription-status-for-you] Remember that the Superwall SDK uses its [audience filters](/docs/dashboard/dashboard-campaigns/campaigns-audience#matching-to-entitlements) for determining when to show paywalls. You generally don't need to wrap your calls to register placements with subscription status checks: ```kotlin // ❌ Unnecessary if (Superwall.instance.subscriptionStatus.value !is SubscriptionStatus.Active) { Superwall.instance.register("campaign_trigger") } // ✅ Just register the placement Superwall.instance.register("campaign_trigger") ``` In your [audience filters](/docs/dashboard/dashboard-campaigns/campaigns-audience#matching-to-entitlements), you can specify whether the subscription state should be considered, which keeps your codebase cleaner and puts the "Should this paywall show?" logic where it belongs—in the Superwall dashboard. # User Management Anonymous Users [#anonymous-users] Superwall automatically generates a random user ID that persists internally until the user deletes/reinstalls your app. You can call `Superwall.shared.reset()` to reset this ID and clear any paywall assignments. Identified Users [#identified-users] If you use your own user management system, call `identify(userId:options:)` when you have a user's identity. This will alias your `userId` with the anonymous Superwall ID enabling us to load the user’s assigned paywalls. Calling `Superwall.shared.reset()` will reset the on-device userId to a random ID and clear the paywall assignments. :::android Google Play receives the identifier you pass to `BillingFlowParams.Builder.setObfuscatedAccountId` as both the **`obfuscatedExternalAccountId`** and the `externalAccountId` we forward to your Superwall backend. By default we SHA-256 hash your `userId` before sending it. If you want the raw `appUserId` to appear in Play Console and downstream server events, set `SuperwallOptions().passIdentifiersToPlayStore = true` before configuring the SDK. Make sure the value complies with [Google's policies](https://developer.android.com/reference/com/android/billingclient/api/BillingFlowParams.Builder#setObfuscatedAccountId)—it must not contain personally identifiable information. ::: :::android ```kotlin Kotlin // After retrieving a user's ID, e.g. from logging in or creating an account Superwall.instance.identify(user.id) // When the user signs out Superwall.instance.reset() ``` :::
**Advanced Use Case** You can supply an `IdentityOptions` object, whose property `restorePaywallAssignments` you can set to `true`. This tells the SDK to wait to restore paywall assignments from the server before presenting any paywalls. This should only be used in advanced use cases. If you expect users of your app to switch accounts or delete/reinstall a lot, you'd set this when users log in to an existing account. Best Practices for a Unique User ID [#best-practices-for-a-unique-user-id] * Do NOT make your User IDs guessable – they are public facing. * Do NOT set emails as User IDs – this isn't GDPR compliant. * Do NOT set IDFA or DeviceIds as User IDs – these are device specific / easily rotated by the operating system. * Do NOT hardcode strings as User IDs – this will cause every user to be treated as the same user by Superwall. Identifying users from App Store server events [#identifying-users-from-appstore-server-events] On iOS, Superwall always supplies an [`appAccountToken`](https://developer.apple.com/documentation/storekit/product/purchaseoption/3749440-appaccounttoken) with every StoreKit 2 transaction: | Scenario | Value used for `appAccountToken` | | -------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------- | | You’ve called `Superwall.shared.identify(userId:)` | The exact `userId` you passed | | You *haven’t* called `identify` yet | The UUID automatically generated for the anonymous user (the **alias ID**), **without** the `$SuperwallAlias:` prefix | | You passed a non‑UUID `userId` to `identify` | StoreKit rejects it; Superwall falls back to the alias UUID | Because the SDK falls back to the alias UUID, purchase notifications sent to your server always include a stable, unique identifier—even before the user signs in. ```swift // Generate and use a UUID user ID in Swift let userId = UUID().uuidString Superwall.shared.identify(userId: userId) ``` # PaywallOptions `PaywallOptions` is provided via the `paywalls` property on [`SuperwallOptions`](/docs/android/sdk-reference/SuperwallOptions) and is passed to the SDK when you call [`configure`](/docs/android/sdk-reference/configure). Purpose [#purpose] Customize how paywalls look and behave, including preload behavior, alerts, dismissal, and haptics. Signature [#signature] ```kotlin import com.superwall.sdk.analytics.Tier import kotlin.time.Duration class PaywallOptions { var isHapticFeedbackEnabled: Boolean = true class RestoreFailed { var title: String = "No Subscription Found" var message: String = "We couldn't find an active subscription for your account." var closeButtonTitle: String = "Okay" } var restoreFailed: RestoreFailed = RestoreFailed() var shouldShowPurchaseFailureAlert: Boolean = true var shouldPreload: Boolean = true var preloadDeviceOverrides: Map = emptyMap() var useCachedTemplates: Boolean = false var automaticallyDismiss: Boolean = true enum class TransactionBackgroundView { SPINNER } var transactionBackgroundView: TransactionBackgroundView? = TransactionBackgroundView.SPINNER var overrideProductsByName: Map = emptyMap() var optimisticLoading: Boolean = false var timeoutAfter: Duration? = null var onBackPressed: ((PaywallInfo?) -> Boolean)? = null } ``` Parameters [#parameters] Usage [#usage] ```kotlin val paywallOptions = PaywallOptions().apply { isHapticFeedbackEnabled = true shouldShowPurchaseFailureAlert = false shouldPreload = true preloadDeviceOverrides = mapOf( Tier.ULTRA_LOW to false, Tier.LOW to false, ) useCachedTemplates = false automaticallyDismiss = true transactionBackgroundView = PaywallOptions.TransactionBackgroundView.SPINNER overrideProductsByName = mapOf( "primary" to "com.example.premium_monthly", "tertiary" to "com.example.premium_annual", ) optimisticLoading = false timeoutAfter = null onBackPressed = { paywallInfo -> // Custom back button handling // Return true to consume the back press, false to use SDK default false } } val options = SuperwallOptions().apply { paywalls = paywallOptions } Superwall.configure( application = this, apiKey = "pk_your_api_key", options = options, ) ``` `preloadDeviceOverrides` is useful when you want to keep preloading enabled by default but disable it on lower-end devices. Tiers you do not specify continue to use the `shouldPreload` value. Related [#related] * [`SuperwallOptions`](/docs/android/sdk-reference/SuperwallOptions) # PurchaseController **This interface is not required.** By default, Superwall handles all subscription-related logic automatically using Google Play Billing. When implementing PurchaseController, you must manually update [`subscriptionStatus`](/docs/android/sdk-reference/subscriptionStatus) whenever the user's entitlements change. Purpose [#purpose] Use this interface only if you want complete control over purchase handling, such as when using RevenueCat or other third-party purchase frameworks. Signature [#signature] ```kotlin interface PurchaseController { suspend fun purchase( activity: Activity, product: StoreProduct ): PurchaseResult suspend fun restorePurchases(): RestorationResult } ``` ```java // Java public interface PurchaseController { CompletableFuture purchase( Activity activity, StoreProduct product ); CompletableFuture restorePurchases(); } ``` Parameters [#parameters] Returns / State [#returns--state] * `purchase()` returns a `PurchaseResult` (`.Purchased`, `.Failed(Throwable)`, `.Cancelled`, or `.Pending`) * `restorePurchases()` returns a `RestorationResult` (`.Restored` or `.Failed(Throwable?)`) When using a PurchaseController, you must also manage [`subscriptionStatus`](/docs/android/sdk-reference/subscriptionStatus) yourself. Usage [#usage] For implementation examples and detailed guidance, see [Using RevenueCat](/docs/android/guides/using-revenuecat). # Superwall You must call [`configure()`](/docs/android/sdk-reference/configure) before accessing `Superwall.instance`, otherwise your app will crash. Purpose [#purpose] Provides access to the configured Superwall instance after calling [`configure()`](/docs/android/sdk-reference/configure). Signature [#signature] ```kotlin companion object { val instance: Superwall } ``` ```java // Java public static Superwall getInstance() ``` Parameters [#parameters] This is a companion object property with no parameters. Returns / State [#returns--state] Returns the shared `Superwall` instance that was configured via [`configure()`](/docs/android/sdk-reference/configure). Usage [#usage] Configure first (typically in Application class): ```kotlin class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure( application = this, apiKey = "pk_your_api_key" ) } } ``` Then access throughout your app: ```kotlin Superwall.instance.register("feature_access") { // Feature code here } ``` Set user identity and attributes: ```kotlin Superwall.instance.identify("user123") Superwall.instance.setUserAttributes(mapOf( "plan" to "premium", "signUpDate" to System.currentTimeMillis() )) ``` Reset the user: ```kotlin Superwall.instance.reset() ``` Avoid calling `Superwall.instance.reset()` repeatedly. Resetting rotates the anonymous user ID, clears local paywall assignments, and requires the SDK to re-download configuration state. Only trigger a reset when a user explicitly logs out or you intentionally need to forget their identity. See [User Management](/docs/android/quickstart/user-management) for more guidance. Set delegate: ```kotlin Superwall.instance.delegate = this ``` Consume a purchase (2.6.2+): ```kotlin // Using coroutines lifecycleScope.launch { val result = Superwall.instance.consume(purchaseToken) result.fold( onSuccess = { token -> println("Purchase consumed: $token") }, onFailure = { error -> println("Failed to consume: ${error.message}") } ) } // Using callback Superwall.instance.consume(purchaseToken) { result -> result.fold( onSuccess = { token -> println("Purchase consumed: $token") }, onFailure = { error -> println("Failed to consume: ${error.message}") } ) } ``` Show an alert over the current paywall (2.5.3+): ```kotlin Superwall.instance.showAlert( title = "Important Notice", message = "Your subscription will renew soon", actionTitle = "View Details", closeActionTitle = "Dismiss", action = { // Handle action button tap navigateToSubscriptionSettings() }, onClose = { // Handle close/dismiss println("Alert dismissed") } ) ``` Set integration attributes for analytics (2.5.3+): ```kotlin import com.superwall.sdk.models.attribution.AttributionProvider Superwall.instance.setIntegrationAttributes( mapOf( AttributionProvider.ADJUST to "adjust_user_id_123", AttributionProvider.MIXPANEL to "mixpanel_distinct_id_456", AttributionProvider.META to "meta_user_id_789", AttributionProvider.GOOGLE_ADS to "google_ads_id_101", AttributionProvider.GOOGLE_APP_SET to "google_app_set_id_202", AttributionProvider.APPSTACK to "appstack_user_id_303" ) ) ``` Observe customer info (2.6.6+) [#observe-customer-info-266] Superwall now exposes purchase history and entitlement snapshots via a `StateFlow`. Each emission contains merged device, web, and external purchase controller data so you can react to subscription changes without wiring up your own polling layer. ```kotlin lifecycleScope.launch { Superwall.instance.customerInfo.collect { info -> val activeProductIds = info.activeSubscriptionProductIds val activeEntitlementIds = info.entitlements .filter { it.isActive } .map { it.id } updateUi( subscriptions = activeProductIds, entitlements = activeEntitlementIds ) } } ``` Need an immediate snapshot (for example during cold start)? Call `Superwall.instance.getCustomerInfo()` to synchronously read the latest cached value, or wire both together: ```kotlin val cachedInfo = Superwall.instance.getCustomerInfo() render(cachedInfo) lifecycleScope.launch { Superwall.instance.customerInfo.collect { render(it) } } ``` Pair the flow with [`SuperwallDelegate.customerInfoDidChange(from:to:)`](/docs/android/sdk-reference/SuperwallDelegate#customerinfodidchangefrom-customerinfo-to-customerinfo) when you need to mirror changes into analytics. Java usage: ```java // Access the instance Superwall.getInstance().register("feature_access", () -> { // Feature code here }); // Set user identity Superwall.getInstance().identify("user123"); ``` # SuperwallDelegate Set the delegate using `Superwall.instance.delegate = this` to receive these callbacks. For Java, use `setJavaDelegate()` for better Java interop. Use `handleSuperwallEvent(eventInfo)` to track Superwall analytics events in your own analytics platform for a complete view of user behavior. Purpose [#purpose] Provides callbacks for Superwall lifecycle events, analytics tracking, and custom paywall interactions. Signature [#signature] ```kotlin interface SuperwallDelegate { fun subscriptionStatusDidChange( from: SubscriptionStatus, to: SubscriptionStatus ) {} fun customerInfoDidChange( from: CustomerInfo, to: CustomerInfo ) {} fun userAttributesDidChange(newAttributes: Map) {} fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) {} fun handleCustomPaywallAction(name: String) {} fun willDismissPaywall(paywallInfo: PaywallInfo) {} fun willPresentPaywall(paywallInfo: PaywallInfo) {} fun didDismissPaywall(paywallInfo: PaywallInfo) {} fun didPresentPaywall(paywallInfo: PaywallInfo) {} fun paywallWillOpenURL(url: String) {} fun paywallWillOpenDeepLink(url: String) {} fun handleLog( level: LogLevel, scope: LogScope, message: String, info: Map?, error: Throwable? ) {} } ``` ```java // Java - Use SuperwallDelegateJava for better Java interop public interface SuperwallDelegateJava { default void subscriptionStatusDidChange( SubscriptionStatus from, SubscriptionStatus to ) {} default void customerInfoDidChange( CustomerInfo from, CustomerInfo to ) {} default void userAttributesDidChange(Map newAttributes) {} default void handleSuperwallEvent(SuperwallEventInfo eventInfo) {} default void handleCustomPaywallAction(String name) {} default void willDismissPaywall(PaywallInfo paywallInfo) {} default void willPresentPaywall(PaywallInfo paywallInfo) {} default void didDismissPaywall(PaywallInfo paywallInfo) {} default void didPresentPaywall(PaywallInfo paywallInfo) {} default void paywallWillOpenURL(String url) {} default void paywallWillOpenDeepLink(String url) {} default void handleLog( LogLevel level, LogScope scope, String message, Map info, Throwable error ) {} } ``` Parameters [#parameters] All methods are optional to implement. Key methods include: Returns / State [#returns--state] All delegate methods return `Unit`. They provide information about Superwall events and state changes. Usage [#usage] Basic delegate setup: ```kotlin class MainActivity : AppCompatActivity(), SuperwallDelegate { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) Superwall.instance.delegate = this } } ``` Track subscription status changes: ```kotlin override fun subscriptionStatusDidChange( from: SubscriptionStatus, to: SubscriptionStatus ) { println("Subscription changed from $from to $to") updateUI(to) } ``` Mirror merged purchase history: ```kotlin override fun customerInfoDidChange( from: CustomerInfo, to: CustomerInfo ) { if (from.activeSubscriptionProductIds != to.activeSubscriptionProductIds) { Analytics.track("customer_info_updated", mapOf( "old_products" to from.activeSubscriptionProductIds.joinToString(), "new_products" to to.activeSubscriptionProductIds.joinToString() )) } refreshEntitlementBadge(to.entitlements.filter { it.isActive }.map { it.id }) } ``` Capture remote attribute changes: ```kotlin override fun userAttributesDidChange(newAttributes: Map) { // Paywall forms or surveys can set attributes directly. // Forward them to your analytics platform or local cache. analytics.identify(newAttributes) } ``` Forward analytics events: ```kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when (val event = eventInfo.event) { is SuperwallEvent.PaywallOpen -> { Analytics.track("paywall_opened", mapOf( "paywall_id" to event.paywallInfo.id, "placement" to event.paywallInfo.placement )) } is SuperwallEvent.TransactionComplete -> { Analytics.track("subscription_purchased", mapOf( "product_id" to event.product.id, "paywall_id" to event.paywallInfo.id )) } else -> { // Handle other events } } } ``` Handle custom paywall actions: ```kotlin override fun handleCustomPaywallAction(name: String) { when (name) { "help" -> presentHelpScreen() "contact" -> presentContactForm() else -> println("Unknown custom action: $name") } } ``` Handle paywall lifecycle: ```kotlin override fun willPresentPaywall(paywallInfo: PaywallInfo) { // Pause video, hide UI, etc. pauseBackgroundTasks() } override fun didDismissPaywall(paywallInfo: PaywallInfo) { // Resume video, show UI, etc. resumeBackgroundTasks() } ``` Java usage: ```java public class MainActivity extends AppCompatActivity implements SuperwallDelegateJava { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Superwall.getInstance().setJavaDelegate(this); } @Override public void subscriptionStatusDidChange( SubscriptionStatus from, SubscriptionStatus to ) { System.out.println("Subscription changed from " + from + " to " + to); updateUI(to); } @Override public void customerInfoDidChange( CustomerInfo from, CustomerInfo to ) { Logger.i("Superwall", "Customer info updated: " + to.getActiveSubscriptionProductIds()); syncUserPurchases(to); } @Override public void userAttributesDidChange(Map newAttributes) { analytics.identify(newAttributes); } } ``` # SuperwallEvent These events provide comprehensive analytics about user behavior and paywall performance. Use them to track conversion funnels, user engagement, and revenue metrics in your analytics platform. Common events to track for conversion analysis include `TriggerFire`, `PaywallOpen`, `TransactionStart`, and `TransactionComplete`. Purpose [#purpose] Represents internal analytics events tracked by Superwall and sent to the [`SuperwallDelegate`](/docs/android/sdk-reference/SuperwallDelegate) for forwarding to your analytics platform. Signature [#signature] ```kotlin sealed class SuperwallEvent { // User lifecycle events object FirstSeen : SuperwallEvent() object AppOpen : SuperwallEvent() object AppLaunch : SuperwallEvent() object AppClose : SuperwallEvent() object SessionStart : SuperwallEvent() object IdentityAlias : SuperwallEvent() object AppInstall : SuperwallEvent() // Deep linking data class DeepLink(val url: String) : SuperwallEvent() // Paywall events data class TriggerFire( val placementName: String, val result: TriggerResult ) : SuperwallEvent() data class PaywallOpen(val paywallInfo: PaywallInfo) : SuperwallEvent() data class PaywallPageView( val paywallInfo: PaywallInfo, val data: PageViewData ) : SuperwallEvent() data class PaywallClose(val paywallInfo: PaywallInfo) : SuperwallEvent() data class PaywallDecline(val paywallInfo: PaywallInfo) : SuperwallEvent() // Transaction events data class TransactionStart( val product: StoreProduct, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class TransactionComplete( val transaction: StoreTransaction?, val product: StoreProduct, val type: TransactionType, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class TransactionFail( val error: TransactionError, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class TransactionAbandon( val product: StoreProduct, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class TransactionRestore( val restoreType: RestoreType, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class TransactionTimeout(val paywallInfo: PaywallInfo) : SuperwallEvent() // Subscription events data class SubscriptionStart( val product: StoreProduct, val paywallInfo: PaywallInfo ) : SuperwallEvent() data class FreeTrialStart( val product: StoreProduct, val paywallInfo: PaywallInfo ) : SuperwallEvent() object SubscriptionStatusDidChange : SuperwallEvent() // Restore events sealed class Restore : SuperwallEvent() { object Start : Restore() data class Fail(val error: String) : Restore() object Complete : Restore() } // Customer and permission events data class CustomerInfoDidChange( val from: CustomerInfo, val to: CustomerInfo ) : SuperwallEvent() data class PermissionRequested( val permissionName: String, val paywallIdentifier: String ) : SuperwallEvent() data class PermissionGranted( val permissionName: String, val paywallIdentifier: String ) : SuperwallEvent() data class PermissionDenied( val permissionName: String, val paywallIdentifier: String ) : SuperwallEvent() // Preloading events data class PaywallPreloadStart(val paywallIdentifier: String) : SuperwallEvent() data class PaywallPreloadComplete(val paywallCount: Int) : SuperwallEvent() // Test mode events class TestModeModalOpen : SuperwallEvent() class TestModeModalClose : SuperwallEvent() // System events data class DeviceAttributes(val attributes: Map) : SuperwallEvent() data class SurveyResponse(val survey: Survey, val selectedOption: SurveyOption, val customResponse: String?, val paywallInfo: PaywallInfo) : SuperwallEvent() class SurveyClose : SuperwallEvent() data class CustomPlacement(val name: String, val params: Map, val paywallInfo: PaywallInfo) : SuperwallEvent() object Reset : SuperwallEvent() // And more... } ``` ```java // Java - SuperwallEvent is a sealed class hierarchy // Access via pattern matching or instanceof checks ``` Parameters [#parameters] Each event contains associated values with relevant information for that event type. Common parameters include: * `paywallInfo: PaywallInfo` - Information about the paywall * `data: PageViewData` - Metadata for multi-page paywall navigation, including the page name, position in the flow, navigation type, and previous-page timing when available * `product: StoreProduct` - The product involved in transactions * `url: String` - Deep link URLs * `attributes: Map` - Device or user attributes Returns / State [#returns--state] This is a sealed class that represents different event types. Events are received via [`SuperwallDelegate.handleSuperwallEvent(eventInfo)`](/docs/android/sdk-reference/SuperwallDelegate). Usage [#usage] These events are received via [`SuperwallDelegate.handleSuperwallEvent(eventInfo)`](/docs/android/sdk-reference/SuperwallDelegate) for forwarding to your analytics platform. Deprecations in 2.7.0 [#deprecations-in-270] * `PaywallWebviewLoadTimeout` is deprecated. This event was causing confusion due to its naming and has been removed from internal tracking. It will no longer fire. New events in 2.7.10 [#new-events-in-2710] * `PaywallPageView` fires when a user navigates within a multi-page paywall. Use `event.data.pageName`, `event.data.flowPosition`, and `event.data.navigationType` to understand how they moved through the flow, and inspect the optional previous-page fields when you need timing context. New events in 2.6.6+ [#new-events-in-266] * `CustomerInfoDidChange` fires whenever the SDK merges device, web, and external purchase controller data into a new [`CustomerInfo`](/docs/android/quickstart/tracking-subscription-state#reading-detailed-purchase-history-2-6-6) snapshot. The event includes the previous and next objects so you can diff entitlements or transactions. * `PermissionRequested`, `PermissionGranted`, and `PermissionDenied` correspond to the new **Request permission** action in the paywall editor. Each event carries the `permissionName` and `paywallIdentifier`. * `PaywallPreloadStart` and `PaywallPreloadComplete` track when preloading kicks off and how many paywalls finished warming the cache. Example handler: ```kotlin override fun handleSuperwallEvent(eventInfo: SuperwallEventInfo) { when (val event = eventInfo.event) { is SuperwallEvent.CustomerInfoDidChange -> { analytics.track("customer_info_updated", mapOf( "old_products" to event.from.activeSubscriptionProductIds.joinToString(), "new_products" to event.to.activeSubscriptionProductIds.joinToString() )) } is SuperwallEvent.PermissionRequested -> { analytics.track("permission_requested", mapOf( "permission" to event.permissionName, "paywall_id" to event.paywallIdentifier )) } is SuperwallEvent.PermissionGranted -> { featureFlags.unlock(event.permissionName) } is SuperwallEvent.PermissionDenied -> { showPermissionHelpSheet(event.permissionName) } is SuperwallEvent.PaywallPageView -> { analytics.track("paywall_page_view", mapOf( "paywall_id" to event.paywallInfo.id, "page_name" to event.data.pageName, "flow_position" to event.data.flowPosition, "navigation_type" to event.data.navigationType )) } is SuperwallEvent.PaywallPreloadComplete -> { Logger.i("Superwall", "Preloaded ${event.paywallCount} paywalls") } else -> Unit } } ``` # SuperwallOptions Only modify `networkEnvironment` if explicitly instructed by the Superwall team. Use `.RELEASE` (default) for production apps. Use different `SuperwallOptions` configurations for debug and release builds to optimize logging, purchasing behavior, and Google Play settings for each environment. Purpose [#purpose] Configures paywall presentation, logging, Google Play purchase behavior, and other global SDK settings. Pass an instance in to [`Superwall.configure`](/docs/android/sdk-reference/configure). Signature [#signature] ```kotlin class SuperwallOptions { var paywalls: PaywallOptions = PaywallOptions() var shouldObservePurchases: Boolean = false var networkEnvironment: NetworkEnvironment = NetworkEnvironment.Release() var isExternalDataCollectionEnabled: Boolean = true var localeIdentifier: String? = null var isGameControllerEnabled: Boolean = false var passIdentifiersToPlayStore: Boolean = false var enableExperimentalDeviceVariables: Boolean = false var logging: Logging = Logging() var useMockReviews: Boolean = false var testModeBehavior: TestModeBehavior = TestModeBehavior.AUTOMATIC } ``` ```java // Java public class SuperwallOptions { public PaywallOptions paywalls = new PaywallOptions(); public boolean shouldObservePurchases = false; public NetworkEnvironment networkEnvironment = new NetworkEnvironment.Release(); public boolean isExternalDataCollectionEnabled = true; public @Nullable String localeIdentifier = null; public boolean isGameControllerEnabled = false; public boolean passIdentifiersToPlayStore = false; public boolean enableExperimentalDeviceVariables = false; public Logging logging = new Logging(); public boolean useMockReviews = false; public TestModeBehavior testModeBehavior = TestModeBehavior.AUTOMATIC; } ``` Parameters [#parameters] How `passIdentifiersToPlayStore` affects Google Play [#how-passidentifierstoplaystore-affects-google-play] Superwall always calls `BillingFlowParams.Builder.setObfuscatedAccountId(Superwall.instance.externalAccountId)` when launching a billing flow. * **Default (`false`)** – `externalAccountId` is a SHA-256 hash of the `userId`. Google Play displays the hashed value as `obfuscatedExternalAccountId`, and the same hash is sent back to your servers. * **Enabled (`true`)** – Superwall forwards the exact `appUserId` you passed to [`identify()`](/docs/android/sdk-reference/identify). This makes it easier to correlate Google Play purchases with your users, but the value must comply with [Google's policy](https://developer.android.com/reference/com/android/billingclient/api/BillingFlowParams.Builder#setObfuscatedAccountId) and must not contain PII. Configure it as part of your application startup: ```kotlin val options = SuperwallOptions().apply { passIdentifiersToPlayStore = true logging.level = LogLevel.info } Superwall.configure( application = this, apiKey = "pk_your_api_key", options = options ) ``` Related [#related] * [`PaywallOptions`](/docs/android/sdk-reference/PaywallOptions) # PaywallBuilder You're responsible for managing the lifecycle of the returned PaywallView. Do not use the same PaywallView instance in multiple places simultaneously. The remotely configured presentation style is ignored when using this method. You must handle presentation styling programmatically. Purpose [#purpose] Creates a PaywallView that you can present however you want, bypassing Superwall's automatic presentation logic. Signature [#signature] ```kotlin class PaywallBuilder(private val placement: String) { fun params(params: Map?): PaywallBuilder fun overrides(overrides: PaywallOverrides?): PaywallBuilder fun delegate(delegate: PaywallViewCallback): PaywallBuilder fun activity(activity: Activity): PaywallBuilder suspend fun build(): Result fun buildSync(): PaywallView fun build(onSuccess: (PaywallView) -> Unit, onError: (Throwable) -> Unit) } ``` Parameters [#parameters] Returns / State [#returns--state] Returns a `Result` that you can add to your view hierarchy. If presentation should be skipped, returns a failure result. Usage [#usage] Using with coroutines: ```kotlin lifecycleScope.launch { val result = PaywallBuilder("premium_feature") .params(mapOf("source" to "settings")) .delegate(object : PaywallViewCallback { override fun onFinished( paywall: PaywallView, result: PaywallResult, shouldDismiss: Boolean ) { // Handle paywall completion } }) .activity(this@MainActivity) .build() result.fold( onSuccess = { paywallView -> binding.container.addView(paywallView) }, onFailure = { error -> println("Error creating paywall: ${error.message}") } ) } ``` Jetpack Compose integration: ```kotlin @Composable fun PaywallScreen() { PaywallComposable( placement = "premium_feature", params = mapOf("source" to "settings"), delegate = object : PaywallViewCallback { override fun onFinished( paywall: PaywallView, result: PaywallResult, shouldDismiss: Boolean ) { // Handle completion } }, errorComposable = { error -> Text("Failed to load paywall: ${error.message}") }, loadingComposable = { CircularProgressIndicator() } ) } ``` # setSubscriptionStatus() This function should only be used when implementing a custom [`PurchaseController`](/docs/android/sdk-reference/PurchaseController). When using Superwall's built-in purchase handling, the subscription status is managed automatically. You must call this function whenever the user's entitlements change to keep Superwall's subscription status synchronized with your purchase system. Purpose [#purpose] Manually updates the subscription status when using a custom [`PurchaseController`](/docs/android/sdk-reference/PurchaseController) to ensure paywall gating and analytics work correctly. Signature [#signature] ```kotlin fun Superwall.setSubscriptionStatus(status: SubscriptionStatus) ``` ```java // Java public void setSubscriptionStatus(SubscriptionStatus status) ``` Parameters [#parameters] Returns / State [#returns--state] This function returns `Unit`. The new status will be reflected in the [`subscriptionStatus`](/docs/android/sdk-reference/subscriptionStatus) StateFlow and will trigger the [`SuperwallDelegate.subscriptionStatusDidChange`](/docs/android/sdk-reference/SuperwallDelegate) callback. Usage [#usage] Set active subscription with entitlements: ```kotlin // User purchased premium subscription Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(setOf("premium", "pro_features")) ) ``` Set inactive subscription: ```kotlin // User's subscription expired or was cancelled Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) ``` Set unknown status during initialization: ```kotlin // While checking subscription status on app launch Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Unknown) ``` Usage with RevenueCat: ```kotlin class RevenueCatPurchaseController : PurchaseController { override suspend fun purchase( activity: Activity, product: StoreProduct ): PurchaseResult { return try { val result = Purchases.sharedInstance.purchase(activity, product.sku) // Update Superwall subscription status based on RevenueCat result if (result.isSuccessful) { val entitlements = result.customerInfo.entitlements.active.keys Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(entitlements) ) PurchaseResult.Purchased } else { PurchaseResult.Failed(Exception("Purchase failed")) } } catch (e: Exception) { PurchaseResult.Failed(e) } } override suspend fun restorePurchases(): RestorationResult { return try { val customerInfo = Purchases.sharedInstance.restorePurchases() val activeEntitlements = customerInfo.entitlements.active.keys if (activeEntitlements.isNotEmpty()) { Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(activeEntitlements) ) } else { Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) } RestorationResult.Restored } catch (e: Exception) { RestorationResult.Failed(e) } } } ``` Listen for external subscription changes: ```kotlin class SubscriptionManager { fun onSubscriptionStatusChanged(isActive: Boolean, entitlements: Set) { val status = if (isActive) { SubscriptionStatus.Active(entitlements) } else { SubscriptionStatus.Inactive } // Update Superwall whenever subscription status changes externally Superwall.instance.setSubscriptionStatus(status) } } ``` Java usage: ```java // Set active subscription Set entitlements = Set.of("premium", "pro_features"); Superwall.getInstance().setSubscriptionStatus( new SubscriptionStatus.Active(entitlements) ); // Set inactive subscription Superwall.getInstance().setSubscriptionStatus( SubscriptionStatus.Inactive.INSTANCE ); ``` # configure() This is a static method called on the `Superwall` class itself, not on the shared instance. The Android SDK requires an Application context for initialization. Purpose [#purpose] Configures the shared instance of Superwall with your API key and optional configurations, making it ready for use throughout your Android app. ```kotlin Android public fun configure( applicationContext: Application, apiKey: String, purchaseController: PurchaseController? = null, options: SuperwallOptions? = null, activityProvider: ActivityProvider? = null, completion: ((Result) -> Unit)? = null ) ``` Parameters [#parameters] Returns / State [#returns--state] Configures the Superwall instance which is accessible via [`Superwall.instance`](/docs/android/sdk-reference/Superwall). ```kotlin Android class MyApplication : Application() { override fun onCreate() { super.onCreate() Superwall.configure( applicationContext = this, apiKey = "pk_your_api_key" ) } } ``` With custom options: ```kotlin class MyApplication : Application() { override fun onCreate() { super.onCreate() val options = SuperwallOptions().apply { paywalls.shouldShowPurchaseFailureAlert = false } Superwall.configure( applicationContext = this, apiKey = "pk_your_api_key", options = options ) { println("Superwall configured successfully") } } } ``` # getPresentationResult() Purpose [#purpose] Retrieves the presentation result for a placement without presenting the paywall. Call this when you need to know whether a placement would show a paywall, send the user to a holdout, or fail due to missing configuration before you decide how to render UI. Signature [#signature] ```kotlin suspend fun Superwall.getPresentationResult( placement: String, params: Map? = null, ): Result ``` ```kotlin fun Superwall.getPresentationResultSync( placement: String, params: Map? = null, ): Result ``` `getPresentationResultSync` blocks the calling thread. Prefer the suspend function inside a coroutine whenever possible. Parameters [#parameters] Returns / State [#returns--state] Returns a Kotlin `Result`. * On success, the wrapped `PresentationResult` can be: * `PresentationResult.PlacementNotFound` – Placement is missing from any live campaign. * `PresentationResult.NoAudienceMatch` – No audience matched, so nothing would show. * `PresentationResult.Paywall(experiment)` – A paywall would be presented; inspect the `experiment`. * `PresentationResult.Holdout(experiment)` – The user is in a holdout group for that experiment. * `PresentationResult.PaywallNotAvailable` – The SDK could not present (no activity, already showing, offline, etc.). * On failure, the `Result` contains the thrown exception (for example, the SDK is not configured yet). Inspect it with `exceptionOrNull()` or `onFailure`. Usage [#usage] ```kotlin lifecycleScope.launch { val result = Superwall.instance.getPresentationResult( placement = "premium_feature", params = mapOf("source" to "settings") ) result .onSuccess { presentation -> when (presentation) { is PresentationResult.Paywall -> { logExperiment(presentation.experiment) showLockedState() } is PresentationResult.Holdout -> showHoldoutBanner() is PresentationResult.NoAudienceMatch -> unlockFeature() is PresentationResult.PlacementNotFound -> Timber.w("Missing placement configuration") is PresentationResult.PaywallNotAvailable -> showOfflineMessage() } } .onFailure { error -> Timber.e(error, "Unable to fetch presentation result") } } ``` ```kotlin // Blocking usage (for example, inside a worker) val result = Superwall.instance.getPresentationResultSync("premium_feature") val presentation = result.getOrNull() ?: return ``` Related [#related] * [`register()`](/docs/android/sdk-reference/register) – Registers a placement and may present a paywall. * [`SuperwallDelegate`](/docs/android/sdk-reference/SuperwallDelegate) – Receive callbacks when paywalls are presented. # handleDeepLink() Configure deep link campaigns on the Superwall dashboard by adding the `deepLink` event to a campaign trigger. Deep link events are also tracked via [`SuperwallEvent.DeepLink`](/docs/android/sdk-reference/SuperwallEvent) and sent to your [`SuperwallDelegate`](/docs/android/sdk-reference/SuperwallDelegate). Purpose [#purpose] Processes a deep link URL and triggers any associated paywall campaigns configured on the Superwall dashboard. Signature [#signature] ```kotlin fun Superwall.handleDeepLink(url: String) ``` ```java // Java public void handleDeepLink(String url) ``` Parameters [#parameters] Returns / State [#returns--state] This function returns `Unit`. If the URL matches a campaign configured on the dashboard, it may trigger a paywall presentation. Usage [#usage] In your Activity: ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) // Handle deep link on app launch intent?.let { handleIntent(it) } } override fun onNewIntent(intent: Intent?) { super.onNewIntent(intent) intent?.let { handleIntent(it) } } private fun handleIntent(intent: Intent) { val action = intent.action val data = intent.data if (Intent.ACTION_VIEW == action && data != null) { val url = data.toString() // Handle the deep link with Superwall Superwall.instance.handleDeepLink(url) // Continue with your app's deep link handling handleAppDeepLink(url) } } } ``` In your manifest (declare your deep link schemes): ```xml ``` Handle deep links in different scenarios: ```kotlin fun handleDeepLinkFromNotification(url: String) { // Handle deep link from push notification Superwall.instance.handleDeepLink(url) navigateToContent(url) } fun handleDeepLinkFromWebView(url: String) { // Handle deep link from web view or external browser Superwall.instance.handleDeepLink(url) processWebViewDeepLink(url) } ``` Java usage: ```java public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Intent intent = getIntent(); handleIntent(intent); } @Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); handleIntent(intent); } private void handleIntent(Intent intent) { String action = intent.getAction(); Uri data = intent.getData(); if (Intent.ACTION_VIEW.equals(action) && data != null) { String url = data.toString(); // Handle the deep link with Superwall Superwall.getInstance().handleDeepLink(url); // Continue with your app's deep link handling handleAppDeepLink(url); } } } ``` Example deep link campaign setup: ```kotlin // When users click a deep link like "myapp://premium-feature" // The dashboard campaign can be configured to: // 1. Track the deepLink event // 2. Show a paywall if user is not subscribed // 3. Direct to premium content if user is subscribed Superwall.instance.handleDeepLink("myapp://premium-feature?source=email") ``` # identify() Call this as soon as you have a user ID, typically after login or when the user's identity becomes available. Purpose [#purpose] Links a user ID to Superwall's automatically generated alias, creating an account for analytics and personalization. Signature [#signature] ```kotlin fun Superwall.identify( userId: String, options: IdentityOptions? = null ) ``` ```java // Java public void identify( String userId, @Nullable IdentityOptions options ) ``` Parameters [#parameters] Returns / State [#returns--state] This function returns `Unit`. After calling, [`isLoggedIn`](/docs/android/sdk-reference/userId) will return `true` and [`userId`](/docs/android/sdk-reference/userId) will return the provided user ID. Usage [#usage] Basic identification: ```kotlin Superwall.instance.identify("user_12345") ``` With options for account switching scenarios: ```kotlin val options = IdentityOptions().apply { restorePaywallAssignments = true } Superwall.instance.identify( userId = "returning_user_67890", options = options ) ``` Call as soon as you have a user ID: ```kotlin fun userDidLogin(user: User) { Superwall.instance.identify(user.id) // Set additional user attributes Superwall.instance.setUserAttributes(mapOf( "email" to user.email, "plan" to user.subscriptionPlan, "signUpDate" to user.createdAt )) } ``` Java usage: ```java // Basic identification Superwall.getInstance().identify("user_12345"); // With options IdentityOptions options = new IdentityOptions(); options.setRestorePaywallAssignments(true); Superwall.getInstance().identify("returning_user_67890", options); ``` # Overview Welcome to the Superwall Android SDK Reference [#welcome-to-the-superwall-android-sdk-reference] You can find the source code for the SDK [on GitHub](https://github.com/superwall/Superwall-Android) along with our [example app](https://github.com/superwall/Superwall-Android/tree/develop/example). Feedback [#feedback] We are always improving our SDKs and documentation! If you have feedback on any of our docs, please leave a rating and message at the bottom of the page. If you have any issues with the SDK, please [open an issue on GitHub](https://github.com/superwall/superwall-android/issues). # localResources Available in Android SDK `2.7.7+`. Purpose [#purpose] `localResources` lets you map paywall asset IDs to local Android resources or file URIs. Paywalls can then request those assets with `swlocal://resource-id` instead of downloading them from a remote URL. Signature [#signature] ```kotlin var localResources: Map ``` ```java public Map getLocalResources() public void setLocalResources(Map localResources) ``` Schema [#schema] PaywallResource variants [#paywallresource-variants] Returns / State [#returns--state] This is a mutable property on [`Superwall.instance`](/docs/android/sdk-reference/Superwall). Set it before presenting paywalls that depend on local assets. Related [#related] * [Local Resources guide](/docs/android/guides/local-resources) # register() Purpose [#purpose] Registers a placement so that when it's added to a campaign on the Superwall Dashboard, it can trigger a paywall and optionally gate access to a feature. Signature [#signature] ```kotlin fun Superwall.register( placement: String, params: Map? = null, handler: PaywallPresentationHandler? = null, feature: () -> Unit, ) ``` ```kotlin fun Superwall.register( placement: String, params: Map? = null, handler: PaywallPresentationHandler? = null, ) ``` Parameters [#parameters] Returns / State [#returns--state] This function returns `Unit`. If you supply a `feature` lambda, it will be executed according to the paywall's gating configuration, as described above. Usage [#usage] ```swift iOS func pressedWorkoutButton() { // remotely decide if a paywall is shown and if // navigation.startWorkout() is a paid-only feature Superwall.shared.register(placement: "StartWorkout") { navigation.startWorkout() } } ``` ```kotlin Android fun pressedWorkoutButton() { // remotely decide if a paywall is shown and if // navigation.startWorkout() is a paid-only feature Superwall.instance.register("StartWorkout") { navigation.startWorkout() } } ``` ```dart Flutter void pressedWorkoutButton() { // remotely decide if a paywall is shown and if // navigation.startWorkout() is a paid-only feature Superwall.shared.registerPlacement('StartWorkout', feature: () { navigation.startWorkout(); }); } ``` ```typescript React Native // remotely decide if a paywall is shown and if // navigation.startWorkout() is a paid-only feature Superwall.shared.register({ placement: 'StartWorkout', feature: () => { navigation.navigate('LaunchedFeature', { value: 'Non-gated feature launched', }); } }); ``` Register without feature gating: ```kotlin Superwall.instance.register( placement = "onboarding_complete", params = mapOf("source" to "onboarding"), handler = this ) ``` # setUserAttributes() These attributes should not be used as a source of truth for sensitive information. Keys beginning with `$` are reserved for Superwall internal use and will be ignored. Arrays and nested maps are not supported as values. Purpose [#purpose] Sets custom user attributes that can be used in paywall personalization, audience filters, and analytics on the Superwall dashboard. Signature [#signature] ```kotlin fun Superwall.setUserAttributes(attributes: Map) ``` ```java // Java public void setUserAttributes(Map attributes) ``` Parameters [#parameters] Returns / State [#returns--state] This function returns `Unit`. If an attribute already exists, its value will be overwritten while other attributes remain unchanged. Usage [#usage] Set multiple user attributes: ```kotlin val attributes = mapOf( "name" to "John Doe", "email" to "john@example.com", "plan" to "premium", "signUpDate" to System.currentTimeMillis(), "profilePicUrl" to "https://example.com/pic.jpg", "isVip" to true, "loginCount" to 42 ) Superwall.instance.setUserAttributes(attributes) ``` Set individual attributes over time: ```kotlin Superwall.instance.setUserAttributes(mapOf("lastActiveDate" to System.currentTimeMillis())) Superwall.instance.setUserAttributes(mapOf("featureUsageCount" to 15)) ``` Remove an attribute by setting it to null: ```kotlin Superwall.instance.setUserAttributes(mapOf("temporaryFlag" to null)) ``` Real-world example after user updates profile: ```kotlin fun updateUserProfile(user: User) { Superwall.instance.setUserAttributes(mapOf( "name" to user.displayName, "avatar" to user.avatarURL, "preferences" to user.notificationPreferences, "lastUpdated" to System.currentTimeMillis() )) } ``` Java usage: ```java // Set multiple attributes Map attributes = new HashMap<>(); attributes.put("name", "John Doe"); attributes.put("email", "john@example.com"); attributes.put("plan", "premium"); attributes.put("loginCount", 42); Superwall.getInstance().setUserAttributes(attributes); ``` # subscriptionStatus If you're using a custom [`PurchaseController`](/docs/android/sdk-reference/PurchaseController), you must update this property whenever the user's entitlements change. You can also observe changes via the [`SuperwallDelegate`](/docs/android/sdk-reference/SuperwallDelegate) method `subscriptionStatusDidChange(from, to)`. Purpose [#purpose] Indicates the current subscription status of the user and can be observed for changes using Kotlin StateFlow. Signature [#signature] ```kotlin val subscriptionStatus: StateFlow // For setting the status (when using custom PurchaseController) fun setSubscriptionStatus(status: SubscriptionStatus) ``` ```java // Java public StateFlow getSubscriptionStatus() public void setSubscriptionStatus(SubscriptionStatus status) ``` Parameters [#parameters] This property accepts a `SubscriptionStatus` sealed class value: * `SubscriptionStatus.Unknown` - Status is not yet determined * `SubscriptionStatus.Active(Set)` - User has active entitlements (set of entitlement identifiers) * `SubscriptionStatus.Inactive` - User has no active entitlements Returns / State [#returns--state] Returns a `StateFlow` that emits the current subscription status. When using a [`PurchaseController`](/docs/android/sdk-reference/PurchaseController), you must set this property yourself using `setSubscriptionStatus()`. Otherwise, Superwall manages it automatically. Usage [#usage] Set subscription status (when using PurchaseController): ```kotlin Superwall.instance.setSubscriptionStatus( SubscriptionStatus.Active(setOf("premium", "pro_features")) ) Superwall.instance.setSubscriptionStatus(SubscriptionStatus.Inactive) ``` Get current subscription status: ```kotlin val status = Superwall.instance.subscriptionStatus.value when (status) { is SubscriptionStatus.Unknown -> println("Subscription status unknown") is SubscriptionStatus.Active -> println("User has active entitlements: ${status.entitlements}") is SubscriptionStatus.Inactive -> println("User has no active subscription") } ``` Observe changes with StateFlow: ```kotlin class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) lifecycleScope.launch { Superwall.instance.subscriptionStatus.collect { status -> updateUI(status) } } } private fun updateUI(status: SubscriptionStatus) { when (status) { is SubscriptionStatus.Active -> showPremiumContent() is SubscriptionStatus.Inactive -> showFreeContent() is SubscriptionStatus.Unknown -> showLoadingState() } } } ``` Jetpack Compose observation: ```kotlin @Composable fun ContentScreen() { val subscriptionStatus by Superwall.instance.subscriptionStatus .collectAsState() Column { when (subscriptionStatus) { is SubscriptionStatus.Active -> { Text("Premium user with: ${subscriptionStatus.entitlements.joinToString()}") } is SubscriptionStatus.Inactive -> { Text("Free user") } is SubscriptionStatus.Unknown -> { Text("Loading...") } } } } ``` Java usage: ```java // Get current status SubscriptionStatus status = Superwall.getInstance() .getSubscriptionStatus().getValue(); // Observe changes Superwall.getInstance().getSubscriptionStatus() .observe(this, status -> { updateUI(status); }); // Set status (when using PurchaseController) Superwall.getInstance().setSubscriptionStatus( new SubscriptionStatus.Active(Set.of("premium")) ); ``` # userId The anonymous user ID is automatically generated and persisted to disk, so it remains consistent across app launches until the user is identified. Purpose [#purpose] Returns the current user's unique identifier, either from a previous call to [`identify()`](/docs/android/sdk-reference/identify) or an anonymous ID if not identified. Signature [#signature] ```kotlin // Accessed via Superwall.instance val userId: String ``` ```java // Java public String getUserId() ``` Parameters [#parameters] This is a read-only property on the [`Superwall.instance`](/docs/android/sdk-reference/Superwall) with no parameters. Returns / State [#returns--state] Returns a `String` representing the user's ID. If [`identify()`](/docs/android/sdk-reference/identify) has been called, returns that user ID. Otherwise, returns an automatically generated anonymous user ID that is cached to disk. Usage [#usage] Get the current user ID: ```kotlin val currentUserId = Superwall.instance.userId println("User ID: $currentUserId") ``` Check if user is identified: ```kotlin if (Superwall.instance.isLoggedIn) { println("User is identified with ID: ${Superwall.instance.userId}") } else { println("User is anonymous with ID: ${Superwall.instance.userId}") } ``` Example usage in analytics: ```kotlin fun trackAnalyticsEvent() { val userId = Superwall.instance.userId Analytics.track("feature_used", mapOf( "user_id" to userId, "timestamp" to System.currentTimeMillis() )) } ``` Example usage in custom logging: ```kotlin fun logError(error: Throwable) { Logger.log("Error for user ${Superwall.instance.userId}: ${error.message}") } ``` Java usage: ```java // Get current user ID String currentUserId = Superwall.getInstance().getUserId(); System.out.println("User ID: " + currentUserId); // Check if user is identified if (Superwall.getInstance().isLoggedIn()) { System.out.println("User is identified with ID: " + Superwall.getInstance().getUserId()); } else { System.out.println("User is anonymous with ID: " + Superwall.getInstance().getUserId()); } ``` # Changelog # Capacitor **Community SDK** This SDK is developed and maintained by [Capawesome](https://capawesome.io), not by Superwall. For support, please visit the [official documentation](https://capawesome.io/plugins/superwall/). The [Capacitor Superwall plugin](https://capawesome.io/plugins/superwall/) by [Capawesome](https://capawesome.io) enables you to integrate Superwall into your Capacitor apps for iOS and Android. Installation [#installation] Install the plugin via npm: ```bash npm install @capawesome/capacitor-superwall npx cap sync ``` Android Setup [#android-setup] Add the following activities to your `AndroidManifest.xml` inside the `` tag: ```xml ``` Configuration [#configuration] Initialize the SDK with your API key: ```typescript import { Superwall } from '@capawesome/capacitor-superwall'; await Superwall.configure({ apiKey: 'YOUR_API_KEY', options: { paywalls: { shouldPreload: true, automaticallyDismiss: true }, logging: { level: 'WARN', scopes: ['ALL'] } } }); ``` Core Methods [#core-methods] | Method | Description | | ------------------------- | ----------------------------------------- | | `configure()` | Initialize the SDK with your API key | | `register()` | Present a paywall for a placement | | `getPresentationResult()` | Check if a paywall would display | | `identify()` | Associate a user ID with the current user | | `reset()` | Clear user identity on logout | | `setUserAttributes()` | Set custom user attributes | | `getSubscriptionStatus()` | Get the current subscription status | Event Listeners [#event-listeners] The plugin supports the following event listeners: * `superwallEvent` - Analytics events * `subscriptionStatusDidChange` - Subscription status changes * `paywallPresented` - Paywall displayed * `paywallWillDismiss` - Before paywall closes * `paywallDismissed` - After paywall closes * `customPaywallAction` - Custom paywall element interactions Full Documentation [#full-documentation] View the complete API reference and detailed documentation on capawesome.io # Community SDKs **Community SDKs** These SDKs are developed and maintained by the community, not by Superwall. For support, please contact the SDK maintainer or visit their official documentation. Available SDKs [#available-sdks] Superwall plugin for Capacitor apps by Capawesome. Supports iOS and Android. # Creating Paywalls The legacy editor is deprecated. Please visit the docs covering our new [editor](/docs/dashboard/dashboard-creating-paywalls/paywall-editor-overview). On the Superwall dashboard under **Paywalls**, click **+ New Paywall** and select **From Template**: This will redirect you to the templates page where you can browse different types of paywall templates. Clicking on one will allow you to preview it: Click **Try it**, give it a name, and click **Create**. This will bring you to the paywall editor: The Paywall Editor consists of 3 sections: 1. [The settings sidebar](/docs/paywall-settings-sidebar) - General paywall settings, designs and products to show on the paywall. 2. [The device preview](/docs/interactive-paywall-preview) - An interactive preview of your paywall in a device of your choosing. 3. [The variable editor](/docs/interactive-paywall-preview#adding-variables) - Fully customise items, such as text or images, on your paywall. # Assets Assets is a media library for all of your aywalls. It lets you upload images and videos once, then reuse them. This helps you avoid uploading the same file multiple times or copying URLs out of older paywalls when you want to reuse media. To open it, click **Assets** in the **sidebar**: How assets get into the library [#how-assets-get-into-the-library] * **Upload from Assets:** Open **Assets** from the sidebar, then click **Add Assets** or drag files onto the page. * **Upload from the editor:** Upload an image or video from a paywall field and Superwall adds it to **Assets** automatically. * **Discover existing media:** Use **Discover Assets** to scan existing paywalls and add images and videos that are already in use. Upload assets from the Assets page [#upload-assets-from-the-assets-page] * **Open Assets:** Click **Assets** in the sidebar for the currently selected app. * **Add files:** Click **Add Assets**, or drag and drop image and video files into the page. * **Wait for the upload to finish:** Uploaded files appear in the library when the upload completes. Discover assets from existing paywalls [#discover-assets-from-existing-paywalls] Use this if you already have paywalls with media and want to build the library from what is already in use. * **Open the menu:** Click the menu in the top-right corner of the **Assets** page. * **Run discovery:** Click **Discover Assets**. * **Review the results:** Superwall scans existing unarchived paywalls and adds any images and videos it finds to the library. Discovering assets does not change any existing paywalls. It only creates asset entries so those files can be reused later. Use an asset in a paywall [#use-an-asset-in-a-paywall] * **Select a component:** In the paywall editor, select an image or video component. * **Open the asset picker:** Click the bookmark button next to the source field to open **Pick from Assets**. * **Choose an asset:** Search or browse the library, then select the asset you want to use. * **Save the paywall:** The selected asset is applied to that component. For videos, the asset picker sets the main video source. Configure the thumbnail separately in the video settings if needed. Find and review assets [#find-and-review-assets] * **Filter by type:** Switch between **All**, **Images**, and **Videos**. * **Search by name:** Use the search field to find a specific asset. * **Preview an asset:** Click an asset card to open a larger preview and view its details. * **Download an asset:** Use the asset menu or preview dialog to download the file. Delete assets [#delete-assets] * **Remove it from the library:** Use the asset menu and click **Delete**. * **Keep existing paywalls working:** Deleting an asset does not remove it from paywalls that already use it. It only removes it from the reusable library. Things to know [#things-to-know] * **Supported media:** Assets supports images and videos. * **Selected app context:** The library you see is based on the app you have selected. * **Project sharing:** If the selected app belongs to a project, the same asset library is shared across the apps in that project. * **No automatic replacements:** Uploading, discovering, or deleting an asset does not update existing paywalls automatically. # Active Subscriptions Active Subscriptions chart What it shows [#what-it-shows] Active Subscriptions shows the count of unexpired, paid subscriptions over time. How to use it [#how-to-use-it] Use this chart to track the size of your paying subscriber base. It is one of the clearest ways to see whether new subscriptions and renewals are outpacing expirations. Good to know [#good-to-know] Active Subscriptions does not tell the full story by itself. Pair it with MRR, Subscriber Churn, and Auto Renew Status to understand revenue and retention quality. # Active Users Active Users chart What it shows [#what-it-shows] Active Users shows the count of users who were active during each period. How to use it [#how-to-use-it] Use this chart to understand audience size and engagement. It helps explain changes in paywall exposure, conversion volume, and revenue that may be driven by app usage. Good to know [#good-to-know] Active Users is a useful denominator when reviewing monetization. If revenue is flat while active users grow, look at paywall exposure and conversion rates next. # ARR ARR chart What it shows [#what-it-shows] ARR shows annualized recurring revenue from subscriptions. It normalizes subscription revenue into an annual view. How to use it [#how-to-use-it] Use ARR to understand your recurring revenue run rate and how it changes over time. It is helpful when you want a longer-range view than monthly movement can provide. Good to know [#good-to-know] ARR is an annualized metric, so use it for directional recurring revenue trends instead of short-term cash timing. # Auto Renew Status Auto Renew Status chart What it shows [#what-it-shows] Auto Renew Status shows how much MRR is currently set to renew versus churn based on subscription renewal status. How to use it [#how-to-use-it] Use this chart to understand renewal risk before subscriptions expire. It is useful for spotting periods where a larger share of recurring revenue is at risk. Good to know [#good-to-know] Use Auto Renew Status alongside Subscriber Churn. One shows renewal intent before expiration, while the other shows subscriptions that actually expired. # Checkout Conversion Checkout Conversion chart What it shows [#what-it-shows] Checkout Conversion shows the percentage of users who converted after starting checkout. How to use it [#how-to-use-it] Use this chart to spot purchase flow friction. If users start checkout but do not complete it, the issue may be closer to product selection, store purchase flow, payment state, or purchase restoration. Good to know [#good-to-know] Compare Checkout Conversion with Paywall Conversion. A drop in checkout conversion points later in the funnel than a drop in paywall conversion. # Cohorted Proceeds Cohorted Proceeds chart What it shows [#what-it-shows] Cohorted Proceeds shows net proceeds grouped by install date. Each cohort represents users who installed during the same period. How to use it [#how-to-use-it] Use this chart to understand how install cohorts monetize over time. It is a good fit for comparing acquisition quality, measuring payback windows, and reviewing the impact of product or campaign changes by cohort. Good to know [#good-to-know] Pair this chart with acquisition spend when you want to understand whether new cohorts are earning back what you spent to acquire them. # Conversions Conversions chart What it shows [#what-it-shows] Conversions shows the count of completed transactions. How to use it [#how-to-use-it] Use this chart to track purchase volume over time. It is useful for reviewing campaign launches, product changes, offer changes, and seasonal traffic patterns. Good to know [#good-to-know] Conversion count is volume, not rate. Pair it with Paywall Conversion and New Users when you need to understand whether performance changed because of traffic or efficiency. # Charts To view charts breaking down your app's performance, click the **Charts** button in the **sidebar**: Check out a video overview of our charts on [YouTube](https://youtu.be/7UIO99LSvTQ). Chart types [#chart-types] Choose between different charts by making a selection from the left sidebar: You can also toggle which charts are showing by using the chevrons on the left-hand side. Currently, we offer the following charts: Revenue Charts [#revenue-charts] * [**Proceeds**](/docs/dashboard/charts/proceeds): Revenue after refunds, store fees, and taxes. * [**Sales**](/docs/dashboard/charts/sales): Revenue before refunds, taxes, and fees. * [**Cohorted Proceeds**](/docs/dashboard/charts/cohorted-proceeds): Net proceeds cohorted by install date, after refunds, taxes, and fees. * [**ARR**](/docs/dashboard/charts/arr): Annualized recurring revenue from subscriptions. * [**MRR**](/docs/dashboard/charts/mrr): Monthly recurring revenue from subscriptions. * [**Realized LTV per new user**](/docs/dashboard/charts/realized-ltv-per-new-user): Net proceeds per new user, cohorted by install date. * [**Realized LTV per paid user**](/docs/dashboard/charts/realized-ltv-per-paid-user): Net proceeds per paying user, cohorted by install date. Subscription Charts [#subscription-charts] * [**Active Subscriptions**](/docs/dashboard/charts/active-subscriptions): Count of unexpired, paid subscriptions. * [**Paid Conversion**](/docs/dashboard/charts/paid-conversion): Percent of installs who became paying users. * [**New Trials**](/docs/dashboard/charts/new-trials): Trial starts, cohorted by trial start date. * [**Trial Conversion**](/docs/dashboard/charts/trial-conversion): Percentage of trials that converted to paid subscriptions. * [**New Subscriptions**](/docs/dashboard/charts/new-subscriptions): New subscriptions, cohorted by subscription start date. * [**Auto Renew Status**](/docs/dashboard/charts/auto-renew-status): How much of your MRR is set to renew versus churn. Paywall Charts [#paywall-charts] * [**Initial Conversion**](/docs/dashboard/charts/initial-conversion): Percent of new users who converted on a paywall. * [**Paywalled Users**](/docs/dashboard/charts/paywalled-users): Count of unique users who opened paywalls. * [**Paywall Rate**](/docs/dashboard/charts/paywall-rate): Percent of new users who opened paywalls. * [**Paywall Conversion**](/docs/dashboard/charts/paywall-conversion): Percent of users who converted after opening a paywall. * [**Conversions**](/docs/dashboard/charts/conversions): Count of completed transactions. * [**Checkout Conversion**](/docs/dashboard/charts/checkout-conversion): Percentage of users who converted after starting checkout. User Charts [#user-charts] * [**New Users**](/docs/dashboard/charts/new-users): Count of new users. * [**Active Users**](/docs/dashboard/charts/active-users): Count of active users. Retention & Churn Charts [#retention--churn-charts] * [**Subscriber Churn**](/docs/dashboard/charts/subscriber-churn): Percentage of paid subscriptions that expired in each period. * [**Refund Rate**](/docs/dashboard/charts/refund-rate): Ratio of refunds to gross proceeds, cohorted by first purchase date. * [**Subscription Retention**](/docs/dashboard/charts/subscription-retention): Subscription retention by cohort over time. Filtering chart data [#filtering-chart-data] To filter data on a chart, **click** the **Filter** button at the top right: Filter data by choosing a filter type and **clicking** on **+ Add Filter** to apply it. You can add one, or several, filters: When you're done, **click** on the **Apply** button, and the chart will refresh with the data filtered by your selections: To remove an individual filter, **click** on the **trash can** icon on the trailing side of it: To remove an individual component that's part of a filter, such as breaking down by **Application** and removing one included app, **click** on the **X** button on its trailing side: To remove all filters, **click** on the **Clear Filters** button: Breaking down chart data [#breaking-down-chart-data] To break down data in a chart, **click** the **Breakdown** toggle at the top right: The breakdowns available are tailored to the type of chart you have selected. After you apply a selection, the chart and the table below it update automatically: Breaking down a **Proceeds** chart by **Placements** is a powerful way to directly correlate features you make, or similar things you've paywalled, to your app's revenue growth. Selecting time ranges [#selecting-time-ranges] To customize the time span and level of detail of the data displayed on the chart, use the two date toggles at the top right: These controls adjust the chart's view interval and data range: **Display Interval:** Sets the interval at which data is displayed on the chart. Choose options like hourly, daily, weekly, or monthly to adjust how granular the chart appears. Selecting **Auto** automatically optimizes the interval based on the selected date range. **Data Fetch Range:** Defines the total date range used to populate the chart. Options include **Yesterday**, **Last 7 Days**, **Last 30 Days**, and more. You can also use natural language to set a data fetch range. For example, "last month", "two weeks ago", or similar ranges. Changing chart formats [#changing-chart-formats] Each chart type can display its data in different chart formats. To change the default display, **click** on the **Chart** button found at the top right: You can toggle the chart format between **Stacked Area**, **Line**, **Stacked Bar**, or **Bar**. Here is the same chart data in each format: Additionally, you can hover any chart element to see more details about the data point: Exporting chart data [#exporting-chart-data] You can export any chart data as a `.csv` file. Just **click** the **Export** button at the bottom-right of any chart: # Initial Conversion Initial Conversion chart What it shows [#what-it-shows] Initial Conversion shows the percentage of new users who converted on a paywall. How to use it [#how-to-use-it] Use this chart to evaluate the first monetization path for new users. It is useful when you are tuning onboarding, first-session paywalls, placements, or introductory offers. Good to know [#good-to-know] If Initial Conversion changes, compare it with New Users and Paywall Rate to make sure the shift is not just a change in acquisition volume or paywall exposure. # MRR MRR chart What it shows [#what-it-shows] MRR shows monthly recurring revenue from subscriptions. It normalizes subscription revenue into a monthly view. How to use it [#how-to-use-it] Use MRR as the baseline for subscription revenue health. It makes it easier to see whether recurring revenue is expanding, flattening, or shrinking over time. Good to know [#good-to-know] Review MRR alongside Active Subscriptions and Auto Renew Status to separate subscriber count changes from renewal risk. # New Subscriptions New Subscriptions chart What it shows [#what-it-shows] New Subscriptions shows new subscription starts, grouped by subscription start date. How to use it [#how-to-use-it] Use this chart to understand subscription acquisition. It is helpful for reviewing launches, pricing tests, offer changes, and campaign experiments. Good to know [#good-to-know] Break this chart down by subscription start type when you want to separate direct paid starts from trial conversions. # New Trials New Trials chart What it shows [#what-it-shows] New Trials shows trial starts, grouped by trial start date. How to use it [#how-to-use-it] Use this chart to understand how often users are entering trials. It is especially helpful after changing offers, pricing, trial eligibility, or paywall presentation. Good to know [#good-to-know] Trial volume is only one side of the story. Review Trial Conversion to see whether those trials are turning into paid subscriptions. # New Users New Users chart What it shows [#what-it-shows] New Users shows the count of users who were first seen during each period. How to use it [#how-to-use-it] Use this chart to understand acquisition volume. It gives context for revenue, conversion, paywall exposure, and subscription metrics that may move because traffic changed. Good to know [#good-to-know] When conversion or revenue rates change, check New Users to see whether the app's audience mix or volume changed at the same time. # Paid Conversion What it shows [#what-it-shows] Paid Conversion shows the percentage of installs that became paying users. How to use it [#how-to-use-it] Use this chart to connect acquisition to paid outcomes. It is useful when you want to know whether new users are becoming customers at a healthy rate. Good to know [#good-to-know] Paid Conversion sits downstream of several steps. If it drops, compare it against Paywall Rate, Paywall Conversion, Trial Conversion, and Checkout Conversion to find where the funnel changed. # Paywall Conversion Paywall Conversion chart What it shows [#what-it-shows] Paywall Conversion shows the percentage of users who converted after opening a paywall. How to use it [#how-to-use-it] Use this chart to judge how well paywalls convert the users who actually see them. It is helpful for comparing paywall designs, products, pricing, and experiments. Good to know [#good-to-know] Review Paywall Conversion together with Paywall Rate. One tells you how often users see paywalls, and the other tells you how often exposed users convert. # Paywall Rate Paywall Rate chart What it shows [#what-it-shows] Paywall Rate shows the percentage of new users who opened paywalls. How to use it [#how-to-use-it] Use this chart to understand how much of your new user base is reaching a monetization moment. It is useful when tuning placements, campaign audiences, onboarding flows, or feature gating. Good to know [#good-to-know] A low Paywall Rate can hide a strong paywall. If too few users see it, pair this chart with Paywall Conversion before changing the paywall itself. # Paywalled Users Paywalled Users chart What it shows [#what-it-shows] Paywalled Users shows the count of unique users who opened paywalls. How to use it [#how-to-use-it] Use this chart to understand paywall exposure. It helps answer whether enough users are reaching monetization moments before you judge paywall conversion performance. Good to know [#good-to-know] Break this chart down by placement or paywall when you want to see which surfaces are driving the most paywall views. # Proceeds Proceeds chart What it shows [#what-it-shows] Proceeds shows the revenue your app keeps after refunds, store fees, and taxes. How to use it [#how-to-use-it] Use this as your grounded revenue view. It is the chart to reach for when you want to understand how much revenue is actually making it through after deductions, instead of only looking at top-line sales. Good to know [#good-to-know] Break proceeds down by renewal type, placement, product, or country when you need to understand which parts of the business are driving net revenue. # Realized LTV per new user What it shows [#what-it-shows] Realized LTV per new user shows net proceeds per new user, cohorted by install date. How to use it [#how-to-use-it] Use this chart to understand how much revenue new users have actually generated so far. It is useful for evaluating acquisition quality, pricing changes, and onboarding or paywall changes across install cohorts. Good to know [#good-to-know] Newer cohorts have had less time to generate revenue, so compare them carefully against older cohorts. # Realized LTV per paid user What it shows [#what-it-shows] Realized LTV per paid user shows net proceeds per paying user, cohorted by install date. How to use it [#how-to-use-it] Use this chart to understand how valuable paying users are after they convert. It helps separate paid customer quality from broader acquisition volume. Good to know [#good-to-know] Compare this chart with Realized LTV per new user to see whether changes are coming from conversion rate, paid user value, or both. # Refund Rate Refund Rate chart What it shows [#what-it-shows] Refund Rate shows refunds as a share of gross proceeds, cohorted by first purchase date. How to use it [#how-to-use-it] Use this chart to detect refund spikes and understand whether specific purchase cohorts are refunding at a higher rate. It is helpful after offer, pricing, product, or audience changes. Good to know [#good-to-know] If Refund Rate rises while Sales stays healthy, review the matching Proceeds chart to understand the net revenue impact. # Sales Sales chart What it shows [#what-it-shows] Sales shows gross revenue before refunds, taxes, and store fees are deducted. How to use it [#how-to-use-it] Use Sales to understand top-line purchase volume and demand. It is especially useful when you want to compare how much users purchased against what eventually became proceeds. Good to know [#good-to-know] If Sales is growing but Proceeds is not, look at refunds, store fees, taxes, and product mix to understand where the gap is coming from. # Subscriber Churn Subscriber Churn chart What it shows [#what-it-shows] Subscriber Churn shows the percentage of paid subscriptions that expired during each period. How to use it [#how-to-use-it] Use this chart to monitor retention risk and understand when paid subscriptions are dropping out. It is useful after pricing changes, product changes, offer changes, or onboarding updates. Good to know [#good-to-know] Compare Subscriber Churn with Auto Renew Status to see whether renewal risk is turning into actual expired subscriptions. # Subscription Retention Subscription Retention chart What it shows [#what-it-shows] Subscription Retention shows how subscription cohorts retain over time. How to use it [#how-to-use-it] Use this chart to understand long-term subscriber retention by cohort. It helps you see whether newer cohorts are holding onto subscriptions better or worse than earlier cohorts. Good to know [#good-to-know] Newer cohorts have less elapsed time, so their later retention periods may not be complete yet. Use wider date ranges when you need a more mature retention view. # Trial Conversion Trial Conversion chart What it shows [#what-it-shows] Trial Conversion shows the percentage of trials that converted to paid subscriptions. How to use it [#how-to-use-it] Use this chart to understand trial quality and renewal behavior. It helps you see whether trial starts are leading to paid outcomes after the trial period ends. Good to know [#good-to-know] When trial conversion moves, check the trial length, offer type, onboarding experience, cancellation behavior, and product usage during the trial window. # Creating Projects Projects can contain one or more applications. For example, a project for one "app" could have an iOS, Android and web checkout app in the same project. To create a new project, follow these steps: Open the menu by selecting your existing project from the top-level side of the sidebar select application
*You may need to scroll down if you have many apps* New Project From there, name your app and choose the platform you're building for. You can always add more platforms later. Start typing Superwall can prefill existing iOS apps live on the App Store. If you haven't launched yet, simply choose "Not released yet" and you'll be good to go. Once you're all done, you should be able to see your new project and its app and switch between them using the project switcher on the top left that we used to get started! 🎉
# Rules This page is outdated. Please visit this [one](/docs/dashboard/dashboard-campaigns/campaigns-audience) for the most relevant information.