I am setting up embedded forms in my rails app.
This does not work
<h1>PlayersToTeams#edit</h1>
<%= form_for @players_to_teams do |field| %>
<%= field.fields_for @players_to_teams.player do |f| %>
<%= f.label :IsActive %>
<%= f.text_field :IsActive %>
<% end %>
<%= field.label :BT %>
<%= field.text_field :BT %>
<br/>
<%= field.submit "Save", class: 'btn btn-primary' %>
<% end %>
Gives me a ActiveRecord::AssociationTypeMismatch error. Notice the @players_to_teams.player in the forms_for line.
This does work:
<h1>PlayersToTeams#edit</h1>
<%= form_for @players_to_teams do |field| %>
<%= field.fields_for :player do |f| %>
<%= f.label :IsActive %>
<%= f.text_field :IsActive %>
<% end %>
<%= field.label :BT %>
<%= field.text_field :BT %>
<br/>
<%= field.submit "Save", class: 'btn btn-primary' %>
<% end %>
Notice the :player call in the fields_for line.
Whats the difference between using a symbol and using an instance? I would think I would want to use an instance in this case, but I guess not?
Edit
models:
class Player < ActiveRecord::Base
has_many :players_to_teams
has_many :teams, through: :players_to_teams
end
class PlayersToTeam < ActiveRecord::Base
belongs_to :player
belongs_to :team
accepts_nested_attributes_for :player
end
controller:
class PlayersToTeamsController < ApplicationController
def edit
@players_to_teams=PlayersToTeam.find(params[:id])
end
def update
@players_to_teams=PlayersToTeam.find(params[:id])
respond_to do |format|
if @players_to_teams.update_attributes(params[:players_to_team])
format.html { redirect_to @players_to_teams, notice: 'Player_to_Team was successfully updated.' }
format.json { head :no_content }
else
format.html { render action: "edit" }
format.json { render json: @players_to_teams.errors, status: :unprocessable_entity }
end
end
end
end
Example Project
Github: https://github.com/workmaster2n/embedded-form-errors
It seems like
fields_forcan’t figure out what theplayerassociation is for the@players_to_teamsinstance variable. You bypass that by explicitly specifying the name of the association (i.e.,:player). Are you defining the association in both directions? I.e.:Additionally, looking at the documentation for
fields_for, the first parameter is calledrecord_name, so it seems like Rails is expecting the name of the association rather than the association itself. But it’s hard to tell exactly what that method does without looking further into the code, and some of their examples do indeed pass the association directly intofields_for.Okay, I was able to clone your sample project and reproduce the error. I think I understand what’s happening.
After your call to
accepts_nested_attributes_for, you now have an instance method on your model namedplayer_attributes=. This is in addition to theplayer=method that’s normally defined for ahas_oneassociation. Theplayer_attributes=method accepts a hash of attributes, whereas theplayer=method only accepts an actualPlayerobject.Here’s an example of the text input generated when you called
fields_for @players_to_teams.player:and here’s that same input when calling
fields_for :player:When you call
update_attributesin your controller, the first example will callplayer=, while the second example will callplayer_attributes=. In both cases, the argument passed to the method is a hash (becauseparamsis ultimately just a hash of hashes).That’s why you were getting an
AssociationTypeMismatch: you can’t pass a hash toplayer=, only aPlayerobject.It appears that the only safe way to use
fields_forwithaccepts_nested_attributes_foris by passing the name of the association and not the association itself.So to answer your original question, the difference is that one works and the other doesn’t 🙂