# Personalization event

Often, when choosing a personalization option, complex JavaScript logic needs to be executed. This can be achieved by subscribing to the personalization event.

# General Idea

Subscription is a special method call within JavaScript code. You can subscribe to either a personalization or a combination of personalization and personalization option.

The result of subscribing is the execution of custom code inside the handler.

The simplest subscription looks like this:

// Show the best review for the product
impact('onPersonalization', {
personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',

    // called when personalization is applied
    handler: (personalization, personalizationOption, dataContext) => {
      switch (personalizationOption.id){
        case ("10337247-6fa7-4ac3-92ed-bb2506d79ffa"):
          // Да
          break;
        case ("fb8211d7-5aa3-40df-a3ed-54a67d72f850"):
          // Нет
          break;
        default:
          // ...
      }
    },
})

# Types of subscription

There are three main types of subscription:

# Only personalization (Id is specified).

In this case, the subscription will be triggered whenever any personalization is applied, regardless of the personalization option.

Effectively, this means the subscription will be triggered whenever the personalization is applicable based on display conditions.

Example:

// Show the best review for the product
impact('onPersonalization', {
personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',

    handler: (personalization, personalizationOption) => {
      switch (personalizationOption.id){
        // called for any option of personalization
      }
    },
})

# Both personalizationId + personalizationOptionId (personalization ID or и personalization option ID) are specified.

In this case, the subscription will be triggered only when the specified personalization option is applied.

Example:

// Show the best review for the product -> Yes
impact('onPersonalization', {
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: '10337247-6fa7-4ac3-92ed-bb2506d79ffa',

    handler: (personalization, personalizationOption) => {
        // Triggered if the personalization option Show the best review for the product is selected = Yes
    },

})

# Neither personalizationId nor personalizationOptionId is specified

In this case, the subscription will be triggered for every applied personalization on the current page.

This type of subscription can be useful for client-side infrastructure tasks, for example, if you want to add all applied personalizations to a custom analytics system.

Example:

impact('onPersonalization', {

    handler: (personalization, personalizationOption) => {
        // Called for every applied personalization
    },
})

# Timeout

In some cases, personalization is critical in terms of execution time.

For example, if there are two landing page options and an intermediate page with personalization that redirects the user to the appropriate landing page option based on the selected personalization result.

In this case, if:

  1. The request to fetch personalizations from the server "fails"
  2. The request to fetch personalizations from the server "did not have time to execute"

Within the specified timeout in the handler, another special handler onTimeoutExceeded is called.

Example:

// Show the best review for the product
impact('onPersonalization', {
personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',

    // Main personalization handler
    handler: (personalization, personalizationOption) => {
        // ...
    },

    // how many milliseconds to wait for personalizations to be applied
    timeout: 2000

    // called if the subscription does not apply within the specified time
    onTimeoutExceeded: (personalization, personalizationOption) => {
      // ...
    },
})

# Multiple subscriptions personalizationId + personalizationOptionId with timeout

If there are multiple subscriptions to different variants of the same personalization, and each subscription has a specified execution timeout, then upon successful execution of any of these subscriptions, the timeout will NOT be triggered in the others.

Example:

// Show the best review for the product -> No
impact('onPersonalization', {
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: 'fb8211d7-5aa3-40df-a3ed-54a67d72f850',

    handler: (personalization, personalizationOption) => {
    },

    timeout: 1000

    onTimeoutExceeded: (personalization, personalizationOption) => {
        // Not triggered if Show the best review for the product -> Yes
        // Successfully "executed"
    },

})

// Show the best review for the product -> Yes
impact('onPersonalization', {
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: '10337247-6fa7-4ac3-92ed-bb2506d79ffa',

    handler: (personalization, personalizationOption) => {
    },

    timeout: 1000

    onTimeoutExceeded: (personalization, personalizationOption) => {
        // Not triggered if Show the best review for the product -> No
        // Successfully executed
    },

})

# Default Timeout

By default, the timeout is set to 5 seconds. Therefore, if a subscription has onTimeoutExceeded but no specified timeout, the value of 5 seconds will be used.

Example:

