# frozen_string_literal: true
module RSpecHTML
# Provides element/attribute/text searching for HTML entities
# rubocop:disable Metrics/ClassLength
class Search
attr_reader :siblings
def initialize(element, siblings, element_wrapper)
@element = element
@siblings = siblings
@element_wrapper = element_wrapper
end
def to_s
@element&.to_s
end
def include?(val)
text.include?(val)
end
def css(*args)
self.class.new(@element&.css(*args), :css)
end
def xpath(*args)
self.class.new(@element&.xpath(*args), :xpath)
end
def present?
!@element.nil?
end
alias exist? present?
def checked?
raise ElementNotFoundError, "Element does not exist: #{element_wrapper.reconstituted}" if @element.nil?
@element.attributes.key?('checked')
end
# rubocop:disable Naming/PredicateName
def has_css?(*args)
!@element&.css(*args)&.empty?
end
def has_xpath?(*args)
!@element&.xpath(*args)&.empty?
end
# rubocop:enable Naming/PredicateName
def [](val)
return index(val) if val.is_a?(Integer)
return range(val) if val.is_a?(Range)
@element&.attr(val.to_s)
end
def text
@element&.text&.gsub(/\s+/, ' ')&.strip || ''
end
def truncated_text
max = RSpec::Support::ObjectFormatter.default_instance.max_formatted_output_length
return text if text.size <= max
"#{text[0..max]}...#{text[-max..-1]}"
end
def attributes
@element&.attributes&.to_h { |key, val| [key.to_sym, val.to_s] } || {}
end
def size
return @element.size if @element.respond_to?(:size)
@siblings&.size || 0
end
alias length size
def new_from_find(tag, options)
Element.new(
find(tag),
tag,
options: options,
siblings: find(tag, all: true)
)
end
def new_from_where(tag, options)
Element.new(
where(tag, options),
tag,
options: options,
siblings: where(tag, options, all: true)
)
end
private
attr_reader :element_wrapper
def index(val)
zero_index_error if val.zero?
self.class.new(@siblings[val - 1], @element.name, element_wrapper)
end
def range(val)
zero_index_error if val.first.zero?
self.class.new(@siblings[(val.first - 1)..(val.last - 1)], :range, element_wrapper)
end
def zero_index_error
raise ArgumentError, 'Index for matched sets starts at 1, not 0.'
end
def where(tag, query, all: false)
matched = matched_from_query(tag, query)
return nil unless matched || all
return matched&.first unless all
matched
end
def matched_from_query(tag, query)
if query.is_a?(String)
@element&.css("#{tag}#{query}")
elsif query[:class]
where_class(tag, query[:class]) & where_xpath(tag, query.merge(class: nil))
else
where_xpath(tag, query)
end
end
def where_xpath(tag, query)
conditions = "[#{where_conditions(query)}]" unless query.compact.empty?
@element&.xpath("//#{tag}#{conditions}")
end
def where_conditions(query)
query.compact.map do |key, value|
next if value.nil?
next %(@#{key}="#{value}") unless key == :text
%[contains(text(),"#{value}")]
end.join ' and '
end
def where_class(tag, class_or_classes)
classes = class_or_classes.is_a?(Array) ? class_or_classes : class_or_classes.to_s.split
selector = classes.map(&:to_s).join('.')
@element&.css("#{tag}.#{selector}")
end
def find(tag, all: false)
return @element&.css(tag.to_s)&.first unless all
@element&.css(tag.to_s)
end
end
# rubocop:enable Metrics/ClassLength
end