class Ransack::Helpers::FormBuilder

def association_array(obj, prefix = nil)

def association_array(obj, prefix = nil)
  ([prefix] + association_object(obj))
  .compact
  .flat_map { |v| [prefix, v].compact.join(Constants::UNDERSCORE) }
end

def association_hash(obj)

def association_hash(obj)
  obj.map do |key, value|
    case value
    when Array, Hash
      association_array(value, key.to_s)
    else
      [key.to_s, [key, value].join(Constants::UNDERSCORE)]
    end
  end
end

def association_object(obj)

def association_object(obj)
  case obj
  when Array
    obj
  when Hash
    association_hash(obj)
  else
    [obj]
  end
end

def attr_from_base_and_column(base, column)

def attr_from_base_and_column(base, column)
  [base, column].reject(&:blank?).join(Constants::UNDERSCORE)
end

def attribute_collection_for_base(attributes, base = nil)

def attribute_collection_for_base(attributes, base = nil)
  attributes.map do |c|
    [
      attr_from_base_and_column(base, c),
      Translate.attribute(
        attr_from_base_and_column(base, c), context: object.context
      )
    ]
  end
end

def attribute_collection_for_bases(action, bases)

def attribute_collection_for_bases(action, bases)
  bases.map { |base| get_attribute_element(action, base) }.compact
end

def attribute_fields(*args, &block)

def attribute_fields(*args, &block)
  search_fields(:a, args, block)
end

def attribute_select(options = nil, html_options = nil, action = nil)

def attribute_select(options = nil, html_options = nil, action = nil)
  options ||= {}
  html_options ||= {}
  action ||= Constants::SEARCH
  default = options.delete(:default)
  raise ArgumentError, formbuilder_error_message(
    "#{action}_select") unless object.respond_to?(:context)
  options[:include_blank] = true unless options.has_key?(:include_blank)
  bases = [''.freeze].freeze + association_array(options[:associations])
  if bases.size > 1
    collection = attribute_collection_for_bases(action, bases)
    object.name ||= default if can_use_default?(
      default, :name, mapped_values(collection.flatten(2))
      )
    template_grouped_collection_select(collection, options, html_options)
  else
    collection = collection_for_base(action, bases.first)
    object.name ||= default if can_use_default?(
      default, :name, mapped_values(collection)
      )
    template_collection_select(:name, collection, options, html_options)
  end
end

def can_use_default?(default, attribute, values)

def can_use_default?(default, attribute, values)
  object.respond_to?("#{attribute}=") && default &&
    values.include?(default.to_s)
end

def collection_for_base(action, base)

def collection_for_base(action, base)
  attribute_collection_for_base(
    object.context.send("#{action}able_attributes", base), base)
end

def combinator_choices

def combinator_choices
  if Nodes::Condition === object
    [
      [Constants::OR,  Translate.word(:any)],
      [Constants::AND, Translate.word(:all)]
    ]
  else
    [
      [Constants::AND, Translate.word(:all)],
      [Constants::OR,  Translate.word(:any)]
    ]
  end
end

def combinator_select(options = {}, html_options = {})

def combinator_select(options = {}, html_options = {})
  template_collection_select(
    :m, combinator_choices, options, html_options)
end

def condition_fields(*args, &block)

def condition_fields(*args, &block)
  search_fields(:c, args, block)
end

def formbuilder_error_message(action)

def formbuilder_error_message(action)
  "#{action.sub(Constants::SEARCH, Constants::ATTRIBUTE)
    } must be called inside a search FormBuilder!"
end

def get_attribute_element(action, base)

def get_attribute_element(action, base)
  begin
    [
      Translate.association(base, context: object.context),
      collection_for_base(action, base)
    ]
  rescue UntraversableAssociationError
    nil
  end
end

def grouping_fields(*args, &block)

def grouping_fields(*args, &block)
  search_fields(:g, args, block)
end

def label(method, *args, &block)

