r123454321
r123454321

Reputation: 3403

Can't render checkboxes in a Rails 4 many_to_many association?

I have a many_to_many association between Articles and Categories, using has_and_belongs_to_many in a Rails 4 app:

Here are the corresponding migration and classes:

class CategoriesArticles < ActiveRecord::Migration
  def change
    create_table :articles_categories, id: false do |t|
      t.belongs_to :category, index: true
      t.belongs_to :article, index: true
    end
    add_index :articles_categories, [:category_id, :article_id]
  end
end

class Category < ActiveRecord::Base
  has_and_belongs_to_many :articles
end

class Article < ActiveRecord::Base
  has_and_belongs_to_many :categories
end

When a user creates a new article, I simply want to give him or her the option to select categories that he/she wants to associate with the new article. I want the user to be able to select these categories with checkboxes.

Here's the ArticlesController:

class ArticlesController < ApplicationController
  before_action :set_article, only: [:show, :edit, :update, :destroy]
  before_action :authenticate_user!, only: [:new, :create, :edit, :destroy, :update]
  before_action :verify_own_article, only: [:destroy]
  respond_to :html

  ...

  def new
    @categories = Category.all
    @article = Article.new
    respond_with(@article)
  end

  def create
    # Creates article object with current_user_id, initial_comment, and URL
    @article = current_user.articles.build(article_params)

    # Uses Pismo (gem) to grab title, content, photo of URL
    @article.populate_url_fields
    if @article.save
      flash[:success] = "Article created!"

      # Might need to change the location of this redirect
      redirect_to root_url
    else
      flash[:notice] = "Invalid article."
      redirect_to new_article_path
    end

  end

  def update
    @article.update(article_params)
    flash[:notice] = "Article successfully updated."
    respond_with(@article)
  end

  private
    def set_article
      @article = Article.find(params[:id])
    end

    def article_params
      params.require(:article).permit(:url, :title, :datetime, :content, :photo, :initial_comment)
    end

    # Ensure that a signed in user can only delete articles that they have posted
    def verify_own_article
      @article = current_user.articles.find_by_id(params[:id])
    end
end

Here's the article new.html.erb view:

<h1>New article</h1>

<%= render 'form' %>

<%= link_to 'Back', articles_path %>

... and the form partial:

<%= form_for(@article) do |f| %>
  <% if @article.errors.any? %>
    <div id="error_explanation">
      <h2><%= pluralize(@article.errors.count, "error") %> prohibited this article from being saved:</h2>

      <ul>
      <% @article.errors.full_messages.each do |message| %>
        <li><%= message %></li>
      <% end %>
      </ul>
    </div>
  <% end %>

  <div class="field">
    <%= f.label :url %><br>
    <%= f.text_field :url %>
  </div>
  <div class="field">
    <%= f.label :initial_comment %><br>
    <%= f.text_field :initial_comment %>
  </div>


  <% @categories.each do |t| %>
    <div class="field">
      <%= f.label t.name %>
      <%= f.check_box "categories[#{t.id}]" %>
      <br />
    </div>
  <% end %>

  <div class="actions">
    <%= f.submit %>
  </div>
<% end %>

However, this is erroring for me, specifically the lines:

<% @categories.each do |t| %>
        <div class="field">
          <%= f.label t.name %>
          <%= f.check_box "categories[#{t.id}]" %>
          <br />
        </div>
      <% end %>

Specifically, it's telling me:

undefined method 'categories[1]' for #<Article:0x007f401193d520> when I try to render the New Article page. How do I fix this? Thanks.

Upvotes: 0

Views: 42

Answers (1)

Guilherme Franco
Guilherme Franco

Reputation: 1483

It is better to use Rails collection_check_boxes helper instead of trying to create those checkboxes by hand. This helper already creates all the parameter / markup stuff you need in order to add or exclude items of a HABTM relation, all under the hood. So you might change you view to include the following:

<%= f.collection_check_boxes :categories_ids, @categories, :id, :name %>

Don't forget to add this in your strong parameters declaration (since you'll have to receive the selected categories ids and bind them to your Article model):

params.require(:article).permit(
  :url, :title, :datetime, :content, 
  :photo, :initial_comment, categories_ids: []
)

For further customizations (html styling or structure for each checkbox), please refer to the complete documentation

I hope it helps :)

Upvotes: 2

Related Questions