Advanced Analytics

Analytics listener

You can listen to various events on the triggers to generate you own analytics. These events can come from a Bubble or Card component.

Example

import { JoinStoriesView, JoinStories } from '@join-stories/react-native-widgets';

export default function Home() {
  useEffect(() => {
    const analyticsSubscription = JoinStories.addAnalyticsListener((data) => {
      console.log('analytics event', data.event_type);
    });

    return () => {
      analyticsSubscription.remove();
    };
  }, [])
  
  return (
  	<SafeAreaView>
    	<Text>Welcome !</Text>
      <JoinStoriesView alias="<your_join_alias>" />
    </SafeAreaView>)
}

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.

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 NameTrigger Scenario
story-chapter-viewA 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-viewThe first chapter of the story becomes active.
story-last-chapter-viewThe last chapter of the story becomes active.
story-chapter-exitThe active chapter exits either manually or automatically.
story-viewA story become active: Opening a player, Change of story (swipe,…), Coming back to a tab with a story opened ...
story-exitThe active story exits, either manually or automatically.
story-completedThe last chapter of the story is fully read or ends due to a tap to next.
story-tap-nextUser taps to next on a story.
story-tap-previousUser taps to previous on a story.
story-swipe-nextUser swipes to the right or clicks on the next story button.
story-swipe-previousUser swipes to the left or clicks on the previous story button.
story-pauseLong press on mobile or on click of the pause button on desktop.
story-unpauseLong 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 NameTrigger Scenario
widget-initializedThe widget has finished initialization before loading data. In case of changing audiences, it is not re-initialized.
widget-data-loadedThe widget loads data (from internal JSON, API fetch, cache, or fallback).
widget-displayedThe widget is inserted in the page and displayed or ready to be displayed.
widget-thumbnail-readyAll thumbnails of visible elements of the widget are loaded.
widget-view-50At least 50% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it).
widget-view-75At least 75% of the widget’s height is visible(in the viewport and not obstructed by another element on top of it).
widget-view-100100% of the widget’s height is visible (in the viewport and not obstructed by another element on top of it).
widget-clickUser clicks on a widget.
widget-player-closeThe player is closed (e.g., swipe down, tap next when no story, or programmatically).
widget-scrolledUser scrolls (mobile) or clicks on an arrow (desktop) to see next stories.
widget-closeThe 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

NameDescription
event_categorystory, widget, interaction, …
event_typeEvent name. See above for a list of event names
event_versionEvent analytics version. (Should be v2)
event_date_timezoneTimezone of the visitor triggering the event. Number in minutes from UTC
event_date_creationTimestamp (in s) the event was created client side.
event_ownerJOIN owner
event_user_idAnonymous user id to monitor sessions
event_device_screen_width
event_device_screen_height
event_device_screen_color
event_device_screen_ratio
event_device_pixel_ratio
event_device_network_typeWi-Fi, cellular, bluetooth, ethernet, unknown
event_device_network_effective_typeWi-Fi, 4G, 5G, Ethernet, unknown
event_device_orientationportrait, landscape
event_device_battery_levelbattery level. Between 0 and 1
event_integration_environmentapp
event_integration_location_name
event_integration_app_id
event_integration_app_name
event_integration_app_version
event_integration_app_language
event_integration_app_language_locale
event_integration_app_sdk_version
event_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

NameDescription
interaction_idInteraction technical Id
interaction_typeInteraction type (CTA, Quiz, Shopping, ...)
interaction_engage_typeredirectform-input-focus| form-input-changeform-submit| poll-clicked| quiz-click | open-product-config| continue-shopping
interaction_engage_value_1First generic value
interaction_engage_value_2Second generic value
interaction_engage_urlOnly used in case of CTA (replacing value 1-2)
interaction_engage_labelOnly used in case of CTA (replacing value 1-2)
interaction_engaging_timeTime between first interactionEngage and interactionEngageComplete or interactionExit
interaction_viewing_timeTime between interactionView and interactionExit
interaction_has_been_blockingIn case the interaction has blocked the auto-advance in the chapter/story, waiting for an engagement
interaction_is_already_completedtrue if it was already completed when the interaction became active
interaction_is_first_engageinteraction-engage event is the first engagement of the page.
interaction_is_story_first_engageinteraction-engage event is the first engagement of the story.
interaction_is_completion_engageThe event is the one that has completed the interaction.
interaction_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 V1 (Deprecated)

Trigger Analytics

You can listen to various events on the triggers to generate you own analytics. These events can come from a Bubble or Card component.

Example

import { JoinStoriesView, JoinStories } from '@join-stories/react-native-widgets';

export default function Home() {
  useEffect(() => {
    const triggerAnalyticsSubscription = JoinStories.addTriggerAnalyticsListener((data) => {
      console.log(data);
    });

    return () => {
      triggerAnalyticsSubscription.remove();
    };
  }, [])
  
  return (
  	<SafeAreaView>
    	<Text>Welcome !</Text>
      <JoinStoriesView alias="<your_join_alias>" />
    </SafeAreaView>)
}

Event payload

Each event has the following basic payload:

{
  type: string;
  cpWidgetAlias: string;
  date: string;
  cpWidgetVersion: string;
  eventOwner: string;
  eventType: string;
  eventCategory: string;
  storyClicked: string;
}

type field is one of the events below.

List of events

Event nameadditional parametersDescription
StoriesFetchedEach time the trigger fetch the list of stories from the API.
WidgetMountedEach time the trigger is instantiated.
ComponentVisible50After instantiation, the first time the trigger is visible at least on 50% of its height.
ComponentVisible75After instantiation, the first time the trigger is visible at least on 75% of its height.
FirstClickOnWidgetstoryClicked(string) : storyIdAfter instantiation, the first time the trigger is clicked to open the player.
AdditionalClickOnWidgetstoryClicked (string) : storyIdEach time the trigger is clicked after the first click (see previous event).

Player Analytics

You can listen to various events on the player to generate you own analytics. These events can come from a player opened via a Bubble or Card component or via a Standalone Player.

Example

import { JoinStoriesView, JoinStories } from '@join-stories/react-native-widgets';

export default function Home() {
  useEffect(() => {
  	const playerAnalyticsSubscription = JoinStories.addPlayerAnalyticsListener((data) => {
      console.log(data);
    });

    return () => {
      playerAnalyticsSubscription.remove();
    };
  }, [])
  
  return (
  	<SafeAreaView>
    	<Text>Welcome !</Text>
      <JoinStoriesView alias="<your_join_alias>" />
    </SafeAreaView>)
}

Event payload

Each event has the following payload:

{
  type: string;
  storyId: string;
  cpIndex: string;
  cpTitle: string;
  cpScreenColor: string;
  eventOwner: string;
  date: string;
  eventCategory: string;
  storyPageCount: string;
  ctaUrl: string;
}

type field is one of the events below.

List of events

Event nameDescription
StoryPageVisibleEach time a chapter (equivalent to story page) starts playing.
StoryLastPageVisibleEach time the last chapter (last story page) starts playing.
StoryErrorSomething went wrong with the player.
StoryClickOnCallToActionEach time the user clicks on a CTA. The url of CTA added on this event only.