mirror of
https://github.com/teamcapybara/capybara.git
synced 2022-11-09 12:08:07 -05:00
276 lines
7.8 KiB
Ruby
276 lines
7.8 KiB
Ruby
require 'forwardable'
|
|
require 'capybara/wait_until'
|
|
|
|
module Capybara
|
|
class Session
|
|
extend Forwardable
|
|
include Searchable
|
|
|
|
DSL_METHODS = [
|
|
:all, :attach_file, :body, :check, :choose, :click, :click_button, :click_link, :current_url, :drag, :evaluate_script,
|
|
:field_labeled, :fill_in, :find, :find_button, :find_by_id, :find_field, :find_link, :has_content?, :has_css?,
|
|
:has_no_content?, :has_no_css?, :has_no_xpath?, :has_xpath?, :locate, :save_and_open_page, :select, :source, :uncheck,
|
|
:visit, :wait_until, :within, :within_fieldset, :within_table, :within_frame, :has_link?, :has_no_link?, :has_button?,
|
|
:has_no_button?, :has_field?, :has_no_field?, :has_checked_field?, :has_unchecked_field?, :has_no_table?, :has_table?,
|
|
:unselect, :has_select?, :has_no_select?, :current_path, :scope_to
|
|
]
|
|
|
|
attr_reader :mode, :app
|
|
|
|
def initialize(mode, app=nil)
|
|
@mode = mode
|
|
@app = app
|
|
end
|
|
|
|
def driver
|
|
@driver ||= begin
|
|
string = mode.to_s
|
|
string.gsub!(%r{(^.)|(_.)}) { |m| m[m.length-1,1].upcase }
|
|
Capybara::Driver.const_get(string.to_sym).new(app)
|
|
rescue NameError
|
|
raise Capybara::DriverNotFoundError, "no driver called #{mode} was found"
|
|
end
|
|
end
|
|
|
|
def_delegator :driver, :cleanup!
|
|
def_delegator :driver, :current_url
|
|
def_delegator :driver, :current_path
|
|
def_delegator :driver, :response_headers
|
|
def_delegator :driver, :status_code
|
|
def_delegator :driver, :visit
|
|
def_delegator :driver, :body
|
|
def_delegator :driver, :source
|
|
|
|
def click(locator)
|
|
msg = "no link or button '#{locator}' found"
|
|
locate(:xpath, XPath.link(locator).button(locator), msg).click
|
|
end
|
|
|
|
def click_link(locator)
|
|
msg = "no link with title, id or text '#{locator}' found"
|
|
locate(:xpath, XPath.link(locator), msg).click
|
|
end
|
|
|
|
def click_button(locator)
|
|
msg = "no button with value or id or text '#{locator}' found"
|
|
locate(:xpath, XPath.button(locator), msg).click
|
|
end
|
|
|
|
def drag(source_locator, target_locator)
|
|
source = locate(:xpath, source_locator, "drag source '#{source_locator}' not found on page")
|
|
target = locate(:xpath, target_locator, "drag target '#{target_locator}' not found on page")
|
|
source.drag_to(target)
|
|
end
|
|
|
|
def fill_in(locator, options={})
|
|
msg = "cannot fill in, no text field, text area or password field with id, name, or label '#{locator}' found"
|
|
raise "Must pass a hash containing 'with'" if not options.is_a?(Hash) or not options.has_key?(:with)
|
|
locate(:xpath, XPath.fillable_field(locator), msg).set(options[:with])
|
|
end
|
|
|
|
def choose(locator)
|
|
msg = "cannot choose field, no radio button with id, name, or label '#{locator}' found"
|
|
locate(:xpath, XPath.radio_button(locator), msg).set(true)
|
|
end
|
|
|
|
def check(locator)
|
|
msg = "cannot check field, no checkbox with id, name, or label '#{locator}' found"
|
|
locate(:xpath, XPath.checkbox(locator), msg).set(true)
|
|
end
|
|
|
|
def uncheck(locator)
|
|
msg = "cannot uncheck field, no checkbox with id, name, or label '#{locator}' found"
|
|
locate(:xpath, XPath.checkbox(locator), msg).set(false)
|
|
end
|
|
|
|
def select(value, options={})
|
|
msg = "cannot select option, no select box with id, name, or label '#{options[:from]}' found"
|
|
locate(:xpath, XPath.select(options[:from]), msg).select(value)
|
|
end
|
|
|
|
def unselect(value, options={})
|
|
msg = "cannot unselect option, no select box with id, name, or label '#{options[:from]}' found"
|
|
locate(:xpath, XPath.select(options[:from]), msg).unselect(value)
|
|
end
|
|
|
|
def attach_file(locator, path)
|
|
msg = "cannot attach file, no file field with id, name, or label '#{locator}' found"
|
|
locate(:xpath, XPath.file_field(locator), msg).set(path)
|
|
end
|
|
|
|
def within(kind, scope=nil)
|
|
kind, scope = Capybara.default_selector, kind unless scope
|
|
scope = XPath.from_css(scope) if kind == :css
|
|
locate(:xpath, scope, "scope '#{scope}' not found on page")
|
|
begin
|
|
scopes.push(scope)
|
|
yield
|
|
ensure
|
|
scopes.pop
|
|
end
|
|
end
|
|
|
|
def within_fieldset(locator)
|
|
within :xpath, XPath.fieldset(locator) do
|
|
yield
|
|
end
|
|
end
|
|
|
|
def within_table(locator)
|
|
within :xpath, XPath.table(locator) do
|
|
yield
|
|
end
|
|
end
|
|
|
|
def within_frame(frame_id)
|
|
driver.within_frame(frame_id) do
|
|
yield
|
|
end
|
|
end
|
|
|
|
def scope_to(*locator)
|
|
scoped_session = self.clone
|
|
scoped_session.instance_eval do
|
|
@scopes = scopes + locator
|
|
end
|
|
scoped_session
|
|
end
|
|
|
|
def has_xpath?(path, options={})
|
|
wait_conditionally_until do
|
|
results = all(:xpath, path, options)
|
|
|
|
if options[:count]
|
|
results.size == options[:count]
|
|
else
|
|
results.size > 0
|
|
end
|
|
end
|
|
rescue Capybara::TimeoutError
|
|
return false
|
|
end
|
|
|
|
def has_no_xpath?(path, options={})
|
|
wait_conditionally_until do
|
|
results = all(:xpath, path, options)
|
|
|
|
if options[:count]
|
|
results.size != options[:count]
|
|
else
|
|
results.empty?
|
|
end
|
|
end
|
|
rescue Capybara::TimeoutError
|
|
return false
|
|
end
|
|
|
|
def has_css?(path, options={})
|
|
has_xpath?(XPath.from_css(path), options)
|
|
end
|
|
|
|
def has_no_css?(path, options={})
|
|
has_no_xpath?(XPath.from_css(path), options)
|
|
end
|
|
|
|
def has_content?(content)
|
|
has_xpath?(XPath.content(content))
|
|
end
|
|
|
|
def has_no_content?(content)
|
|
has_no_xpath?(XPath.content(content))
|
|
end
|
|
|
|
def has_link?(locator)
|
|
has_xpath?(XPath.link(locator))
|
|
end
|
|
|
|
def has_no_link?(locator)
|
|
has_no_xpath?(XPath.link(locator))
|
|
end
|
|
|
|
def has_button?(locator)
|
|
has_xpath?(XPath.button(locator))
|
|
end
|
|
|
|
def has_no_button?(locator)
|
|
has_no_xpath?(XPath.button(locator))
|
|
end
|
|
|
|
def has_field?(locator, options={})
|
|
has_xpath?(XPath.field(locator, options))
|
|
end
|
|
|
|
def has_no_field?(locator, options={})
|
|
has_no_xpath?(XPath.field(locator, options))
|
|
end
|
|
|
|
def has_checked_field?(locator)
|
|
has_xpath?(XPath.field(locator, :checked => true))
|
|
end
|
|
|
|
def has_unchecked_field?(locator)
|
|
has_xpath?(XPath.field(locator, :unchecked => true))
|
|
end
|
|
|
|
def has_select?(locator, options={})
|
|
has_xpath?(XPath.select(locator, options))
|
|
end
|
|
|
|
def has_no_select?(locator, options={})
|
|
has_no_xpath?(XPath.select(locator, options))
|
|
end
|
|
|
|
def has_table?(locator, options={})
|
|
has_xpath?(XPath.table(locator, options))
|
|
end
|
|
|
|
def has_no_table?(locator, options={})
|
|
has_no_xpath?(XPath.table(locator, options))
|
|
end
|
|
|
|
def save_and_open_page
|
|
require 'capybara/save_and_open_page'
|
|
Capybara::SaveAndOpenPage.save_and_open_page(body)
|
|
end
|
|
|
|
#return node identified by locator or raise ElementNotFound(using desc)
|
|
def locate(kind_or_locator, locator=nil, fail_msg = nil)
|
|
node = wait_conditionally_until { find(kind_or_locator, locator) }
|
|
ensure
|
|
raise Capybara::ElementNotFound, fail_msg || "Unable to locate '#{locator || kind_or_locator}'" unless node
|
|
return node
|
|
end
|
|
|
|
def wait_until(timeout = Capybara.default_wait_time)
|
|
WaitUntil.timeout(timeout,driver) { yield }
|
|
end
|
|
|
|
def execute_script(script)
|
|
driver.execute_script(script)
|
|
end
|
|
|
|
def evaluate_script(script)
|
|
driver.evaluate_script(script)
|
|
end
|
|
|
|
private
|
|
|
|
def wait_conditionally_until
|
|
if driver.wait? then wait_until { yield } else yield end
|
|
end
|
|
|
|
def all_unfiltered(locator)
|
|
XPath.wrap(locator).scope(current_scope).paths.map do |path|
|
|
driver.find(path)
|
|
end.flatten
|
|
end
|
|
|
|
def current_scope
|
|
scopes.join('')
|
|
end
|
|
|
|
def scopes
|
|
@scopes ||= []
|
|
end
|
|
end
|
|
end
|