module Searchlogic module NamedScopes # Handles dynamically creating named scopes for associations. See the README for a detailed explanation. module AssociationConditions def condition?(name) # :nodoc: super || association_condition?(name) end private def association_condition?(name) !association_condition_details(name).nil? unless name.to_s.downcase.match("_or_") end def method_missing(name, *args, &block) if !local_condition?(name) && details = association_condition_details(name) create_association_condition(details[:association], details[:condition], args) send(name, *args) else super end end def association_condition_details(name, last_condition = nil) assocs = reflect_on_all_associations.reject { |assoc| assoc.options[:polymorphic] }.sort { |a, b| b.name.to_s.size <=> a.name.to_s.size } return nil if assocs.empty? name_with_condition = [name, last_condition].compact.join('_') if name_with_condition.to_s =~ /^(#{assocs.collect(&:name).join("|")})_(\w+)$/ association_name = $1 condition = $2 association = reflect_on_association(association_name.to_sym) klass = association.klass if klass.condition?(condition) {:association => $1, :condition => $2} else nil end end end def create_association_condition(association, condition, args) named_scope("#{association}_#{condition}", association_condition_options(association, condition, args)) end def association_condition_options(association_name, association_condition, args) association = reflect_on_association(association_name.to_sym) scope = association.klass.send(association_condition, *args) scope_options = association.klass.named_scope_options(association_condition) arity = association.klass.named_scope_arity(association_condition) if !arity || arity == 0 # The underlying condition doesn't require any parameters, so let's just create a simple # named scope that is based on a hash. options = scope.scope(:find) prepare_named_scope_options(options, association) options else proc_args = arity_args(arity) arg_type = (scope_options.respond_to?(:searchlogic_arg_type) && scope_options.searchlogic_arg_type) || :string eval <<-"end_eval" searchlogic_lambda(:#{arg_type}) { |#{proc_args.join(",")}| scope = association.klass.send(association_condition, #{proc_args.join(",")}) options = scope ? scope.scope(:find) : {} prepare_named_scope_options(options, association) options } end_eval end end # Used to match the new scopes parameters to the underlying scope. This way we can disguise the # new scope as best as possible instead of taking the easy way out and using *args. def arity_args(arity) args = [] if arity > 0 arity.times { |i| args << "arg#{i}" } else positive_arity = arity * -1 positive_arity.times do |i| if i == (positive_arity - 1) args << "*arg#{i}" else args << "arg#{i}" end end end args end def prepare_named_scope_options(options, association) options.delete(:readonly) # AR likes to set :readonly to true when using the :joins option, we don't want that options[:conditions] = association.klass.sanitize_sql_for_conditions(options[:conditions]) if options[:conditions].is_a?(Hash) if options[:joins].is_a?(String) || array_of_strings?(options[:joins]) options[:joins] = [inner_joins(association.name), options[:joins]].flatten else options[:joins] = options[:joins].blank? ? association.name : {association.name => options[:joins]} end end end end end