class Capybara::Selenium::SafariNode

def _send_keys(keys, actions = browser_action, down_keys = ModifierKeysStack.new)

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

def click(keys = [], **options)

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 disabled?

def disabled?
  driver.evaluate_script("arguments[0].matches(':disabled, select:disabled *')", self)
end

def hover

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

def select_option

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 send_keys(*args)

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_file(value) # rubocop:disable Naming/AccessorMethodName

rubocop:disable Naming/AccessorMethodName
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 set_text(value, clear: nil, **_unused)

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 unselect_option

def unselect_option
  driver.execute_script("arguments[0].closest('select').scrollIntoView()", self)
  super
end

def visible_text

def visible_text
  return '' unless visible?
  vis_text = driver.execute_script('return arguments[0].innerText', self)
  vis_text.squeeze(' ')
          .gsub(/[\ \n]*\n[\ \n]*/, "\n")
          .gsub(/\A[[:space:]&&[^\u00a0]]+/, '')
          .gsub(/[[:space:]&&[^\u00a0]]+\z/, '')
          .tr("\u00a0", ' ')
end