class Capybara::RackTest::Form < Capybara::RackTest::Node # This only needs to inherit from Rack::Test::UploadedFile because Rack::Test checks for # the class specifically when determing whether to consturct the request as multipart. # That check should be based solely on the form element's 'enctype' attribute value, # which should probably be provided to Rack::Test in its non-GET request methods. class NilUploadedFile < Rack::Test::UploadedFile def initialize @empty_file = Tempfile.new("nil_uploaded_file") @empty_file.close end def original_filename; ""; end def content_type; "application/octet-stream"; end def path; @empty_file.path; end end def params(button) params = {} form_element_types=[:input, :select, :textarea] form_elements_xpath=XPath.generate do |x| xpath=x.descendant(*form_element_types).where(~x.attr(:form)) xpath=xpath.union(x.anywhere(*form_element_types).where(x.attr(:form) == native[:id])) if native[:id] xpath.where(~x.attr(:disabled)) end.to_s native.xpath(form_elements_xpath).map do |field| case field.name when 'input' if %w(radio checkbox).include? field['type'] merge_param!(params, field['name'].to_s, field['value'].to_s) if field['checked'] elsif %w(submit image).include? field['type'] # TO DO identify the click button here (in document order, rather # than leaving until the end of the params) elsif field['type'] =='file' if multipart? file = \ if (value = field['value']).to_s.empty? NilUploadedFile.new else content_type = MIME::Types.type_for(value).first.to_s Rack::Test::UploadedFile.new(value, content_type) end merge_param!(params, field['name'].to_s, file) else merge_param!(params, field['name'].to_s, File.basename(field['value'].to_s)) end else merge_param!(params, field['name'].to_s, field['value'].to_s) end when 'select' if field['multiple'] == 'multiple' options = field.xpath(".//option[@selected]") options.each do |option| merge_param!(params, field['name'].to_s, (option['value'] || option.text).to_s) end else option = field.xpath(".//option[@selected]").first option ||= field.xpath('.//option').first merge_param!(params, field['name'].to_s, (option['value'] || option.text).to_s) if option end when 'textarea' merge_param!(params, field['name'].to_s, field.text.to_s) end end merge_param!(params, button[:name], button[:value] || "") if button[:name] params end def submit(button) driver.submit(method, native['action'].to_s, params(button)) end def multipart? self[:enctype] == "multipart/form-data" end private def method self[:method] =~ /post/i ? :post : :get end def merge_param!(params, key, value) Rack::Utils.normalize_params(params, key, value) end end