# frozen_string_literal: true module Matchi class Change # *Change by at most* matcher. class ByAtMost # Initialize the matcher with an object and a block. # # @example # require "matchi/change/by_at_most" # # object = [] # # Matchi::Change::ByAtMost.new(1) { object.length } # # @param expected [#object_id] An expected delta. # @param state [Proc] A block of code to execute to get the # state of the object. def initialize(expected, &state) @expected = expected @state = state end # Boolean comparison on the expected change by comparing the value # before and after the code execution. # # @example # require "matchi/change/by_at_most" # # object = [] # # matcher = Matchi::Change::ByAtMost.new(1) { object.length } # matcher.matches? { object << "foo" } # => true # # @yieldreturn [#object_id] The block of code to execute. # # @return [Boolean] Comparison between the value before and after the # code execution. def matches?(*, **) value_before = @state.call yield value_after = @state.call @expected >= (value_after - value_before) end # A string containing a human-readable representation of the matcher. def inspect "#{self.class}(#{@expected.inspect})" end # Returns a string representing the matcher. def to_s "change by at most #{@expected.inspect}" end end end end