Skip to main content
To get a better understanding of how paywalls are served, visit here.

Present a Paywall

Install the SDK

The minimum version of Flutter supported by this SDK is 3.24.0.
To add the Helium package, run:
flutter pub add helium_flutter
Or if you prefer, you can add the helium_flutter package to your pubspec.yaml:
dependencies:
  helium_flutter: ^3.1.3
then run:
flutter pub get
Recommended - Make sure that Swift Package Manager support is enabled:
flutter upgrade
flutter config --enable-swift-package-manager
See this Flutter documentation for more details about Swift Package Manager.
You can still use Cocoapods for your dependencies if preferred. If you need to disable Swift Package Manager dependencies after having enabled it, refer to that same Flutter documentation.

iOS Settings

Helium requires iOS 15+. If your app already has a minimum of iOS 15 or higher, you’re all set. This can be specified in your ios/Podfile with:
platform :ios, '15.0'
If you still see errors related to minimum iOS version, consider updating to 15.0 or higher directly in the Xcode project.

Initialize Helium

Find your API key here. If you do not have a Helium account set up yet, you can still integrate but will not be able to show a real paywall.
In your app’s initialization code (typically in main.dart or your root widget):
import 'package:helium_flutter/helium_flutter.dart';

Future<void> main() async {
  WidgetsFlutterBinding.ensureInitialized();

  final heliumFlutter = HeliumFlutter();
  await heliumFlutter.initialize(
    apiKey: "<your-helium-api-key>",
  );

  runApp(const MyApp());
}
initialize
method

Show Your First Paywall 🎉

Set up a trigger and workflow in the dashboard to show your desired paywall.
Call presentUpsell wherever you want to show a full-screen paywall:
ElevatedButton(
  onPressed: () {
    final heliumFlutter = HeliumFlutter();
    heliumFlutter.presentUpsell(context: context, trigger: 'insert-trigger-here');
  },
  child: Text('Show Premium Features'),
),
Do not call presentUpsell in Widget build() as this can have unpredictable behavior.
You should now be able to see Helium paywalls in your app! Well done! 🎉
Looking for alternative presentation methods? Check out the guide on Ways to Show a Paywall.

Recommended Setup

Here are some common additional steps that you may want to consider.

Identifying Users

Identifying users is optional but can help with targeting and when forwarding events to external analytics platforms. If you are not sure, you probably do not need to identify your users.
Identify users as early as you can to maximize consistency in metrics and targeting. Ideally in your initialize call!
await heliumFlutter.initialize(
  apiKey: "<your-helium-api-key>",
  customUserId: "your-custom-user-id",
  customUserTraits: {
    "exampleTrait": "value",
    "userType": "premium",
  },
);

Helium Events

Helium dispatches various events during paywall presentation and purchase flow. You can optionally handle these events in your mobile app. You can also configure Helium to forward them to your existing analytics provider.

PaywallEventHandlers

When displaying a paywall you can pass in event handlers to listen for select events:
HeliumFlutter().presentUpsell(
  trigger: 'my_paywall',
  context: context,
  eventHandlers: PaywallEventHandlers(
    onOpen: (event) {
      log('${event.type} - trigger: ${event.triggerName}');
    },
    onClose: (event) {
      log('${event.type} - trigger: ${event.triggerName}');
    },
    onDismissed: (event) {
      log('${event.type} - trigger: ${event.triggerName}');
    },
    onPurchaseSucceeded: (event) {
      log('${event.type} - trigger: ${event.triggerName}');
    },
    onAnyEvent: (event) {
      // A handler for all paywall-related events.
      // Note that if you have other handlers (i.e. onOpen) set up,
      // both that handler AND this one will fire during paywall open.
    },
  ),
);

Global Event Listener

You can also listen for all Helium events globally by implementing HeliumCallbacks and passing it to initialize:
abstract class HeliumCallbacks {
  Future<void> onPaywallEvent(HeliumPaywallEvent heliumPaywallEvent);
}
For example:
class LogCallbacks implements HeliumCallbacks {
  @override
  Future<void> onPaywallEvent(HeliumPaywallEvent heliumPaywallEvent) async {
    log('onPaywallEvent: ${heliumPaywallEvent.type} - trigger: ${heliumPaywallEvent.triggerName}');
  }
}
Then pass it to initialize:
await heliumFlutter.initialize(
  apiKey: "<your-helium-api-key>",
  callbacks: LogCallbacks(),
);

Fallback Paywalls

It is highly recommended that you set up “fallbacks” to handle the rare case when a paywall fails to display. Please follow this guide to do so.
Do this after you have a paywall created that you want to use in production.

Checking Subscription Status & Entitlements

Check entitlements before showing paywalls to avoid showing a paywall to a user who should not see it.
heliumFlutter.presentUpsell(
  trigger: 'my_paywall',
  context: context,
  dontShowIfAlreadyEntitled: true
);
/// Checks if the user has any active subscription (including non-renewable)
Future<bool> hasAnyActiveSubscription();

/// Checks if the user has any entitlement
Future<bool> hasAnyEntitlement();

/// Checks if the user has an active entitlement for any product attached to
/// the paywall that will show for the provided trigger.
/// Returns `null` if not known (i.e. the paywall is not downloaded yet).
Future<bool?> hasEntitlementForPaywall(String trigger);

Advanced

RevenueCat

By default, Helium will handle purchases for you! This section is typically for users who already use RevenueCat in their app.
Helium integrates seamlessly with RevenueCat so you can continue to let RevenueCat handle your purchases and entitlements.

Install HeliumRevenueCat

Add the helium_revenuecat package alongside helium_flutter:
flutter pub add helium_revenuecat

Configure Helium to use RevenueCat for Purchases

Use the built-in RevenueCatPurchaseDelegate:
import 'package:helium_revenuecat/helium_revenuecat.dart';

final heliumFlutter = HeliumFlutter();
await heliumFlutter.initialize(
  apiKey: "<your-helium-api-key>",
  purchaseDelegate: RevenueCatPurchaseDelegate(),
);
Make sure to configure RevenueCat before initializing Helium.

RevenueCat appUserID

If you ever change the appUserID of a user, keep Helium in sync with:
HeliumFlutter().setRevenueCatAppUserId(await Purchases.appUserID);

Custom Purchase Handling

By default, Helium will handle purchases for you! This section is for those who want to implement custom purchase logic.
Implement HeliumPurchaseDelegate and pass it to initialize:
abstract class HeliumPurchaseDelegate {
  Future<HeliumPurchaseResult> makePurchase(String productId);

  Future<bool> restorePurchases();
}

Additional Features

For the full public API and detailed parameter documentation, see the inline docstrings in the SDK source. Import Helium in your project and use your IDE’s autocomplete or jump-to-definition to explore all available methods and types.
In most cases there is no need to check download status. Helium will display a loading indication if a paywall is presented before download has completed.
You can listen to the status of the paywalls download via a stream:
HeliumFlutter.downloadStatus.listen((status) {
  print('Download status: ${status.name}');
});
The download status will be one of the following:
  • notDownloadedYet: The download has not been initiated.
  • inProgress: The download is in progress.
  • downloadSuccess: The download was successful.
  • downloadFailure: The download failed.
You can also check if paywalls have been downloaded with await heliumFlutter.paywallsLoaded()
Reset Helium entirely so you can call initialize again, for example after changing user traits that can affect the paywalls a user might see via targeting.
await HeliumFlutter().resetHelium();