两个model,Product
and Category
;彼此之间有has_and_belongs_to_many
的关系;期望在编辑project时,同时编辑project与category之间的关系;编辑project的界面如下:
添加 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
执行后,发现提交后,被选中的内容已被更新完成,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