首页 > 代码库 > RailsCasts17 HABTM Checkboxes 更新多对多的关系

RailsCasts17 HABTM Checkboxes 更新多对多的关系

两个model,Product and Category;彼此之间有has_and_belongs_to_many的关系;期望在编辑project时,同时编辑project与category之间的关系;编辑project的界面如下:

http://asciicasts.com/system/photos/13/original/E017I01.png

添加 checkboxes

check_box and check_box_tag方法均可向form中添加checkboxes;由于需要重写checkbox的name,故选用check_box_tag,编辑project的代码如下:

ruby" style="clear: both;

<% for category in Category.find(:all) %>
    <div>
      <%= check_box_tag "product[category_ids][]", category.id, @product.categories.include?(category) %>
      <%= category.name %>
    </div>
<% end %>
check_box_tag 需要checkbox的name、value、是否被选中三个参数,之后说明为何name如此命名;

Seeing if it works

The edit form with the checkboxes added.

执行后,发现提交后,被选中的内容已被更新完成,rails是如何工作的呢?如下的log会给出答案:

terminal

Processing ProductsController#update (for 127.0.0.1 at 2009-01-15 20:57:56) [PUT]
Parameters: {"commit"=>"Edit", "authenticity_token"=>"31b711f2c24ae7cea5abf3f758eef46b472eebf3", "product"=>{"price"=>"99.0", "name"=>"Television Stand", "category_ids"=>["2", "4"]}, "id"=>"1"}

提交后,edit form将category 参数放在project hash中作为array传递出来。将checkboxes的name置为project[category_ids][]是正常工作的关键。名字的第一部分告诉rails category_ids将作为project hash的一部分,名字的第二部分[]说明传递的方式是array。当更新project时,category_ids的方法从何而来?来自project model的has_and_belongs_to_many。在script/console中手动更新的log如下:

terminal

>> p = Product.first
=> #<Product id: 1, name: "Television Stand", price: 99.0, created_at: "2009-01-11 21:32:12", updated_at: "2009-01-11 21:32:12">
>> p.category_ids
=> [2, 3]
>> p.category_ids = [1,4]
=> [1, 4]
>>

产生的sql如下:
sql

DELETE FROM "categories_products" WHERE product_id = 1 AND category_id IN (2,3)
INSERT INTO "categories_products" ("product_id", "id", "category_id") VALUES (1, 1, 1)
INSERT INTO "categories_products" ("product_id", "id", "category_id") VALUES (1, 4, 4)

一个小问题

在更新方法中仍有个小问题。若checkboxes中任何项都不选,则将project对应的所有category的操作会失败。主要原因是若checkboxes未有被选中的项,category_ids将不会出现在product的参数中,意味着category_ids不会被更新。

可以使用ruby的 ||=操作解决这个问题。

params[:product][:category_ids] ||= []

这保证了,当category不被选中时,置params[:product][:category_ids]为空,进而可更新category。


原文:http://railscasts.com/episodes/17-habtm-checkboxes?view=asciicast

RailsCasts17 HABTM Checkboxes 更新多对多的关系