Попытка использовать acceptpts_nested_attributes_for и has_and_belongs_to_many, но таблица соединения не заполняется - PullRequest
11 голосов
/ 04 декабря 2009

Я изучаю RoR и пытаюсь использовать acceptpts_nested_attributes_for и has_and_belongs_to_many для отправки информации, которая традиционно была бы двумя формами. Я читал, что на некоторых сайтах они совместимы, на некоторых сайтах они несовместимы, а некоторые сайты не знают. Для справки я использую Rails 2.3.4. Я попытался смоделировать свое решение из учебника Райана "Записки" на вложенных моделях

Из того, что я пытался отладить, кажется, что у меня есть две проблемы, но я не уверен, почему.

  1. Когда я отправляю форму с вложенными моделями, публикуется только часть информации о вложенных моделях. Я получаю только первое поле, а не "n" других, которые пользователь мог выбрать
  2. Из единственного поля, которое публикуется, нет ни одной строки, вставленной в таблицу соединения, которую я создал для отношения HABTM.

Вот фрагмент кода и соответствующие журналы для моей попытки вставки:

Модель адвоката:

class Attorney < ActiveRecord::Base 
  has_and_belongs_to_many :associations
  accepts_nested_attributes_for :associations, :reject_if => proc { |a| a['name'].blank? }
end

Модель ассоциации:

class Association < ActiveRecord::Base
  has_and_belongs_to_many :attorneys
  accepts_nested_attributes_for :attorneys
  validates_presence_of :name, :message => "Please enter an association name."
end

Контролер адвокатов:

def new
  @attorney = Attorney.new
  @attorney.associations.build

  respond_to do |format|
    format.html # new.html.erb
    format.xml  { render :xml => @attorney }
  end
end

def create
  @attorney = Attorney.new(params[:attorney])

  respond_to do |format|
    if @attorney.save
      flash[:notice] = 'Attorney was successfully created.'
      format.html { redirect_to(@attorney) }
      format.xml  { render :xml => @attorney, :status => :created, :location => @attorney }
    else
      format.html { render :action => "new" }
      format.xml  { render :xml => @attorney.errors, :status => :unprocessable_entity }
    end
  end
end

Новый взгляд адвоката:

<% form_for(@attorney, :html => {:multipart => true}) do |f| %>
  <%= f.error_messages %>
 <%= f.label :"First name" %> 
 <%= f.text_field :firstname %><br>

 <%= f.label :"Last Name" %> 
 <%= f.text_field :lastname %><br>

 <%= f.label :"Attorney Type" %> 
 <%= f.collection_select :member_type_id, MemberType.all, :id, :name %><br>

 <%= f.text_area :bio, :cols => 70, :rows => 20 %><br><br>

 <%= f.label :"Attorney Location" %> 
 <%= f.collection_select :office_location_id, OfficeLocation.all, :id, :location %><br>

 <div id="associations">
      <%= render :partial => 'shared/membership' %>
 </div>
 <%= add_association_link "Add Association" %>
    <%= f.submit 'Create' %>
<% end %>

Частичное членство:

<div class="association">
  <% fields_for :associations do |assoc_form| %>
    <%= assoc_form.collection_select(:association_id, Association.find(:all), :id, :name, :include_blank => true) %>

<% = link_to_function "remove", "$ (this) .up ('. Association'). Remove ()"%> <% = link_to 'Новая ассоциация', new_association_path%> <% end%>

Ссылка помощника адвоката:

def add_association_link(name)
  link_to_function name do |page|
    page.insert_html :bottom, :associations, :partial => 'shared/membership', :object => AssociationsAttorneys.new
  end
end

Миграция за присоединительным столом:

class CreateAssociationsAttorneys < ActiveRecord::Migration
  def self.up
    create_table :associations_attorneys do |t|
      t.references :attorney, :null => false
      t.references :association, :null => false
      t.timestamps
    end
  end

  def self.down
    drop_table :associations_attorneys
  end
end

