我为@miniature
模型开发了新的/创建操作和表单,它是嵌套模型@scales
。我无法正确进行更新/编辑操作。应该很简单,但我很困难。
@miniature
has_many @scales
到@sizes
。
在我的@miniature
模型中
has_many :sizes, dependent: :destroy
has_many :scales, :through => :sizes
accepts_nested_attributes_for :sizes, allow_destroy: true
在控制器中我有
def new
@miniature = Miniature.new
@all_scales = Scale.all
@size = @miniature.sizes.build
end
def create
@miniature = Miniature.new(miniature_params)
params[:scales][:id].each do |scale|
if !scale.empty?
@miniature.sizes.build(:scale_id => scale)
end
end
if @miniature.save
redirect_to miniature
else
render 'new'
end
end
private
def miniature_params
params.require(:miniature).permit(:name, :release_date, :material, :pcode, :notes, sizes_attributes: [:id, :scale_id, :miniset_id])
end
虽然有效,但编辑和更新操作却没有。我的编辑设置与我的编辑相同。
def edit
@miniature = Miniature.find(params[:id])
@all_scales = Scale.all
@size = @miniature.sizes.build
end
我原以为更新miniature params
会更新@sizes
模型,但它不会
def update
@miniature = Miniature.find(params[:id])
if @miniature.update_attributes(miniature_params)
flash[:success] = "Miniature updated"
redirect_to @miniature
else
render 'edit'
end
end
它目前更新@miniature
但不更新@sizes
信息。我的问题出现在编辑或更新中,还是两者都有?
表格的相关部分:
<%= f.fields_for(@size) do |sf| %>
<%= sf.label simple_pluralize(@miniature.scales.count, 'Scale') %>
<%= collection_select( :scales, :id, @all_scales, :id, :name,
{:selected => @miniature.scales.map(&:id)},
{class: 'multiselect', multiple: true}) %>
<% end %>
任何帮助或指示进一步阅读非常感谢。即使只是说“你忽略了这个显而易见的事情,去做更多的阅读/工作”。
似乎我需要update
行动更类似于我的create
行动的if
声明?
更新
这是提交编辑时来自我的服务器日志的PATCH请求:
Started PATCH "/miniatures/21" for 127.0.0.1 at 2014-04-02 16:00:10 +0100
Processing by MiniaturesController#update as HTML
Parameters: {"utf8"=>"✓", "authenticity_token"=>"jQ79L1Exx83C47jnCF3nsWQ2tV07tRwKfI8wNeLzojo=", "miniature"=>{"name"=>"Test Miniature", "material"=>"Metal", "pcode"=>"123123123123123", "release_date(1i)"=>"2013", "release_date(2i)"=>"2", "release_date(3i)"=>"2", "notes"=>""}, "scales"=>{"id"=>["", "2"]}, "commit"=>"Save changes", "id"=>"21"}
User Load (0.7ms) SELECT "users".* FROM "users" WHERE "users"."id" = 4 ORDER BY "users"."id" ASC LIMIT 1
Miniature Load (0.2ms) SELECT "miniatures".* FROM "miniatures" WHERE "miniatures"."id" = ? LIMIT 1 [["id", "21"]]
(0.1ms) begin transaction
(0.2ms) commit transaction
Redirected to http://localhost:3000/miniatures/21
Completed 302 Found in 12ms (ActiveRecord: 1.3ms)
观察者更新
我正在使用fields_for,因为这是我设法让新的/ create操作使用的。如果有更好的方法,至少据我所知,我并不依赖它。
我的路线有
resources :miniatures do
collection do
get :scales
get :collections
get :lines
get :contents
post :import
end
member do
get :minisets, :setminis
end
end
并进一步向下
resources :sizes
resources :scales
我认为我的路由文件通常有点混乱,需要进行重构。
答案 0 :(得分:5)
根据服务器日志条目(见下文),我看到在params
哈希中,scales
(以粗体突出显示)作为单独的哈希传递而不是 miniature
:
参数:{“utf8”=&gt;“✓”,“authenticity_token”=&gt;“jQ79L1Exx83C47jnCF3nsWQ2tV07tRwKfI8wNeLzojo =”, “微型”=&gt; {“名称”=&gt;“测试微型”,“材料”=&gt;“金属”, “pcode”=&gt;“123123123123123”,“release_date(1i)”=&gt;“2013”, “release_date(2i)”=&gt;“2”,“release_date(3i)”=&gt;“2”,“notes”=&gt;“”}, “scales”=&gt; {“id”=&gt; [“”,“2”]} ,“commit”=&gt;“保存更改”,“id”=&gt;“21” }
将update
操作更改为:
def update
@miniature = Miniature.find(params[:id])
if params[:scales][:id]
## Convert ["", "1","2","4","8"] to [1,2,4,8]
params[:scales][:id] = params[:scales][:id].reject(&:empty?).map(&:to_i)
## Get the scale_id from sizes already present in database [1,2,5,6]
old_scales = @miniature.sizes.pluck(:scale_id)
## Find the new scales to be added [1,2,4,8] - [1,2,5,6] = [4,8]
new_scales = params[:scales][:id] - old_scales
## Find the old_scales to be deleted [1,2,5,6] - [1,2,4,8] = [5,6]
old_scales = old_scales - params[:scales][:id]
## Build new_scales [4,8]
new_scales.each do |scale|
@miniature.sizes.build(:scale_id => scale)
end
## Delete old_scales [5,6]
Size.delete_all(:scale_id => old_scales)
end
if @miniature.update_attributes(miniature_params)
flash[:success] = "Miniature updated"
redirect_to @miniature
else
render 'edit'
end
end
更新create
操作,以便将scale_id
作为整数传递而不是字符串:
def create
@miniature = Miniature.new(miniature_params)
if params[:scales][:id]
## Convert ["", "1","2","4","8"] to [1,2,4,8]
params[:scales][:id] = params[:scales][:id].reject(&:empty?).map(&:to_i)
params[:scales][:id].each do |scale|
@miniature.sizes.build(:scale_id => scale)
end
end
if @miniature.save
redirect_to miniature
else
render 'new'
end
end