A bit of history: this gem was inspired by digging deeper into Draper with an eye on a refactoring.
It implements a general decorator logic. It’s not meant to be a presenter.
Install the gem and add to the application's Gemfile by executing:
$ bundle add magic-decorator
If bundler is not being used to manage dependencies, install the gem by executing:
$ gem install magic-decorator
Magic::Decorator::Base
is a basic decorator class to be inherited by any other decorator.
It further inherits from SimpleDelegator
and is straightforward like that.
class PersonDecorator < Magic::Decorator::Base
def name = "#{first_name} #{last_name}"
end
Person = Struct.new :first_name, :last_name do
include Magic::Decoratable
end
person = Person.new('John', 'Smith').decorate
person.name # => "John Smith"
This module adds three methods to decorate an object. Decorator class is being inferred automatically. When no decorator is found,
#decorate
returnsnil
,#decorate!
raisesMagic::Lookup::Error
,#decorated
returns the original object.
One can test for the object is actually decorated with #decorated?
.
'with no decorator for String'.decorated
.decorated? # => false
['with a decorator for Array'].decorated
.decorated? # => true
When extending Magic::Decoratable
, one may override #decorator_base
to be used for lookup.
class Special::Decorator < Magic::Decorator::Base
def self.name_for object_class
"Special::#{object_class}Decorator"
end
end
module Special::Decoratable
include Magic::Decoratable
private
def decorator_base = Special::Decorator
end
class Special::Model
include Special::Decoratable
end
Special::Model.new.decorate # looks for Special::Decorator descendants
Magic::Decoratable
is mixed into Object
by default. It means that effectively any object is magically decoratable.
One can use Magic::Decoratable.classes
to see all the decoratable classes.
For almost any method called on a decorated object, both its result and yield
ed arguments get decorated.
'with no decorator for String'.decorated.chars
.decorated? # => false
['with a decorator for Array'].decorated.map(&:chars).first.grep(/\S/).group_by(&:upcase).transform_values(&:size).sort_by(&:last).reverse.first(5).map(&:first)
.decorated? # => true
Some methods aren’t meant to be decorated though:
deconstruct
&deconstruct_keys
for pattern matching,- converting methods: those starting with
to_
, - system methods: those starting with
_
.
Magic::Decorator::Base.undecorated
can be used to exclude methods from being decorated automagically.
class MyDecorator < Magic::Decorator::Base
undecorated %i[to_s inspect]
undecorated :raw_method
undecorated :m1, :m2
end
Decorators provide automatic class inference for any object based on its class name powered by Magic Lookup.
For example, MyNamespace::MyModel.new.decorate
looks for MyNamespace::MyModelDecorator
first.
When missing, it further looks for decorators for its ancestor classes, up to ObjectDecorator
.
It automagically decorates all its decoratable items.
[1, [2], { 3 => 4 }, '5'].decorated
.map &:decorated? # => [false, true, true, false]
{ 1 => 2, [3] => [4] }.decorated.keys
.map &:decorated? # => [false, true]
{ 1 => 2, [3] => [4] }.decorated.values
.map &:decorated? # => [false, true]
{ 1 => 2, [3] => [4] }.decorated[1]
.decorated? # => false
{ 1 => 2, [3] => [4] }.decorated[[3]]
.decorated? # => true
- enables splat operator:
*decorated
, - enables double-splat operator:
**decorated
, - enumerating methods yield decorated items.
When one needs more complicated behavior than the default one or feels like explicit is better than implicit.
One may override #decorator
for any decoratable class, to be used instead of Magic Lookup.
-
That could be as straightforward as a constant:
class Guest private def decorator = UserDecorator end guest.decorate # => instance of UserDecorator
-
Or, that could be virtually any logic:
class User private def decorator = admin? ? AdminDecorator : super end user.decorate # => instance of UserDecorator admin.decorate # => instance of AdminDecorator
Testing a decorator is much like testing any other class.
To test whether an object is decorated one can use #decorated?
method.
Note
A decorated object equals the original one (object.decorated == object
).
Thus, any existing tests shouldn’t break when the objects being tested get decorated.
After checking out the repo, run bin/setup
to install dependencies. Then, run rake spec
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
.
Bug reports and pull requests are welcome on GitHub at https://github.com/Alexander-Senko/magic-decorator. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the code of conduct.
The gem is available as open source under the terms of the MIT License.
Everyone interacting in the Magic Decorator project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the code of conduct.