AndyObtiva/glimmer-dsl-swt

View on GitHub
lib/glimmer/swt/custom/shape/line.rb

Summary

Maintainability
A
3 hrs
Test Coverage
# Copyright (c) 2007-2024 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 'glimmer/swt/custom/shape'
require 'glimmer/swt/custom/shape/path_segment'
require 'glimmer/swt/swt_proxy'
require 'glimmer/swt/display_proxy'
require 'glimmer/swt/color_proxy'
require 'glimmer/swt/font_proxy'
require 'glimmer/swt/transform_proxy'

module Glimmer
  module SWT
    module Custom
      # Represents a shape (graphics) to be drawn on a control/widget/canvas/display
      # That is because Shape is drawn on a parent as graphics and doesn't have an SWT widget for itself
      class Shape
        class Line < Shape
          include PathSegment
          
          class << self
            def include?(x1, y1, x2, y2, x, y)
              distance1 = Math.sqrt((x - x1)**2 + (y - y1)**2)
              distance2 = Math.sqrt((x2 - x)**2 + (y2 - y)**2)
              distance = Math.sqrt((x2 - x1)**2 + (y2 - y1)**2)
              (distance1 + distance2).to_i == distance.to_i
            end
          end
        
          def parameter_names
            [:x1, :y1, :x2, :y2]
          end
          
          def location_parameter_names
            parameter_names
          end
          
          def bounds
            shape_bounds = geometry.getBounds2D
            org.eclipse.swt.graphics.Rectangle.new(shape_bounds.x, shape_bounds.y, shape_bounds.width, shape_bounds.height)
          end
          
          def size
            shape_bounds = geometry.getBounds2D
            org.eclipse.swt.graphics.Point.new(shape_bounds.width, shape_bounds.height)
          end
          
          def geometry
            java.awt.geom.Line2D::Double.new(absolute_x1, absolute_y1, absolute_x2, absolute_y2)
          end
          
          # Logical x coordinate relative to parent
          def x
            x_value = bounds.x
            x_value -= parent.absolute_x if parent.is_a?(Shape) && parent.class != Shape
            x_value
          end
          
          # Logical y coordinate relative to parent
          def y
            y_value = bounds.y
            y_value -= parent.absolute_y if parent.is_a?(Shape) && parent.class != Shape
            y_value
          end
          
          def width
            size.x
          end
          
          def height
            size.y
          end
          
          def absolute_x1
            if parent.is_a?(Shape) && parent.class != Shape
              parent.absolute_x + x1
            else
              x1
            end
          end
          
          def absolute_y1
            if parent.is_a?(Shape) && parent.class != Shape
              parent.absolute_y + y1
            else
              y1
            end
          end
          
          def absolute_x2
            if parent.is_a?(Shape) && parent.class != Shape
              parent.absolute_x + x2.to_f
            else
              x2
            end
          end
          
          def absolute_y2
            if parent.is_a?(Shape) && parent.class != Shape
              parent.absolute_y + y2.to_f
            else
              y2
            end
          end
          
          def include?(x, y)
            x, y = inverse_transform_point(x, y)
            # TODO must account for line width
            Line.include?(absolute_x1, absolute_y1, absolute_x2, absolute_y2, x, y)
          end
          alias contain? include?
            
          def move_by(x_delta, y_delta)
            self.x1 += x_delta
            self.y1 += y_delta
            self.x2 += x_delta
            self.y2 += y_delta
          end
          
          def irregular?
            true
          end
                    
          def path_segment_method_name
            'lineTo'
          end
          
          def path_segment_args
            # TODO make args auto-infer first point if previous_point_connected is true or if there is only x1,y1 or x2,y2 (but not both), or if there is an x, y, or if there is a point_array with 1 point
            @args
          end
          
          def default_path_segment_arg_count
            4
          end
          
          def default_connected_path_segment_arg_count
            2
          end
          
          def path_segment_geometry_args
            # TODO make args auto-infer first point if previous_point_connected is true or if there is only x1,y1 or x2,y2 (but not both), or if there is an x, y, or if there is a point_array with 1 point
            @args
          end
          
          def previous_point_connected?
            @args.compact.count == 2 && !first_path_segment?
          end
          
          def eql?(other)
            other.is_a?(Line) &&
              x1 == (other && other.respond_to?(:x1) && other.x1) &&
              y1 == (other && other.respond_to?(:y1) && other.y1) &&
              x2 == (other && other.respond_to?(:x2) && other.x2) &&
              y2 == (other && other.respond_to?(:y2) && other.y2)
          end
          alias == eql?
          
          def hash
            [x1, y1, x2, y2].hash
          end
                    
        end
      end
    end
  end
end