0

我有一个产品表和一个相关的产品表,以一种方式进行相关工作:

Product(id: int, name: string)
RelatedProduct(id: int, product1: int, product2: int)

使用以下型号:

class Product < ActiveRecord::Base
  validates :name, :uniqueness => true, :presence => true
  has_many :relations, :class_name => "RelatedProducts", :foreign_key => :product1, inverse_of: :source
  has_many :related_products, through: :relations, :source => :destination
end

class RelatedProducts < ActiveRecord::Base
  belongs_to :source, :class_name => "Product", inverse_of: :relations
  belongs_to :destination, :foreign_key => :product2, :class_name => "Product"
end

如果我预填充 RelatedProducts 表,这将有效,显示视图将正确显示所有相关产品。我无法弄清楚如何从 html 填充相关产品?

<%= form_for(@product) do |f| %>

  <div class="field">
    <%= f.label :name %>
    <%= f.text_field :name %>
  </div>

  <div class="related-products field">
    <h3>Related Products</h3>
    <%= f.collection_check_boxes :related_products, Product.where.not(id: @product.id), :self, :name %>
  </div>

  <div class="actions">
    <%= f.submit 'Update Products' %>
  </div>
<% end %>

强参数定义为:

params.require(:product).permit(:name, related_products: [:id, :name])
4

1 回答 1

0

我的解决方案是让视图使用 id 然后提交它们

<div class="related-products field">
  <h3>Related Products</h3>
  <%= f.collection_check_boxes :related_products, Product.where.not(id: @product.id), :id, :name %>
</div>

然后我将它与模型中的一个函数结合起来,将 id 列表转换为产品列表

def related_products_list=(ids)
  self.related_products = ids.reject(&:empty?).collect { |id| Product.find(id) }
end

这适用于提交,但不会在编辑时预先检查复选框。这是因为模型谈论产品,但视图有一个 id 列表,因此不会发现匹配项。我使用的解决方案是手动循环遍历集合以创建复选框,而不是使用 collection_check_boxes。

<% Product.where.not(id: @product.id).each do |p| %>
  <%= check_box_tag 'product[related_products][]', p.id, @product.is_related_to?(p), id: "product_related_products_" + p.id.to_s %>
  <%= label_tag "product_related_products_" + p.id.to_s, p.name %>
<% end %>
于 2013-08-01T20:30:54.220 回答