module Watir # # The main class through which you control the browser. # class Browser include Container include HasWindow include Waitable attr_writer :default_context, :original_window attr_reader :driver attr_reader :after_hooks alias_method :wd, :driver # ensures duck typing with Watir::Element class << self # # Creates a Watir::Browser instance and goes to URL. # # @example # browser = Watir::Browser.start "www.google.com", :chrome # #=> # # # @param [String] url # @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance # @return [Watir::Browser] # def start(url, browser = :chrome, *args) b = new(browser, *args) b.goto url b end end # # Creates a Watir::Browser instance. # # @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance # @param args Passed to the underlying driver # def initialize(browser = :chrome, *args) case browser when ::Symbol, String selenium_args = Watir::Capabilities.new(browser, *args).to_args @driver = Selenium::WebDriver.for *selenium_args when Selenium::WebDriver::Driver @driver = browser else raise ArgumentError, "expected Symbol or Selenium::WebDriver::Driver, got #{browser.class}" end @after_hooks = AfterHooks.new(self) @closed = false @default_context = true end def inspect '#<%s:0x%x url=%s title=%s>' % [self.class, hash*2, url.inspect, title.inspect] rescue Errno::ECONNREFUSED '#<%s:0x%x closed=true>' % [self.class, hash*2] rescue Selenium::WebDriver::Error::UnhandledAlertError, Selenium::WebDriver::Error::UnexpectedAlertOpenError '#<%s:0x%x alert=true>' % [self.class, hash*2] end alias selector_string inspect # # Goes to the given URL. # # @example # browser.goto "watir.github.io" # # @param [String] uri The url. # @return [String] The url you end up at. # def goto(uri) uri = "http://#{uri}" unless uri =~ URI.regexp @driver.navigate.to uri @after_hooks.run uri end # # Navigates back in history. # def back @driver.navigate.back end # # Navigates forward in history. # def forward @driver.navigate.forward end # # Returns URL of current page. # # @example # browser.goto "watir.com" # browser.url # #=> "http://watir.com/" # # @return [String] # def url assert_exists @driver.current_url end # # Returns title of current page. # # @example # browser.goto "watir.github.io" # browser.title # #=> "Watir is... – Watir Project – Watir stands for Web Application Testing In Ruby. It facilitates the writing of automated tests by mimicking the behavior of a user interacting with a website." # # @return [String] # def title @driver.title end # # Closes browser. # def close return if @closed @driver.quit @closed = true end alias_method :quit, :close # TODO: close vs quit # # Handles cookies. # # @return [Watir::Cookies] # def cookies @cookies ||= Cookies.new driver.manage end # # Returns browser name. # # @example # browser = Watir::Browser.new :chrome # browser.name # #=> :chrome # # @return [Symbol] # def name @driver.browser end # # Returns text of page body. # # @return [String] # def text body.text end # # Returns HTML code of current page. # # @return [String] # def html # use body.html instead? @driver.page_source end # # Handles JavaScript alerts, confirms and prompts. # # @return [Watir::Alert] # def alert Alert.new(self) end # # Refreshes current page. # def refresh @driver.navigate.refresh @after_hooks.run end # # Waits until readyState of document is complete. # # @example # browser.wait # # @param [Integer] timeout # @raise [Watir::Wait::TimeoutError] if timeout is exceeded # def wait(timeout = 5) wait_until(timeout: timeout, message: "waiting for document.readyState == 'complete'") do ready_state == "complete" end end # # Returns readyState of document. # # @return [String] # def ready_state execute_script 'return document.readyState' end # # Returns the text of status bar. # # @return [String] # def status execute_script "return window.status;" end # # Executes JavaScript snippet. # # If you are going to use the value snippet returns, make sure to use # `return` explicitly. # # @example Check that Ajax requests are completed with jQuery # browser.execute_script("return jQuery.active") == 0 # #=> true # # @param [String] script JavaScript snippet to execute # @param *args Arguments will be available in the given script in the 'arguments' pseudo-array # def execute_script(script, *args) args.map! { |e| e.kind_of?(Watir::Element) ? e.wd : e } returned = @driver.execute_script(script, *args) wrap_elements_in(self, returned) end # # Sends sequence of keystrokes to currently active element. # # @example # browser.goto "www.google.com" # browser.send_keys "Watir", :return # # @param [String, Symbol] *args # def send_keys(*args) @driver.switch_to.active_element.send_keys(*args) end # # Handles screenshots of current pages. # # @return [Watir::Screenshot] # def screenshot Screenshot.new driver end # # Returns true if browser is not closed and false otherwise. # # @return [Boolean] # def exist? !@closed && window.present? end alias_method :exists?, :exist? # # Protocol shared with Watir::Element # # @api private # def assert_exists ensure_context return if window.present? raise Exception::NoMatchingWindowFoundException, "browser window was closed" end def ensure_context raise Exception::Error, "browser was closed" if @closed driver.switch_to.default_content unless @default_context @default_context = true end def browser self end private def wrap_elements_in(scope, obj) case obj when Selenium::WebDriver::Element wrap_element(scope, obj) when Array obj.map { |e| wrap_elements_in(scope, e) } when Hash obj.each { |k,v| obj[k] = wrap_elements_in(scope, v) } obj else obj end end def wrap_element(scope, element) Watir.element_class_for(element.tag_name.downcase).new(scope, element: element) end end # Browser end # Watir