rails/activesupport/lib/active_support/concern.rb

218 lines
5.8 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
module ActiveSupport
# = Active Support \Concern
#
# 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
2010-08-14 15:04:17 +00:00
# ...
# end
# end
#
# By using +ActiveSupport::Concern+ the above module could instead be
2012-09-17 05:22:18 +00:00
# written as:
2010-08-14 15:04:17 +00:00
#
# require "active_support/concern"
#
# module M
# extend ActiveSupport::Concern
#
# included do
# scope :disabled, -> { where(disabled: true) }
# end
#
# class_methods do
2010-08-14 15:04:17 +00:00
# ...
# end
# end
#
2012-09-17 05:22:18 +00:00
# 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
2010-08-14 15:04:17 +00:00
# def self.method_injected_by_foo
# ...
# end
# end
# end
# end
#
# module Bar
# def self.included(base)
2010-08-14 15:04:17 +00:00
# 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
#
2012-09-17 05:22:18 +00:00
# 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)
2010-08-14 15:04:17 +00:00
# base.method_injected_by_foo
# end
# end
#
# class Host
# include Bar
# end
#
2012-09-17 05:22:18 +00:00
# Unfortunately this won't work, since when +Foo+ is included, its <tt>base</tt>
# is the +Bar+ module, not the +Host+ class. With +ActiveSupport::Concern+,
2012-09-17 05:22:18 +00:00
# module dependencies are properly resolved:
#
# require "active_support/concern"
#
# module Foo
# extend ActiveSupport::Concern
# included do
# def self.method_injected_by_foo
# ...
# end
# end
# end
#
# module Bar
# extend ActiveSupport::Concern
# include Foo
#
# included do
2010-08-14 15:04:17 +00:00
# self.method_injected_by_foo
# end
# end
#
# class Host
# include Bar # It works, now Bar takes care of its dependencies
# end
2019-09-12 15:53:24 +00:00
#
# === Prepending concerns
#
2020-11-17 13:12:18 +00:00
# Just like <tt>include</tt>, concerns also support <tt>prepend</tt> with a corresponding
# <tt>prepended do</tt> callback. <tt>module ClassMethods</tt> or <tt>class_methods do</tt> are
# prepended as well.
2019-09-12 15:53:24 +00:00
#
2020-11-17 13:12:18 +00:00
# <tt>prepend</tt> is also used for any dependencies.
module Concern
2021-07-29 21:18:07 +00:00
class MultipleIncludedBlocks < StandardError # :nodoc:
def initialize
super "Cannot define multiple 'included' blocks for a Concern"
end
end
2021-07-29 21:18:07 +00:00
class MultiplePrependBlocks < StandardError # :nodoc:
2019-09-12 15:33:53 +00:00
def initialize
super "Cannot define multiple 'prepended' blocks for a Concern"
end
end
2021-07-29 21:18:07 +00:00
def self.extended(base) # :nodoc:
base.instance_variable_set(:@_dependencies, [])
end
2021-07-29 21:18:07 +00:00
def append_features(base) # :nodoc:
if base.instance_variable_defined?(:@_dependencies)
base.instance_variable_get(:@_dependencies) << self
2017-10-28 08:20:38 +00:00
false
else
return false if base < self
@_dependencies.each { |dep| base.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
2021-07-29 21:18:07 +00:00
def prepend_features(base) # :nodoc:
2019-09-12 15:33:53 +00:00
if base.instance_variable_defined?(:@_dependencies)
base.instance_variable_get(:@_dependencies).unshift self
false
else
return false if base < self
@_dependencies.each { |dep| base.prepend(dep) }
super
base.singleton_class.prepend const_get(:ClassMethods) if const_defined?(:ClassMethods)
2019-09-12 15:33:53 +00:00
base.class_eval(&@_prepended_block) if instance_variable_defined?(:@_prepended_block)
end
end
# Evaluate given block in context of base class,
# so that you can write class macros here.
# When you define more than one +included+ block, it raises an exception.
def included(base = nil, &block)
if base.nil?
if instance_variable_defined?(:@_included_block)
if @_included_block.source_location != block.source_location
raise MultipleIncludedBlocks
end
else
@_included_block = block
end
else
super
end
end
2019-09-12 15:33:53 +00:00
# Evaluate given block in context of base class,
# so that you can write class macros here.
# When you define more than one +prepended+ block, it raises an exception.
def prepended(base = nil, &block)
if base.nil?
if instance_variable_defined?(:@_prepended_block)
if @_prepended_block.source_location != block.source_location
raise MultiplePrependBlocks
end
else
@_prepended_block = block
end
else
super
end
end
# Define class methods from given block.
# You can define private class methods as well.
#
# module Example
# extend ActiveSupport::Concern
#
# class_methods do
# def foo; puts 'foo'; end
#
# private
# def bar; puts 'bar'; end
# end
# end
#
# class Buzz
# include Example
# end
#
# Buzz.foo # => "foo"
# Buzz.bar # => private method 'bar' called for Buzz:Class(NoMethodError)
def class_methods(&class_methods_module_definition)
mod = const_defined?(:ClassMethods, false) ?
const_get(:ClassMethods) :
const_set(:ClassMethods, Module.new)
mod.module_eval(&class_methods_module_definition)
end
end
end