接受的答案需要手动修改 main_app 的父模型,以便将 has_many 关系设置为引擎的子模型。因此,每次将引擎添加到 main_apps 时,您都必须进入 main_apps 模型并手动设置所有必需的关系。
一个更健壮但更复杂的解决方案是在引擎中使用装饰器模式,以便引擎自动配置 main_app 的父模型及其所需的关系。
通过使用此方法,您只需在 main_app 中向引擎初始化程序添加一个设置,引擎将处理其余部分。
在引擎中:
blog.gemspec.rb
s.add_dependency 'decorators' #this will install the decorators gem for use in engine
lib/blog/blog.rb
module Blog
class Engine < ::Rails::Engine
isolate_namespace Blog
engine_name 'blog'
#to set up main_app objects via decorators in engine
config.to_prepare do
Decorators.register! Engine.root, Rails.root
end
end
end
lib/blog.rb
require 'decorators'
module Blog
mattr_accessor :user_class #Can now reference this setting as Blog.user_class
class << self
#the following lets us add functionality to main_app user model
def decorate_user_class!
Blog.user_class.class_eval do
has_many :articles, :class_name => "Blog::Article", :foreign_key => "user_id"
end
end
end
end
app/decorators/lib/blog/user_class_decorator.rb
if Blog.user_class
Blog.decorate_user_class!
else
raise "Blog.user_class must be set in main_app blog.rb initializer"
end
在主应用中:
app/initializers/blog.rb
Blog.user_class = User
如果您从主应用程序运行 rails 控制台,您将看到已正确设置关系。引擎中的装饰器模式也可用于以不同方式扩展 main_app 的模型和控制器,而不仅仅是 Activerecord 关系。实现了几乎完全的解耦!