module Remarkable module DSL module Optionals OPTIONAL_KEYS = [ :positive, :negative, :not_given ] def self.included(base) base.extend ClassMethods end module ClassMethods protected # Creates optional handlers for matchers dynamically. The following # statement: # # optional :range, :default => 0..10 # # Will generate: # # def range(value=0..10) # @options ||= {} # @options[:range] = value # self # end # # Options: # # * :default - The default value for this optional # * :alias - An alias for this optional # * :splat - Should be true if you expects multiple arguments # # Examples: # # optional :name, :title # optional :range, :default => 0..10, :alias => :within # # Optionals will be included in description messages if you assign them # properly on your locale file. If you have a validate_uniqueness_of # matcher with the following on your locale file: # # description: validate uniqueness of {{attributes}} # optionals: # scope: # positive: scoped to {{value}} # case_sensitive: # positive: case sensitive # negative: case insensitive # # When invoked like below will generate the following messages: # # validate_uniqueness_of :project_id, :scope => :company_id # #=> "validate uniqueness of project_id scoped to company_id" # # validate_uniqueness_of :project_id, :scope => :company_id, :case_sensitive => true # #=> "validate uniqueness of project_id scoped to company_id and case sensitive" # # validate_uniqueness_of :project_id, :scope => :company_id, :case_sensitive => false # #=> "validate uniqueness of project_id scoped to company_id and case insensitive" # # The interpolation options available are "value" and "inspect". Where # the first is the optional value transformed into a string and the # second is the inspected value. # # Three keys are available to be used in I18n files and control how # optionals are appended to your description: # # * positive - When the optional is given and it evaluates to true (everything but false and nil). # * negative - When the optional is given and it evaluates to false (false or nil). # * not_given - When the optional is not given. # def optionals(*names) options = names.extract_options! @matcher_optionals += names splat = options[:splat] ? '*' : '' default = options[:default] ? "=#{options[:default].inspect}" : "" names.each do |name| class_eval <<-END, __FILE__, __LINE__ def #{name}(#{splat}value#{default}) @options ||= {} @options[:#{name}] = value self end END end class_eval "alias_method(:#{options[:alias]}, :#{names.last})" if options[:alias] # Call unique to avoid duplicate optionals. @matcher_optionals.uniq! end alias :optional :optionals # Instead of appending, prepend optionals to the beginning of optionals # array. This is important because this decide how the description # message is generated. # def prepend_optionals(*names) current_optionals = @matcher_optionals.dup @matcher_optionals = [] optional(*names) @matcher_optionals += current_optionals @matcher_optionals.uniq! end alias :prepend_optional :prepend_optionals end # Overwrites description to support optionals. Check optional for # more information. # def description(options={}) message = super(options) message.strip! optionals = self.class.matcher_optionals.map do |optional| scope = matcher_i18n_scope + ".optionals.#{optional}" if @options.key?(optional) value = @options[optional] defaults = [ (value ? :positive : :negative) ] # If optional is a symbol and it's not any to any of the reserved symbols, search for it also defaults.unshift(value) if value.is_a?(Symbol) && !OPTIONAL_KEYS.include?(value) Remarkable.t defaults.shift, :default => defaults, :raise => true, :scope => scope, :inspect => value.inspect, :value => value.to_s else Remarkable.t :not_given, :raise => true, :scope => scope end rescue nil end.compact message << ' ' << array_to_sentence(optionals) message.strip! message end end end end