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 JOINStoriesListenerDelegate to redirect the JOIN Stories events to your data platform :

import JOINStoriesSDK

class YourViewController: UIViewController {

   override func viewDidLoad() {
        super.viewDidLoad()
        JOINStoriesListener.setDelegate(delegate: self)
   }
}

extension YourViewController: JOINStoriesListenerDelegate {
    func joinStoriesAnalyticsCallback(event: JOINTrackingEvent, payload: [String: Any?]) {}
}

import JOINStoriesSDK

struct YourView : View {
    
   @StateObject var viewModel = YourViewModel()
    
    var body: some View {
        Text("Hello, world!")
    }
}

class YourViewModel: ObservableObject {
    init() {
        JOINStoriesListener.setDelegate(delegate: self)
    }
}

extension YourViewModel: JOINStoriesListenerDelegate {
    func joinStoriesAnalyticsCallback(event: JOINTrackingEvent, payload: [String: Any?]) {}
}

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.

import JOINStoriesSDK

class YourViewController: UIViewController {

   override func viewDidLoad() {
        super.viewDidLoad()
     		JOINStories.setTrackingUserId("<custom_user_id>") // if you wish to use your personnalize 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 NameJOINTrackingEventTrigger Scenario
story-chapter-vieweventStoryChapterViewA 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-vieweventStoryFirstChapterViewThe first chapter of the story becomes active.
story-last-chapter-vieweventStoryLastChapterViewThe last chapter of the story becomes active.
story-chapter-exiteventStoryChapterExitThe active chapter exits either manually or automatically.
story-vieweventStoryViewA story become active: Opening a player, Change of story (swipe,…), Coming back to a tab with a story opened ...
story-exiteventStoryExitThe active story exits, either manually or automatically.
story-completedeventStoryCompletedThe last chapter of the story is fully read or ends due to a tap to next.
story-tap-nexteventStoryTapNextUser taps to next on a story.
story-tap-previouseventStoryTapPreviousUser taps to previous on a story.
story-swipe-nexteventStorySwipeNextUser swipes to the right or clicks on the next story button.
story-swipe-previouseventStorySwipePreviousUser swipes to the left or clicks on the previous story button.
story-pauseeventStoryPauseLong press on mobile or on click of the pause button on desktop.
story-unpauseeventStoryUnpauseLong press is released on mobile or on click of the play button on desktop.

Category : Interaction

Event NameJOINTrackingEventTrigger Scenario
interaction-vieweventInteractionViewAn interaction becomes active (displayed/viewed) in an active story.
interaction-completedeventInteractionCompletedUser completes an interaction. It must be triggered only once.
interaction-engageeventInteractionEngageUser executes an action on an interaction (e.g., form -> fill fields, tooltip click, CTA click).
interaction-exiteventInteractionExitThe interaction becomes inactive (from an active state).

Category : Widget

Event NameJOINTrackingEventTrigger Scenario
widget-initializedeventWidgetInitializedThe widget has finished initialization before loading data. In case of changing audiences, it is not re-initialized.
widget-data-loadedeventWidgetDataLoadedThe widget loads data (from internal JSON, API fetch, cache, or fallback).
widget-displayedeventWidgetDisplayedThe widget is inserted in the page and displayed or ready to be displayed.
widget-thumbnail-readyeventWidgetThumbnailReadyAll thumbnails of visible elements of the widget are loaded.
widget-view-50eventWidgetView50At least 50% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it).
widget-view-75eventWidgetView75At least 75% of the widget’s height is visible(in the viewport and not obstructed by another element on top of it).
widget-view-100eventWidgetView100100% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it).
widget-clickeventWidgetClickUser clicks on a widget.
widget-player-closeeventWidgetPlayerCloseThe player is closed (e.g., swipe down, tap next when no story, or programmatically).
widget-scrolledeventWidgetScrolledUser scrolls (mobile) or clicks on an arrow (desktop) to see next stories.
widget-closeeventWidgetCloseThe 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

