119 lines
4.0 KiB
Ruby
119 lines
4.0 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# require 'capybara/selenium/extensions/html5_drag'
|
|
require 'capybara/selenium/extensions/modifier_keys_stack'
|
|
|
|
class Capybara::Selenium::SafariNode < Capybara::Selenium::Node
|
|
# include Html5Drag
|
|
|
|
def click(keys = [], **options)
|
|
# driver.execute_script('arguments[0].scrollIntoViewIfNeeded({block: "center"})', self)
|
|
super
|
|
rescue ::Selenium::WebDriver::Error::ElementNotInteractableError
|
|
if tag_name == 'tr'
|
|
warn 'You are attempting to click a table row which has issues in safaridriver - '\
|
|
'Your test should probably be clicking on a table cell like a user would. '\
|
|
'Clicking the first cell in the row instead.'
|
|
return find_css('th:first-child,td:first-child')[0].click(keys, options)
|
|
end
|
|
raise
|
|
rescue ::Selenium::WebDriver::Error::WebDriverError => e
|
|
raise unless e.instance_of? ::Selenium::WebDriver::Error::WebDriverError
|
|
|
|
# Safari doesn't return a specific error here - assume it's an ElementNotInteractableError
|
|
raise ::Selenium::WebDriver::Error::ElementNotInteractableError,
|
|
'Non distinct error raised in #click, translated to ElementNotInteractableError for retry'
|
|
end
|
|
|
|
def select_option
|
|
# To optimize to only one check and then click
|
|
selected_or_disabled = driver.execute_script(<<~JS, self)
|
|
arguments[0].closest('select').scrollIntoView();
|
|
return arguments[0].matches(':disabled, select:disabled *, :checked');
|
|
JS
|
|
click unless selected_or_disabled
|
|
end
|
|
|
|
def unselect_option
|
|
driver.execute_script("arguments[0].closest('select').scrollIntoView()", self)
|
|
super
|
|
end
|
|
|
|
def visible_text
|
|
return '' unless visible?
|
|
|
|
vis_text = driver.execute_script('return arguments[0].innerText', self)
|
|
vis_text.gsub(/\ +/, ' ')
|
|
.gsub(/[\ \n]*\n[\ \n]*/, "\n")
|
|
.gsub(/\A[[:space:]&&[^\u00a0]]+/, '')
|
|
.gsub(/[[:space:]&&[^\u00a0]]+\z/, '')
|
|
.tr("\u00a0", ' ')
|
|
end
|
|
|
|
def disabled?
|
|
driver.evaluate_script("arguments[0].matches(':disabled, select:disabled *')", self)
|
|
end
|
|
|
|
def set_file(value) # rubocop:disable Naming/AccessorMethodName
|
|
# By default files are appended so we have to clear here if its multiple and already set
|
|
native.clear if multiple? && driver.evaluate_script('arguments[0].files', self).any?
|
|
super
|
|
end
|
|
|
|
def send_keys(*args)
|
|
if args.none? { |arg| arg.is_a?(Array) || (arg.is_a?(Symbol) && MODIFIER_KEYS.include?(arg)) }
|
|
return super(*args.map { |arg| arg == :space ? ' ' : arg })
|
|
end
|
|
|
|
native.click
|
|
_send_keys(args).perform
|
|
end
|
|
|
|
def set_text(value, clear: nil, **_unused)
|
|
value = value.to_s
|
|
if clear == :backspace
|
|
# Clear field by sending the correct number of backspace keys.
|
|
backspaces = [:backspace] * self.value.to_s.length
|
|
send_keys(*([[:control, 'e']] + backspaces + [value]))
|
|
else
|
|
super.tap do
|
|
# React doesn't see the safaridriver element clear
|
|
send_keys(:space, :backspace) if value.to_s.empty? && clear.nil?
|
|
end
|
|
end
|
|
end
|
|
|
|
def hover
|
|
# Workaround issue where hover would sometimes fail - possibly due to mouse not having moved
|
|
scroll_if_needed { browser_action.move_to(native, 0, 0).move_to(native).perform }
|
|
end
|
|
|
|
private
|
|
|
|
def _send_keys(keys, actions = browser_action, down_keys = ModifierKeysStack.new)
|
|
case keys
|
|
when *MODIFIER_KEYS
|
|
down_keys.press(keys)
|
|
actions.key_down(keys)
|
|
when String
|
|
keys = keys.upcase if down_keys&.include?(:shift)
|
|
actions.send_keys(keys)
|
|
when Symbol
|
|
actions.send_keys(keys)
|
|
when Array
|
|
down_keys.push
|
|
keys.each { |sub_keys| _send_keys(sub_keys, actions, down_keys) }
|
|
down_keys.pop.reverse_each { |key| actions.key_up(key) }
|
|
else
|
|
raise ArgumentError, 'Unknown keys type'
|
|
end
|
|
actions
|
|
end
|
|
|
|
MODIFIER_KEYS = %i[control left_control right_control
|
|
alt left_alt right_alt
|
|
shift left_shift right_shift
|
|
meta left_meta right_meta
|
|
command].freeze
|
|
end
|