MJML-Rails allows you to render HTML e-mails from an MJML template.
Note: As of MJML 4.3.0 you can no longer use <mj-text>
directly inside an <mj-body>
, wrap it in <mj-section><mj-column>
.
An example template might look like:
<!-- ./app/views/user_mailer/email.html.mjml -->
<mjml>
<mj-head>
<mj-preview>Hello World</mj-preview>
</mj-head>
<mj-body>
<mj-section>
<mj-column>
<mj-text>Hello World</mj-text>
<%= render partial: "info" %>
</mj-column>
</mj-section>
</mj-body>
</mjml>
And the partial:
<!-- ./app/views/user_mailer/_info.html.erb -->
<mj-text>This is <%= @user.username %></mj-text>
- Notice you can use ERB and partials inside the template.
Your user_mailer.rb
might look like this:
# ./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
- MJML with Rails 6: Renders HTML emails with MJML layout, view, and partial.
Add it to your Gemfile.
gem 'mjml-rails'
Run the following command to install it:
bundle install
Install the MJML parser (optional -g to install it globally):
npm install -g mjml
Note that you'll need at least Node.js version 6 for MJML to function properly.
If you're using :haml
or any other Rails template language, create an initializer to set it up:
# config/initializers/mjml.rb
Mjml.setup do |config|
config.template_language = :erb # :erb (default), :slim, :haml, or any other you are using
end
Note: If you’re using Haml/Slim layouts, please don’t put <mjml>
in comments in your partial. Read more: #34.
If there are configurations you'd like change:
- render errors: defaults to
true
(errors raised) - minify: defaults to
false
(not minified) - beautify: defaults to
true
(beautified)
Mjml.setup do |config|
# set to `false` to ignore errors silently
config.raise_render_exception = true
# optimize the size of your email
config.beautify = false
config.minify = true
end
If you’d like to specify a different MJML binary to run other than 4.
:
Mjml.setup do |config|
config.mjml_binary_version_supported = "3.3.5"
end
# If you set a different MJML binary version, you need to re-discover the binary
Mjml::BIN = Mjml.discover_mjml_bin
Note: If you set a different MJML binary you’ll see warnings in your logs of already initialized constant Mjml::BIN
. Read more: #39
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:
gem 'mjml-rails', '2.2.0'
For MJML 3.x lock the mjml-rails gem:
gem 'mjml-rails', '2.4.3'
And then to install MJML 3.x
npm install -g [email protected]
Hugo Giraudel wrote a post on using MJML in Rails.
Note: Aleksandrs Ļedovskis 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:
# 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:
<!-- views/layouts/default.html.mjml -->
<mjml>
<mj-body>
<%= yield %>
</mj-body>
</mjml>
Email view:
<!-- views/my_mailer/foo_bar.html.mjml (or foo_bar.html.erb) -->
<%= render partial: "to" %>
<mj-section>
<mj-column>
<mj-text>
Something foo regarding bar!
</mj-text>
</mj-column>
</mj-section>
Email partial:
<!-- views/my_mailer/_to.html.mjml (or _to.html.erb) -->
<mj-section>
<mj-column>
<mj-text>
Hello <%= @recipient.name %>,
</mj-text>
</mj-column>
</mj-section>
If you use Devise for user authentication and want to send user emails with MJML templates, here's how to override the devise mailer:
# 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 if you need them.
To deploy with Heroku you'll need to setup multiple buildpacks so that Heroku first builds Node for MJML and then the Ruby environment for your app.
Once you've installed the Heroku Toolbelt 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:
{
"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.
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
- Simon Loffler github.com/sighmon
- Steven Pickles github.com/thatpixguy
- The Rails community. :-)
MIT License. Copyright 2018 Simon Loffler. sighmon.com
Lovingly built on github.com/plataformatec/markerb