class Object

def acts_like?(duck)


Stringish.new.acts_like?(:string) # => true

Then client code can query for duck-type-safeness this way:

end
end
def acts_like_string?
class Stringish

This class may define:

==== Example: A class that provides the same interface as String

value are irrelevant.
Note that the marker method is only expected to exist. It isn't called, so its body or return

method to interoperate.
and classes that are able to act like Time can also define an acts_like_time?
x.acts_like?(:time) and x.acts_like?(:date) to test duck-type compatibility,
and extends Time to define acts_like_time?. As a result, developers can call
For example, Active Support extends Date to define an acts_like_date? method,

acts_like?(:some_class).
acts_like_some_class? to signal its compatibility to callers of
A class that provides the same interface as SomeClass may define a marker method named

an appropriately-named marker method.
Provides a way to check whether some class acts like some other class based on the existence of
def acts_like?(duck)
  case duck
  when :time
    respond_to? :acts_like_time?
  when :date
    respond_to? :acts_like_date?
  when :string
    respond_to? :acts_like_string?
  else
    respond_to? :"acts_like_#{duck}?"
  end
end

def as_json(options = nil) # :nodoc:

:nodoc:
def as_json(options = nil) # :nodoc:
  if respond_to?(:to_hash)
    to_hash.as_json(options)
  else
    instance_values.as_json(options)
  end
end

def blank?

Returns:
  • (true, false) -
def blank?
  respond_to?(:empty?) ? !!empty? : !self
end

def deep_dup

dup.instance_variable_defined?(:@a) # => true
object.instance_variable_defined?(:@a) # => false

dup.instance_variable_set(:@a, 1)
dup = object.deep_dup
object = Object.new

not duplicable, returns +self+.
Returns a deep copy of object if it's duplicable. If it's
def deep_dup
  duplicable? ? dup : self
end

def duplicable?

true otherwise.
False for method objects;

Can you safely dup this object?
def duplicable?
  true
end

def html_safe?

def html_safe?
  false
end

def in?(another_object)

to +#include?+.
This will throw an +ArgumentError+ if the argument doesn't respond

"Konata".in?(characters) # => true
characters = ["Konata", "Kagami", "Tsukasa"]

any object which responds to +#include?+. Usage:
Returns true if this object is included in the argument. Argument must be
def in?(another_object)
  another_object.include?(self)
rescue NoMethodError
  raise ArgumentError.new("The parameter passed to #in? must respond to #include?")
end

def instance_values

C.new(0, 1).instance_values # => {"x" => 0, "y" => 1}

end
end
@x, @y = x, y
def initialize(x, y)
class C

corresponding values.
Returns a hash with string keys that maps instance variable names without "@" to their
def instance_values
  Hash[instance_variables.map { |name| [name[1..-1], instance_variable_get(name)] }]
end

def instance_variable_names

C.new(0, 1).instance_variable_names # => ["@y", "@x"]

end
end
@x, @y = x, y
def initialize(x, y)
class C

Returns an array of instance variable names as strings including "@".
def instance_variable_names
  instance_variables.map(&:to_s)
end

def presence

Returns:
  • (Object) -
def presence
  self if present?
end

def presence_in(another_object)

Returns:
  • (Object) -
def presence_in(another_object)
  in?(another_object) ? self : nil
end

def present?

Returns:
  • (true, false) -
def present?
  !blank?
end

def to_param

Alias of to_s.
def to_param
  to_s
end

def to_query(key)

using the given key as the param name.
Converts an object into a string suitable for use as a URL query string,
def to_query(key)
  "#{CGI.escape(key.to_param)}=#{CGI.escape(to_param.to_s)}"
end

def with_options(options, &block)


# #=>
# styled.button_tag "I'm red too!"

# #=> I'm red
# styled.link_to "I'm red", "/"

end
end
with_options style: "color: red;"
def styled
module MyStyledHelpers

When the block argument is omitted, the decorated Object instance is returned:

end
end
validates :phone_number_type, inclusion: { in: Phone.phone_number_types.keys }
with_options presence: true do

enum phone_number_type: { home: 0, office: 1, mobile: 2 }
class Phone < ActiveRecord::Base

You can access these methods using the class name instead:
NOTE: You cannot call class methods implicitly inside of with_options.

Hence the inherited default for +if+ key is ignored.

validates :content, length: { minimum: 50 }, if: -> { content.present? }

The code is equivalent to:

end
end
validates :content, if: -> { content.present? }
with_options if: :persisted?, length: { minimum: 50 } do
class Post < ActiveRecord::Base

NOTE: Each nesting level will merge inherited defaults in addition to their own.

with_options can also be nested since the call is forwarded to its receiver.

end
end
has_many :expenses
has_many :invoices
has_many :products
has_many :customers
with_options dependent: :destroy do
class Account < ActiveRecord::Base

in merging options context:
When you don't pass an explicit receiver, it executes the whole block

end
body i18n.t :body, user_name: user.name
subject i18n.t :subject
I18n.with_options locale: user.locale, scope: 'newsletter' do |i18n|

It can also be used with an explicit receiver:

end
end
assoc.has_many :expenses
assoc.has_many :invoices
assoc.has_many :products
assoc.has_many :customers
with_options dependent: :destroy do |assoc|
class Account < ActiveRecord::Base

Using with_options, we can remove the duplication:

end
has_many :expenses, dependent: :destroy
has_many :invoices, dependent: :destroy
has_many :products, dependent: :destroy
has_many :customers, dependent: :destroy
class Account < ActiveRecord::Base

Without with_options, this code contains duplication:

hash as its final argument.
provided. Each method called on the block variable must take an options
the receiver, will have its options merged with the default +options+ hash
method calls. Each method called in the block, with the block variable as
An elegant way to factor duplication out of options passed to a series of
def with_options(options, &block)
  option_merger = ActiveSupport::OptionMerger.new(self, options)
  if block
    block.arity.zero? ? option_merger.instance_eval(&block) : block.call(option_merger)
  else
    option_merger
  end
end