Has Many Through Forms Rails
Objectives
Construct a bi-directional has many through.
Identify the join model in a has many through.
Construct a nested params hash with data about the primary object and a has many through association.
Use the conventional key names for associated data (association_attributes).
Name form inputs correctly to create a nested params hash with has many through association data.
Define a conventional association writer for the primary model to properly instantiate associations based on the nested params association data.
Define a custom association writer for the primary model to properly instantiate associations with custom logic (like unique by name) on the nested params association data.
Use fields_for to generate the association fields.
Overview
We've looked at the different ways we can interact with our associated models through nested forms as well as displaying data from more complex associations. In this lesson, we'll look at some different ways we can create data from our complex associations to make for a great user experience.
Join Model Forms
Sometimes, it may be appropriate for a user to create an instance of our join model directly. Think back to the hospital domain from our previous lab. It makes perfect sense that a user would go to appointments/new
and fill out a form to create a new appointment.
In this example, a user is filling out a form, entering the date and time they'd like to come, and choosing their doctor and their name from a dropdown. We're assigning these properties directly to the appointment as it's created.
Other times, we need to be more abstract. Let's return to our blog example, but this time we'll say that a post can have many categories and categories can have many posts. For this, we'll need a join table –– let's call it post_categories
. If our user wants to associate a post with a category, it doesn't make sense for them to go to /post_categories/new
and fill out a "new post category form." That's confusing! Let's look at a more abstract way that we can do this thanks to the magic of Active Record.
Setting up our Posts and Categories
Now, let's make it so that our user can assign categories to a post when the post is created. We did this in a previous example without a join table. Our post was directly related to its category, and the posts
table had the foreign key for its category. Because of this, instances of our Post
class responded to a method called category_ids=
. We called upon this method from our form helpers to build out a nested form.
Luckily, has_many, through
functions exactly the same as a has_many
relationship. Instances of our Post
class still respond to a method called category_ids=
. We'll use a helper method very similar to the collection_select
we used previously.
This will create a checkbox field for each Category
in our database. The HTML generated looks something like this:
In our controller, we've setup our post_params
to expect a key of :category_ids
with a value of an array.
After submitting the form, we end up with post_params
that look something like:
Let's check out the SQL that fires from creating our new post.
First, we're creating a new row in our posts
table with title
and content
. Next, we create a row in our post_categories
table for each ID number that was stored in our category_ids
array. This functions just like it did with a has_many
relationship, but, instead of creating a new record in our categories
table, Active Record is creating two new rows in our post_categories
table. This means that we can interact with our higher-level models directly without having to think too much at all about our join table - ActiveRecord will manage that relationship for us behind the scenes.
Creating New Categories
We can now associate categories with our posts, but what about creating new categories? If I'm posting about baby elephants and no one has created the category "Super Cute!" yet, I want to be able to create it at the same time as my post. Again, this will be very similar to the way we've done things before.
First, we want a text field to enter the name of our new category. The value of the name should be nested under our post_params
, so we don't have to add too much code to our controller. We can use the fields_for
helper to do this very easily.
The fields_for
helper takes two arguments: the associated model that we're creating and an object to wrap around. In this case, we've passed in the :categories
association and built an empty category associated with the post.
Let's look at the html that this generated for us.
Our params hash will now have a key of :categories_attributes
nested under the key of post
. Let's add that to our strong params and tell it to expect a key of name
inside for the category's name.
Now, when we do mass assignment, our Post
model will call a method called categories_attributes=
. Let's add that method to our model using the accepts_nested_attributes_for
macro.
Voila! Just like when our models were directly related, we can now create categories that are automatically associated with our new post. Looking at the SQL that gets generated, we can see that it's creating new instances of PostCategory
without us ever having to interact with them.
Still, there's a problem. We're creating a new category each time, regardless of whether or not it exists. It will look pretty weird if three people type in "Super Cute!" and we get three different categories. In this case, we need to customize the way our category is created. Luckily, we can easily do this by creating our own categories_attributes=
method.
Now, we're only creating a new category if it doesn't already exist with the current name. We're also using a cool method called categories<<
. What's great about this is you can mentally think of it as two steps. First, we call self.categories
, which returns an array of Category
objects, and then we call the shovel (<<
) method to add our newly found or created Category
object to the array. We could imagine later calling save
on the Post
object and this then creating the post_categories
join record for us. In reality, this is syntactic sugar for the categories<<
method. That's the actual method name, and behind the scenes it will create the join record for us. It's one of the methods dynamically created for us whenever we use a has_many
association. The end result is this method doing exactly what Active Record was doing for us before; we're just customizing the behavior a little bit.
Conclusion/So What?
As you can see, it doesn't really matter how complex our associations are –– Active Record is really good at managing that complexity for us. We can always drop down a level of abstraction if needed to customize the way our application behaves.
Last updated