lib/ransack/translate.rb



require 'i18n'

I18n.load_path += Dir[
  File.join(File.dirname(__FILE__), 'locale'.freeze, '*.yml'.freeze)
]

module Ransack
  module Translate
    def self.word(key, options = {})
      I18n.translate(:"ransack.#{key}", :default => key.to_s)
    end

    def self.predicate(key, options = {})
      I18n.translate(:"ransack.predicates.#{key}", :default => key.to_s)
    end

    def self.attribute(key, options = {})
      unless context = options.delete(:context)
        raise ArgumentError, "A context is required to translate attributes"
      end

      original_name = key.to_s
      base_class = context.klass
      base_ancestors = base_class.ancestors.select {
        |x| x.respond_to?(:model_name)
      }
      predicate = Predicate.detect_from_string(original_name)
      attributes_str = original_name.sub(/_#{predicate}$/, ''.freeze)
      attribute_names = attributes_str.split(/_and_|_or_/)
      combinator = attributes_str.match(/_and_/) ? :and : :or
      defaults = base_ancestors.map do |klass|
        "ransack.attributes.#{i18n_key(klass)}.#{original_name}".to_sym
      end

      translated_names = attribute_names.map do |name|
        attribute_name(context, name, options[:include_associations])
      end

      interpolations = {
        :attributes => translated_names.join(" #{Translate.word(combinator)} ")
      }

      if predicate
        defaults << "%{attributes} %{predicate}".freeze
        interpolations[:predicate] = Translate.predicate(predicate)
      else
        defaults << "%{attributes}".freeze
      end

      defaults << options.delete(:default) if options[:default]
      options.reverse_merge! :count => 1, :default => defaults
      I18n.translate(defaults.shift, options.merge(interpolations))
    end

    def self.association(key, options = {})
      unless context = options.delete(:context)
        raise ArgumentError, "A context is required to translate associations"
      end

      defaults =
        if key.blank?
          [:"ransack.models.#{i18n_key(context.klass)}",
           :"#{context.klass.i18n_scope}.models.#{i18n_key(context.klass)}"]
        else
          [:"ransack.associations.#{i18n_key(context.klass)}.#{key}"]
        end
      defaults << context.traverse(key).model_name.human
      options = { :count => 1, :default => defaults }
      I18n.translate(defaults.shift, options)
    end

  private

    def self.attribute_name(context, name, include_associations = nil)
      @context, @name = context, name
      @assoc_path = context.association_path(name)
      @attr_name = @name.sub(/^#{@assoc_path}_/, ''.freeze)
      associated_class = @context.traverse(@assoc_path) if @assoc_path.present?
      @include_associated = include_associations && associated_class

      defaults = default_attribute_name << fallback_args
      options = { :count => 1, :default => defaults }
      interpolations = build_interpolations(associated_class)

      I18n.translate(defaults.shift, options.merge(interpolations))
    end

    def self.default_attribute_name
      ["ransack.attributes.#{i18n_key(@context.klass)}.#{@name}".to_sym]
    end

    def self.fallback_args
      if @include_associated
        '%{association_name} %{attr_fallback_name}'.freeze
      else
        '%{attr_fallback_name}'.freeze
      end
    end

    def self.build_interpolations(associated_class)
      {
        :attr_fallback_name => attr_fallback_name(associated_class),
        :association_name   => association_name
      }
      .reject { |_, value| value.nil? }
    end

    def self.attr_fallback_name(associated_class)
      I18n.t(
        :"ransack.attributes.#{fallback_class(associated_class)}.#{@attr_name}",
        :default => default_interpolation(associated_class)
        )
    end

    def self.fallback_class(associated_class)
      i18n_key(associated_class || @context.klass)
    end

    def self.association_name
      association(@assoc_path, :context => @context) if @include_associated
    end

    def self.default_interpolation(associated_class)
      [
        associated_attribute(associated_class),
        ".attributes.#{@attr_name}".to_sym,
        @attr_name.humanize
      ]
      .flatten
    end

    def self.associated_attribute(associated_class)
      if associated_class
        translated_attribute(associated_class)
      else
        translated_ancestor_attributes
      end
    end

    def self.translated_attribute(associated_class)
      key = "#{associated_class.i18n_scope}.attributes.#{
        i18n_key(associated_class)}.#{@attr_name}"
      ["#{key}.one".to_sym, key.to_sym]
    end

    def self.translated_ancestor_attributes
      @context.klass.ancestors
      .select { |ancestor| ancestor.respond_to?(:model_name) }
      .map { |ancestor| translated_attribute(ancestor) }
    end

    def self.i18n_key(klass)
      raise "not implemented"
    end
  end
end