After this lesson, you should be able to...
- Create a
has_many
andbelongs_to
association. - Build associated data through the console and
db/seeds.rb
. - Query for associated data using methods provided by association.
- Embed association data within views.
- Iterate over associated data within a view displaying individual instances.
In this lesson, we'll be setting up a blog admin panel so that Posts can be created, associated with Categories, and listed by category.
First, we'll set up associated models, just like in the preceding lesson:
# app/models/post.rb
class Post < ActiveRecord::Base
belongs_to :category
end
# app/models/category.rb
class Category < ActiveRecord::Base
has_many :posts
end
Once you start working with more and more complicated data sets, you will realize that there is a lot of stuff you have to set up just to be able to play with your methods. The associations are so vast that you need to make many posts with many categories and all of that! How frustrating. What you are doing is called "seeding" the database. Pretty much putting in some test data so that you can play with your app. In Rails we set up our seed data in db/seeds.rb
. Then we'll be able to just seed (or re-seed) the database with a quick rake db:seed
.
# db/seeds.rb
clickbait = Category.create!(name: "Motivation")
clickbait.posts.create!(title: "10 Ways You Are Already Awesome")
clickbait.posts.create!(title: "This Yoga Stretch Cures Procrastination, Maybe")
clickbait.posts.create!(title: "The Power of Positive Thinking and 100 Gallons of Coffee")
movies = Category.create!(name: "Movies")
movies.posts.create!(title: "Top 20 Summer Blockbusters Featuring a Cute Dog")
Woot! The best thing about the seeds.rb
file is that it's just Ruby! There is no magic. Look, super standard Ruby. To run the seed file in the development environment, you can activate the rake task:
rake db:seed
If you want to play around with the data, of course, it's always possible to take the create statements exactly as written above and type them into rails console
.
When viewing a single post, we'll want to have a link to its category available.
<%# app/views/posts/show.html.erb %>
<h2><%= @post.title %></h2>
Category: <%= link_to @post.category.name, category_path(@post.category) %>
<p><%= @post.description %></p>
@post.category
is the Category
model itself, so we can use it anywhere we would use @category
on a view for that object.
In this domain, the primary use of a category is as a bucket for posts. So we'll definitely have to make heavy use of associations when designing the view.
<%# app/views/categories/show.html.erb %>
<h2><%= @category.name %></h2>
<%= @category.posts.count %> post(s):
<ul>
<% @category.posts.each do |post| %>
<li><%= link_to post.title, post_path(post) %></li>
<% end %>
</ul>
The object returned by an association method (posts
in this case) is a CollectionProxy, and it responds to most of the methods you can use on an array. Pretty much think of it like an array.
If we open up rails console
, we can confirm that the count
results are
accurate:
Post.count
#=> 4
clickbait = Category.find_by(name: "Motivation")
#=> #<Category id=1>
clickbait.posts.count
#=> 3
Meanwhile, for listing a category's posts, we write a loop very similar to the loops we've been writing on index
actions, which makes sense, since a category is essentially an index for its posts. Let's compare them side-by-side:
<%# app/views/categories/show.html.erb %>
<% @category.posts.each do |post| %>
<li><%= link_to post.title, post_path(post) %></li>
<% end %>
Versus:
<%# app/views/posts/index.html.erb %>
<% @posts.each do |post| %>
<li><%= link_to post.title, post_path(post) %></li>
<% end %>
In fact, the only difference is what we call each
on.
With ActiveRecord's powerful association macros and instance methods, we can treat related models exactly the same as we treat directly-accessed models. As long as the database and classes are set up correctly, ActiveRecord will figure the rest out for us!
View Displaying Associations Data on Learn.co and start learning to code for free.