假设我有一个像这样的多态结构。
map.resources :bar, :has_many => :foo
map.resources :baz, :has_many => :foo
map.resources :qux, :has_many => :foo
class Foo
belongs_to :parent, :polymorphic => true
end
class FooController < AC
before_filter :find_parent
...
private
def find_parent
# ugly way:
@parent = if params[:bar_id]
Bar.find(params[:bar_id])
elsif params[:baz_id]
Baz.find(params[:baz_id])
elsif params[:qux_id]
Qux.find(params[:qux_id])
end
end
end
那是相当丑陋的。每当我们添加一个它可能属于的新事物时,我们都需要将它 un-DRYly 添加到 before_filter 中。
它也变得更糟。假设 Foos真的是多态的东西,可以出现在任何地方,比如评论或标签。并假设您有以下路线:
map.resources :bar, :has_many => :foo do |bar|
bar.resources :baz, :has_many => :foo
end
map.resources :qux, :has_many => :foo do |qux|
qux.resources :baz, :has_many => :foo
end
...现在我们必须担心是先检查 bar_id 还是 baz_id。
对于更复杂的资源,这甚至可能不足以确保您获得父母的 ID。
如果我们可以做这样的事情,那将是理想的:
def get_parent
# fetch the parameter that immediately preceeded :id
@parent = if x = params.before(:id)
# polymorphic find
x.key.to_s[0..-4].classify.constantize.find x.value
end
end
毕竟,我们的路由已经根据 URL 中参数的顺序对父级进行了编码。为什么要丢弃这些信息?
那么:如何做到这一点?