require 'selenium-webdriver' class Capybara::Driver::Selenium < Capybara::Driver::Base DEFAULT_OPTIONS = { :resynchronize => true, :resynchronization_timeout => 10, :browser => :firefox } SPECIAL_OPTIONS = [:browser, :resynchronize, :resynchronization_timeout] class Node < Capybara::Driver::Node def text native.text end def [](name) if name == :value value else native.attribute(name.to_s) end rescue Selenium::WebDriver::Error::WebDriverError nil end def value if tag_name == "select" and self[:multiple] and not self[:multiple] == "false" native.find_elements(:xpath, ".//option").select { |n| n.selected? }.map { |n| n.value || n.text } else native.value end end def set(value) if tag_name == 'input' and type == 'radio' click elsif tag_name == 'input' and type == 'checkbox' click if value ^ native.attribute('checked').to_s.eql?("true") elsif tag_name == 'textarea' or tag_name == 'input' resynchronize do native.clear native.send_keys(value.to_s) end end end def select_option resynchronize { native.select } end def unselect_option if select_node['multiple'] != 'multiple' and select_node['multiple'] != 'true' raise Capybara::UnselectNotAllowed, "Cannot unselect option from single select box." end resynchronize { native.toggle } if selected? end def click resynchronize { native.click } end def drag_to(element) resynchronize { native.drag_and_drop_on(element.native) } end def tag_name native.tag_name end def visible? displayed = native.displayed? displayed and displayed != "false" end def selected? selected = native.selected? selected and selected != "false" end alias :checked? :selected? def find(locator) native.find_elements(:xpath, locator).map { |n| self.class.new(driver, n) } end private def resynchronize driver.resynchronize { yield } end # a reference to the select node if this is an option node def select_node find('./ancestor::select').first end def type self[:type] end end attr_reader :app, :rack_server, :options def browser unless @browser @browser = Selenium::WebDriver.for(options[:browser], options.reject { |key,val| SPECIAL_OPTIONS.include?(key) }) at_exit do @browser.quit end end @browser end def initialize(app, options={}) @app = app @options = DEFAULT_OPTIONS.merge(options) @rack_server = Capybara::Server.new(@app) @rack_server.boot if Capybara.run_server end def visit(path) browser.navigate.to(url(path)) end def source browser.page_source end def body browser.page_source end def current_url browser.current_url end def find(selector) browser.find_elements(:xpath, selector).map { |node| Node.new(self, node) } end def wait?; true; end def resynchronize if options[:resynchronize] load_wait_for_ajax_support yield Capybara.timeout(options[:resynchronization_timeout], self, "failed to resynchronize, ajax request timed out") do evaluate_script("!window.capybaraRequestsOutstanding") end else yield end end def execute_script(script) browser.execute_script script end def evaluate_script(script) browser.execute_script "return #{script}" end def reset! # Use instance variable directly so we avoid starting the browser just to reset the session @browser.manage.delete_all_cookies if @browser end def within_frame(frame_id) old_window = browser.window_handle browser.switch_to.frame(frame_id) yield browser.switch_to.window old_window end def find_window( selector ) original_handle = browser.window_handle browser.window_handles.each do |handle| browser.switch_to.window handle if( selector == browser.execute_script("return window.name") || browser.title.include?(selector) || browser.current_url.include?(selector) || (selector == handle) ) browser.switch_to.window original_handle return handle end end raise Capybara::ElementNotFound, "Could not find a window identified by #{selector}" end def within_window(selector, &blk) handle = find_window( selector ) browser.switch_to.window(handle, &blk) end private def load_wait_for_ajax_support browser.execute_script <<-JS window.capybaraRequestsOutstanding = 0; (function() { // Overriding XMLHttpRequest var oldXHR = window.XMLHttpRequest; function newXHR() { var realXHR = new oldXHR(); window.capybaraRequestsOutstanding++; realXHR.addEventListener("readystatechange", function() { if( realXHR.readyState == 4 ) { setTimeout( function() { window.capybaraRequestsOutstanding--; if(window.capybaraRequestsOutstanding < 0) { window.capybaraRequestsOutstanding = 0; } }, 500 ); } }, false); return realXHR; } window.XMLHttpRequest = newXHR; })(); JS end def url(path) rack_server.url(path) end end