# `MJML` email templating This directory contains `MJML` templates for emails. `MJML` is a markup language designed to reduce the pain of coding responsive email templates. Component-based development features in `MJML` improve code quality and reusability. > [!TIP] > `MJML` stands for MailJet Markup Language. ## Implementation considerations `MJML` templates are compiled into `HTML`, and those outputs are then consumed by Handlebars to render the final email for delivery. It builds on top of our existing infrastructure and means we can continue to use the double brace (`{{}}`) syntax within `MJML`, since Handlebars will assign values to those `{{variables}}`. To do this, there is an added step where we compile `*.mjml` to `*.html.hbs`. `*.html.hbs` is the format we use so the Handlebars service can apply the variables. This build pipeline process is in progress and may need to be manually done at times. ### `*.txt.hbs` There is no change to how we create the `txt.hbs`. MJML does not impact how we create these artifacts. ## Building `MJML` files ```shell npm ci # Build *.html to ./out directory npm run build # To build on changes to *.mjml and *.js files, new files will not be tracked, you will need to run again npm run build:watch # Build *.html.hbs to ./out directory npm run build:hbs # Build minified *.html.hbs to ./out directory npm run build:minify # apply prettier formatting npm run prettier ``` ## Development process `MJML` supports components and you can create your own components by adding them to `.mjmlconfig`. Components are simple JavaScript that return `MJML` markup based on the attributes assigned, see components/mj-bw-hero.js. The markup is not a proper object, but contained in a string. When using `MJML` templating you can use the above [commands](#building-mjml-files) to compile the template and view it in a web browser. Not all `MJML` tags have the same attributes, it is highly recommended to review the documentation on the official MJML website to understand the usages of each of the tags. ### Developing the mail template 1. Create `cool-email.mjml` in appropriate team directory. 2. Run `npm run build:watch`. 3. View compiled `HTML` output in a web browser. 4. Iterate through your development. While running `build:watch` you should be able to refresh the browser page after the `mjml/js` recompile to see the changes. ### Testing the mail template with `IMailer` After the email is developed in the [initial step](#developing-the-mail-template), we need to make sure that the email `{{variables}}` are populated properly by Handlebars. We can do this by running it through an `IMailer` implementation. The `IMailer`, documented [here](../../Platform/Mail/README.md#step-3-create-handlebars-templates), requires that the ViewModel, the `.html.hbs` `MJML` build artifact, and `.text.hbs` files be in the same directory. 1. Run `npm run build:hbs`. 2. Copy built `*.html.hbs` files from the build directory to the directory that the `IMailer` expects. All files in the `Core/MailTemplates/Mjml/out` directory should be copied to the `/src/Core/MailTemplates/Mjml` directory, ensuring that the files are in the same directory as the corresponding ViewModels. If a shared component is modified it is important to copy and overwrite all files in that directory to capture changes in the `*.html.hbs` files. 3. Run code that will send the email. The minified `html.hbs` artifacts are deliverables and must be placed into the correct `/src/Core/MailTemplates/Mjml` directories in order to be used by `IMailer` implementations, see step 2 above. ### Testing the mail template with `IMailService` > [!WARNING] > The `IMailService` has been deprecated. The [IMailer](#testing-the-mail-template-with-imailer) should be used instead. After the email is developed from the [initial step](#developing-the-mail-template), make sure the email `{{variables}}` are populated properly by running it through an `IMailService` implementation. 1. Run `npm run build:hbs` 2. Copy built `*.html.hbs` files from the build directory to a location the mail service can consume them. 1. All files in the `Core/MailTemplates/Mjml/out` directory should be copied to the `src/Core/MailTemplates/Handlebars/MJML` directory. If a shared component is modified it is important to copy and overwrite all files in that directory to capture changes in the `*.html.hbs`. 3. Run code that will send the email. The minified `html.hbs` artifacts are deliverables and must be placed into the correct `src/Core/MailTemplates/Handlebars/` directories in order to be used by `IMailService` implementations, see 2.1 above. ### Custom tags There is currently a `mj-bw-hero` tag you can use within your `*.mjml` templates. This is a good example of how to create a component that takes in attribute values allowing us to be more DRY in our development of emails. Since the attribute's input is a string we are able to define whatever we need into the component, in this case `mj-bw-hero`. In order to view the custom component you have written you will need to include it in the `.mjmlconfig` and reference it in a `.mjml` template file. ```html ``` Attributes in custom components are defined by the developer. They can be required or optional depending on implementation. See the official `MJML` [documentation](https://documentation.mjml.io/#components) for more information. ```js static allowedAttributes = { "img-src": "string", // REQUIRED: Source for the image displayed in the right-hand side of the blue header area title: "string", // REQUIRED: large text stating primary purpose of the email "button-text": "string", // OPTIONAL: text to display in the button "button-url": "string", // OPTIONAL: URL to navigate to when the button is clicked "sub-title": "string", // OPTIONAL: smaller text providing additional context for the title }; static defaultAttributes = {}; ``` Custom components, such as `mj-bw-hero`, must be defined in the `.mjmlconfig` in order for them to be compiled and rendered properly in the templates. ```json { "packages": ["components/mj-bw-hero"] } ``` ### `mj-include` You are also able to reference other more static `MJML` templates in your `MJML` file simply by referencing the file within the `MJML` template. ```html ``` #### `head.mjml` Currently we include the `head.mjml` file in all `MJML` templates as it contains shared styling and formatting that ensures consistency across all email implementations. In the future we may deviate from this practice to support different layouts. At that time we will modify the docs with direction.