teamcapybara--capybara/lib/capybara/xpath.rb

159 lines
4.4 KiB
Ruby
Raw Normal View History

2009-12-09 18:03:55 +00:00
module Capybara
# this is a class for generating XPath queries, use it like this:
# Xpath.text_field('foo').link('blah').to_s
# this will generate an XPath that matches either a text field or a link
class XPath
2009-12-09 18:03:55 +00:00
class << self
def from_css(css)
Nokogiri::CSS.xpath_for(css).first
end
alias_method :for_css, :from_css
2010-01-01 16:48:39 +00:00
def wrap(path)
if path.is_a?(self)
path
else
new(path.to_s)
end
end
2010-01-01 16:48:39 +00:00
2009-12-09 18:03:55 +00:00
def respond_to?(method)
new.respond_to?(method)
end
def method_missing(*args)
new.send(*args)
end
end
attr_reader :paths
def initialize(*paths)
@paths = paths
end
2010-01-18 20:28:06 +00:00
def field(locator, options={})
if options[:with]
fillable_field(locator, options)
else
xpath = fillable_field(locator)
xpath = xpath.input_field(:file, locator, options)
xpath = xpath.checkbox(locator, options)
xpath = xpath.radio_button(locator, options)
xpath.select(locator, options)
2010-01-18 20:28:06 +00:00
end
2009-12-09 18:03:55 +00:00
end
2010-01-18 20:28:06 +00:00
def fillable_field(locator, options={})
[:text, :password, :email, :url, :search, :tel, :color].inject(text_area(locator, options)) do |all, type|
2010-01-18 20:28:06 +00:00
all.input_field(type, locator, options)
2010-01-11 19:36:05 +00:00
end
2009-12-09 18:03:55 +00:00
end
2010-01-01 16:48:39 +00:00
def content(locator)
append("/descendant-or-self::*[contains(.,#{s(locator)})]")
end
2010-01-01 16:48:39 +00:00
def table(locator)
append("//table[@id=#{s(locator)} or contains(caption,#{s(locator)})]")
end
2010-01-01 16:48:39 +00:00
def fieldset(locator)
append("//fieldset[@id=#{s(locator)} or contains(legend,#{s(locator)})]")
end
2010-01-01 16:48:39 +00:00
def link(locator)
xpath = append("//a[@href][@id=#{s(locator)} or contains(.,#{s(locator)}) or contains(@title,#{s(locator)}) or img[contains(@alt,#{s(locator)})]]")
xpath.prepend("//a[@href][text()=#{s(locator)} or @title=#{s(locator)} or img[@alt=#{s(locator)}]]")
end
2010-01-01 16:48:39 +00:00
def button(locator)
xpath = append("//input[@type='submit' or @type='image'][@id=#{s(locator)} or contains(@value,#{s(locator)})]")
xpath = xpath.append("//button[@id=#{s(locator)} or contains(@value,#{s(locator)}) or contains(.,#{s(locator)})]")
xpath = xpath.prepend("//input[@type='submit' or @type='image'][@value=#{s(locator)}]")
xpath = xpath.prepend("//button[@value=#{s(locator)} or text()=#{s(locator)}]")
end
2009-12-09 18:03:55 +00:00
def text_area(locator, options={})
add_field(locator, "//textarea", options)
2009-12-09 18:03:55 +00:00
end
def select(locator, options={})
add_field(locator, "//select", options)
2009-12-09 18:03:55 +00:00
end
2010-01-18 20:28:06 +00:00
def input_field(type, locator, options={})
add_field(locator, "//input[@type='#{type}']", options)
2009-12-09 18:03:55 +00:00
end
2010-01-01 16:48:39 +00:00
def scope(scope)
XPath.new(*paths.map { |p| scope + p })
end
2009-12-09 18:03:55 +00:00
def to_s
@paths.join(' | ')
end
2010-01-01 16:48:39 +00:00
def append(path)
XPath.new(*[@paths, XPath.wrap(path).paths].flatten)
end
2010-01-01 16:48:39 +00:00
def prepend(path)
XPath.new(*[XPath.wrap(path).paths, @paths].flatten)
end
2009-12-09 18:03:55 +00:00
def checkbox(locator, options={})
input_field(:checkbox, locator, options)
2010-01-11 19:36:05 +00:00
end
def radio_button(locator, options={})
input_field(:radio, locator, options)
2010-01-11 19:36:05 +00:00
end
[:text, :password, :email, :url, :search, :tel, :color, :file].each do |type|
class_eval <<-RUBY, __FILE__, __LINE__+1
def #{type}_field(locator)
input_field(:#{type}, locator)
end
RUBY
end
protected
2010-01-01 16:48:39 +00:00
2010-01-18 20:28:06 +00:00
def add_field(locator, field, options={})
postfix = extract_postfix(options)
xpath = append("#{field}[@id=#{s(locator)}]#{postfix}")
xpath = xpath.append("#{field}[@name=#{s(locator)}]#{postfix}")
xpath = xpath.append("#{field}[@id=//label[contains(.,#{s(locator)})]/@for]#{postfix}")
xpath = xpath.append("//label[contains(.,#{s(locator)})]#{field}#{postfix}")
xpath.prepend("#{field}[@id=//label[text()=#{s(locator)}]/@for]#{postfix}")
end
def extract_postfix(options)
options.inject("") do |postfix, (key, value)|
case key
when :with then postfix += "[@value=#{s(value)}]"
when :checked then postfix += "[@checked]"
when :unchecked then postfix += "[not(@checked)]"
2010-01-18 20:28:06 +00:00
end
postfix
end
end
2010-01-01 16:48:39 +00:00
# Sanitize a String for putting it into an xpath query
def s(string)
if string.include?("'")
string = string.split("'", -1).map do |substr|
"'#{substr}'"
end.join(%q{,"'",})
"concat(#{string})"
else
"'#{string}'"
end
2010-01-01 16:48:39 +00:00
end
2009-12-09 18:03:55 +00:00
end
end