def label(method, *args, &block)
  options = args.extract_options!
  text = args.first
  i18n = options[:i18n] || {}
  text ||= object.translate(
    method, i18n.reverse_merge(include_associations: true)
    ) if object.respond_to? :translate
  super(method, text, options, &block)
end

def mapped_values(values)

def mapped_values(values)
  values.map { |v| v.is_a?(Array) ? v.first : nil }.compact
end

def predicate_fields(*args, &block)

def predicate_fields(*args, &block)
  search_fields(:p, args, block)
end

def predicate_select(options = {}, html_options = {})

def predicate_select(options = {}, html_options = {})
  options[:compounds] = true if options[:compounds].nil?
  default = options.delete(:default) || Constants::CONT
  keys =
  if options[:compounds]
    Predicate.names
  else
    Predicate.names.reject { |k| k.match(/_(any|all)$/) }
  end
  if only = options[:only]
    if only.respond_to? :call
      keys = keys.select { |k| only.call(k) }
    else
      only = Array.wrap(only).map(&:to_s)
      keys = keys.select {
        |k| only.include? k.sub(/_(any|all)$/, ''.freeze)
      }
    end
  end
  collection = keys.map { |k| [k, Translate.predicate(k)] }
  object.predicate ||= Predicate.named(default) if
    can_use_default?(default, :predicate, keys)
  template_collection_select(:p, collection, options, html_options)
end

def search_fields(name, args, block)

def search_fields(name, args, block)
  args << {} unless args.last.is_a?(Hash)
  args.last[:builder] ||= options[:builder]
  args.last[:parent_builder] = self
  options = args.extract_options!
  objects = args.shift
  objects ||= @object.send(name)
  objects = [objects] unless Array === objects
  name = "#{options[:object_name] || object_name}[#{name}]"
  objects.inject(ActiveSupport::SafeBuffer.new) do |output, child|
    output << @template.fields_for("#{name}[#{options[:child_index] ||
    nested_child_index(name)}]", child, options, &block)
  end
end

def sort_array

def sort_array
  [
    ['asc'.freeze,  object.translate('asc'.freeze)].freeze,
    ['desc'.freeze, object.translate('desc'.freeze)].freeze
  ].freeze
end

def sort_direction_select(options = {}, html_options = {})

def sort_direction_select(options = {}, html_options = {})
  unless object.respond_to?(:context)
    raise ArgumentError,
    formbuilder_error_message('sort_direction'.freeze)
  end
  template_collection_select(:dir, sort_array, options, html_options)
end

def sort_fields(*args, &block)

def sort_fields(*args, &block)
  search_fields(:s, args, block)
end

def sort_link(attribute, *args)

def sort_link(attribute, *args)
  @template.sort_link @object, attribute, *args
end

def sort_select(options = {}, html_options = {})

def sort_select(options = {}, html_options = {})
  attribute_select(options, html_options, 'sort'.freeze) +
  sort_direction_select(options, html_options)
end

def sort_url(attribute, *args)

def sort_url(attribute, *args)
  @template.sort_url @object, attribute, *args
end

def submit(value = nil, options = {})

def submit(value = nil, options = {})
  value, options = nil, value if value.is_a?(Hash)
  value ||= Translate.word(:search).titleize
  super(value, options)
end

def template_collection_select(name, collection, options, html_options)

def template_collection_select(name, collection, options, html_options)
  @template.collection_select(
    @object_name, name, collection, :first, :last,
    objectify_options(options), @default_options.merge(html_options)
    )
end

def template_grouped_collection_select(collection, options, html_options)

def template_grouped_collection_select(collection, options, html_options)
  @template.grouped_collection_select(
    @object_name, :name, collection, :last, :first, :first, :last,
    objectify_options(options), @default_options.merge(html_options)
    )
end

def value_fields(*args, &block)

def value_fields(*args, &block)
  search_fields(:v, args, block)
end