Rails View Helpers

Everyone can use a little help, sometimes...

2016-09-10

It isn’t always easy to know when to use Rails view helpers. You don’t want to sit down and try to write a bunch of helpers before you have a UI spec, or even before you’ve started to write a rough layout of your markup. But at some point, hopefully earlier in the implementation phase, it’s good to remember that helpers are there to help.

Consider the following markup for displaying a user’s profile image:

Traditional markup

1
2
3
4
5
<% if @current_user.avatar_uid.blank? %>
<div class="Profile-image" style="background-image: url(<%= image_url('profile-image-default.png') %>)"></div>
<% else %>
<div class="Profile-image" style="background-image: url(<%= @current_user.avatar.thumb('100x100#').url %>)"></div>
<% end %>

This is functional code, but it has quite a bit of redundancy and, honestly, it’s not that easy to see what it’s doing at a glance. Now, let’s imagine what this part of our view would look like with a rails helper:

Rails Helper Version

1
<% profile_image_for(@current_user) %>

Hopefully, the merit of simplicity is readily apparent. Replacing seven lines of html with a simple function call can be a big win, even if you’re only calling it once. If you end up using the same markup across many views in your app, the savings just multiply.

So, how should we think about view helpers? I like to follow a few basic principles that guide how I write markup.

Basic Principles

  1. Don’t Repeat Yourself
  2. Separate Semantics from Style
  3. Maintain Testability and Clarity of Intent

It would be cheating if I just showed you the code that used the helper without providing an example of the helpers implementation. Even though it’s quite powerful, it isn’t actually much more code:

Example Implementation

1
2
3
4
5
6
7
8
9
PROFILE_CLASSNAME = 'Profile-image'
PROFILE_THUMB_FORMAT = '100x100#'
DEFAULT_PROFILE_IMAGE = 'profile-image-default.png'
def profile_image_for(user)
url = user.avatar_uid && user.avatar.thumb(PROFILE_THUMB_FORMAT).url
url ||= image_url(DEFAULT_PROFILE_IMAGE)
tag(:div, class: PROFILE_CLASSNAME, style: { 'background-image': "url(#{url})" })
end

When you use a helper like this, you get clean, testable code that doesn’t need to be repeated every time you want to render a profile image in a view. And if you need to change how profile images are marked up later, only one simple method needs to be updated.

A Next Level Developer invests a little effort up-front so they can avoid complexity and save time later in the project.