Rails编辑/更新操作不适用于嵌套项目

时间:2014-03-05 12:08:49

标签: ruby-on-rails ruby-on-rails-4 controller nested-attributes

我为@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声明?

JKen13579的

更新

这是提交编辑时来自我的服务器日志的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

我认为我的路由文件通常有点混乱,需要进行重构。

1 个答案:

答案 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