teamcapybara--capybara/lib/capybara/rspec/matchers.rb

297 lines
8.7 KiB
Ruby
Raw Normal View History

2016-03-08 00:52:19 +00:00
# frozen_string_literal: true
module Capybara
module RSpecMatchers
2013-02-16 09:02:32 +00:00
class Matcher
include ::RSpec::Matchers::Composable if defined?(::RSpec::Expectations::Version) && (Gem::Version.new(RSpec::Expectations::Version::STRING) >= Gem::Version.new('3.0'))
2013-02-16 09:02:32 +00:00
def wrap(actual)
if actual.respond_to?("has_selector?")
actual
else
Capybara.string(actual.to_s)
end
end
2016-11-18 18:56:47 +00:00
def matches?(actual)
yield(wrap(actual))
rescue Capybara::ExpectationNotMet => e
@failure_message = e.message
return false
end
def does_not_match?(actual)
yield(wrap(actual))
rescue Capybara::ExpectationNotMet => e
@failure_message_when_negated = e.message
return false
end
2013-02-16 09:02:32 +00:00
end
class HaveSelector < Matcher
attr_reader :failure_message, :failure_message_when_negated
def initialize(*args, &filter_block)
@args = args
@filter_block = filter_block
end
def matches?(actual)
2016-11-18 18:56:47 +00:00
super(actual){ |el| el.assert_selector(*@args) }
end
def does_not_match?(actual)
2016-11-18 18:56:47 +00:00
super(actual){ |el| el.assert_no_selector(*@args) }
2011-02-06 19:07:12 +00:00
end
def description
"have #{query.description}"
end
def query
@query ||= Capybara::Queries::SelectorQuery.new(*@args, &@filter_block)
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
end
2013-02-16 09:02:32 +00:00
class HaveText < Matcher
2013-03-03 23:04:23 +00:00
attr_reader :type, :content, :options
attr_reader :failure_message, :failure_message_when_negated
2013-03-03 23:04:23 +00:00
def initialize(*args)
@args = args.dup
# are set just for backwards compatability
@type = args.shift if args.first.is_a?(Symbol)
@content = args.shift
@options = (args.first.is_a?(Hash))? args.first : {}
end
def matches?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_text(*@args) }
end
def does_not_match?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_no_text(*@args) }
end
def description
2013-03-03 23:04:23 +00:00
"text #{format(content)}"
end
2013-03-03 23:04:23 +00:00
def format(content)
content = Capybara::Helpers.normalize_whitespace(content) unless content.is_a? Regexp
content.inspect
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
end
2013-02-16 09:02:32 +00:00
class HaveTitle < Matcher
attr_reader :title
attr_reader :failure_message, :failure_message_when_negated
def initialize(*args)
@args = args
# are set just for backwards compatability
@title = args.first
2013-02-16 09:02:32 +00:00
end
def matches?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_title(*@args) }
2013-02-16 09:02:32 +00:00
end
def does_not_match?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_no_title(*@args) }
2013-02-16 09:02:32 +00:00
end
def description
"have title #{title.inspect}"
2013-02-16 09:02:32 +00:00
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
2013-02-16 09:02:32 +00:00
end
2015-08-24 05:14:48 +00:00
class HaveCurrentPath < Matcher
attr_reader :current_path
attr_reader :failure_message, :failure_message_when_negated
def initialize(*args)
@args = args
# are set just for backwards compatability
@current_path = args.first
end
def matches?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_current_path(*@args) }
2015-08-24 05:14:48 +00:00
end
def does_not_match?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_no_current_path(*@args) }
2015-08-24 05:14:48 +00:00
end
def description
"have current path #{current_path.inspect}"
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
end
class BecomeClosed
def initialize(options)
@wait_time = Capybara::Queries::BaseQuery.wait(options)
end
def matches?(window)
@window = window
start_time = Capybara::Helpers.monotonic_time
2014-06-26 18:55:45 +00:00
while window.exists?
return false if (Capybara::Helpers.monotonic_time - start_time) > @wait_time
sleep 0.05
end
2014-06-26 18:55:45 +00:00
true
end
def failure_message
"expected #{@window.inspect} to become closed after #{@wait_time} seconds"
end
def failure_message_when_negated
"expected #{@window.inspect} not to become closed after #{@wait_time} seconds"
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
end
class MatchSelector < Matcher
attr_reader :failure_message, :failure_message_when_negated
def initialize(*args)
@args = args
end
def matches?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_matches_selector(*@args) }
end
def does_not_match?(actual)
2016-11-18 18:56:47 +00:00
super(actual) { |el| el.assert_not_matches_selector(*@args) }
end
def description
"match #{query.description}"
end
def query
@query ||= Capybara::Queries::MatchQuery.new(*@args)
end
# RSpec 2 compatibility:
alias_method :failure_message_for_should, :failure_message
alias_method :failure_message_for_should_not, :failure_message_when_negated
end
def have_selector(*args, &optional_filter_block)
HaveSelector.new(*args, &optional_filter_block)
end
def match_selector(*args)
MatchSelector.new(*args)
end
# defined_negated_matcher was added in RSpec 3.1 - it's syntactic sugar only since a user can do
# expect(page).not_to match_selector, so not sure we really need to support not_match_selector for prior to RSpec 3.1
::RSpec::Matchers.define_negated_matcher :not_match_selector, :match_selector if defined?(::RSpec::Expectations::Version) && (Gem::Version.new(RSpec::Expectations::Version::STRING) >= Gem::Version.new('3.1'))
def have_xpath(xpath, options={}, &optional_filter_block)
HaveSelector.new(:xpath, xpath, options, &optional_filter_block)
end
def match_xpath(xpath, options={})
MatchSelector.new(:xpath, xpath, options)
end
def have_css(css, options={}, &optional_filter_block)
HaveSelector.new(:css, css, options, &optional_filter_block)
end
2011-02-10 16:15:28 +00:00
def match_css(css, options={})
MatchSelector.new(:css, css, options)
end
2013-03-03 23:04:23 +00:00
def have_text(*args)
HaveText.new(*args)
end
2013-03-03 23:04:23 +00:00
alias_method :have_content, :have_text
2011-02-13 16:04:13 +00:00
def have_title(title, options = {})
HaveTitle.new(title, options)
2013-02-16 09:02:32 +00:00
end
2015-08-24 05:14:48 +00:00
def have_current_path(path, options = {})
HaveCurrentPath.new(path, options)
end
def have_link(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
HaveSelector.new(:link, locator, options, &optional_filter_block)
2011-02-13 16:04:13 +00:00
end
2011-03-25 09:54:52 +00:00
def have_button(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
HaveSelector.new(:button, locator, options, &optional_filter_block)
end
def have_field(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
HaveSelector.new(:field, locator, options, &optional_filter_block)
end
def have_checked_field(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
2016-10-04 18:10:29 +00:00
HaveSelector.new(:field, locator, options.merge(checked: true), &optional_filter_block)
end
def have_unchecked_field(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
2016-10-04 18:10:29 +00:00
HaveSelector.new(:field, locator, options.merge(unchecked: true), &optional_filter_block)
2011-03-25 09:54:52 +00:00
end
def have_select(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
HaveSelector.new(:select, locator, options, &optional_filter_block)
end
def have_table(locator=nil, options={}, &optional_filter_block)
locator, options = nil, locator if locator.is_a? Hash
HaveSelector.new(:table, locator, options, &optional_filter_block)
end
##
# Wait for window to become closed.
# @example
# expect(window).to become_closed(wait: 0.8)
# @param options [Hash] optional param
# @option options [Numeric] :wait (Capybara.default_max_wait_time) Maximum wait time
def become_closed(options = {})
BecomeClosed.new(options)
end
end
end