Components¶
Client-side web frameworks, like Angular and React, have popularized “components” as a means of composing HTML interfaces from smaller parts. Until recently, this approach was not as popular or widespread when rendering on the server-side. Communities around server-side frameworks like Laravel or Ruby on Rails are beginning to pick up these concepts as well.
As the hipster kids we are, we already started doing this before it was cool. 😎
After starting with a home-grown solution, we later switched to the matured and now widespread ViewComponent library by GitHub.
Purpose¶
By default, Rails suggests structuring your views into partials. They help keep the file size under control, and when you want to re-use certain snippets of HTML. However, they do not help you in keeping logic (e.g. business rules) out of your templates. By itself, Rails does not suggest any mechanism to deal with this. In many Rails projects, an additional “presenter” layer is introduced to implement the view-specific logic.
Building on these established patterns, components combine the small templates from partials and the presentation-specific code from presenters into one cohesive unit.
Therefore, when dealing with components, you have a named unit consisting of a dynamic HTML snippet combined with the logic and rules to render it.
In other words
The fine folks at GitHub also explained this well when they announced their ViewComponent library.
Usage¶
What follows is only a short primer on ViewComponent basics. Do check out the extensive ViewComponent guide to learn more about the library and its features!
Defining a component¶
A component consists of two files:
- a component class defining the public interface of the component - it stores state (constructor arguments) and holds the logic to determine values needed for rendering the template
- a Slim template that takes the internal state and calls methods in order to generate HTML
The class for a very simple component could look like this:
Example
##
# Greet users by name and give them the current time
#
class SayHelloWithTime < ApplicationComponent
##
# A plain Ruby class initializer. It defines the required (and optional)
# input for rendering a component instance. Both positional and named
# arguments can be used.
#
def initialize(name)
@name = name
end
private
##
# The "presenter logic" can live in private methods of the component
# class. Here, the input (instance variables) can be turned into
# something useful for rendering.
#
def now
DateTime.now.strftime('%H:%I')
end
end
The corresponding template could then look like this:
Example
p = "Hello, dear #{@name}!"
p = "The current time is #{now}."
The template is evaluated within the context of the component class.
Both instance variables like @name
and (private) methods like now
can be accessed directly.
Note
The template is co-located with its class, in the same directory.
Rendering a component¶
The above component can be rendered in any Rails template (and even in a controller!) using the render
helper:
Example
= render SayHelloWithTime.new('Claus')
Previews¶
We use lookbook to catalog, document and preview the different variants of our components. This serves as our living style guide for UI components.
When developing, the lookbook UI is available at http://localhost:3000/rails/components.
Like templates, component previews are co-located with the component class. Define a preview class with (at least) one action - et voilà : The previews will be automatically detected and picked up by lookbook.
Example
class SayHelloWithTimePreview < ViewComponent::Preview
def default
render SayHelloWithTimePreview.new('Grace')
end
end
Testing¶
Because components are limited in size and self-contained, they can be unit-tested very nicely:
Example
RSpec.describe SayHelloWithTime, type: :component do
subject(:component) do
described_class.new 'Claus'
end
it 'greets the user by name' do
render_inline(component)
expect(rendered_content).to have_content 'Hello, dear Claus'
end
end
ViewComponent offers some useful test helpers and wraps the rendered HTML in a Capybara fragment to offer useful assertions (#have_content
, #have_selector
, …).
Component tests should run assertions against the rendered HTML, and describe how that HTML changes when the input changes.
Tip
Even though it may have other public methods, do not run assertions against those public methods.
Those are just implementation details.
The only public interface of components is their initialization with constructor arguments (the input) and the final HTML generated by #call
or the template (the output).