# coding: utf-8 # typed: true # frozen_string_literal: true module PDF class Reader # PDFs represent rectangles all over the place. They're 4 element arrays, like this: # # [A, B, C, D] # # Four element arrays are yucky to work with though, so here's a class that's better. # Initialize it with the 4 elements, and get utility functions (width, height, etc) # for free. # # By convention the first two elements are x1, y1, the co-ords for the bottom left corner # of the rectangle. The third and fourth elements are x2, y2, the co-ords for the top left # corner of the rectangle. It's valid for the alternative corners to be used though, so # we don't assume which is which. # class Rectangle attr_reader :bottom_left, :bottom_right, :top_left, :top_right def initialize(x1, y1, x2, y2) set_corners(x1, y1, x2, y2) end def self.from_array(arr) if arr.size != 4 raise ArgumentError, "Only 4-element Arrays can be converted to a Rectangle" end PDF::Reader::Rectangle.new( arr[0].to_f, arr[1].to_f, arr[2].to_f, arr[3].to_f, ) end def ==(other) to_a == other.to_a end def height top_right.y - bottom_right.y end def width bottom_right.x - bottom_left.x end def contains?(point) point.x >= bottom_left.x && point.x <= top_right.x && point.y >= bottom_left.y && point.y <= top_right.y end # A pdf-style 4-number array def to_a [ bottom_left.x, bottom_left.y, top_right.x, top_right.y, ] end def apply_rotation(degrees) return if degrees != 90 && degrees != 180 && degrees != 270 if degrees == 90 new_x1 = bottom_left.x new_y1 = bottom_left.y - width new_x2 = bottom_left.x + height new_y2 = bottom_left.y elsif degrees == 180 new_x1 = bottom_left.x - width new_y1 = bottom_left.y - height new_x2 = bottom_left.x new_y2 = bottom_left.y elsif degrees == 270 new_x1 = bottom_left.x - height new_y1 = bottom_left.y new_x2 = bottom_left.x new_y2 = bottom_left.y + width end set_corners(new_x1, new_y1, new_x2, new_y2) end private def set_corners(x1, y1, x2, y2) @bottom_left = PDF::Reader::Point.new( [x1, x2].min, [y1, y2].min, ) @bottom_right = PDF::Reader::Point.new( [x1, x2].max, [y1, y2].min, ) @top_left = PDF::Reader::Point.new( [x1, x2].min, [y1, y2].max, ) @top_right = PDF::Reader::Point.new( [x1, x2].max, [y1, y2].max, ) end end end end