Simple, powerful, and explicit decorators for Ruby on Rails, inspired (and forked) from ActiveDecorator. It has the following goals:
- Fast and efficient.
- Implicit decorators.
- Explicit decoration.
Gifted has an almost identical feature set to ActiveDecorator...
- Automatically mixes decorator module into corresponding model only when:
- Passing a model or collection of models or an instance of ActiveRecord::Relation from controllers to views
- rendering partials with models (using
:collection
or:object
or:locals
explicitly or implicitly) - fetching already decorated Active Record model object's association
- the decorator module runs in the model's context. So, you can directly call any attributes or methods in the decorator module
- since decorators are considered as sort of helpers, you can also call any ActionView's helper methods such as
content_tag
orlink_to
But with the following differences...
- Decorator methods are name-spaced to a single
gift
method. This avoids confusion and collision. - Decorators are mixed into SimpleDelegator classes, which themselves delegate to the decorated object.
- Decorator Views allow you to create child decorators.
- Only works with Rails.
Gifted requires Rails >= 5.
Add this line to your application's Gemfile:
gem 'gifted'
And then execute:
$ bundle
Or install it yourself as:
$ gem install gifted
Create your Decorators into /app/decorators
for each model you want decorated, and define methods. For example, a model User
, can have a decorator UserDecorator
:
- Model
class User < ActiveRecord::Base
# first_name:string last_name:string
end
- Controller
class UsersController < ApplicationController
def show
@user = User.find(params[:id])
end
end
- Decorator
# I can call any method from the decorated object, and because I am mixed into SimpleDelegator, I
# can access the decorated object directly at __getobj__.
module UserDecorator
# This is a Decorator View, and can be access by passing the name of the view (:admin) to `#gift`.
module AdminDecorator
def full_name
"#{first_name} #{last_initial} (administrator)"
end
end
def full_name
"#{first_name} #{last_initial}"
end
def first_name
first_name.titleize
end
def last_initial
last_name[0]
end
end
- View
<!-- @user here is auto-decorated in between the controller and the view -->
<!-- @user.gift.full_name will use the method #full_name from the decorator. -->
<p><%= @user.gift.full_name %></p>
<!--
@user.gift.last_name will use the method #last_name from the model, because @user.gift delegates
to the model.
-->
<p><%= @user.gift.last_name %></p>
<!--
Calling gift with a symbol as an argument, will decorate the object with a Decorator view.
So @user.gift(:admin).last_name will use the method #full_name from UserDecorator::AdminDecorator.
-->
<p><%= @user.gift(:admin).full_name %></p>
<!-- @user.last_name will use the method #last_name from the model. -->
<p><%= @user.last_name %></p>
After checking out the repo, run bin/setup
to install dependencies. Then, run rake test
to run the tests. You can also run bin/console
for an interactive prompt that will allow you to experiment.
To install this gem onto your local machine, run bundle exec rake install
. To release a new version, update the version number in version.rb
, and then run bundle exec rake release
, which will create a git tag for the version, push git commits and tags, and push the .gem
file to rubygems.org.
Bug reports and pull requests are welcome on GitHub at https://github.com/joelmoss/gifted.
This gem is hugely inspired by and (in parts) copied from ActiveDecorator.