impact('onPersonalization', {
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: '10337247-6fa7-4ac3-92ed-bb2506d79ffa',

    handler: (personalization, personalizationOption) => {
    },

    onTimeoutExceeded: (personalization, personalizationOption) => {
        // Will be triggered after 5 seconds if the conditions for triggering the timeout are met
    },

})

# Subscription method returns a Promise

The onPersonalization method returns a Promise (opens new window).

This means you can use await to synchronously wait for the personalization result.

When using promises, wait for the script to be ready - window.SustainImpact.ready == true

This also allows you to perform some action asynchronously immediately after successful or unsuccessful retrieval of personalization.

Example:

// Default pickup point selection during checkout
impact('onPersonalization', {
personalizationId: 'e9b3f461-0857-4b21-a6fc-1bdcbe429cd4',

    // called when personalization is applied
    handler: (personalization, personalizationOption) => {
      switch (personalizationOption.id){
        case ("a554d770-3ed3-4d07-be0b-98475e27eb44"):
          // Нет
          break;
        case ("cc30da87-338c-49a2-adcc-f2c3b47ee6ed"):
          // Да
          break;
        default:
          // ...
      }
    },
}).then(({personalizationId, personalizationOptionId}) => {
    // After successful execution of the subscription
}, ({personalizationId, personalizationOptionId}) => {
    // After unsuccessful execution of the subscription
    // personalizationId and personalizationOptionId may not be filled!
})

# Third Argument in handler - dataContext

Inside the handler function, you have access to the current context through the dataContext variable.

Example:

impact('onPersonalization', {
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: '10337247-6fa7-4ac3-92ed-bb2506d79ffa',
    handler: (personalization, personalizationOption, dataContext) => {
        
        if(dataContext){
            // Example of working with dataContext
            let cityName = dataContext.geo.city.nativeName;
            let population = dataContext.geo.city.population;
            let cityInfo = "You are in the city of " + cityName + ", which has a population of " + population + " people.";
            if (dataContext.geo.city.isRegionCapital) {
                cityInfo += "This city is the capital of the region.";
            }
            if (dataContext.geo.city.isCountryCapital) {
                cityInfo += "This city is the capital of the country.";
            }

            // The variable cityInfo will contain the string
            // You are in the city of Moscow, which has a population of 12,615,078 people. This city is the capital of the country.
        }
    },

})

# List of available fields in the data variable

Within the handler function, using the dataContext variable, you have access to the following context data:

  • geo.city.nativeName: City name in the native language.
  • geo.city.englishName: City name in English.
  • geo.city.russianName: City name in Russian.
  • geo.city.isRegionCapital: Indicates if the city is a regional capital.
  • geo.city.isCountryCapital: Indicates if the city is a country's capital.
  • geo.city.population: City population.
  • geo.region.nativeName: Region name in the native language.
  • geo.region.englishName: Region name in English.
  • geo.region.russianName: Region name in Russian.
  • geo.region.population: Region population.
  • geo.country.nativeName: Country name in the native language.
  • geo.country.englishName: Country name in English.
  • geo.country.russianName: Country name in Russian.
  • geo.country.population: Country population.
  • page.schemaOrg.brandName: Brand name in Schema.org format.
  • page.schemaOrg.offer.price: Offer price in Schema.org format.
  • page.schemaOrg.offer.availability: Offer availability in Schema.org format.
  • page.schemaOrg.aggregateRating.ratingValue: Rating value in Schema.org format.
  • page.schemaOrg.aggregateRating.reviewCount: Number of reviews in Schema.org format.
  • weather.temp: Current temperature.
  • weather.tempFeelsLike: Feels like temperature.
  • weather.humidity: Humidity.

# Deprecated subscription method

For backward compatibility, the deprecated method window.SustainImpact.onPersonalization is also available, which fully replicates the functionality described above.

Example:

window.SustainImpact.onPersonalization({
    personalizationId: '6fbf1dd7-3771-45ce-a7a0-bd08187549ee',
    personalizationOptionId: '10337247-6fa7-4ac3-92ed-bb2506d79ffa',
    handler: (personalization, personalizationOption, dataContext) => {
        // Your handler code
    },

})