This is a great form building method, one of the many awesome ActionView::Helper objects. Ideal for interfacing with RESTful routes, and nested resources. It will wrap around an instance of a Model object, and give all kinds of useful methods for quickly generating otherwise lengthy HTML, and have all kinds of cool shortcuts since it is directly referencing a model. I will be using pry, a simple ruby gem that bounces into console at a given point.

So, first a little setup:

Here is my model schema:

 create_table "users", force: :cascade do |t|
    t.string "username"
    t.string "password_digest"
    t.datetime "created_at", null: false
    t.datetime "updated_at", null: false
    t.string "email"
  end

Here is my model:

class User < ApplicationRecord
	has_secure_password
	validates :username, presence: true
end

And here is my controller action, serving up a nice warm @user instance:

def new
	# notice here just sending a blank, unsaved instance
	@user = User.new
end

Ok, without further ado, our form_for:

<!-- app/views/users/new.html.erb-->

<%= form_for @user do |f| %>
	<% binding.pry %>
<% end %>

pry pops us into console, now what the heck is f :

<pry> f.class
<pry> ActionView::Helpers::FormBuilder

Shows us the f is an ActionView:Helpers::FormBuilder object: Here are some methods it gives us access to:

<pry> f.methods
 :fields_for,
 :text_field,
 :password_field,
 :hidden_field,
 :file_field,
 :text_area,
 :check_box,
 :radio_button,
 :color_field,
 :search_field,
 :telephone_field,
 :phone_field,
 etc.....

There are a bunch, scope them here. So, let’s build a sign up form:

<%= form_for @user do |f| %>
	<p>
	<%= f.label :username %>
	<%= f.text_field :username %>
	</p>

	<p>
	<%= f.label :password %>
	<%= f.password_field :password %>
	</p>
	
	<p>
	<%= f.label "Password Confirmation" %>
	<%= f.password_field :password_confirmation %>
	</p>
	
	<%= f.submit "Sign Up"%>
<% end %>

That generates this HTML:

<form class="new_user" id="new_user" action="/users" accept-charset="UTF-8" method="post"><input name="utf8" type="hidden" value="✓"><input type="hidden" name="authenticity_token" value="T36sH3iVwQCE2Frn2ERlOU8S62ev+zR6mRc9/cMT75fAvmOqGWqS+TrJ8uoRafb04YdaounpUJzXmu+eBV+pbg==">
	<p>
	<label for="user_Username">Username</label>
	<input type="text" name="user[username]" id="user_username" style="background-image: url(&quot;&quot;); background-repeat: no-repeat; background-attachment: scroll; background-size: 16px 18px; background-position: 98% 50%; cursor: auto;">
	</p>

	<p>
	<label for="user_Password">Password</label>
	<input type="password" name="user[password]" id="user_password" style="background-image: url(&quot;&quot;); background-repeat: no-repeat; background-attachment: scroll; background-size: 16px 18px; background-position: 98% 50%;">
	</p>
	
	<p>
	<label for="user_Password Confirmation">Password confirmation</label>
	<input type="password" name="user[password_confirmation]" id="user_password_confirmation" style="background-image: url(&quot;&quot;); background-repeat: no-repeat; background-attachment: scroll; background-size: 16px 18px; background-position: 98% 50%;">
	</p>
	

	<input type="submit" name="commit" value="Sign Up" data-disable-with="Sign Up">
</form>

Holy crap, that is alot of HTML! Gives us some awesome freebies like:

Here are a few of my favorite methods:

checkboxes:

<%= f.check_box :status, :class => "toggle", :checked => true  %>

collection of checkboxes:

<%= f.collection_check_boxes :id, User.all, :id, :username do |b| %>
	  <p>
	  <%= b.label class: "Whatever" %>
	  <%= b.check_box %>
	  </p>
	<% end %>

dropdown menus:

<%= f.collection_check_boxes :id, User.all, :id, :username do |b| %>
	  <p>
	  <%= b.label class: "Whatever" %>
	  <%= b.check_box %>
	  </p>
	<% end %>

ability to create nested resources for an entire different model:

<%= f.fields_for @post do |b| %>
	<% b.text_area :content %>
<% end %>

Will need to send an instance of that @post though:

def new
	# notice here just sending a blank, unsaved instance
	@user = User.new
	@post = Post.new
end

Basically params will give you a double nested resource, which you can use to create a new user and a post in the same form:

params = {
	:user => {
		:username => "CooolUsername"
		:password => "password"
		:post => {:content => "what ever was entered into the text area above"}
	}
}

It will also automatically know if your are creating a new object, via a post request, or updating an object, and will automagically make it a patch request. Cool!

Now, need to use accepts_nested_attributes in your user model:

class User < ApplicationRecord
	has_secure_password
	validates :username, presence: true
	accepts_nested_attributes_for :posts
end

Which gives you a .posts= setter and .post getter method, create your nested post in your user controller:

def create
	@user = User.create(params[:user])
	@user.post.create(params[:user][:post])
end

Pretty sweet, tons where that came from as well. Check the docs and utilize form_for for all your form building needs, your fingers will thank you.

Thanks for reading!!