Module: ActiveSupport::Concern
- Included in:
- AbstractController::AssetPaths, AbstractController::Callbacks, AbstractController::Helpers, AbstractController::Layouts, AbstractController::Logger, AbstractController::Rendering, AbstractController::UrlFor, AbstractController::ViewPaths, ActionController::Caching, ActionController::ConditionalGet, ActionController::Cookies, ActionController::DataStreaming, ActionController::Flash, ActionController::ForceSSL, ActionController::Head, ActionController::Helpers, ActionController::HideActions, ActionController::HttpAuthentication::Basic::ControllerMethods, ActionController::Instrumentation, ActionController::MimeResponds, ActionController::ParamsWrapper, ActionController::RackDelegation, ActionController::Redirecting, ActionController::Renderers, ActionController::Renderers::All, ActionController::Rendering, ActionController::RequestForgeryProtection, ActionController::Rescue, ActionController::Streaming, ActionController::StrongParameters, ActionController::TemplateAssertions, ActionController::TestCase::Behavior, ActionController::Testing, ActionController::UrlFor, ActionDispatch::Assertions, ActionDispatch::Http::MimeNegotiation, ActionDispatch::Routing::RouteSet::MountedHelpers, ActionDispatch::Routing::UrlFor, ActionMailer::DeliveryMethods, ActionMailer::TestCase::Behavior, ActionView::Helpers, ActionView::Helpers::AssetTagHelper, ActionView::Helpers::FormHelper, ActionView::Helpers::FormTagHelper, ActionView::Helpers::SanitizeHelper, ActionView::Helpers::TagHelper, ActionView::Helpers::TextHelper, ActionView::Helpers::UrlHelper, ActionView::TestCase::Behavior, ActiveModel::AttributeMethods, ActiveModel::Conversion, ActiveModel::Dirty, ActiveModel::SecurePassword, ActiveModel::Serializers::JSON, ActiveModel::Serializers::Xml, ActiveModel::Validations, ActiveModel::Validations::Callbacks, ActiveRecord::Aggregations, ActiveRecord::Associations, ActiveRecord::AttributeAssignment, ActiveRecord::AttributeMethods, ActiveRecord::AttributeMethods::BeforeTypeCast, ActiveRecord::AttributeMethods::Dirty, ActiveRecord::AttributeMethods::PrimaryKey, ActiveRecord::AttributeMethods::Query, ActiveRecord::AttributeMethods::Read, ActiveRecord::AttributeMethods::Serialization, ActiveRecord::AttributeMethods::Serialization::Behavior, ActiveRecord::AttributeMethods::TimeZoneConversion, ActiveRecord::AttributeMethods::Write, ActiveRecord::AutosaveAssociation, ActiveRecord::Callbacks, ActiveRecord::Core, ActiveRecord::CounterCache, ActiveRecord::Delegation, ActiveRecord::Delegation::ClassSpecificRelation, ActiveRecord::Inheritance, ActiveRecord::Integration, ActiveRecord::Locking::Optimistic, ActiveRecord::ModelSchema, ActiveRecord::NestedAttributes, ActiveRecord::Persistence, ActiveRecord::QueryMethods, ActiveRecord::Railties::ControllerRuntime, ActiveRecord::ReadonlyAttributes, ActiveRecord::Reflection, ActiveRecord::Sanitization, ActiveRecord::Scoping, ActiveRecord::Scoping::Default, ActiveRecord::Scoping::Named, ActiveRecord::Serialization, ActiveRecord::Store, ActiveRecord::TestFixtures, ActiveRecord::Timestamp, ActiveRecord::Transactions, ActiveRecord::Validations, Callbacks, Configurable, Rescuable, Testing::ConstantLookup, Testing::SetupAndTeardown, LoggerSilence, Rails::Generators::Migration, Rails::Generators::Testing::Behaviour, Rails::Railtie::Configurable
- Defined in:
- activesupport/lib/active_support/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 ActiveSupport::Concern
the above module could instead be written as:
require 'active_support/concern'
module M
extend ActiveSupport::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 ActiveSupport::Concern
, 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
self.method_injected_by_foo
end
end
class Host
include Bar # works, Bar takes care now of its dependencies
end
Defined Under Namespace
Classes: MultipleIncludedBlocks
Class Method Summary collapse
-
.extended(base) ⇒ Object
:nodoc:.
Instance Method Summary collapse
Class Method Details
.extended(base) ⇒ Object
:nodoc:
107 108 109 |
# File 'activesupport/lib/active_support/concern.rb', line 107 def self.extended(base) #:nodoc: base.instance_variable_set(:@_dependencies, []) end |
Instance Method Details
#append_features(base) ⇒ Object
111 112 113 114 115 116 117 118 119 120 121 122 |
# File 'activesupport/lib/active_support/concern.rb', line 111 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
124 125 126 127 128 129 130 131 132 |
# File 'activesupport/lib/active_support/concern.rb', line 124 def included(base = nil, &block) if base.nil? raise MultipleIncludedBlocks if instance_variable_defined?(:@_included_block) @_included_block = block else super end end |