DEV Community

Cover image for How to set up Ruby on Rails 6 and TailwindCSS 1.1.4
Andrew Mason
Andrew Mason

Posted on • Edited on

How to set up Ruby on Rails 6 and TailwindCSS 1.1.4

Tutorial

For the purpose of this tutorial, we will assume you have Ruby and the Rails gem installed. Please visit the Getting Started with Rails Guide if you do not.

I will also be working through the creation of code in this repo. Please use the repo as a resource to help you in case you get stuck or open issues if it's broken!!

I created this demo using the following:

  • Ruby 2.7
  • Node 13.7.0
  • Rails 6.0.2.1
  • Webpacker 4.2.2
  • TailwindCSS 1.1.4
  • psql (PostgreSQL) 12.1

Create a new Rails project

rails new tailwind_css_rails_demo -d postgresql
cd tailwind_css_rails_demo
rails db:create
Enter fullscreen mode Exit fullscreen mode

This will create a new Ruby on Rails project with PostgreSQL configured for you. You can omit the -d postgresql flag if you would prefer to use SQLite or MySQL.

Running Rails and Webpack

I prefer to run the Rails server in one command line tab and webpack-dev-server in another since it's much faster. In development, Rails can tell whether the webpack-dev-server has compiled your packs and will compile them inline if it has not been done.

Let's go ahead and get the app running:

# Terminal tab 1
rails s
Enter fullscreen mode Exit fullscreen mode

And webpack-dev-server:

# Terminal tab 2
./bin/webpack-dev-server
Enter fullscreen mode Exit fullscreen mode

You should now see Rails welcome page if you navigate to localhost:3000 in your browser.

Rails default information page

Generate a new resource

I personally like to see more than one record in the database for tutorials to make the app seem more "real". You can skip this part if you are not interested in adding some seed data, and would rather create a Home controller or something similar like in the previous tutorial.

If you would like some records in your database, lets scaffold out a small resource:

rails generate scaffold Post title:string content:text
Enter fullscreen mode Exit fullscreen mode

Rails will then generate several files for us, but we will only focus on a few.

Use the following command to run the generated migration:

rails db:migrate
Enter fullscreen mode Exit fullscreen mode

Now, lets add some seed data. Open db/seeds.rb and add the following:

# db/seeds.rb

10.times do |n|
  Post.create!(title:  "Post title - ##{n}", content: "This is the content for the #{n.ordinalize} post.")
end
Enter fullscreen mode Exit fullscreen mode

It's not important for this tutorial for you to fully grok that code, but I would be happy to explain it in more detail if you reach out or let me know in the comments. The TL;DR is that we now have 10 unique Post records in our database.

The last thing we need to do before getting to the fun part is to update our config/routes.rb file to make the root path for the app the index page for posts.

# config/routes.rb

Rails.application.routes.draw do
  resources :posts
  root to: "posts#index"
end
Enter fullscreen mode Exit fullscreen mode

Restart the Rails server, navigate to localhost:3000, and you should see a table with our random data, with links to other CRUD actions.

Install TailwindCSS

Now to the fun stuff.

Run the following command in your terminal to install TailwindCSS

yarn add tailwindcss
Enter fullscreen mode Exit fullscreen mode

Let's also add the Tailwind config file:

./node_modules/.bin/tailwind init
Enter fullscreen mode Exit fullscreen mode

This should create a tailwind.config.js file at the root of your project. This file can be used to customize the TailwindCSS defaults, add plugins, and more. You can learn more about this from Tailwind's docs

We also need to update our PostCSS config that comes default with Rails 6 with two new requires:

require('tailwindcss'),
require('autoprefixer'),
Enter fullscreen mode Exit fullscreen mode

I have been told the best order for these requires is as I have them below, but I think just adding them to the top of your PostCSS config will work for the majority of people:

// postcss.config.js

module.exports = {
  plugins: [
    require('autoprefixer'),
    require('postcss-import'),
    require('tailwindcss'),
    require('postcss-flexbugs-fixes'),
    require('postcss-preset-env')({
      autoprefixer: {
        flexbox: 'no-2009'
      },
      stage: 3
    })
  ]
}
Enter fullscreen mode Exit fullscreen mode

Configure Tailwind

There are a few ways you can do this but this is my personal preference.

Remove the assets folder, we won't be needing it since we will rely fully on Webpacker:

rm -rf app/assets
Enter fullscreen mode Exit fullscreen mode

Create a new stylesheet file:

touch app/javascript/src/application.scss
Enter fullscreen mode Exit fullscreen mode

Since we are using postcss-import and Webpack, the Tailwind docs instruct us to add the following to our stylesheet file:

// app/javascript/src/application.scss

@import "tailwindcss/base";
@import "tailwindcss/components";
@import "tailwindcss/utilities";
Enter fullscreen mode Exit fullscreen mode

We also need to add following line in app/javascript/packs/application.js:

import '../src/application.scss'
Enter fullscreen mode Exit fullscreen mode

The last step is to tell Rails to use our pack files. In app/views/layouts/application.html.erb, change:

<%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
<%= javascript_include_tag 'application', 'data-turbolinks-track': 'reload' %>
Enter fullscreen mode Exit fullscreen mode

to:

<%= stylesheet_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
<%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
Enter fullscreen mode Exit fullscreen mode

Restart the Rails server and webpack-dev-server and you should now see the following on localhost:3000:

tailwind home index

Tailwind should now be working so lets tweak our views to see some Tailwind goodness.

Update views to use TailwindCSS

In app/views/layouts/application.html.erb change:

<body>
  <%= yield %>
</body>
Enter fullscreen mode Exit fullscreen mode

to:

<body class="min-h-screen bg-gray-100">
  <div class="container mx-auto">
    <%= yield %>
  </div>
