Sha256: 736f508393bed973720f03cfd266f981941483617d9d7a0e5d7eaa321454c706

Contents?: true

Size: 1.86 KB

Versions: 27

Compression:

Stored size: 1.86 KB

Contents

# Based on ActiveRecord::NamedScope
module MongoDoc
  module Scope
    def scopes
      read_inheritable_attribute(:scopes) || write_inheritable_attribute(:scopes, {})
    end

    def scope(name, *args, &block)
      options = args.extract_options!
      raise ArgumentError if args.size != 1
      criteria = args.first
      name = name.to_sym
      scopes[name] = lambda do |parent_scope, *args|
        CriteriaProxy.new(parent_scope, Mongoid::Criteria === criteria ? criteria : criteria.call(*args), options, &block)
      end
      (class << self; self; end).class_eval <<-EOT
        def #{name}(*args)
          scopes[:#{name}].call(self, *args)
        end
      EOT
    end

    class CriteriaProxy
      attr_accessor :criteria, :klass, :parent_scope

      delegate :scopes, :to => :parent_scope

      def initialize(parent_scope, criteria, options, &block)
        [options.delete(:extend)].flatten.each { |extension| extend extension } if options.include?(:extend)
        extend Module.new(&block) if block_given?
        if CriteriaProxy === parent_scope
          chained = Mongoid::Criteria.new(klass)
          chained.merge(parent_scope)
          chained.merge(criteria)
          self.criteria = chained
          self.klass = criteria.klass
        else
          self.criteria = criteria
          self.klass = parent_scope
        end

        self.parent_scope = parent_scope
      end

      def respond_to?(method, include_private = false)
        return true if scopes.include?(method)
        criteria.respond_to?(method, include_private)
      end

      private

      def method_missing(method, *args, &block)
        if scopes.include?(method)
          scopes[method].call(self, *args)
        else
          chained = Mongoid::Criteria.new(klass)
          chained.merge(criteria)
          chained.send(method, *args, &block)
        end
      end
    end
  end
end

Version data entries

27 entries across 27 versions & 3 rubygems

Version Path
mongo_doc-0.6.16 lib/mongo_doc/scope.rb
mongo_doc-0.6.15 lib/mongo_doc/scope.rb
mongo_doc-0.6.14 lib/mongo_doc/scope.rb
mongo_doc-0.6.13 lib/mongo_doc/scope.rb
mongo_doc-0.6.12 lib/mongo_doc/scope.rb
mongo_doc-0.6.11 lib/mongo_doc/scope.rb
mongo_doc-0.6.10 lib/mongo_doc/scope.rb
mongo_doc-0.6.9 lib/mongo_doc/scope.rb
mongo_doc-0.6.8 lib/mongo_doc/scope.rb
mongo_doc-0.6.7 lib/mongo_doc/scope.rb
mongo_doc-0.6.6 lib/mongo_doc/scope.rb
mongo_doc-0.6.5 lib/mongo_doc/scope.rb
mongo_doc-0.6.4 lib/mongo_doc/scope.rb
mongo_doc_rails2-0.6.2 lib/mongo_doc/scope.rb
mongo_doc-0.6.3 lib/mongo_doc/scope.rb
mongo_doc-0.6.2 lib/mongo_doc/scope.rb
mongo_doc-0.6.1 lib/mongo_doc/scope.rb
mongo_doc-0.6.0 lib/mongo_doc/scope.rb
mongo_doc_rails2-0.6.1 lib/mongo_doc/scope.rb
mongo_doc-0.5.5 lib/mongo_doc/scope.rb