It uses an internal queue to make calls fast and non-blocking. It also batches requests and flushes asynchronously, making it perfect to use in any part of your mobile app.
Installation
The best way to install the PostHog Android library is with a build system like Gradle. This ensures you can easily upgrade to the latest versions.
All you need to do is add the posthog-android
module to your App's build.gradle
or build.gradle.kts
:
dependencies {implementation("com.posthog:posthog-android:3.+")}
Configuration
The best place to initialize the client is in your Application
subclass.
import android.app.Applicationimport com.posthog.android.PostHogAndroidimport com.posthog.android.PostHogAndroidConfigclass SampleApp : Application() {companion object {const val POSTHOG_API_KEY = "<ph_project_api_key>"// usually 'https://us.i.posthog.com' or 'https://eu.i.posthog.com'const val POSTHOG_HOST = "https://us.i.posthog.com"}override fun onCreate() {super.onCreate()val config = PostHogAndroidConfig(apiKey = POSTHOG_API_KEY,host = POSTHOG_HOST)PostHogAndroid.setup(this, config)}}
Capturing events
You can send custom events using capture
:
import com.posthog.PostHogPostHog.capture(event = "user_signed_up")
Tip: We recommend using a
[object] [verb]
format for your event names, where[object]
is the entity that the behavior relates to, and[verb]
is the behavior itself. For example,project created
,user signed up
, orinvite sent
.
Setting event properties
Optionally, you can also include additional information in the event by setting the properties value:
import com.posthog.PostHogPostHog.capture(event = "user_signed_up",properties = mapOf("login_type" to "email","is_free_trial" to true))
Autocapture
PostHog autocapture automatically tracks the following events for you:
- Application Opened - when the app is opened from a closed state or when the app comes to the foreground (e.g. from the app switcher)
- Deep Link Opened - when the app is opened from a deep link.
- Application Backgrounded - when the app is sent to the background by the user
- Application Installed - when the app is installed.
- Application Updated - when the app is updated.
- $screen - when the user navigates (if using
android.app.Activity
)
Capturing screen views
With captureScreenViews = true
, PostHog will try to record all screen changes automatically.
The screenTitle
will be the <activity>
's android:label
, if not set it'll fallback to the <application>
's android:label
or the <activity>
's android:name
.
<activityandroid:name="com.example.app.ChildActivity"android:label="@string/title_child_activity"...</activity>
If you want to manually send a new screen capture event, use the screen
function.
This function requires a screenTitle
. You may also pass in an optional properties
object.
import com.posthog.PostHogPostHog.screen(screenTitle = "Dashboard",properties = mapOf("background" to "blue","hero" to "superhog"))
Identifying users
We highly recommend reading our section on Identifying users to better understand how to correctly use this method.
Using identify
, you can associate events with specific users. This enables you to gain full insights as to how they're using your product across different sessions, devices, and platforms.
An identify
call has the following arguments:
- distinctId: Required. A unique identifier for your user. Typically either their email or database ID.
- userProperties: Optional. A dictionary with key:value pairs to set the person properties
- userPropertiesSetOnce: Optional. Similar to
userProperties
. See the difference betweenuserProperties
anduserPropertiesSetOnce
import com.posthog.PostHogPostHog.identify(distinctId = distinctID,userProperties = mapOf("name" to "Max Hedgehog","email" to "max@hedgehogmail.com"),userPropertiesSetOnce = mapOf("date_of_first_log_in" to "2024-03-01"),)
You should call identify
as soon as you're able to. Typically, this is every time your app loads for the first time as well as directly after your user logs in. This ensures that events sent during your user's sessions are correctly associated with them.
When you call identify
, all previously tracked anonymous events will be linked to the user.
Get the current user's distinct ID
You may find it helpful to get the current user's distinct ID. For example, to check whether you've already called identify
for a user or not.
To do this, call distinctId()
. This returns either the ID automatically generated by PostHog or the ID that has been passed by a call to identify()
.
Alias
Sometimes, you want to assign multiple distinct IDs to a single user. This is helpful when your primary distinct ID is inaccessible. For example, if a distinct ID used on the frontend is not available in your backend.
In this case, you can use alias
to assign another distinct ID to the same user.
/*** Create an alias for the current user.*/PostHog.alias("distinct_id")
We strongly recommend reading our docs on alias to best understand how to correctly use this method.
Anonymous and identified events
PostHog captures two types of events: anonymous and identified
Identified events enable you to attribute events to specific users, and attach person properties. They're best suited for logged-in users.
Scenarios where you want to capture identified events are:
- Tracking logged-in users in B2B and B2C SaaS apps
- Doing user segmented product analysis
- Growth and marketing teams wanting to analyze the complete conversion lifecycle
Anonymous events are events without individually identifiable data. They're best suited for web analytics or apps where users aren't logged in.
Scenarios where you want to capture anonymous events are:
- Tracking a marketing website
- Content-focused sites
- B2C apps where users don't sign up or log in
Under the hood, the key difference between identified and anonymous events is that for identified events we create a person profile for the user, whereas for anonymous events we do not.
💡 Tip: Under our current pricing, anonymous events can be up to 4x cheaper than identified ones (due to the cost of processing them), so it's recommended you only capture identified events when needed.
How to capture anonymous events
PostHog captures identified events by default. To change this and capture anonymous events, change the personProfiles
config when initializing PostHog:
personProfiles: PersonProfiles.ALWAYS
- Capture identified events for all events.personProfiles: PersonProfiles.IDENTIFIED_ONLY
(default) - Anonymous events are captured by default. PostHog only captures identified events for users where person profiles have already been created.
For example:
val config = PostHogAndroidConfig(apiKey = POSTHOG_API_KEY,host = POSTHOG_HOST,).apply {personProfiles = PersonProfiles.IDENTIFIED_ONLY}
How to capture identified events
Identified events are captured by default if you've set the personProfiles
config to ALWAYS
(the default option).
If you've set the personProfiles
to IDENTIFIED_ONLY
, anonymous events are captured by default. Then, to capture identified events, call any of the following functions:
When you call any of these functions, it creates a person profile for the user. Once this profile is created, all subsequent events for this user will be captured as identified events.
Setting person properties
To set properties on your users via an event, you can leverage the event properties userProperties
and userPropertiesSetOnce
.
When capturing an event, you can pass a property called userProperties
as an event property, and specify its value to be an object with properties to be set on the user that will be associated with the user who triggered the event.
import com.posthog.PostHogPostHog.capture(event = "button_b_clicked",properties = mapOf("color" to "blue"),userProperties = mapOf("string" to "value1","integer" to 2))
userPropertiesSetOnce
works just like userProperties
, except that it will only set the property if the user doesn't already have that property set.
import com.posthog.PostHogPostHog.capture(event = "button_b_clicked",properties = mapOf("color" to "blue"),userPropertiesSetOnce = mapOf("string" to "value1","integer" to 2))
Super Properties
Super Properties are properties associated with events that are set once and then sent with every capture
call, be it a $screen
, or anything else.
They are set using PostHog.register
, which takes a key and value, and they persist across sessions.
For example, take a look at the following call:
import com.posthog.PostHogPostHog.register("team_id", 22)
The call above ensures that every event sent by the user will include "team_id": 22
. This way, if you filtered events by property using team_id = 22
, it would display all events captured on that user after the PostHog.register
call, since they all include the specified Super Property.
However, please note that this does not store properties against the User, only against their events. To store properties against the User object, you should use PostHog.identify
. More information on this can be found on the Sending User Information section.
Removing stored Super Properties
Super Properties are persisted across sessions so you have to explicitly remove them if they are no longer relevant. In order to stop sending a Super Property with events, you can use PostHog.unregister
, like so:
import com.posthog.PostHogPostHog.unregister("team_id")
This will remove the Super Property and subsequent events will not include it.
If you are doing this as part of a user logging out you can instead simply use PostHog.reset
which takes care of clearing all stored Super Properties and more.
Opt out of data capture
You can completely opt-out users from data capture. To do this, there are two options:
- Opt users out by default by setting
optOut
totrue
in your PostHog config:
val config = PostHogAndroidConfig(apiKey = <ph_project_api_key>,host = https://us.i.posthog.com)config.optOut = truePostHogAndroid.setup(this, config)
- Opt users out on a per-person basis by calling
optOut()
:
PostHog.optOut()
Similarly, you can opt users in:
PostHog.optIn()
To check if a user is opted out:
PostHog.isOptOut()
Flush
You can set the number of events in the configuration that should queue before flushing.
Setting this to 1
will send events immediately and will use more battery. The default value for this is 20
.
You can also configure the flush interval. By default we flush all events after 30
seconds,
no matter how many events have been gathered.
import com.posthog.android.PostHogAndroidConfigval config = PostHogAndroidConfig(apiKey = POSTHOG_API_KEY, host = POSTHOG_HOST).apply {flushAt = 20flushIntervalSeconds = 30}
You can also manually flush the queue:
import com.posthog.PostHogPostHog.flush()
Reset after logout
To reset the user's ID and anonymous ID, call reset
. Usually you would do this right after the user logs out.
import com.posthog.PostHogPostHog.reset()
Feature Flags
PostHog's feature flags enable you to safely deploy and roll back new features.
Boolean feature flags
import com.posthog.PostHogif (PostHog.isFeatureEnabled("flag-key")) {// Do something differently for this user// Optional: fetch the payloadval matchedFlagPayload = PostHog.getFeatureFlagPayload("flag-key")}
Multivariate feature flags
import com.posthog.PostHogif (PostHog.getFeatureFlag("flag-key") == "variant-key") { // replace 'variant-key' with the key of your variant// Do something differently for this user// Optional: fetch the payloadval matchedFlagPayload = PostHog.getFeatureFlagPayload("flag-key")}
Ensuring flags are loaded before usage
Every time a user opens the app, we send a request in the background to fetch the feature flags that apply to that user. We store those flags in the storage.
This means that for most screens, the feature flags are available immediately – except for the first time a user visits.
To handle this, you can use the onFeatureFlags
callback to wait for the feature flag request to finish:
import com.posthog.PostHogimport com.posthog.android.PostHogAndroidConfigimport com.posthog.PostHogOnFeatureFlags// During SDK initializationval config = PostHogAndroidConfig("<ph_project_api_key>").apply {onFeatureFlags = PostHogOnFeatureFlags {if (PostHog.isFeatureEnabled("flag-key")) {// do something}}}// And/Or manually the SDK is initializedPostHog.reloadFeatureFlags {if (PostHog.isFeatureEnabled("flag-key")) {// do something}}
Reloading feature flags
Feature flag values are cached. If something has changed with your user and you'd like to refetch their flag values, call:
import com.posthog.PostHogPostHog.reloadFeatureFlags()
Experiments (A/B tests)
Since experiments use feature flags, the code for running an experiment is very similar to the feature flags code:
import com.posthog.PostHogif (PostHog.getFeatureFlag("experiment-feature-flag-key") == "variant-name") {// do something}
It's also possible to run experiments without using feature flags.
Group analytics
Group analytics allows you to associate the events for that person's session with a group (e.g. teams, organizations, etc.). Read the Group Analytics guide for more information.
Note: This is a paid feature and is not available on the open-source or free cloud plan. Learn more here.
- Associate the events for this session with a group
import com.posthog.PostHog// organization is the group type, company_id_in_your_db is the group IDPostHog.group(type = "company",key = "company_id_in_your_db")
- Associate the events for this session with a group AND update the properties of that group
import com.posthog.PostHogPostHog.group(type = "company",key = "company_id_in_your_db",groupProperties = mapOf("name" to "Awesome Inc."))
The name
is a special property which is used in the PostHog UI for the name of the Group. If you don't specify a name
property, the group ID will be used instead.
Session replay
To set up session replay in your project, all you need to do is install the Android SDK, enable "Record user sessions" in your project settings and enable the sessionReplay
option.
All configuration options
When creating the PostHog client, there are many options you can set:
val config = PostHogAndroidConfig(apiKey = POSTHOG_API_KEY, host = POSTHOG_HOST).apply {// Capture certain application events automatically. (on/true by default)captureApplicationLifecycleEvents = true// Capture screen views automatically. (on/true by default)captureScreenViews = true // (on/true by default)// Capture deep links as part of the screen call. (on/true by default)captureDeepLinks = true// Maximum number of events to keep in queue before flushing (20 by default)flushAt = 20// Number of maximum events in memory and disk, when the maximum is exceed, the oldest event is deleted and the new one takes place. (1000 by default)maxQueueSize = 1000// Number of maximum events in a batch call. (50 by default)maxBatchSize = 50// Maximum delay before flushing the queue (30 seconds)flushIntervalSeconds = 30// Logs the SDK messages into Logcat. (off/false by default)debug = false// Prevents capturing any data if enabled. (off/false by default)optOut = false// Send a '$feature_flag_called' event when a feature flag is used automatically. (on/true by default)sendFeatureFlagEvent = true// Preload feature flags automatically. (on/true by default)preloadFeatureFlags = true// Callback that is called when feature flags are loaded (not set by default)onFeatureFlags = { ... }// Callback that allows to sanitize the event properties (not set by default)propertiesSanitizer = { properties -> ... }// Hook for encrypt and decrypt events// Devices are sandbox already// Defaults to no encryptionencryption = object : PostHogEncryption { ... }// Hook that allows for modification of the default mechanism for// generating anonymous id (which as of now is just random UUID v7)getAnonymousId = { ... }// Determines the behavior for processing user profiles.// Defaults to PersonProfiles.IDENTIFIED_ONLYpersonProfiles = PersonProfiles.IDENTIFIED_ONLY// Enable Recording of Session Replay. (off/false by default)// Experimental supportsessionReplay = false// Session Replay configuration// Experimental supportsessionReplayConfig = PostHogSessionReplayConfig(...)}