require "active_support/core_ext/class/attribute"
module Datagrid
module Filters
require "datagrid/filters/base_filter"
require "datagrid/filters/enum_filter"
require "datagrid/filters/boolean_enum_filter"
require "datagrid/filters/extended_boolean_filter"
require "datagrid/filters/boolean_filter"
require "datagrid/filters/date_filter"
require "datagrid/filters/date_time_filter"
require "datagrid/filters/default_filter"
require "datagrid/filters/integer_filter"
require "datagrid/filters/composite_filters"
require "datagrid/filters/string_filter"
require "datagrid/filters/float_filter"
require "datagrid/filters/dynamic_filter"
FILTER_TYPES = {
:date => Filters::DateFilter,
:datetime => Filters::DateTimeFilter,
:string => Filters::StringFilter,
:default => Filters::DefaultFilter,
:eboolean => Filters::BooleanEnumFilter ,
:xboolean => Filters::ExtendedBooleanFilter ,
:boolean => Filters::BooleanFilter ,
:integer => Filters::IntegerFilter,
:enum => Filters::EnumFilter,
:float => Filters::FloatFilter,
:dynamic => Filters::DynamicFilter
}
class DefaultFilterScope
end
def self.included(base) #:nodoc:
base.extend ClassMethods
base.class_eval do
include Datagrid::Core
include Datagrid::Filters::CompositeFilters
class_attribute :filters_array
self.filters_array = []
end
base.send :include, InstanceMethods
end # self.included
module ClassMethods
# Returns filter definition object by name
def filter_by_name(attribute)
return attribute if attribute.is_a?(Datagrid::Filters::BaseFilter)
self.filters.find do |filter|
filter.name.to_sym == attribute.to_sym
end
end
# Defines new datagrid filter.
# This method automatically generates attr_accessor for filter name
# and adds it to the list of datagrid attributes.
#
# Arguments:
#
# * name - filter name
# * type - filter type that defines type case and GUI representation of a filter
# * options - hash of options
# * block - proc to apply the filter
#
# Available options:
#
# * :header - determines the header of the filter
# * :default - the default filter value. Able to accept a Proc in case default should be recalculated
# * :range - if true, filter can accept two values that are treated as a range that will be used for filtering
# Not all of the filter types support this option. Here are the list of types that do:
# :integer, :float, :date, :datetime, :string
# * :multiple - if true multiple values can be assigned to this filter.
# If String is assigned as a filter value, it is parsed from string using a separator symbol (`,` by default).
# But you can specify a different separator as option value. Default: false.
# * :allow_nil - determines if the value can be nil
# * :allow_blank - determines if the value can be blank
# * :before - determines the position of this filter,
# by adding it before the filter passed here (when using datagrid_form_for helper)
# * :after - determines the position of this filter,
# by adding it after the filter passed here (when using datagrid_form_for helper)
# * :dummy - if true, this filter will not be applied automatically
# and will be just displayed in form. In case you may want to apply it manually.
# * :if - specify the condition when the filter can be dislayed and used.
# Accepts a block or a symbol with an instance method name
# * :unless - specify the reverse condition when the filter can be dislayed and used.
# Accepts a block or a symbol with an instance method name
#
# See: https://github.com/bogdan/datagrid/wiki/Filters for examples
def filter(name, type = :default, options = {}, &block)
if type.is_a?(Hash)
options = type
type = :default
end
klass = type.is_a?(Class) ? type : FILTER_TYPES[type]
raise ConfigurationError, "filter class #{type.inspect} not found" unless klass
position = Datagrid::Utils.extract_position_from_options(filters_array, options)
filter = klass.new(self, name, options, &block)
filters_array.insert(position, filter)
datagrid_attribute(name) do |value|
filter.parse_values(value)
end
end
def default_filter
DefaultFilterScope.new
end
def inspect
"#{super}(#{filters_inspection})"
end
def filters
filters_array
end
protected
def inherited(child_class)
super(child_class)
child_class.filters_array = self.filters_array.clone
end
def filters_inspection
return "no filters" if filters.empty?
filters.map do |filter|
"#{filter.name}: #{filter.type}"
end.join(", ")
end
end # ClassMethods
module InstanceMethods
def initialize(*args, &block) # :nodoc:
self.filters_array = self.class.filters_array.clone
self.filters_array.each do |filter|
self[filter.name] = filter.default(self)
end
super(*args, &block)
end
def assets # :nodoc:
apply_filters(super, filters)
end
# Returns filter value for given filter definition
def filter_value(filter)
self[filter.name]
end
# Returns string representation of filter value
def filter_value_as_string(name)
filter = filter_by_name(name)
value = filter_value(filter)
if value.is_a?(Array)
value.map {|v| filter.format(v) }.join(filter.separator)
else
filter.format(value)
end
end
# Returns filter object with the given name
def filter_by_name(name)
self.class.filter_by_name(name)
end
# Returns assets filtered only by specified filters
# Allows partial filtering
def filter_by(*filters)
apply_filters(scope, filters.map{|f| filter_by_name(f)})
end
# Returns select options for specific filter or filter name
# If given filter doesn't support select options raises `ArgumentError`
def select_options(filter)
filter = filter_by_name(filter)
unless filter.class.included_modules.include?(::Datagrid::Filters::SelectOptions)
raise ::Datagrid::ArgumentError, "#{filter.name} with type #{FILTER_TYPES.invert[filter.class].inspect} can not have select options"
end
filter.select(self)
end
def default_filter
self.class.default_filter
end
# Returns all currently enabled filters
def filters
self.class.filters.select do |filter|
filter.enabled?(self)
end
end
protected
def apply_filters(current_scope, filters)
filters.inject(current_scope) do |result, filter|
filter.apply(self, result, filter_value(filter))
end
end
end # InstanceMethods
end
end