# Copyright (c) 2021-2022 Andy Maleh # # Permission is hereby granted, free of charge, to any person obtaining # a copy of this software and associated documentation files (the # "Software"), to deal in the Software without restriction, including # without limitation the rights to use, copy, modify, merge, publish, # distribute, sublicense, and/or sell copies of the Software, and to # permit persons to whom the Software is furnished to do so, subject to # the following conditions: # # The above copyright notice and this permission notice shall be # included in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. require 'perfect_shape/shape' require 'perfect_shape/point_location' module PerfectShape # Point class includes point-specific operations like `#==`, `point_distance` and a fuzzy `contain?` matcher class Point < Shape class << self def point_distance(x, y, px, py) x = BigDecimal(x.to_s) y = BigDecimal(y.to_s) px = BigDecimal(px.to_s) py = BigDecimal(py.to_s) BigDecimal(Math.sqrt((px - x)**2 + (py - y)**2).to_s) end end include PointLocation include Equalizer.new(:x, :y) def initialize(x_or_point = nil, y_arg = nil, x: nil, y: nil) if x_or_point.is_a?(Array) x, y = x_or_point super(x: x, y: y) elsif x_or_point && y_arg super(x: x_or_point, y: y_arg) else x ||= 0 y ||= 0 super(x: x, y: y) end end def max_x x end def max_y y end # Checks if points match, with distance tolerance (0 by default) # # @param x The X coordinate of the point to test. # @param y The Y coordinate of the point to test. # @param distance_tolerance The distance from point to tolerate (0 by default) # # @return {@code true} if the point is close enough within distance tolerance, # {@code false} if the point is too far. def contain?(x_or_point, y = nil, outline: false, distance_tolerance: 0) x, y = normalize_point(x_or_point, y) return unless x && y distance_tolerance = BigDecimal(distance_tolerance.to_s) point_distance(x, y) <= distance_tolerance end def point_distance(x_or_point, y = nil) x, y = normalize_point(x_or_point, y) return unless x && y Point.point_distance(self.x, self.y, x, y) end # Convert to pair Array of x,y coordinates def to_a [self.x, self.y] end end end