Advanced Analytics
Sending Events
This section shows you how to send the JOIN Stories events to specific data platforms. You can use the following function of JOINStoriesAnalyticsListener
to redirect the JOIN Stories events to your data platform :
val analyticsListener: JOINStoriesAnalyticsListener = object : JOINStoriesAnalyticsListener {
override fun joinStoriesAnalyticsCallback(event: JOINTrackingEvent, payload: Map<String, Any?>)) {}
}
...
trigger.joinAnalyticsListener = analyticsListener
// OR
JOINStories.startPlayer(context, "<you_join_alias>", analyticsListener)
Tracking user ID
In stories analytics, you can track user activity by adding a user ID. You can use the setTrackingUserId() method to enter the user's ID, or set it to empty if the user does not accept consent.
val userId = AdvertisingIdClient.getAdvertisingIdInfo(this).id
JOINStories.setTrackingUserId(userId) // if you wish to use your personnalize advertising Id
// or
JOINStories.setTrackingUserId("") // if user does not accept consent
Analytics Events
During its life on a mobile, a widget can trigger various analytics events. There are events related to the widget itself and events related to the stories inside of it. They track how the users interact with the widget and its content.
Analytics v2
A new Analytics stack has been implemented. Old events have been renamed, with more properties. New events have been added. They are split by categories. Each category has its own properties.
Category : Story
Event Name | JOINTrackingEvent | Trigger Scenario |
---|---|---|
story-chapter-view | EVENT_STORY_CHAPTER_VIEW | A chapter of the story become active: Opening a player, Change of story (swipe,…), Change of chapter, Coming back to a tab with a story opened ... |
story-first-chapter-view | EVENT_STORY_FIRST_CHAPTER_VIEW | The first chapter of the story becomes active. |
story-last-chapter-view | EVENT_STORY_LAST_CHAPTER_VIEW | The last chapter of the story becomes active. |
story-chapter-exit | EVENT_STORY_CHAPTER_EXIT | The active chapter exits either manually or automatically. |
story-view | EVENT_STORY_VIEW | A story become active: Opening a player, Change of story (swipe,…), Coming back to a tab with a story opened ... |
story-exit | EVENT_STORY_EXIT | The active story exits, either manually or automatically. |
story-completed | EVENT_STORY_COMPLETED | The last chapter of the story is fully read or ends due to a tap to next. |
story-tap-next | EVENT_STORY_TAP_NEXT | User taps to next on a story. |
story-tap-previous | EVENT_STORY_TAP_PREVIOUS | User taps to previous on a story. |
story-swipe-next | EVENT_STORY_SWIPE_NEXT | User swipes to the right or clicks on the next story button. |
story-swipe-previous | EVENT_STORY_SWIPE_PREVIOUS | User swipes to the left or clicks on the previous story button. |
story-pause | EVENT_STORY_PAUSE | Long press on mobile or on click of the pause button on desktop. |
story-unpause | EVENT_STORY_UNPAUSE | Long press is released on mobile or on click of the play button on desktop. |
Category : Interaction
Event Name | JOINTrackingEvent | Trigger Scenario |
---|---|---|
interaction-view | EVENT_INTERACTION_VIEW | An interaction becomes active (displayed/viewed) in an active story. |
interaction-completed | EVENT_INTERACTION_COMPLETED | User completes an interaction. It must be triggered only once. |
interaction-engage | EVENT_INTERACTION_ENGAGE | User executes an action on an interaction (e.g., form -> fill fields, tooltip click, CTA click). |
interaction-exit | EVENT_INTERACTION_EXIT | The interaction becomes inactive (from an active state). |
Category : Widget
Event Name | JOINTrackingEvent | Trigger Scenario |
---|---|---|
widget-initialized | EVENT_WIDGET_INITIALIZED | The widget has finished initialization before loading data. In case of changing audiences, it is not re-initialized. |
widget-data-loaded | EVENT_WIDGET_DATA_LOADED | The widget loads data (from internal JSON, API fetch, cache, or fallback). |
widget-displayed | EVENT_WIDGET_DISPLAYED | The widget is inserted in the page and displayed or ready to be displayed. |
widget-thumbnail-ready | EVENT_WIDGET_THUMBNAIL_READY | All thumbnails of visible elements of the widget are loaded. |
widget-view-50 | EVENT_WIDGET_VIEW_50 | At least 50% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it). |
widget-view-75 | EVENT_WIDGET_VIEW_75 | At least 75% of the widget’s height is visible(in the viewport and not obstructed by another element on top of it). |
widget-view-100 | EVENT_WIDGET_VIEW_100 | 100% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it). |
widget-click | EVENT_WIDGET_CLICK | User clicks on a widget. |
widget-player-close | EVENT_WIDGET_PLAYER_CLOSE | The player is closed (e.g., swipe down, tap next when no story, or programmatically). |
widget-scrolled | EVENT_WIDGET_SCROLLED | User scrolls (mobile) or clicks on an arrow (desktop) to see next stories. |
widget-close | EVENT_WIDGET_CLOSE | The widget is closed in certain cases (e.g., sticky widget). |
Properties
All events have at least the common properties. Each category has its own properties.
Common properties
Name | JOINEventParams | Description |
---|---|---|
event_category | EVENT_CATEGORY | story, widget, interaction, … |
event_type | EVENT_TYPE | Event name. See above for a list of event names |
event_version | EVENT_VERSION | Event analytics version. (Should be v2) |
event_date_timezone | EVENT_DATE_TIMEZONE | Timezone of the visitor triggering the event. Number in minutes from UTC |
event_date_creation | EVENT_DATE_CREATION | Timestamp (in s) the event was created client side. |
event_owner | EVENT_OWNER | JOIN owner |
event_user_id | EVENT_USER_ID | Anonymous user id to monitor sessions |
event_device_screen_width | EVENT_DEVICE_SCREEN_WIDTH | |
event_device_screen_height | EVENT_DEVICE_SCREEN_HEIGHT | |
event_device_screen_color | EVENT_DEVICE_SCREEN_COLOR | |
event_device_screen_ratio | EVENT_DEVICE_SCREEN_RATIO | |
event_device_pixel_ratio | EVENT_DEVICE_PIXEL_RATIO | |
event_device_network_type | EVENT_DEVICE_NETWORK_TYPE | Wi-Fi, cellular, bluetooth, ethernet, unknown |
event_device_network_effective_type | EVENT_DEVICE_NETWORK_EFFECTIVE_TYPE | Wi-Fi, 4G, 5G, Ethernet, unknown |
event_device_orientation | EVENT_DEVICE_ORIENTATION | portrait, landscape |
event_device_battery_level | EVENT_DEVICE_BATTERY_LEVEL | battery level. Between 0 and 1 |
event_integration_environment | EVENT_INTEGRATION_ENVIRONMENT | app |
event_integration_location_name | EVENT_INTEGRATION_LOCATION_NAME | |
event_integration_app_id | EVENT_INTEGRATION_APP_ID | |
event_integration_app_name | EVENT_INTEGRATION_APP_NAME | |
event_integration_app_version | EVENT_INTEGRATION_APP_VERSION | |
event_integration_app_language | EVENT_INTEGRATION_APP_LANGUAGE | |
event_integration_app_language_locale | EVENT_INTEGRATION_APP_LANGUAGE_LOCALE | |
event_integration_app_sdk_version | EVENT_INTEGRATION_APP_SDK_VERSION | |
event_integration_app_sdk_type | EVENT_INTEGRATION_APP_SDK_TYPE |
Story properties
Name | JOINEventParams | Description |
---|---|---|
story_id | STORY_ID | Story technical id |
story_title | STORY_TITLE | Story's title |
story_label | STORY_LABEL | Story's displayed label (may be different from title) |
story_last_published_date | STORY_LAST_PUBLISHED_DATE | Last published date of the story |
story_has_subtitles | STORY_HAS_SUBTITLES | Wether story has subtitles or not |
story_has_interactions | STORY_HAS_INTERACTIONS | Wether story has interactions or not |
story_has_sound | STORY_HAS_SOUND | If at least one chapter has sound on a story, then yes |
story_chapter_count | STORY_CHAPTER_COUNT | Total count of chapters |
story_interaction_count | STORY_INTERACTION_COUNT | Total count of interactions on the story |
story_duration | STORY_DURATION | Total duration of a story (in ms) |
story_chapter_index | STORY_CHAPTER_INDEX | The index of the chapter responsible for sending the event. Indexes start at 0 |
story_chapter_duration | STORY_CHAPTER_DURATION | Total duration of the chapter |
story_previous_chapter_index | STORY_PREVIOUS_CHAPTER_INDEX | The index of the previous chapter |
story_index_in_widget | STORY_INDEX_IN_WIDGET | Index of the story in the list of stories. Indexes start at 0. |
story_muted | STORY_MUTED | Is Story muted by the user |
story_displayed_format | STORY_DISPLAYED_FORMAT | mobile |
story_completion | STORY_COMPLETION | Ratio of completion between 0 and 1 |
story_chapter_completion | STORY_CHAPTER_COMPLETION | Ratio of completion between 0 and 1 |
story_viewing_time | STORY_VIEWING_TIME | Duration in ms |
story_chapter_viewing_time | STORY_CHAPTER_VIEWING_TIME | Duration in ms |
story_is_active | STORY_IS_ACTIVE | Property that should always be true for now, but can be used to dissociate events in the future |
story_chapter_is_active | STORY_CHAPTER_IS_ACTIVE | Property that should always be true for now, but can be used to dissociate events in the future |
story_exit_reason | STORY_EXIT_REASON | Enum: swipeDown, swipeNext, swipePrevious, tapNext, tapPrevious, closeButton, ctaClick, autoAdvance |
story_chapter_exit_reason | STORY_CHAPTER_EXIT_REASON | Enum: swipeDown, swipeNext, swipePrevious, tapNext, tapPrevious, closeButton, ctaClick, autoAdvance |
story_pause_duration | STORY_PAUSE_DURATION | Duration in ms |
story_version | STORY_VERSION | Story's technical version. |
Interaction properties
Name | JOINEventParams | Description |
---|---|---|
interaction_id | INTERACTION_ID | Interaction technical Id |
interaction_type | INTERACTION_TYPE | Interaction type (CTA, Quiz, Shopping, ...) |
interaction_engage_type | INTERACTION_ENGAGE_TYPE | redirect | form-input-focus | form-input-change | form-submit | poll-clicked | quiz-click | open-product-config | continue-shopping |
interaction_engage_value_1 | INTERACTION_ENGAGE_VALUE_1 | First generic value |
interaction_engage_value_2 | INTERACTION_ENGAGE_VALUE_2 | Second generic value |
interaction_engage_url | INTERACTION_ENGAGE_URL | Only used in case of CTA (replacing value 1-2) |
interaction_engage_label | INTERACTION_ENGAGE_LABEL | Only used in case of CTA (replacing value 1-2) |
interaction_engaging_time | INTERACTION_ENGAGING_TIME | Time between first interactionEngage and interactionEngageComplete or interactionExit |
interaction_viewing_time | INTERACTION_VIEWING_TIME | Time between interactionView and interactionExit |
interaction_has_been_blocking | INTERACTION_HAS_BEEN_BLOCKING | In case the interaction has blocked the auto-advance in the chapter/story, waiting for an engagement |
interaction_is_already_completed | INTERACTION_IS_ALREADY_COMPLETED | true if it was already completed when the interaction became active |
interaction_is_first_engage | INTERACTION_IS_FIRST_ENGAGE | interaction-engage event is the first engagement of the page. |
interaction_is_story_first_engage | INTERACTION_IS_STORY_FIRST_ENGAGE | interaction-engage event is the first engagement of the story. |
interaction_is_completion_engage | INTERACTION_IS_COMPLETION_ENGAGE | The event is the one that has completed the interaction. |
interaction_exit_state | INTERACTION_EXIT_STATE | completed |canceled : engaged at least once |ignored : exited interaction without engage |already-completed |
Widget properties
Name | JOINEventParams | Description |
---|---|---|
widget_id | WIDGET_ID | Widget technical id |
widget_alias | WIDGET_ALIAS | Widget alias |
widget_type | WIDGET_TYPE | Widget type (list, standalone) |
widget_shape | WIDGET_SHAPE | (Only applies for type list) Widget shape (bubble, card, square) |
widget_layout | WIDGET_LAYOUT | Widget layout (list, grid) |
widget_integration_type | WIDGET_INTEGRATION_TYPE | sdk |
widget_story_count | WIDGET_STORY_COUNT | Number of stories in the widget, after widget-data-loaded (with audience conditions applied) |
widget_duration_displayed | WIDGET_DURATION_DISPLAYED | Duration (in ms) since widget is displayed, starting at widget_display event |
widget_click_count | WIDGET_CLICK_COUNT | Number of clicks on the widget since it has been displayed, also to set for all story events |
widget_player_close_reason | WIDGET_PLAYER_CLOSE_REASON | close-button | background | auto-advance | swipe-down | tap-next |
widget_close_reason | WIDGET_CLOSE_REASON | close reason (manual or auto ) |
widget_version | WIDGET_VERSION | Widget version |
widget_audience_slots_matched | WIDGET_AUDIENCE_SLOTS_MATCHED | For every widget event except widget initialized, and also for stories when available |
widget_visible_duration_threshold | WIDGET_VISIBLE_DURATION_THRESHOLD | Used in the event widgetView100Extended, to know the threshold of visibility |
widget_is_scrollable | WIDGET_IS_SCROLLABLE | According to integration, if the widget is too large to be displayed fully and is made scrollable |
widget_scroll_type | WIDGET_SCROLL_TYPE | natural/button – to know in case of widgetScroll event, if the scroll is from button or natural gesture |
widget_data_loading_time | WIDGET_DATA_LOADING_TIME | Duration of the data loading (from API) if instant if coming from JSON = null |
widget_data_loaded_source | WIDGET_DATA_LOADED_SOURCE | The source from where the data has been loaded (from API, embedded-data, cache, fallback, ...) |
widget_video_cover_count | WIDGET_VIDEO_COVER_COUNT | Count of the number of video covers |
widget_static_cover_count | WIDGET_STATIC_COVER_COUNT | Count of the number of static covers |
widget_element_height | WIDGET_ELEMENT_HEIGHT | Height of the widget (element not full bloc) configured in the studio (or in app) in px |
Analytics sender
When integrating, you need to be aware of the widget's lifecycle, as the widget may send events several times during use if it is destroyed and recreated, in the case of a dynamic list (RecyclerView, for example).
JOIN Stories Events (Deprecated)
In order to get notification about JOIN Stories events, you should override the following functions in JOINStoriesAnalyticsListener. The events are separated between trigger events (onTriggerAnalyticsCallback
called) and player events (onPlayerAnalyticsCallback
called).
Trigger Events
onTriggerAnalyticsCallback
returns the following payload for each event:
data class JOINAnalyticsTrigger(
val eventOwner: String, // owner of the story (equal teamId)
val cpWidgetAlias: String, // Alias of the trigger
val cpWidgetVersion: String, // Version of the SDK
val date: Long, // Current UNIX Timestamp in ms
val eventCategory: String = "widget",
val eventTypeName: String, // The event type name
var eventType: JOINStoriesTriggerEventType?, // The event type
val storyClicked: String? // The story id - only for click events
)
Stories Fetched Event
Each time the trigger fetch the list of stories from the API, a Stories fetched event is sent:
event: JOINStoriesTriggerEventType.StoriesFetched
Trigger Mounted Event
Each time the trigger is instantiated, a Trigger Mounted event is sent:
event: JOINStoriesTriggerEventType.WidgetMounted
Trigger Visibility 50% Event
After instantiation, the first time the trigger is visible at least on 50% of its height, a Trigger Visible 50% event is sent:
event: JOINStoriesTriggerEventType.ComponentVisible50
Trigger Visibility 75% Event
After instantiation, the first time the trigger is visible at least on 75% of its height, a Trigger Visible 75% event is sent:
event: JOINStoriesTriggerEventType.ComponentVisible75
First Click on Trigger Event
After instantiation, the first time the trigger is clicked to open the player, a First Click on Trigger event is sent:
event: JOINStoriesTriggerEventType.FirstClickOnWidget
It also contains the following value: storyClicked: STORY_ID
Additional Click on Trigger
Each time the trigger is clicked after the first click (see previous event), an Additional Click on Trigger event is sent:
event: JOINStoriesTriggerEventType.AdditionalClickOnWidget
It also contains the following value: storyClicked: STORY_ID
Player Events
onPlayerAnalyticsCallback
returns the following payload for each event:
data class JOINAnalyticsPlayer(
val storyPageCount: Int, // Number of chapter of the story
val eventOwner: String, // owner of the story (equal teamId)
val storyId: String, // ID of the story
val date: Long, // Current UNIX Timestamp in ms
val eventCategory: String = "story",
val cpIndex: Int, // Index of the current chapter: from 0 to N
val cpTitle: String, // Title of the story
var eventType: JOINStoriesPlayerEventType?, // The event type
val eventTypeName: String, // The event type name
val ctaUrl: String? // Url of CTA
)
Chapter View Event
Each time a chapter (equivalent to story page) starts playing, a Chapter View event is sent:
event: JOINStoriesPlayerEventType.StoryPageVisible
Last Chapter View Event
Each time the last chapter (last story page) starts playing, a Last Chapter View event is sent:
event: JOINStoriesPlayerEventType.StoryLastPageVisible
Click Call To Action Event
Each time the user clicks on a CTA, a Click Call To Action event is sent:
event: JOINStoriesPlayerEventType.StoryClickOnCallToAction
The parameter ctaUrl
is set on this event only.
Updated 7 days ago