module Watir
#
# Base class for HTML elements.
#
class Element
extend AttributeHelper
include Exception
include Container
include EventuallyPresent
include Waitable
include Adjacent
include JSExecution
include Locators::ClassHelpers
attr_accessor :keyword
attr_reader :selector
#
# temporarily add :id and :class_name manually since they're no longer specified in the HTML spec.
#
# @see http://html5.org/r/6605
# @see http://html5.org/r/7174
#
# TODO: use IDL from DOM core - http://dvcs.w3.org/hg/domcore/raw-file/tip/Overview.html
#
attribute String, :id, :id
attribute String, :class_name, :className
def initialize(query_scope, selector)
@query_scope = query_scope
unless selector.kind_of? Hash
raise ArgumentError, "invalid argument: #{selector.inspect}"
end
@element = selector.delete(:element)
@selector = selector
end
#
# Returns true if element exists.
#
# @return [Boolean]
#
def exists?
assert_exists
true
rescue UnknownObjectException, UnknownFrameException
false
end
alias_method :exist?, :exists?
def inspect
string = "#<#{self.class}: "
string << "keyword: #{keyword} " if keyword
string << "located: #{!!@element}; "
if @selector.empty?
string << '{element: (selenium element)}'
else
string << selector_string
end
string << '>'
string
end
#
# Returns true if two elements are equal.
#
# @example
# browser.text_field(name: "new_user_first_name") == browser.text_field(name: "new_user_first_name")
# #=> true
#
def ==(other)
other.is_a?(self.class) && wd == other.wd
end
alias_method :eql?, :==
def hash
@element ? @element.hash : super
end
#
# Returns the text of the element.
#
# @return [String]
#
def text
element_call { @element.text }
end
#
# Returns tag name of the element.
#
# @return [String]
#
def tag_name
element_call { @element.tag_name.downcase }
end
#
# Clicks the element, optionally while pressing the given modifier keys.
# Note that support for holding a modifier key is currently experimental,
# and may not work at all.
#
# @example Click an element
# browser.element(name: "new_user_button").click
#
# @example Click an element with shift key pressed
# browser.element(name: "new_user_button").click(:shift)
#
# @example Click an element with several modifier keys pressed
# browser.element(name: "new_user_button").click(:shift, :control)
#
# @param [:shift, :alt, :control, :command, :meta] modifiers to press while clicking.
#
def click(*modifiers)
element_call(:wait_for_enabled) do
if modifiers.any?
action = driver.action
modifiers.each { |mod| action.key_down mod }
action.click @element
modifiers.each { |mod| action.key_up mod }
action.perform
else
@element.click
end
end
browser.after_hooks.run
end
#
# Simulates JavaScript click event on element.
#
# @example Click an element
# browser.element(name: "new_user_button").click!
#
def click!
fire_event :click
browser.after_hooks.run
end
#
# Double clicks the element.
# Note that browser support may vary.
#
# @example
# browser.element(name: "new_user_button").double_click
#
def double_click
element_call(:wait_for_present) { driver.action.double_click(@element).perform }
browser.after_hooks.run
end
#
# Simulates JavaScript double click event on element.
#
# @example
# browser.element(name: "new_user_button").double_click!
#
def double_click!
fire_event :dblclick
browser.after_hooks.run
end
#
# Right clicks the element.
# Note that browser support may vary.
#
# @example
# browser.element(name: "new_user_button").right_click
#
def right_click
element_call(:wait_for_present) { driver.action.context_click(@element).perform }
browser.after_hooks.run
end
#
# Moves the mouse to the middle of this element.
# Note that browser support may vary.
#
# @example
# browser.element(name: "new_user_button").hover
#
def hover
element_call(:wait_for_present) { driver.action.move_to(@element).perform }
end
#
# Drag and drop this element on to another element instance.
# Note that browser support may vary.
#
# @example
# a = browser.div(id: "draggable")
# b = browser.div(id: "droppable")
# a.drag_and_drop_on b
#
def drag_and_drop_on(other)
assert_is_element other
value = element_call(:wait_for_present) do
driver.action.
drag_and_drop(@element, other.wd).
perform
end
browser.after_hooks.run
value
end
#
# Drag and drop this element by the given offsets.
# Note that browser support may vary.
#
# @example
# browser.div(id: "draggable").drag_and_drop_by 100, -200
#
# @param [Integer] right_by
# @param [Integer] down_by
#
def drag_and_drop_by(right_by, down_by)
element_call(:wait_for_present) do
driver.action.
drag_and_drop_by(@element, right_by, down_by).
perform
end
end
#
# Returns value of the element.
#
# @return [String]
#
def value
attribute_value('value') || ''
rescue Selenium::WebDriver::Error::InvalidElementStateError
''
end
#
# Returns given attribute value of element.
#
# @example
# browser.a(id: "link_2").attribute_value "title"
# #=> "link_title_2"
#
# @param [String] attribute_name
# @return [String, nil]
#
def attribute_value(attribute_name)
element_call { @element.attribute attribute_name }
end
alias_method :attribute, :attribute_value
#
# Sends sequence of keystrokes to element.
#
# @example
# browser.text_field(name: "new_user_first_name").send_keys "Watir", :return
#
# @param [String, Symbol] args
#
def send_keys(*args)
element_call(:wait_for_writable) { @element.send_keys(*args) }
end
#
# Returns true if this element is focused.
#
# @return [Boolean]
#
def focused?
element_call { @element == driver.switch_to.active_element }
end
#
# Scroll until the element is in the view screen
#
# @example
# browser.button(name: "new_user_button").scroll_into_view
#
# @return [Selenium::WebDriver::Point]
#
def scroll_into_view
element_call { @element.location_once_scrolled_into_view }
end
#
# location of element (x, y)
#
# @example
# browser.button(name: "new_user_button").location
#
# @return [Selenium::WebDriver::Point]
#
def location
element_call { @element.location }
end
#
# size of element (width, height)
#
# @example
# browser.button(name: "new_user_button").size
#
# @return [Selenium::WebDriver::Dimension]
#
def size
element_call { @element.size }
end
#
# Get height of element
#
# @example
# browser.button(name: "new_user_button").height
#
# @return [Selenium::WebDriver::Dimension]
#
def height
size['height']
end
#
# Get width of element
#
# @example
# browser.button(name: "new_user_button").width
#
# @return [Selenium::WebDriver::Dimension]
#
def width
size['width']
end
#
# Get centre coordinates of element
#
# @example
# browser.button(name: "new_user_button").centre
#
# @return [Selenium::WebDriver::Point]
#
def center
point = location
dimensions = size
Selenium::WebDriver::Point.new(point.x + (dimensions['width']/2),
point.y + (dimensions['height']/2))
end
alias_method :centre, :center
#
# @api private
#
def driver
@query_scope.driver
end
#
# @api private
#
def wd
return driver if @element.is_a? FramedDriver
assert_exists if @element.nil?
@element
end
#
# Returns true if this element is visible on the page.
# Raises exception if element does not exist
#
# @return [Boolean]
#
def visible?
element_call(:assert_exists) { @element.displayed? }
end
#
# Returns true if this element is present and enabled on the page.
#
# @return [Boolean]
# @see Watir::Wait
#
def enabled?
element_call(:assert_exists) { @element.enabled? }
end
#
# Returns true if the element exists and is visible on the page.
# Returns false if element does not exist or exists but is not visible
#
# @return [Boolean]
# @see Watir::Wait
#
def present?
visible?
rescue UnknownObjectException, UnknownFrameException
false
end
#
# Returns given style property of this element.
#
# @example
# browser.button(value: "Delete").style #=> "border: 4px solid red;"
# browser.button(value: "Delete").style("border") #=> "4px solid rgb(255, 0, 0)"
#
# @param [String] property
# @return [String]
#
def style(property = nil)
if property
element_call { @element.style property }
else
attribute_value("style").to_s.strip
end
end
#
# Cast this Element instance to a more specific subtype.
#
# @example
# browser.element(xpath: "//input[@type='submit']").to_subtype
# #=> #
#
def to_subtype
tag = tag_name()
klass = if tag == "input"
case attribute_value(:type)
when *Button::VALID_TYPES
Button
when 'checkbox'
CheckBox
when 'radio'
Radio
when 'file'
FileField
else
TextField
end
else
Watir.element_class_for(tag)
end
klass.new(@query_scope, @selector.merge(element: wd))
end
#
# Returns browser.
#
# @return [Watir::Browser]
#
def browser
@query_scope.browser
end
#
# Returns true if a previously located element is no longer attached to DOM.
#
# @return [Boolean]
# @see Watir::Wait
#
def stale?
raise Watir::Exception::Error, "Can not check staleness of unused element" unless @element
@element.enabled? # any wire call will check for staleness
false
rescue Selenium::WebDriver::Error::ObsoleteElementError
true
end
def reset!
@element = nil
end
protected
def wait_for_exists
return assert_exists unless Watir.relaxed_locate?
return if exists? # Performance shortcut
begin
@query_scope.wait_for_exists unless @query_scope.is_a? Browser
wait_until(&:exists?)
rescue Watir::Wait::TimeoutError
msg = "timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to be located"
raise unknown_exception, msg
end
end
def wait_for_present
return visible? unless Watir.relaxed_locate?
return if present?
begin
@query_scope.wait_for_present unless @query_scope.is_a? Browser
wait_until_present
rescue Watir::Wait::TimeoutError
msg = "element located, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to be present"
raise unknown_exception, msg
end
end
def wait_for_enabled
return assert_enabled unless Watir.relaxed_locate?
wait_for_exists
return if [Input, Button, Select, Option].none? { |c| self.is_a? c }
begin
wait_until(&:enabled?)
rescue Watir::Wait::TimeoutError
message = "element present, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to be enabled"
raise ObjectDisabledException, message
end
end
def wait_for_writable
wait_for_exists
wait_for_enabled
unless Watir.relaxed_locate?
raise_writable unless !respond_to?(:readonly?) || !readonly?
end
begin
wait_until { !respond_to?(:readonly?) || !readonly? }
rescue Watir::Wait::TimeoutError
message = "element present and enabled, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to not be readonly"
raise ObjectReadOnlyException, message
end
end
# Ensure that the element exists, making sure that it is not stale and located if necessary
def assert_exists
if @element && @selector.empty?
@query_scope.ensure_context
reset! if stale?
elsif @element && !stale?
return
else
@element = locate
end
assert_element_found
end
def assert_element_found
return if @element
raise unknown_exception, "unable to locate element: #{inspect}"
end
def locate
@query_scope.ensure_context
element_validator = element_validator_class.new
selector_builder = selector_builder_class.new(@query_scope, @selector.dup, self.class.attribute_list)
@locator = locator_class.new(@query_scope, @selector.dup, selector_builder, element_validator)
@element = @locator.locate
end
def selector_string
return @selector.inspect if @query_scope.is_a?(Browser)
"#{@query_scope.send :selector_string} --> #{@selector.inspect}"
end
# Ensure the driver is in the desired browser context
def ensure_context
assert_exists
end
private
def unknown_exception
Watir::Exception::UnknownObjectException
end
def raise_writable
message = "element present and enabled, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to not be readonly"
raise ObjectReadOnlyException, message
end
def raise_disabled
message = "element present and enabled, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to not be disabled"
raise ObjectDisabledException, message
end
def raise_present
raise unknown_exception, "element located, but timed out after #{Watir.default_timeout} seconds, waiting for #{inspect} to be present"
end
def element_class
self.class
end
def attribute?(attribute_name)
!attribute_value(attribute_name).nil?
end
def assert_enabled
unless element_call { @element.enabled? }
raise ObjectDisabledException, "object is disabled #{inspect}"
end
end
def assert_is_element(obj)
unless obj.kind_of? Watir::Element
raise TypeError, "execpted Watir::Element, got #{obj.inspect}:#{obj.class}"
end
end
def element_call(exist_check = :wait_for_exists)
already_locked = Wait.timer.locked?
caller = caller_locations(1,1)[0].label
if already_locked
Watir.logger.info "-> `#{inspect}##{caller}` after `##{exist_check}` (as a prerequisite for a previously specified execution)"
else
Watir.logger.info "-> `#{inspect}##{caller}` after `##{exist_check}`"
Wait.timer = Wait::Timer.new(timeout: Watir.default_timeout)
end
begin
send exist_check
yield
rescue unknown_exception => ex
msg = ex.message
msg += "; Maybe look in an iframe?" if @query_scope.iframes.count > 0
custom_attributes = @locator.selector_builder.custom_attributes
msg += "; Watir treated #{custom_attributes} as a non-HTML compliant attribute, ensure that was intended" unless custom_attributes.empty?
raise unknown_exception, msg
rescue Selenium::WebDriver::Error::StaleElementReferenceError
retry
rescue Selenium::WebDriver::Error::ElementNotVisibleError, Selenium::WebDriver::Error::ElementNotInteractableError
raise_present unless Wait.timer.remaining_time > 0
raise_present unless exist_check == :wait_for_present || exist_check == :wait_for_enabled
retry
rescue Selenium::WebDriver::Error::InvalidElementStateError
raise_disabled unless Wait.timer.remaining_time > 0
raise_disabled unless exist_check == :wait_for_writable || exist_check == :wait_for_enabled
retry
rescue Selenium::WebDriver::Error::NoSuchWindowError
raise Exception::NoMatchingWindowFoundException, "browser window was closed"
ensure
Watir.logger.info "<- `#{inspect}##{caller}` has been completed"
Wait.timer.reset! unless already_locked
end
end
def method_missing(meth, *args, &blk)
method = meth.to_s
if method =~ Locators::Element::SelectorBuilder::WILDCARD_ATTRIBUTE
attribute_value(method.tr('_', '-'), *args)
else
super
end
end
def respond_to_missing?(meth, *)
Locators::Element::SelectorBuilder::WILDCARD_ATTRIBUTE === meth.to_s || super
end
end # Element
end # Watir