</body>
Enter fullscreen mode Exit fullscreen mode

and in app/views/posts/index.html.erb lets replace the scaffolded page with the following:

<p id="notice"><%= notice %></p>
<h1 class="font-semibold text-4xl text-gray-700 my-8">Posts</h1>
<div class="w-full xl:w-8/12 mb-12 xl:mb-0 px-4">
  <div class="relative flex flex-col min-w-0 break-words bg-white w-full mb-6 shadow-lg rounded">
    <table class="items-center w-full bg-transparent border-collapse">
      <thead>
        <tr>
          <th class="px-6 bg-gray-100 text-gray-600 align-middle border border-solid border-gray-200 py-3 text-xs uppercase border-l-0 border-r-0 whitespace-no-wrap font-semibold text-left">Title</th>
          <th class="px-6 bg-gray-100 text-gray-600 align-middle border border-solid border-gray-200 py-3 text-xs uppercase border-l-0 border-r-0 whitespace-no-wrap font-semibold text-left">Content</th>
          <th class="px-6 bg-gray-100 text-gray-600 align-middle border border-solid border-gray-200 py-3 text-xs uppercase border-l-0 border-r-0 whitespace-no-wrap font-semibold text-left", colspan="3"></th>
        </tr>
      </thead>
      <tbody>
        <%% @posts.each do |post| %>
          <tr>
            <td class="border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left"><%= post.title %></td>
            <td class="border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left"><%= post.content %></td>
            <td class="border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left"><%= link_to 'Show', post %></td>
            <td class="border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left"><%= link_to 'Edit', edit_post_path(post) %></td>
            <td class="border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left"><%= link_to 'Destroy', post, method: :delete, data: { confirm: 'Are you sure?' } %></td>
          </tr>
        <% end %>
      </tbody>
    </table>
  </div>
</div>
<%= link_to 'New Post', new_post_path %>
Enter fullscreen mode Exit fullscreen mode

You should now see the following page when you navigate to localhost:3000

updated tailwind home index

Abstraction

This table looks so much better, but the classes for these elements are long and repetitive. Let's clean that up a bit by abstracting them to our scss file.

Create two new classes in app/javascript/src/application.scss:

.table-header {
  @apply px-6 bg-gray-100 text-gray-900 align-middle border border-solid border-gray-200 py-3 text-xs uppercase border-l-0 border-r-0 whitespace-no-wrap font-semibold text-left;
}

.table-content {
  @apply border-t-0 px-6 align-middle border-l-0 border-r-0 text-xs whitespace-no-wrap p-4 text-left;
}
Enter fullscreen mode Exit fullscreen mode

Then in our HTML, change all of the table head and body element classes accordingly. We also darkened the header text to make the change standout.

When you reload the page, you should see the same screen as before, only with darker table headings.

Summary

Hopefully this is helpful to those of you looking to add TailwindCSS to your Rails app.

If you are interested in more information about this article of using Rails and TailwindCSS, leave a comment or reach out to me on Twitter and I am happy to chat.

View repo for this post

Happy coding! 😄

Top comments (6)

Collapse
 
pavelloz profile image
Paweł Kowalski • Edited

NEVER, ever use Tailwindcss without purgeCSS to clean it up on production. Otherwise you will be shipping 780KB of css (in 2.0 this will be probably over 1MB, and rapidly growing with plugins and customizations), from which you will be using 1% of it.

Those couple lines of code in postcss config is why Tailwind is so great in the performance department.

Example config: github.com/pavelloz/webpack-tailwi...
Official docs explanation: tailwindcss.com/docs/controlling-f...

Collapse
 
andrewmcodes profile image
Andrew Mason

This is a great point. I will be doing a follow up to this for adding purgecss.

Collapse
 
rossta profile image
Ross Kaffenberger

Good callout. It can be easy to miss this concern when first getting started with TailwindCSS.

Collapse
 
belgoros profile image
Serguei Cambour

Thank you very much for the detailed article, Andrew! One point is about removing app/assets folder which contains manifest.js file. If I remove it, I get:

sprockets/railtie.rb:105:in `block in <class:Railtie>': Expected to find a manifest file in `app/assets/config/manifest.js` (Sprockets::Railtie::ManifestNeededError)
But did not, please create this file and use it to link any assets that need
to be rendered by your app:

Example:
  //= link_tree ../images
  //= link_directory ../javascripts .js
  //= link_directory ../stylesheets .css
and restart your server
Enter fullscreen mode Exit fullscreen mode

I'm using Rails 6.0.2 version with Ruby 2.7.0. Another question I can't find the right answer is about the syntax to use in javascript/packs/application.js file. You used import '../src/application.scss'. It worked for me. Another resource es used it as require('../css/application.scss') and it worked as well. Sometimes it is really frustrating to not find the right way to follow as Rails guides still have nothing explaining this point, event after more then 6 month after its release.

Collapse
 
promisepreston profile image
Promise Chukwuenyem • Edited

Firstly, you could use still use the app/assets folder. It's a choice. I often use it for images and stylesheets and the javascript/packs folder for javascript files. So this way I will not need to tweak my app/views/layouts/application.html.erb by changing:

<%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
<%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
Enter fullscreen mode Exit fullscreen mode

to:

<%= stylesheet_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
<%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
Enter fullscreen mode Exit fullscreen mode

Secondly, import '../src/application.scss' and require('../css/application.scss') can be used interchangeably. I often use require as it saves me the stress of specifying the full path of the module. The major difference between require and import , is that require will automatically scan node_modules folder to find modules, but import , which comes from ES6 doesn't so you must specify the path of the module.

Read up more here: The difference between “require(x)” and “import x”

Collapse
 
pabloc profile image
PabloC

Thanks for this recap!