MCP Open Time Email
Personalise your SFMC emails with the next best action recommendations. In real-time. Even after delivery. Magic.
What is Open Time Email Campaign
In short, Open Time Email Campaign in Marketing Cloud Personalization (Interaction Studio) is a solution for providing always up-to-date, personalised recommendations to your customers.
The general rule of email communication is that the moment you send the email, you lose control and cannot make any changes (just like with paper mail). There are, however, two exceptions:
- Links - you cannot change the URL used in the email, but (if you use wrapper/tracking links), you can change the final target URL. In Marketing Cloud, it is done via updating Job Links.
- Images - you cannot change the URL of the image used in the email, but you can change the image that is hosted behind that URL. It is precisely what the Open Time Email feature in MC Personalization is all about.
It consists of two parts. In Open Time Email Template, you define the format of the recommendation (dimensions, content, personalisations). Then in Open Time Email Campaign, you use that template with selected recommendations logic (Einstein Recipe) to automatically generate HTML that will render the ever-updating Next Best Action in your emails. But...
There are three caveats:
- Open Time Email recommendation is an image (that's the only way to enable updating after sending), which means that the inboxes that hide images by default will not display the recommendations.
- Some inboxes cache images. In this case, the customer might not see the updates to the recommendation, or they might be delayed.
- Apple Mail Privacy blocks the magic behind the Open Time Email and falls back to a static recommendation generated on send time.
With that out of the way, let's dive into what is officially possible and what is possible with MCP Open Time Email.
Open Time Email Template
To create an Open Time Email Campaign, first, you need to code the structure of the recommendation with HTML, CSS and personalisations. MC Personalization will use it to generate the image on the fly.
Good thing: because this HTML will be used for image generation and will not be available in the final Campaign, you are not limited to markup supported by Email clients.
Bad thing: because MCP will use this HTML for image generation, you are not able to leverage custom fonts - there are only 17 built-in western and 4 non-western fonts to choose from.
To start building the template, you need to provide the dimensions of the final image. It is super important because once you select and save those, you won't be able to edit them. Those dimensions are for the recommendation image only - you will be configuring the whole campaign dimensions separately in the Open Time Email Campaign configuration, so leave some pixels for whitespace.
Open Time Email code editor isn't very robust and likes to crash completly if there is a "wrong" set of characters for it to preview. I recommend writing code outside and just copy-pasting it into OTE for validation to not lose the work progress.
Basic OTE Template
Building the template requires basic HTML & CSS knowledge and leveraging the built-in Insert {Dynamic}
personalisation option to connect the code boilerplate with catalog data.
The most popular boilerplate will leverage product image, name and price plus a call to action. The basic approach could look like this:
<div style="padding:10px; text-align:center; font-family: Arial;">
<img src="${item.imageUrl}" style="max-width:180px; max-height:150px;">
<div style="text-align:left;">
${item.name}
</div>
<div style="margin-top:5px; text-align:left;">
$currency.format(${item.price})
</div>
<div style="position:absolute; left: 0px; right: 0px; bottom: 10px;">
<button style="border:none; padding: 10px; background:#1D73C9; color:#ffffff; font-size:16px;">
Buy now »
</button>
</div>
</div>
You can see a whole section wrapped in a div that sets the font and padding for proper spacing and three key elements within it:
img
for displaying how the product looks (withmax-width
&max-height
to control non-standard images)div
s with name and price personalisation strings (we will talk about the format difference in a moment)div
with a button (we don't need any link because the whole final image will be a link)
That's it. You can add more personalisation strings - including those using attributes defined on the catalog object. You can also write nicer HTML with a separate CSS style section (and either way - be sure to test the code a lot with edge cases from your catalog). But in the end - it will be just a boilerplate for an image you can preview directly in the editor.
It's okay, but you can quickly come up with the "what if..." scenarios (f.e. showing both current price and listPrice only if there is a promotion or displaying a placeholder if the image is broken) that would require a bit more control over the template. Thankfully, some undocumented features can solve those needs.
Attributes vs Custom Fields
When you select Insert {Dynamic}
, you will find both Attributes and Custom Field options at the bottom of the picklist.
That differentiation is a mess, as depending on the product value, you can use one, two or all three:
// ID
${item.id} // Returns Product ID
${item.attributes.published.value} // Error generating preview image: Unfulfilled variable
${item.custom.customDatetime} // Error generating preview image: Unfulfilled variable
// Name
${item.name} // Returns Product Name
${item.attributes.name.value} // Returns Product Name
${item.custom.name} // Returns Product Name
// Published Date
${item.published} // Returns Product Published Date
${item.attributes.published.value} // Returns Product Published Date
${item.custom.published} // Error generating preview image: Unfulfilled variable
To make it even more chaotic, the formatting of the output might be different:
${item.published} // Wed Nov 30 23:00:00 UTC 2022
${item.attributes.published.value} // 1669849200000
${item.attributes.customDatetime.value} // 1672182000000
${item.custom.customDatetime} // 2022-12-27T23:00:00.000Z
While the Insert {Dynamic}
will suggest you should use ${item.attributes.xxxx.value}
notation for custom attributes, this is not always true.
If you try this style for an Object or MultiString (Array) attribute, it will instead crash with: "Error generating preview image: Failed to render DMC for template [email]".
Marketing Cloud Personalization will display the value correctly if you drop the .value
suffix from the personalisation string: ${item.attributes.MultiStringAttribute}
.
Formatting values
The only working formatting option I was able to find are the currency ones:
$currency.format(${item.price}) // 49,00
$currency.formatNumber(${item.price},0) // 49
$currency.formatNumber(decimal, scale)
lets you provide second argument to define the number of decimal places in the formatted value.
If scale is smaller than original value - it will be rounded. If scale is higher - zeroes will be added.
// For item.price === 39,99
$currency.formatNumber(${item.price},0) // 40
$currency.formatNumber(${item.price},1) // 40,0
$currency.formatNumber(${item.price},2) // 39,99
$currency.formatNumber(${item.price},3) // 39,990
Currency formatting isn't limited to built-in price and listPrice. You can leverage it also for any custom integer or decimal attribute:
$currency.format(${item.attributes.decimalAttribute.value})
There is also a pseudo-formatting hack for datetime fields, but it is lacking:
${item.published} // Wed Nov 30 23:00:00 UTC 2022
${item.attributes.published.value} // 1669849200000
${item.attributes.customDatetime.value} // 1672182000000
${item.custom.customDatetime} // 2022-12-27T23:00:00.000Z
As you can see, the built-in datetime attributes return a semi-readable date, item.attributes
is outputting epoch timestamp (for both built-in and custom attributes) and the item.custom
returns the full ISO format. Neither is remotely useful for Open Time Email...
Applying the .getDate()
or .substring()
methods leads to an error.
Fallback for null values
The first problem you may encounter with the official implementation is crashing on null
values. If you want to leverage a product attribute for personalisation (for example, the brand with ${item.attributes.brand.value}
), products missing this value will error out with "Error generating preview image: Unfulfilled variable".
Thankfully, there is a solution for it - a #field()
function:
#field(${item.attributes.brand.value}, 'Fallback Value')
#field()
functions expect two parameters:
- The personalisation you want to use
- The fallback value. It can be either a string or another variable. However, you cannot nest the
#field()
function as a parameter of the#field()
function, so if you decide to use another variable, be sure it is filled in for all the products.
#field(${item.attributes.customPromotionalImageUrl.value}, ${item.imageUrl})
In most cases, the most useful way to use #field()
will be with fallback to an empty string, as it will just hide the personalisation from the image while protecting against the error:
#field(${item.attributes.brand.value}, '')
If you want to add a fallback for formatted currency value, you need to wrap it around the formatting function:
#field($currency.format(${item.attributes.decimalAttribute.value}), '')
Once again, the MultiString attributes are problematic, as they are not considered null
when empty and will show []
instead (ignoring the fallback).
Empty Object fields fall back correctly when they are truly empty. However, once you fill them with a JSON Object, UI will no longer let you truly empty that field, forcing you to write at least {}
and leading to the same issue as with MultiString.
Conditional content with if/else
Sometimes you need more than just a fallback for a null
value. When it is not about the null
or there are more conditions to check, if
/else
is the best. Thankfully, it also can be leveraged in OTE:
<div style="text-align:left;">
$currency.format(${item.price})
<!-- #if(${item.listPrice} > ${item.price}) -->
<span style="font-size: 14px; text-decoration: line-through;">
$currency.format(${item.listPrice})
</span>
<!-- #end -->
</div>
As you can see, for the if
block you need to use HTML comment notation with #if()
and #end
statements. The #if()
accepts parameters that return a boolean, just as in JavaScript if
.
The Object and MultiString attributes do not work correctly with if
/else
statements.
If you test Object or MultiString with basic #if(${item.attributes.attributeName})
, it will be treated as true even for an empty field. And even though ${item.attributes.MultiStringAttribute}
will output []
doing #if(${item.attributes.attributeName} == [])
or #if(${item.attributes.attributeName} == '[]')
will return false.
You can add more condition-based paths with #else
:
<!-- #if (${item.imageUrl} && ${item.imageUrl.endsWith('.jpg')}) -->
<img src="${item.imageUrl}" style="max-width:180px; max-height:150px;">
<!-- #else -->
<img src="https://mateuszdabrowski.pl/placeholder-image.jpg" style="max-width:180px; max-height:150px;">
<!-- #end -->
While the if
/else
block in the Open Time Email Template works similarly to what you may know from JavaScript or SSJS, it has one huge disadvantage - it is evaluating code in all paths.
It means that even if MC Personalization is not using a path for the final image generation, the code written there can still error out the whole Campaign. Example:
<!-- #if(${item.attributes.brand.value} && !${item.attributes.categoryName.value}) -->
${item.attributes.brand.value}
<!-- #end -->
Even with checking whether there is a value behind the Brand attribute and only then personalising it, a template with such code would still error out for products without a brand.
Always use fallbacks along with if
/else
blocks:
<!-- #if(${item.attributes.brand.value} && !${item.attributes.categoryName.value}) -->
#field(${item.attributes.brand.value}, '')
<!-- #end -->
Related Catalog Object support
Unfortunatelly support for Related Catalog Objects is heavily limited in Open Time Email.
You can leverage ${item.getTagValue("YourRelatedCatalogObject")}
call to return the ID of the Item in RCO but you won't be able to get any other information (like name or URL of that Item). Additionally, if you have one-to-many relationship, you will still get only one ID for the first related Item. Due to those limitations, I recommend using it only if you have RCO with one-to-one relationship and readable IDs (for example, RED
as an ID in the Color RCO).
The workaround for other use cases is to pass the crucial RCO data as custom attributes on the Product (for example,productColorName
attribute if you are using numeric IDs for colors in the RCO). It's not perfect as it duplicates the data and eats up attributes, but it is currently the only way you will be able to leverage that information in OTE.
Example dynamic OTE Template
<div style="padding:10px; text-align:center; font-family:Arial; font-size:16px;">
<!-- #if (${item.imageUrl} && ${item.imageUrl.endsWith('.jpg')}) -->
<img src="${item.imageUrl}" style="max-width:180px; max-height:150px;">
<!-- #else -->
<img src="https://mateuszdabrowski.pl/placeholder-image.jpg" style="max-width:180px; max-height:150px;">
<!-- #end -->
<div style="text-align:left;">
<span style="font-size:14px; color:#212121;">
#field(${item.attributes.categoryName.value}, '')
</span>
<!-- #if (!${item.attributes.categoryName.value}) -->
<br />
<!-- #end -->
<span style="margin-top:5px; overflow:hidden; text-overflow:ellipsis; display:-webkit-box; -webkit-line-clamp:2; -webkit-box-orient:vertical;">
#field(${item.attributes.brand.value}, '') $item.name
</span>
</div>
<div style="margin-top:5px; text-align:left;">
$currency.format(${item.price})
<!-- #if(${item.listPrice} > ${item.price}) -->
<span style="font-size:14px; color:#212121;text-decoration:line-through;">
$currency.format(${item.listPrice})
</span>
<!-- #end -->
</div>
<div style="position:absolute; left:0px; right:0px; bottom:10px;">
<button style="border:none; padding:10px; background-color:#1D73C9; color:#ffffff;">
Buy now »
</button>
</div>
</div>