NameJOINEventParamsDescription
event_categoryEVENT_CATEGORYstory, widget, interaction, …
event_typeEVENT_TYPEEvent name. See above for a list of event names
event_versionEVENT_VERSIONEvent analytics version. (Should be v2)
event_date_timezoneEVENT_DATE_TIMEZONETimezone of the visitor triggering the event. Number in minutes from UTC
event_date_creationEVENT_DATE_CREATIONTimestamp (in s) the event was created client side.
event_ownerEVENT_OWNERJOIN owner
event_user_idEVENT_USER_IDAnonymous user id to monitor sessions
event_device_screen_widthEVENT_DEVICE_SCREEN_WIDTH
event_device_screen_heightEVENT_DEVICE_SCREEN_HEIGHT
event_device_screen_colorEVENT_DEVICE_SCREEN_COLOR
event_device_screen_ratioEVENT_DEVICE_SCREEN_RATIO
event_device_pixel_ratioEVENT_DEVICE_PIXEL_RATIO
event_device_network_typeEVENT_DEVICE_NETWORK_TYPEWi-Fi, cellular, bluetooth, ethernet, unknown
event_device_network_effective_typeEVENT_DEVICE_NETWORK_EFFECTIVE_TYPEWi-Fi, 4G, 5G, Ethernet, unknown
event_device_orientationEVENT_DEVICE_ORIENTATIONportrait, landscape
event_device_battery_levelEVENT_DEVICE_BATTERY_LEVELbattery level. Between 0 and 1
event_integration_environmentEVENT_INTEGRATION_ENVIRONMENTapp
event_integration_location_nameEVENT_INTEGRATION_LOCATION_NAME
event_integration_app_idEVENT_INTEGRATION_APP_ID
event_integration_app_nameEVENT_INTEGRATION_APP_NAME
event_integration_app_versionEVENT_INTEGRATION_APP_VERSION
event_integration_app_languageEVENT_INTEGRATION_APP_LANGUAGE
event_integration_app_language_localeEVENT_INTEGRATION_APP_LANGUAGE_LOCALE
event_integration_app_sdk_versionEVENT_INTEGRATION_APP_SDK_VERSION
event_integration_app_sdk_typeEVENT_INTEGRATION_APP_SDK_TYPE

Story properties

NameJOINEventParamsDescription
story_idSTORY_IDStory technical id
story_titleSTORY_TITLEStory's title
story_labelSTORY_LABELStory's displayed label (may be different from title)
story_last_published_dateSTORY_LAST_PUBLISHED_DATELast published date of the story
story_has_subtitlesSTORY_HAS_SUBTITLESWether story has subtitles or not
story_has_interactionsSTORY_HAS_INTERACTIONSWether story has interactions or not
story_has_soundSTORY_HAS_SOUNDIf at least one chapter has sound on a story, then yes
story_chapter_countSTORY_CHAPTER_COUNTTotal count of chapters
story_interaction_countSTORY_INTERACTION_COUNTTotal count of interactions on the story
story_durationSTORY_DURATIONTotal duration of a story (in ms)
story_chapter_indexSTORY_CHAPTER_INDEXThe index of the chapter responsible for sending the event. Indexes start at 0
story_chapter_durationSTORY_CHAPTER_DURATIONTotal duration of the chapter
story_previous_chapter_indexSTORY_PREVIOUS_CHAPTER_INDEXThe index of the previous chapter
story_index_in_widgetSTORY_INDEX_IN_WIDGETIndex of the story in the list of stories. Indexes start at 0.
story_mutedSTORY_MUTEDIs Story muted by the user
story_displayed_formatSTORY_DISPLAYED_FORMATmobile
story_completionSTORY_COMPLETIONRatio of completion between 0 and 1
story_chapter_completionSTORY_CHAPTER_COMPLETIONRatio of completion between 0 and 1
story_viewing_timeSTORY_VIEWING_TIMEDuration in ms
story_chapter_viewing_timeSTORY_CHAPTER_VIEWING_TIMEDuration in ms
story_is_activeSTORY_IS_ACTIVEProperty that should always be true for now, but can be used to dissociate events in the future
story_chapter_is_activeSTORY_CHAPTER_IS_ACTIVEProperty that should always be true for now, but can be used to dissociate events in the future
story_exit_reasonSTORY_EXIT_REASONEnum: swipeDown, swipeNext, swipePrevious, tapNext, tapPrevious, closeButton, ctaClick, autoAdvance
story_chapter_exit_reasonSTORY_CHAPTER_EXIT_REASONEnum: swipeDown, swipeNext, swipePrevious, tapNext, tapPrevious, closeButton, ctaClick, autoAdvance
story_pause_durationSTORY_PAUSE_DURATIONDuration in ms
story_versionSTORY_VERSIONStory's technical version.

Interaction properties

NameJOINEventParamsDescription
interaction_idINTERACTION_IDInteraction technical Id
interaction_typeINTERACTION_TYPEInteraction type (CTA, Quiz, Shopping, ...)
interaction_engage_typeINTERACTION_ENGAGE_TYPEredirectform-input-focus| form-input-changeform-submit| poll-clicked| quiz-click | open-product-config| continue-shopping
interaction_engage_value_1INTERACTION_ENGAGE_VALUE_1First generic value
interaction_engage_value_2INTERACTION_ENGAGE_VALUE_2Second generic value
interaction_engage_urlINTERACTION_ENGAGE_URLOnly used in case of CTA (replacing value 1-2)
interaction_engage_labelINTERACTION_ENGAGE_LABELOnly used in case of CTA (replacing value 1-2)
interaction_engaging_timeINTERACTION_ENGAGING_TIMETime between first interactionEngage and interactionEngageComplete or interactionExit
interaction_viewing_timeINTERACTION_VIEWING_TIMETime between interactionView and interactionExit
interaction_has_been_blockingINTERACTION_HAS_BEEN_BLOCKINGIn case the interaction has blocked the auto-advance in the chapter/story, waiting for an engagement
interaction_is_already_completedINTERACTION_IS_ALREADY_COMPLETEDtrue if it was already completed when the interaction became active
interaction_is_first_engageINTERACTION_IS_FIRST_ENGAGEinteraction-engage event is the first engagement of the page.
interaction_is_story_first_engageINTERACTION_IS_STORY_FIRST_ENGAGEinteraction-engage event is the first engagement of the story.
interaction_is_completion_engageINTERACTION_IS_COMPLETION_ENGAGEThe event is the one that has completed the interaction.
interaction_exit_stateINTERACTION_EXIT_STATEcompleted |
canceled: engaged at least once |
ignored: exited interaction without engage |
already-completed

