vfonic/liquid4-rails5

View on GitHub
lib/liquid-rails/drops/collection_drop.rb

Summary

Maintainability
A
35 mins
Test Coverage
module Liquid
  module Rails
    class CollectionDrop < ::Liquid::Drop

      class << self
        attr_accessor :_scopes
      end

      def self.inherited(base)
        base._scopes = []
      end

      def self.scope(*scope_names)
        @_scopes.concat scope_names

        scope_names.each do |scope_name|
          define_method(scope_name) do
            value = instance_variable_get("@_#{scope_name}")
            return value if value

            raise ::Liquid::ArgumentError, "#{objects.class.name} doesn't define scope: #{scope_name}" unless objects.respond_to?(scope_name)
            instance_variable_set("@_#{scope_name}", self.class.new(objects.send(scope_name)))
          end
        end
      end

      array_methods = Array.instance_methods - Object.instance_methods
      delegate *array_methods, to: :dropped_collection
      delegate :total_count, :total_pages, to: :objects

      def initialize(objects, options={})
        options.assert_valid_keys(:with, :scope)

        @objects         = options[:scope].nil? ? objects : objects.send(options[:scope])
        @drop_class_name = options[:with]
      end

      def page(number)
        self.class.new(objects.page(number))
      end

      def per(number)
        self.class.new(objects.per(number))
      end

      def dropped_collection
        @dropped_collection ||= @objects.map { |item| drop_item(item) }
      end

      def kind_of?(klass)
        dropped_collection.kind_of?(klass) || super
      end
      alias_method :is_a?, :kind_of?

      ## :[] is invoked by parser before the actual. However, this method has the same name as array method.
      ## Override this, so it will work for both cases.
      ## {{ post_drop.comments[0] }}
      ## {{ post_drop.<other_methods> }}
      def [](method)
        if method.is_a?(Integer)
          dropped_collection.at(method)
        else
          public_send(method)
        end
      end

      ## Need to override this. I don't understand too, otherwise it will return an array of drop objects.
      ## Need to return self so that we can do chaining.
      def to_liquid
        self
      end

      def inspect
        "#<#{self.class.name} of #{drop_class} for #{objects.inspect}>"
      end

      protected

        attr_reader :objects

        def drop_class
          @drop_class ||= @drop_class_name.is_a?(String) ? @drop_class_name.safe_constantize : @drop_class_name
        end

        def drop_item(item)
          liquid_drop_class = drop_class || item.drop_class
          liquid_drop_class.new(item)
        end
    end
  end
end