Запись журнала:

    Processing AttorneysController#new (for 127.0.0.1 at 2009-12-04 08:16:19) [GET]
Rendering template within layouts/default
Rendering attorneys/new
  [4;35;1mMemberType Load (0.4ms)[0m   [0mSELECT * FROM "member_types" [0m
  [4;36;1mOfficeLocation Load (18.6ms)[0m   [0;1mSELECT * FROM "office_locations" [0m
  [4;35;1mAssociation Load (0.6ms)[0m   [0mSELECT * FROM "associations" [0m
Rendered shared/_membership (3.5ms)
  [4;36;1mCACHE (0.0ms)[0m   [0;1mSELECT * FROM "associations" [0m
Rendered shared/_membership (1.5ms)
Rendered shared/_nav (0.6ms)
Rendered shared/_footer (0.1ms)
Completed in 149ms (View: 114, DB: 20) | 200 OK [http://localhost/attorneys/new]

Processing ApplicationController#index (for 127.0.0.1 at 2009-12-04 08:16:19) [GET]

Processing AttorneysController#create (for 127.0.0.1 at 2009-12-04 08:16:57) [POST]
  Parameters: {"commit"=>"Create", "authenticity_token"=>"Jh7aMCcOY7jUu/D1YtiCswg2n6iwqnS98VnVn46psp0=", "associations"=>{"association_id"=>"3"}, "attorney"=>{"birthstate"=>"Alabama", "office_location_id"=>"1", "birthdate"=>"December 3, 2009", "birthcity"=>"Test", "middlename"=>"Test", "lastname"=>"Testing", "image_temp"=>"", "member_type_id"=>"2", "firstname"=>"Test", "bio"=>"testing testing testing", "suffix"=>"", "email"=>"testing@test.com"}}
  [4;35;1mAttorney Load (15.6ms)[0m   [0mSELECT "attorneys".id FROM "attorneys" WHERE ("attorneys"."email" = 'testing@test.com') LIMIT 1[0m
  [4;36;1mAttorney Create (0.8ms)[0m   [0;1mINSERT INTO "attorneys" ("birthstate", "created_at", "birthdate", "office_location_id", "birthcity", "updated_at", "middlename", "lastname", "firstname", "member_type_id", "suffix", "bio", "image", "email") VALUES('Alabama', '2009-12-04 15:16:57', 'December 3, 2009', 1, 'Test', '2009-12-04 15:16:57', 'Test', 'Testing', 'Test', 2, '', 'testing testing testing', NULL, 'testing@test.com')[0m
Redirected to http://localhost:3000/attorneys/11
Completed in 150ms (DB: 16) | 302 Found [http://localhost/attorneys]

Я вижу, что ассоциации "=> {" association_id "=>" 3 "} он получает только последнюю из нескольких ассоциаций, которые у меня были для конкретного человека, и не создает никаких записей в таблице соединений Где мой код мог пойти не так?

1 Ответ

27 голосов
/ 04 декабря 2009

У вас двоих проблемы здесь, к сожалению, один из них замаскирован другим.

Обе проблемы вытекают из этой части представления:

<div class="association">
  <% fields_for :associations do |assoc_form| %>
    <%= assoc_form.collection_select(:association_id, Association.find(:all),
      :id, :name, :include_blank => true) %>

Проблема 1: Вы неправильно поняли, что делает accept_nested_fields_for.

accepts_nested_fields_for используется для создания и изменения связанных объектов в форме. Его можно использовать для заполнения таблицы соединений, что является своего рода тем, что вы пытаетесь сделать. Тем не менее, использование acceptpts_nested_fields_for для заполнения таблицы объединения невозможно с отношением HABTM. Хорошее использование acceptpts_nested_fields_for будет, если вы хотите создать новую ассоциацию, которая будет связана с новым прокурором. Или если у вас была богатая модель соединения, которая требовала дополнительной информации для каждой записи.

Проблема 2: Вы не связываете поля в этой форме с формой адвоката. Что необходимо для использования acceptpts_nested_fields_for.

Мы уже установили, что accepts_nested_fields_for - это не то, что вам нужно для этого, но вы все еще не связываете поле select association_id с формой. Вот почему были установлены параметры [ассоциации] [ассоциация_id], а не параметры [адвокат] [ассоциации] [ассоциация].

Задача 3: Структура формы неверна для того, что вы пытаетесь достичь.

Слишком много всего, что мне нужно исправить, чтобы дать должный расклад. Вам лучше проверить репозиторий с примерами сложных форм . Это рабочий пример acceptpts_nested_attributes_for, он не касается каких-либо отношений HABTM, но он должен научить вас всему, что вам нужно знать. Исправленный код ниже составляет 90% того, что вам нужно. Приведенные выше примеры сложных форм научат вас тому, что вам нужно знать, чтобы заполнить пробелы, такие как add_association_link и create_association_link.

Исправление включает в себя следующие шаги:

  1. Создайте модель соединения и измените отношение «а» имеет множество через одно, принимая вложенные атрибуты в модели соединения.
  2. Произведите небольшую настройку в контроллере с точки зрения построения вещей.
  3. Передача объекта построителя форм частичному.
  4. Переписать форму частично, чтобы она была сосредоточена на только что созданной модели соединения.

Вы можете сделать это с помощью следующих изменений.

class Attorney < ActiveRecord::Base
  has_many :attorney_associations
  has_many :associations, :through => :attorney_associations

  accepts_nested_attributes_for :attorney_associations, :reject_if => proc { |a| 
     a['association_id'].blank? }
  accepts_nested_attributes_for :associations, :reject_if => proc {|a|
     a['name'].blank?}
end

class AttorneyAssociations < ActiveRecord::Base
  belongs_to :attorney
  belongs_to :association
end

Контролер адвоката:

def new
  @attorney = Attorney.new
  @attorney.associations.build
  @attorney.attorney_associations.build


  respond_to do |format|
    format.html # new.html.erb
    format.xml  { render :xml => @attorney }
  end
end

Новый адвокат:

<% form_for(@attorney, :html => {:multipart => true}) do |f| %>
  <%= f.error_messages %>
 <%= f.label :"First name" %> 
 <%= f.text_field :firstname %><br>

 <%= f.label :"Last Name" %> 
 <%= f.text_field :lastname %><br>

 <%= f.label :"Attorney Type" %> 
 <%= f.collection_select :member_type_id, MemberType.all, :id, :name %><br>

 <%= f.text_area :bio, :cols => 70, :rows => 20 %><br><br>

 <%= f.label :"Attorney Location" %> 
 <%= f.collection_select :office_location_id, OfficeLocation.all, :id, :location %><br>

 <div id="associations">
   <% f.fields_for :attorney_association do |aa_form| %>
     <%= render :partial => 'attorney_association', :locals => {:f => aa_form} %>
   <% end %>
   <%= add_association_link "Add Another Existing Association" %>
   <% f.fields_for :associations do |assoc_form| %>
     <%= render :partial => 'attorney', :locals => {:f => assoc_form} %>       
   <%= create_association_link, "Create a New Association for this Attorney" %>
 </div>



 <%= f.submit 'Create' %>
<% end %>

Я предполагаю, что add_association_link является помощником javascript, который создает ссылку для клонирования пустого экземпляра того, что было частичным членства. create_association_link - это заполнитель для аналогичного помощника, который добавит частичное для новой ассоциации.

Частичная ассоциация адвокатов:

  <div class="attorney_association">
    <%= f.collection_select(:association_id, Association.find(:all),
      :id, :name, :include_blank => true) %>
    <%= link_to_function "remove", "$(this).up('.attorney_association').remove()" %>
  </div>

Частичное объединение:

  <div class="association">
    <%= f.label_for :name %>
    <%= f.text_field :name %>
    <%= link_to_function "remove", "$(this).up('.attorney_association').remove()" %>
  </div>
...