6

我在我们的站点中使用了refinerycms来让技术含量较低的人员更新内容。在 gem 中,他们有一个 Page 类,用于映射站点上的每个顶级页面。我想在这个 Page 类上使用acts_as_taggable gem。现在我可以将acts_as_taggle 声明直接添加到page.rb 文件中,但是我必须维护一个单独的git repo 来跟踪我的版本和正式版本之间的差异。

基于 SO 上的其他一些问题,我创建了一个初始化程序和扩展,如下所示:

lib/page_extensions.rb:

module Pants
  module Extensions

    module Page
      module ClassMethods
        def add_taggable
          acts_as_taggable
        end
      end

      def self.included(base)
        base.extend(ClassMethods).add_taggable
      end

    end

  end
end

配置/初始化程序/pants.rb

require 'page_extensions'

Page.send :include, Pants::Extensions::Page

应用程序/视图/布局/application.html.erb

...
Tags: <%= @page.tag_list %>

我第一次从服务器请求页面时,它会正确输出页面上的所有标签。但是,如果我点击刷新,我会得到一个NoMethodError指示 tag_list 未定义。

我是 Rails 新手,所以也许我的假设是错误的,但我预计对 Page.send 的调用会对 Page 类而不是对类的特定实例进行永久性更改。那么如何在每个请求上将acts_as_taggable 添加到 Page 类中呢?

4

1 回答 1

14

您需要将 module_eval 代码放入一个config.to_prepare do块中。最容易做到这一点的地方是config/application.rb或创建一个引擎。代码是相同的,除了它在您每次运行站点时执行,而不仅仅是第一次(尤其适用于开发模式)和仅在初始化过程(也就是需要文件)进入config.before_initialize do块之前执行的代码。

重要的原因config.to_prepare是因为在开发模式下,每次请求都会重新加载代码,但初始化程序通常不会。这意味着Page,您正在运行 module_eval 的,只会让 module_eval 运行一次,但会在每次请求时重新加载自己。config.to_prepare是一个每次运行的 Rails 钩子,为这种情况提供了极大的便利。

config/application.rb 方法

class Application < Rails::Application
  # ... other stuff ...

  config.before_initialize do
    require 'page_extensions'
  end

  config.to_prepare do
    Page.send :include, Pants::Extensions::Page
  end
end

发动机方法

如果您不想修改config/application.rb,则可以在 Refinery CMS 中创建vendor/engines/add_page_extensions/lib/add_page_extensions.rb如下所示的内容:

require 'refinery'

module Refinery
  module AddPageExtensions
    class Engine < Rails::Engine

      config.before_initialize do
        require 'page_extensions'
      end

      config.to_prepare do
        Page.send :include, Pants::Extensions::Page
      end

    end
  end
end

如果您使用引擎方法,您还需要创建vendor/engines/add_page_extensions/add_page_extensions.gemspec应该包含一个简单的 gemspec:

Gem::Specification.new do |s|
  s.name = 'add_page_extensions'
  s.require_paths = %w(lib)
  s.version = 1.0
  s.files = Dir["lib/**/*"]
end

然后在你Gemfile添加这一行:

gem 'add_page_extensions', :path => 'vendor/engines'

如果您采用引擎方法,您可能希望将所有逻辑放在引擎lib目录中,包括Pants::Extensions::Page代码。

希望这可以帮助

于 2010-10-25T21:36:27.403 回答