Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/sighmon/mjml-rails

MJML + ERb/Haml/Slim view template
https://github.com/sighmon/mjml-rails

Last synced: 2 months ago
JSON representation

MJML + ERb/Haml/Slim view template

Awesome Lists containing this project

README

        

# MJML-Rails

[![Build Status](https://github.com/sighmon/mjml-rails/workflows/test/badge.svg?branch=master)](https://github.com/sighmon/mjml-rails/actions?query=workflow%3Atest+branch%3Amaster) [![Gem Version](https://badge.fury.io/rb/mjml-rails.svg)](https://badge.fury.io/rb/mjml-rails)

**MJML-Rails** allows you to render HTML emails from an [MJML](https://mjml.io) template.

**Note**: As of MJML 4.3.0 you can no longer use `` directly inside an ``, wrap it in ``.

An example template might look like:

```erb


Hello World




Hello World
<%= render partial: "info", formats: [:html] %>


```

And the partial:

```erb

This is <%= @user.username %>
```

* Notice you can use ERB and partials inside the template.

Your `user_mailer.rb` might look like this:

```ruby
# ./app/mailers/user_mailer.rb
class UserMailer < ActionMailer::Base
def user_signup_confirmation
mail(to: "[email protected]", from: "[email protected]") do |format|
format.text
format.mjml
end
end
end
```

## Example application

* [MJML with Rails 6](https://github.com/dyanagi/example_mjml_rails): Renders HTML emails with MJML layout, view, and partial.

## Installation

### Using MJML NPM package

Add it to your Gemfile.

```ruby
gem 'mjml-rails'
```

Run the following command to install it:

```console
bundle install
```

Add the MJML parser to your project with your favourite package manager:

```console
# with npm
npm install mjml

# or install it globally (The best way for Rails 7.x.x with importmaps)
npm install -g mjml

# with yarn
yarn add mjml

# with bun
bun add mjml
```

MJML-Rails falls back to a global installation of MJML but it is strongly recommended to add MJML directly to your project.

You'll need at least Node.js version 6 for MJML to function properly.

### Using MRML with included binaries

If for some reason you can't or don't want to run JS code in your production environment, you can use [MRML](https://github.com/hardpixel/mrml-ruby). It ships with already compiled binaries for Rust implementation of MJML so it has no external dependencies.

Add `mjml-rails` and `mrml` to your Gemfile.

```ruby
gem 'mjml-rails'
gem 'mrml'
```

Run the following command to install it:

```console
bundle install
```

Set `use_mrml` option to `true` in your initializer:

```ruby
# config/initializers/mjml.rb
Mjml.setup do |config|
config.use_mrml = true
end
```

**Note**: MRML does not fully support all MJML functionalities, see [Missing implementations](https://github.com/jdrouet/mrml#missing-implementations)

## Configuration

MJML-Rails has the following settings with defaults:

- `template_language: :erb`

ERB can be used inside MJML templates by default. Possible values are all template languages that you have installed, e.g. `:haml` or `:slim`.

**Note:** If you’re using Haml/Slim layouts, please don’t put `` in comments in your partial. Read more at [#34](https://github.com/sighmon/mjml-rails/issues/34).

- `raise_render_exception: true`

Exceptions will be raised and passed to your application by default.

Beware that setting it to `false` leads to an empty html email when an exception is raised, so only set this to `false` if you do not rely on html (e.g. you have a text fallback for your emails).

- `minify: false`

- `beautify: true`

- `validation_level: "strict"`

MJML-Rails will raise an exception on any template error by default.

If set to `soft`, MJML will render invalid templates and ignore invalid parts. This means in case of an invalid template those invalid parts will be missing from the output.

See [MJML validation documentation](https://github.com/mjmlio/mjml/tree/master/packages/mjml-validator#validating-mjml) for all possible values.

- `mjml_binary: nil`

This can be used to specify the path to a custom MJML binary if it is not detected automatically (shouldn't be needed).

- `mjml_binary_version_supported: "4."`

MJML-Rails checks the version of the MJML binary and fails if it does not start with this version, e.g. if an old version is installed by accident.

- `use_mrml: false`
Enabling this will allow you to use Rust implementation of MJML via the `mrml` gem. It comes with prebuilt binaries instead of having to install MJML along with Node. When enabled the options `mjml_binary_version_supported`, `mjml_binary`, `minify`, `beautify` and `validation_level` are ignored.

- `fonts`
By default, MJML-Rails uses MJML default fonts, but enables you to override it.
Example : `config.fonts = { Raleway: 'https://fonts.googleapis.com/css?family=Raleway }`

```ruby
# config/initializers/mjml.rb
Mjml.setup do |config|
# Use :haml as a template language
config.template_language = :haml

# Ignore errors silently
config.raise_render_exception = false

# Optimize the size of your emails
config.beautify = false
config.minify = true

# Render MJML templates with errors
config.validation_level = "soft"

# Use MRML instead of MJML, false by default
config.use_mrml = false

# Use custom MJML binary with custom version
config.mjml_binary = "/path/to/custom/mjml"
config.mjml_binary_version_supported = "3.3.5"

# Use default system fonts instead of google fonts
config.fonts = {}
end
```

### MJML v3.x & v4.x support

Version 4.x of this gem brings support for MJML 4.x

Version 2.3.x and 2.4.x of this gem brings support for MJML 3.x

If you'd rather still stick with MJML 2.x then lock the mjml-rails gem:

```ruby
gem 'mjml-rails', '2.2.0'
```

For MJML 3.x lock the mjml-rails gem:

```ruby
gem 'mjml-rails', '2.4.3'
```

And then to install MJML 3.x

```console
npm install -g [email protected]
```

### How to guides

[Kitty Giraudel](https://twitter.com/KittyGiraudel) wrote a post on [using MJML in Rails](http://dev.edenspiekermann.com/2016/06/02/using-mjml-in-rails/).

## Using Email Layouts

*Note*: [Aleksandrs Ļedovskis](https://github.com/aleksandrs-ledovskis) kindly updated the gem for better Rails Email Layouts support - it should be a non-breaking change, but check the updated file naming below if you experience problems.

Mailer:
```ruby
# mailers/my_mailer.rb
class MyMailer < ActionMailer::Base
layout "default"

def foo_bar(user)
@recipient = user

mail(to: user.email, from: "[email protected]") do |format|
format.html # This will look for "default.html.erb" and then "default.html.mjml"
end
end
end
```

Note: If `default.html.erb` exists, email will be rendered as ERB, and MJML tags will not be compiled.

Email layout:
```html


<%= yield %>

```

Email view:
```html

<%= render partial: "to" %>



Something foo regarding bar!

```

Email partial:
```html



Hello <%= @recipient.name %>,

```

## Sending Devise user emails

If you use [Devise](https://github.com/plataformatec/devise) for user authentication and want to send user emails with MJML templates, here's how to override the [devise mailer](https://github.com/plataformatec/devise/blob/master/app/mailers/devise/mailer.rb):
```ruby
# app/mailers/devise_mailer.rb
class DeviseMailer < Devise::Mailer
def reset_password_instructions(record, token, opts={})
@token = token
@resource = record
# Custom logic to send the email with MJML
mail(
template_path: 'devise/mailer',
from: "[email protected]",
to: record.email,
subject: "Custom subject"
) do |format|
format.text
format.mjml
end
end
end
```

Now tell devise to user your mailer in `config/initializers/devise.rb` by setting `config.mailer = 'DeviseMailer'` or whatever name you called yours.

And then your MJML template goes here: `app/views/devise/mailer/reset_password_instructions.mjml`

Devise also have [more instructions](https://github.com/plataformatec/devise/wiki/How-To:-Use-custom-mailer) if you need them.

## Deploying with Heroku

To deploy with [Heroku](https://heroku.com) you'll need to setup [multiple buildpacks](https://devcenter.heroku.com/articles/using-multiple-buildpacks-for-an-app) so that Heroku first builds Node for MJML and then the Ruby environment for your app.

Once you've installed the [Heroku Toolbelt](https://toolbelt.heroku.com/) you can setup the buildpacks from the commandline:

`$ heroku buildpacks:set heroku/ruby`

And then add the Node buildpack to index 1 so it's run first:

`$ heroku buildpacks:add --index 1 heroku/nodejs`

Check that's all setup by running:

`$ heroku buildpacks`

Next you'll need to setup a `package.json` file in the root, something like this:

```json
{
"name": "your-site",
"version": "1.0.0",
"description": "Now with MJML email templates!",
"main": "index.js",
"directories": {
"doc": "doc",
"test": "test"
},
"dependencies": {
"mjml": "^4.0.0"
},
"repository": {
"type": "git",
"url": "git+https://github.com/your-repo/your-site.git"
},
"keywords": [
"mailer"
],
"author": "Your Name",
"license": "ISC",
"bugs": {
"url": "https://github.com/sighmon/mjml-rails/issues"
},
"homepage": "https://github.com/sighmon/mjml-rails"
}
```

Then `$ git push heroku master` and it should Just WorkTM.

## Bug reports

If you discover any bugs, feel free to create an issue on GitHub. Please add as much information as possible to help us fixing the possible bug. We also encourage you to help even more by forking and sending us a pull request.

[github.com/sighmon/mjml-rails/issues](https://github.com/sighmon/mjml-rails/issues)

## Maintainers

* Simon Loffler [github.com/sighmon](https://github.com/sighmon)
* Steven Pickles [github.com/thatpixguy](https://github.com/thatpixguy)
* [The Rails community](https://github.com/sighmon/mjml-rails/pulls?q=is%3Apr+is%3Aclosed). :-)

## Other similar gems

* [srghma/mjml-premailer](https://github.com/srghma/mjml-premailer)
* [kolybasov/mjml-ruby/](https://github.com/kolybasov/mjml-ruby/)

## License

MIT License. Copyright 2018 Simon Loffler. [sighmon.com](http://sighmon.com)

Lovingly built on [github.com/plataformatec/markerb](https://github.com/plataformatec/markerb)