# encoding: utf-8
require 'spec_helper'
describe 'check_boxes input' do
include FormtasticSpecHelper
describe 'for a has_many association' do
before do
@output_buffer = ''
mock_everything
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :value_as_class => true, :required => true))
end)
end
it_should_have_input_wrapper_with_class("check_boxes")
it_should_have_input_wrapper_with_class(:input)
it_should_have_input_wrapper_with_id("author_posts_input")
it_should_have_a_nested_fieldset
it_should_have_a_nested_fieldset_with_class('choices')
it_should_have_a_nested_ordered_list_with_class('choices-group')
it_should_apply_error_logic_for_input_type(:check_boxes)
it_should_call_find_on_association_class_when_no_collection_is_provided(:check_boxes)
it_should_use_the_collection_when_provided(:check_boxes, 'input[@type="checkbox"]')
it 'should generate a legend containing a label with text for the input' do
output_buffer.should have_tag('form li fieldset legend.label label')
output_buffer.should have_tag('form li fieldset legend.label label', /Posts/)
end
it 'should not link the label within the legend to any input' do
output_buffer.should_not have_tag('form li fieldset legend label[@for^="author_post_ids_"]')
end
it 'should generate an ordered list with an li.choice for each choice' do
output_buffer.should have_tag('form li fieldset ol')
output_buffer.should have_tag('form li fieldset ol li.choice input[@type=checkbox]', :count => ::Post.all.size)
end
it 'should have one option with a "checked" attribute' do
output_buffer.should have_tag('form li input[@checked]', :count => 1)
end
it 'should not generate hidden inputs with default value blank' do
output_buffer.should_not have_tag("form li fieldset ol li label input[@type='hidden'][@value='']")
end
it 'should not render hidden inputs inside the ol' do
output_buffer.should_not have_tag("form li fieldset ol li input[@type='hidden']")
end
it 'should render one hidden input for each choice outside the ol' do
output_buffer.should have_tag("form li fieldset > input[@type='hidden']", :count => 1)
end
describe "each choice" do
it 'should not give the choice label the .label class' do
output_buffer.should_not have_tag('li.choice label.label')
end
it 'should not be marked as required' do
output_buffer.should_not have_tag('li.choice input[@required]')
end
it 'should contain a label for the radio input with a nested input and label text' do
::Post.all.each do |post|
output_buffer.should have_tag('form li fieldset ol li label', /#{post.to_label}/)
output_buffer.should have_tag("form li fieldset ol li label[@for='author_post_ids_#{post.id}']")
end
end
it 'should use values as li.class when value_as_class is true' do
::Post.all.each do |post|
output_buffer.should have_tag("form li fieldset ol li.post_#{post.id} label")
end
end
it 'should have a checkbox input but no hidden field for each post' do
::Post.all.each do |post|
output_buffer.should have_tag("form li fieldset ol li label input#author_post_ids_#{post.id}")
output_buffer.should have_tag("form li fieldset ol li label input[@name='author[post_ids][]']", :count => 1)
end
end
it 'should have a hidden field with an empty array value for the collection to allow clearing of all checkboxes' do
output_buffer.should have_tag("form li fieldset > input[@type=hidden][@name='author[post_ids][]'][@value='']", :count => 1)
end
it 'the hidden field with an empty array value should be followed by the ol' do
output_buffer.should have_tag("form li fieldset > input[@type=hidden][@name='author[post_ids][]'][@value=''] + ol", :count => 1)
end
it 'should not have a hidden field with an empty string value for the collection' do
output_buffer.should_not have_tag("form li fieldset > input[@type=hidden][@name='author[post_ids]'][@value='']", :count => 1)
end
it 'should have a checkbox and a hidden field for each post with :hidden_field => true' do
output_buffer.replace ''
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :hidden_fields => true, :value_as_class => true))
end)
::Post.all.each do |post|
output_buffer.should have_tag("form li fieldset ol li label input#author_post_ids_#{post.id}")
output_buffer.should have_tag("form li fieldset ol li label input[@name='author[post_ids][]']", :count => 2)
output_buffer.should have_tag('form li fieldset ol li label', /#{post.to_label}/)
end
end
it "should mark input as checked if it's the the existing choice" do
::Post.all.include?(@fred.posts.first).should be_truthy
output_buffer.should have_tag("form li fieldset ol li label input[@checked='checked']")
end
end
describe 'and no object is given' do
before(:each) do
output_buffer.replace ''
concat(semantic_form_for(:project, :url => 'http://test.host') do |builder|
concat(builder.input(:author_id, :as => :check_boxes, :collection => ::Author.all))
end)
end
it 'should generate a fieldset with legend' do
output_buffer.should have_tag('form li fieldset legend', /Author/)
end
it 'shold generate an li tag for each item in the collection' do
output_buffer.should have_tag('form li fieldset ol li input[@type=checkbox]', :count => ::Author.all.size)
end
it 'should generate labels for each item' do
::Author.all.each do |author|
output_buffer.should have_tag('form li fieldset ol li label', /#{author.to_label}/)
output_buffer.should have_tag("form li fieldset ol li label[@for='project_author_id_#{author.id}']")
end
end
it 'should generate inputs for each item' do
::Author.all.each do |author|
output_buffer.should have_tag("form li fieldset ol li label input#project_author_id_#{author.id}")
output_buffer.should have_tag("form li fieldset ol li label input[@type='checkbox']")
output_buffer.should have_tag("form li fieldset ol li label input[@value='#{author.id}']")
output_buffer.should have_tag("form li fieldset ol li label input[@name='project[author_id][]']")
end
end
it 'should html escape the label string' do
concat(semantic_form_for(:project, :url => 'http://test.host') do |builder|
concat(builder.input(:author_id, :as => :check_boxes, :collection => [["Item 1", 1], ["Item 2", 2]]))
end)
output_buffer.should have_tag('form li fieldset ol li label') do |label|
label.body.should match /<b>Item [12]<\/b>$/
end
end
end
describe 'when :hidden_fields is set to false' do
before do
@output_buffer = ''
mock_everything
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :value_as_class => true, :hidden_fields => false))
end)
end
it 'should have a checkbox input for each post' do
::Post.all.each do |post|
output_buffer.should have_tag("form li fieldset ol li label input#author_post_ids_#{post.id}")
output_buffer.should have_tag("form li fieldset ol li label input[@name='author[post_ids][]']", :count => ::Post.all.length)
end
end
it "should mark input as checked if it's the the existing choice" do
::Post.all.include?(@fred.posts.first).should be_truthy
output_buffer.should have_tag("form li fieldset ol li label input[@checked='checked']")
end
it 'should not generate empty hidden inputs' do
output_buffer.should_not have_tag("form li fieldset ol li label input[@type='hidden'][@value='']", :count => ::Post.all.length)
end
end
describe 'when :disabled is set' do
before do
@output_buffer = ''
end
describe "no disabled items" do
before do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :disabled => nil))
end)
end
it 'should not have any disabled item(s)' do
output_buffer.should_not have_tag("form li fieldset ol li label input[@disabled='disabled']")
end
end
describe "single disabled item" do
before do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :disabled => @fred.id))
end)
end
it "should have one item disabled; the specified one" do
output_buffer.should have_tag("form li fieldset ol li label input[@disabled='disabled']", :count => 1)
output_buffer.should have_tag("form li fieldset ol li label[@for='post_author_ids_#{@fred.id}']", /fred/i)
output_buffer.should have_tag("form li fieldset ol li label input[@disabled='disabled'][@value='#{@fred.id}']")
end
end
describe "multiple disabled items" do
before do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :disabled => [@bob.id, @fred.id]))
end)
end
it "should have multiple items disabled; the specified ones" do
output_buffer.should have_tag("form li fieldset ol li label input[@disabled='disabled']", :count => 2)
output_buffer.should have_tag("form li fieldset ol li label[@for='post_author_ids_#{@bob.id}']", /bob/i)
output_buffer.should have_tag("form li fieldset ol li label input[@disabled='disabled'][@value='#{@bob.id}']")
output_buffer.should have_tag("form li fieldset ol li label[@for='post_author_ids_#{@fred.id}']", /fred/i)
output_buffer.should have_tag("form li fieldset ol li label input[@disabled='disabled'][@value='#{@fred.id}']")
end
end
end
describe "with i18n of the legend label" do
before do
::I18n.backend.store_translations :en, :formtastic => { :labels => { :post => { :authors => "Translated!" }}}
with_config :i18n_lookups_by_default, true do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes))
end)
end
end
after do
::I18n.backend.reload!
end
it "should do foo" do
output_buffer.should have_tag("legend.label label", /Translated/)
end
end
describe "when :label option is set" do
before do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :label => 'The authors'))
end)
end
it "should output the correct label title" do
output_buffer.should have_tag("legend.label label", /The authors/)
end
end
describe "when :label option is false" do
before do
@output_buffer = ''
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :label => false))
end)
end
it "should not output the legend" do
output_buffer.should_not have_tag("legend.label")
end
it "should not cause escaped HTML" do
output_buffer.should_not include(">")
end
end
describe "when :required option is true" do
before do
@new_post.stub(:author_ids).and_return(nil)
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes, :required => true))
end)
end
it "should output the correct label title" do
output_buffer.should have_tag("legend.label label abbr")
end
end
end
describe 'for a has_and_belongs_to_many association' do
before do
@output_buffer = ''
mock_everything
concat(semantic_form_for(@freds_post) do |builder|
concat(builder.input(:authors, :as => :check_boxes))
end)
end
it 'should render checkboxes' do
# I'm aware these two lines test the same thing
output_buffer.should have_tag('input[type="checkbox"]', :count => 2)
output_buffer.should have_tag('input[type="checkbox"]', :count => ::Author.all.size)
end
it 'should only select checkboxes that are present in the association' do
# I'm aware these two lines test the same thing
output_buffer.should have_tag('input[checked="checked"]', :count => 1)
output_buffer.should have_tag('input[checked="checked"]', :count => @freds_post.authors.size)
end
end
describe 'for an association when a :collection is provided' do
describe 'it should use the specified :member_value option' do
before do
@output_buffer = ''
mock_everything
end
it 'to set the right input value' do
item = double('item')
item.should_not_receive(:id)
item.stub(:custom_value).and_return('custom_value')
item.should_receive(:custom_value).exactly(3).times
@new_post.author.should_receive(:custom_value).exactly(1).times
concat(semantic_form_for(@new_post) do |builder|
concat(builder.input(:author, :as => :check_boxes, :member_value => :custom_value, :collection => [item, item, item]))
end)
output_buffer.should have_tag('input[@type=checkbox][@value="custom_value"]', :count => 3)
end
end
end
describe 'when :collection is provided as an array of arrays' do
before do
@output_buffer = ''
mock_everything
@fred.stub(:genres) { ['fiction', 'biography'] }
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:genres, :as => :check_boxes, :collection => [['Fiction', 'fiction'], ['Non-fiction', 'non_fiction'], ['Biography', 'biography']]))
end)
end
it 'should check the correct checkboxes' do
output_buffer.should have_tag("form li fieldset ol li label input[@value='fiction'][@checked='checked']")
output_buffer.should have_tag("form li fieldset ol li label input[@value='biography'][@checked='checked']")
end
end
describe 'when :collection is a set' do
before do
@output_buffer = ''
mock_everything
@fred.stub(:roles) { Set.new([:reviewer, :admin]) }
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:roles, :as => :check_boxes, :collection => [['User', :user], ['Reviewer', :reviewer], ['Administrator', :admin]]))
end)
end
it 'should check the correct checkboxes' do
output_buffer.should have_tag("form li fieldset ol li label input[@value='user']")
output_buffer.should have_tag("form li fieldset ol li label input[@value='admin'][@checked='checked']")
output_buffer.should have_tag("form li fieldset ol li label input[@value='reviewer'][@checked='checked']")
end
end
describe "when namespace is provided" do
before do
@output_buffer = ''
mock_everything
concat(semantic_form_for(@fred, :namespace => "context2") do |builder|
concat(builder.input(:posts, :as => :check_boxes))
end)
end
it "should have a label for #context2_author_post_ids_19" do
output_buffer.should have_tag("form li label[@for='context2_author_post_ids_19']")
end
it_should_have_input_with_id('context2_author_post_ids_19')
it_should_have_input_wrapper_with_id("context2_author_posts_input")
end
describe "when index is provided" do
before do
@output_buffer = ''
mock_everything
concat(semantic_form_for(@fred) do |builder|
concat(builder.fields_for(@fred.posts.first, :index => 3) do |author|
concat(author.input(:authors, :as => :check_boxes))
end)
end)
end
it 'should index the id of the wrapper' do
output_buffer.should have_tag("li#author_post_3_authors_input")
end
it 'should index the id of the input tag' do
output_buffer.should have_tag("input#author_post_3_author_ids_42")
end
it 'should index the name of the checkbox input' do
output_buffer.should have_tag("input[@type='checkbox'][@name='author[post][3][author_ids][]']")
end
end
describe "when collection is an array" do
before do
@output_buffer = ''
@_collection = [["First", 1], ["Second", 2]]
mock_everything
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :collection => @_collection))
end)
end
it "should use array items for labels and values" do
@_collection.each do |post|
output_buffer.should have_tag('form li fieldset ol li label', /#{post.first}/)
output_buffer.should have_tag("form li fieldset ol li label[@for='author_post_ids_#{post.last}']")
end
end
it "should not check any items" do
output_buffer.should have_tag('form li input[@checked]', :count => 0)
end
describe "and the attribute has values" do
before do
@fred.stub(:posts) { [1] }
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :collection => @_collection))
end)
end
it "should check the appropriate items" do
output_buffer.should have_tag("form li input[@value='1'][@checked]")
end
end
describe "and the collection includes html options" do
before do
@_collection = [["First", 1, {'data-test' => 'test-data'}], ["Second", 2, {'data-test2' => 'test-data2'}]]
concat(semantic_form_for(@fred) do |builder|
concat(builder.input(:posts, :as => :check_boxes, :collection => @_collection))
end)
end
it "should have injected the html attributes" do
@_collection.each do |v|
output_buffer.should have_tag("form li input[@value='#{v[1]}'][@#{v[2].keys[0]}='#{v[2].values[0]}']")
end
end
end
end
end