Widget properties

NameJOINEventParamsDescription
widget_idWIDGET_IDWidget technical id
widget_aliasWIDGET_ALIASWidget alias
widget_typeWIDGET_TYPEWidget type (list, standalone)
widget_shapeWIDGET_SHAPE(Only applies for type list) Widget shape (bubble, card, square)
widget_layoutWIDGET_LAYOUTWidget layout (list, grid)
widget_integration_typeWIDGET_INTEGRATION_TYPEsdk
widget_story_countWIDGET_STORY_COUNTNumber of stories in the widget, after widget-data-loaded (with audience conditions applied)
widget_duration_displayedWIDGET_DURATION_DISPLAYEDDuration (in ms) since widget is displayed, starting at widget_display event
widget_click_countWIDGET_CLICK_COUNTNumber of clicks on the widget since it has been displayed, also to set for all story events
widget_player_close_reasonWIDGET_PLAYER_CLOSE_REASONclose-button | background | auto-advance | swipe-down| tap-next
widget_close_reasonWIDGET_CLOSE_REASONclose reason (manual or auto)
widget_versionWIDGET_VERSIONWidget version
widget_audience_slots_matchedWIDGET_AUDIENCE_SLOTS_MATCHEDFor every widget event except widget initialized, and also for stories when available
widget_visible_duration_thresholdWIDGET_VISIBLE_DURATION_THRESHOLDUsed in the event widgetView100Extended, to know the threshold of visibility
widget_is_scrollableWIDGET_IS_SCROLLABLEAccording to integration, if the widget is too large to be displayed fully and is made scrollable
widget_scroll_typeWIDGET_SCROLL_TYPEnatural/button – to know in case of widgetScroll event, if the scroll is from button or natural gesture
widget_data_loading_timeWIDGET_DATA_LOADING_TIMEDuration of the data loading (from API) if instant if coming from JSON = null
widget_data_loaded_sourceWIDGET_DATA_LOADED_SOURCEThe source from where the data has been loaded (from API, embedded-data, cache, fallback, ...)
widget_video_cover_countWIDGET_VIDEO_COVER_COUNTCount of the number of video covers
widget_static_cover_countWIDGET_STATIC_COVER_COUNTCount of the number of static covers
widget_element_heightWIDGET_ELEMENT_HEIGHTHeight 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.

JOIN Stories Events (Depreacted)

In order to get notification about JOIN Stories events, you should override the following functions in JOINStoriesListenerDelegate. The events are separated between trigger events (onTriggerAnalyticsCallback called) and player events (onPlayerAnalyticsCallback called).

Trigger Events

onTriggerAnalyticsCallback returns the following payload for each event:

public struct TriggerAnalyticsModel {
    public var eventOwner: String? // owner of the story (equal teamId)
    public var cpWidgetAlias: String // Alias of the trigger
    public var cpWidgetVersion: String // Version of the SDK
    public var date: Int64 // Current UNIX Timestamp in ms
    public var eventCategory: String = "widget"
    public var eventType: TriggerAnalyticsEventType // The event type
    public var eventTypeName: String // The event type name
    public var 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: .storiesFetched

Trigger Mounted Event

Each time the trigger is instantiated, a Trigger Mounted event is sent:

event: .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: .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: .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: .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: .additionalClickOnWidget

It also contains the following value: storyClicked: STORY_ID

Player Events

onPlayerAnalyticsCallback returning the following payload for each event:

public struct PlayerAnalyticsModel {
    public var storyPageCount: Int // Number of chapter of the story
    public var eventOwner: String? // owner of the story (equal teamId)
    public var storyId: String? // ID of the story
    public var date: Int64 // Current UNIX Timestamp in ms
    public var eventCategory: String = "story",
    public var cpIndex: Int? // Index of the current chapter: from 0 to N
    public var cpTitle: String? // Title of the story
    public var eventType: PlayerAnalyticsEventType // The event type
    public var eventTypeName: String // The event type name
}

Chapter View Event

Each time a chapter (equivalent to story page) starts playing, a Chapter View event is sent:

event: .storyPageVisible

Last Chapter View Event

Each time the last chapter (last story page) starts playing, a Last Chapter View event is sent:

event: .storyLastPageVisible

Click Call To Action Event

Each time the user clicks on a CTA, a Click Call To Action event is sent:

event: .storyClickOnCallToAction

The parameter ctaUrl is set on this event only.