Accepts_nested_attributes_for with belongs_to полиморфный

Я хотел бы установить полиморфное отношение с accepts_nested_attributes_for. Вот код:

class Contact <ActiveRecord::Base
  has_many :jobs, :as=>:client
end

class Job <ActiveRecord::Base
  belongs_to :client, :polymorphic=>:true
  accepts_nested_attributes_for :client
end

Когда я пытаюсь получить доступ к Job.create(..., :client_attributes=>{...}, мне дают NameError: uninitialized constant Job::Client

Ответ 1

Просто понял, что рельсы не поддерживают такое поведение, поэтому я придумал следующее обходное решение:

class Job <ActiveRecord::Base
  belongs_to :client, :polymorphic=>:true, :autosave=>true
  accepts_nested_attributes_for :client

  def attributes=(attributes = {})
    self.client_type = attributes[:client_type]
    super
  end

  def client_attributes=(attributes)
    self.client = type.constantize.find_or_initialize_by_id(attributes.delete(:client_id)) if client_type.valid?
  end
end

Это дает мне возможность настроить мою форму следующим образом:

<%= f.select :client_type %>
<%= f.fields_for :client do |client|%>
  <%= client.text_field :name %>
<% end %>

Не точное решение, но идея важна.

Ответ 2

У меня также возникла проблема с "ArgumentError: невозможно создать имя модели ассоциации. Вы пытаетесь построить полиморфную ассоциацию" один-к-одному "?"

И я нашел лучшее решение для такого рода проблем. Вы можете использовать собственный метод. Давайте посмотрим на реализацию nested_attributes, внутри Rails3:

elsif !reject_new_record?(association_name, attributes)
  method = "build_#{association_name}"
  if respond_to?(method)
    send(method, attributes.except(*UNASSIGNABLE_KEYS))
  else
    raise ArgumentError, "Cannot build association #{association_name}. Are you trying to build a polymorphic one-to-one association?"
  end
end

Итак, что нам нужно делать здесь? Это просто создать build _ # {association_name} внутри нашей модели. Я сделал полный рабочий пример внизу:

class Job <ActiveRecord::Base
  CLIENT_TYPES = %w(Contact)

  attr_accessible :client_type, :client_attributes

  belongs_to :client, :polymorphic => :true

  accepts_nested_attributes_for :client

  protected

  def build_client(params, assignment_options)
    raise "Unknown client_type: #{client_type}" unless CLIENT_TYPES.include?(client_type)
    self.client = client_type.constantize.new(params)
  end
end

Ответ 3

Наконец-то я получил это для работы с Rails 4.x. Это основано на ответе Дмитрия/Скоттера, поэтому +1 к ним.

ШАГ 1. Чтобы начать, вот полная модель с полиморфной ассоциацией:

# app/models/polymorph.rb
class Polymorph < ActiveRecord::Base
  belongs_to :associable, polymorphic: true

  accepts_nested_attributes_for :associable

  def build_associable(params)
    self.associable = associable_type.constantize.new(params)
  end
end

# For the sake of example:
# app/models/chicken.rb
class Chicken < ActiveRecord::Base
  has_many: :polymorphs, as: :associable
end

Да, это ничего нового. Однако вы можете удивиться, откуда приходит polymorph_type и как его значение установлено? Это часть базовой записи базы данных, так как полиморфные ассоциации добавляют в таблицу столбцы <association_name>_id и <association_name>_type. Как бы то ни было, когда build_associable выполняется, значение _type равно nil.

ШАГ 2. Пропустите и примите тип ребенка

Отправьте форму в форме child_type вместе с типичными данными формы, и ваш контроллер должен разрешить ее при проверке сильных параметров.

# app/views/polymorph/_form.html.erb
<%= form_for(@polymorph) do |form| %>
  # Pass in the child_type - This one has been turned into a chicken!
  <%= form.hidden_field(:polymorph_type, value: 'Chicken' %>
  ...
  # Form values for Chicken
  <%= form.fields_for(:chicken) do |chicken_form| %>
    <%= chicken_form.text_field(:hunger_level) %>
    <%= chicken_form.text_field(:poop_level) %>
    ...etc...
  <% end %>
<% end %>

# app/controllers/polymorph_controllers.erb
...
private
  def polymorph_params
    params.require(:polymorph).permit(:id, :polymorph_id, :polymorph_type)
  end

Конечно, ваши взгляды должны будут обрабатывать различные типы моделей, которые являются "связанными", но это демонстрирует один.

Надеюсь, что это поможет кому-то. (Зачем вам полиморфные цыплята?)

Ответ 4

Вышеупомянутый ответ велик, но не работает с показанной настройкой. Это вдохновило меня, и я смог создать рабочее решение:

работает для создания и обновления

class Job <ActiveRecord::Base
  belongs_to :client, :polymorphic=>:true
  attr_accessible :client_attributes
  accepts_nested_attributes_for :client

  def attributes=(attributes = {})
    self.client_type = attributes[:client_type]
    super
  end

  def client_attributes=(attributes)
    some_client = self.client_type.constantize.find_or_initilize_by_id(self.client_id)
    some_client.attributes = attributes
    self.client = some_client
  end
end