searchize与globalize2?

时间:2009-11-01 20:11:34

标签: ruby-on-rails searchlogic

鉴于有一个模型:

class MenuItem < ActiveRecord::Base
  translates :title
end
插入了

和searchlogic,我希望以下内容能够正常工作:

>> MenuItem.search(:title_like => 'tea')

可悲的是,它没有:

Searchlogic::Search::UnknownConditionError: The title_like is not a valid condition. You may only use conditions that map to a named scope

有没有办法让工作?


P.S。 我最接近工作的是:

>> MenuItem.search(:globalize_translations_title_like => 'tea')

哪个看起来不太好看。

1 个答案:

答案 0 :(得分:1)

我开发了searchlogic。默认情况下,它利用现有的命名范围和数据库列。它实际上不能超越它,因为最终它必须使用有效的列名创建结果SQL。也就是说,searchlogic无法清楚地理解你的:title属性的含义。即使它确实如此,它也将特定于您的翻译库中定义的逻辑。这是一个红旗,这不应该在库本身,而是在您的应用程序中初始化的插件或代码。

为什么不重写method_missing方法并自己进行映射? Searchlogic通过执行alias_scope提供了简单的别名范围:

alias_scope :title_like, lambda { |value| globalize_translations_title_like(value) }

这是一个快速刺(这是未经测试的):

module TranslationsMapping
  def self.included(klass)
    klass.class_eval do
      extend ClassMethods
    end
  end

  module ClassMethods
    protected
      def method_missing(name, *args, &block)
        translation_attributes = ["title"].join("|")
        conditions = (Searchlogic::NamedScopes::Conditions::PRIMARY_CONDITIONS + 
          Searchlogic::NamedScopes::Conditions::ALIAS_CONDITIONS).join("|"))

        if name.to_s =~ /^(#{translation_attributes})_(#{conditions})$/
          attribute_name = $1
          condition_name = $2
          alias_scope "#{attribute_name}_#{condition_name}", lambda { |value| send("globalize_translations_#{attribute_name}_#{condition_name}", value) }
          send(name, *args, &block)
        else
          super
        end
      end
   end
end

ActiveRecord::Base.send(:include, TranslationsMapping)

希望有所帮助。同样,我还没有测试过代码,但你应该得到一般的想法。但我同意,翻译的实现应该在幕后,你真的不应该在你的应用程序的任何地方输入“globalize_translations”,这应该在模型级别上透明地处理。