Module: MotionSupport::Concern
- Included in:
- Callbacks
- Defined in:
- motion/concern.rb
Overview
A typical module looks like this:
module M
def self.included(base)
base.extend ClassMethods
base.class_eval do
scope :disabled, -> { where(disabled: true) }
end
end
module ClassMethods
...
end
end
By using MotionSupport::Concern
the above module could instead be written as:
module M
extend MotionSupport::Concern
included do
scope :disabled, -> { where(disabled: true) }
end
module ClassMethods
...
end
end
Moreover, it gracefully handles module dependencies. Given a Foo
module and a Bar
module which depends on the former, we would typically write the following:
module Foo
def self.included(base)
base.class_eval do
def self.method_injected_by_foo
...
end
end
end
end
module Bar
def self.included(base)
base.method_injected_by_foo
end
end
class Host
include Foo # We need to include this dependency for Bar
include Bar # Bar is the module that Host really needs
end
But why should Host
care about Bar
‘s dependencies, namely Foo
? We could try to hide these from Host
directly including Foo
in Bar
:
module Bar
include Foo
def self.included(base)
base.method_injected_by_foo
end
end
class Host
include Bar
end
Unfortunately this won’t work, since when Foo
is included, its base
is the Bar
module, not the Host
class. With MotionSupport::Concern
, module dependencies are properly resolved:
module Foo
extend MotionSupport::Concern
included do
def self.method_injected_by_foo
...
end
end
end
module Bar
extend MotionSupport::Concern
include Foo
included do
self.method_injected_by_foo
end
end
class Host
include Bar # works, Bar takes care now of its dependencies
end
Class Method Summary collapse
-
.extended(base) ⇒ Object
:nodoc:.
Instance Method Summary collapse
Class Method Details
.extended(base) ⇒ Object
:nodoc:
97 98 99 |
# File 'motion/concern.rb', line 97 def self.extended(base) #:nodoc: base.instance_variable_set("@_dependencies", []) end |
Instance Method Details
#append_features(base) ⇒ Object
101 102 103 104 105 106 107 108 109 110 111 112 |
# File 'motion/concern.rb', line 101 def append_features(base) if base.instance_variable_defined?("@_dependencies") base.instance_variable_get("@_dependencies") << self return false else return false if base < self @_dependencies.each { |dep| base.send(:include, dep) } super base.extend const_get("ClassMethods") if const_defined?("ClassMethods") base.class_eval(&@_included_block) if instance_variable_defined?("@_included_block") end end |
#included(base = nil, &block) ⇒ Object
114 115 116 117 118 119 120 |
# File 'motion/concern.rb', line 114 def included(base = nil, &block) if base.nil? @_included_block = block else super end end |