mirror of https://github.com/docusealco/docuseal
				
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							188 lines
						
					
					
						
							5.7 KiB
						
					
					
				
			
		
		
	
	
							188 lines
						
					
					
						
							5.7 KiB
						
					
					
				# frozen_string_literal: true
 | 
						|
 | 
						|
module LoadBmp
 | 
						|
  module_function
 | 
						|
 | 
						|
  # rubocop:disable Metrics
 | 
						|
  def call(bmp_bytes)
 | 
						|
    bmp_bytes = bmp_bytes.b
 | 
						|
 | 
						|
    header_data = parse_bmp_headers(bmp_bytes)
 | 
						|
 | 
						|
    raw_pixel_data_from_file = extract_raw_pixel_data_blob(
 | 
						|
      bmp_bytes,
 | 
						|
      header_data[:pixel_data_offset],
 | 
						|
      header_data[:bmp_stride],
 | 
						|
      header_data[:height]
 | 
						|
    )
 | 
						|
 | 
						|
    final_pixel_data = prepare_unpadded_pixel_data_string(
 | 
						|
      raw_pixel_data_from_file,
 | 
						|
      header_data[:bpp],
 | 
						|
      header_data[:width],
 | 
						|
      header_data[:height],
 | 
						|
      header_data[:bmp_stride]
 | 
						|
    )
 | 
						|
 | 
						|
    bands = header_data[:bpp] / 8
 | 
						|
 | 
						|
    unless header_data[:bpp] == 24 || header_data[:bpp] == 32
 | 
						|
      raise ArgumentError, "Conversion for #{header_data[:bpp]}-bpp BMP not implemented."
 | 
						|
    end
 | 
						|
 | 
						|
    image = Vips::Image.new_from_memory(final_pixel_data, header_data[:width], header_data[:height], bands, :uchar)
 | 
						|
 | 
						|
    image = image.flip(:vertical) if header_data[:orientation] == -1
 | 
						|
 | 
						|
    image_rgb =
 | 
						|
      if bands == 3
 | 
						|
        image.recomb(band3_recomb)
 | 
						|
      elsif bands == 4
 | 
						|
        image.recomb(band4_recomb)
 | 
						|
      end
 | 
						|
 | 
						|
    image_rgb = image_rgb.copy(interpretation: :srgb) if image_rgb.interpretation != :srgb
 | 
						|
 | 
						|
    image_rgb
 | 
						|
  end
 | 
						|
 | 
						|
  def parse_bmp_headers(bmp_bytes)
 | 
						|
    raise ArgumentError, 'BMP data too short for file header (14 bytes).' if bmp_bytes.bytesize < 14
 | 
						|
 | 
						|
    signature, pixel_data_offset = bmp_bytes.unpack('a2@10L<')
 | 
						|
 | 
						|
    raise ArgumentError, "Not a valid BMP file (invalid signature 'BM')." if signature != 'BM'
 | 
						|
 | 
						|
    raise ArgumentError, 'BMP data too short for info header size field (4 bytes).' if bmp_bytes.bytesize < (14 + 4)
 | 
						|
 | 
						|
    info_header_size = bmp_bytes.unpack1('@14L<')
 | 
						|
 | 
						|
    min_expected_info_header_size = 40
 | 
						|
 | 
						|
    if info_header_size < min_expected_info_header_size
 | 
						|
      raise ArgumentError,
 | 
						|
            "Unsupported BMP info header size: #{info_header_size}. Expected at least #{min_expected_info_header_size}."
 | 
						|
    end
 | 
						|
 | 
						|
    header_and_info_header_min_bytes = 14 + min_expected_info_header_size
 | 
						|
 | 
						|
    if bmp_bytes.bytesize < header_and_info_header_min_bytes
 | 
						|
      raise ArgumentError,
 | 
						|
            'BMP data too short for essential BITMAPINFOHEADER fields ' \
 | 
						|
            "(requires #{header_and_info_header_min_bytes} bytes total)."
 | 
						|
    end
 | 
						|
 | 
						|
    _header_size_check, width, raw_height_from_header, planes, bpp, compression =
 | 
						|
      bmp_bytes.unpack('@14L<l<l<S<S<L<')
 | 
						|
 | 
						|
    height = 0
 | 
						|
    orientation = -1
 | 
						|
 | 
						|
    if raw_height_from_header.negative?
 | 
						|
      height = -raw_height_from_header
 | 
						|
      orientation = 1
 | 
						|
    else
 | 
						|
      height = raw_height_from_header
 | 
						|
    end
 | 
						|
 | 
						|
    raise ArgumentError, 'BMP width must be positive.' if width <= 0
 | 
						|
    raise ArgumentError, 'BMP height must be positive.' if height <= 0
 | 
						|
 | 
						|
    if compression != 0
 | 
						|
      raise ArgumentError,
 | 
						|
            "Unsupported BMP compression type: #{compression}. Only uncompressed (0) is supported."
 | 
						|
    end
 | 
						|
 | 
						|
    unless [24, 32].include?(bpp)
 | 
						|
      raise ArgumentError, "Unsupported BMP bits per pixel: #{bpp}. Only 24-bit and 32-bit are supported."
 | 
						|
    end
 | 
						|
 | 
						|
    raise ArgumentError, "Unsupported BMP planes: #{planes}. Expected 1." if planes != 1
 | 
						|
 | 
						|
    bytes_per_pixel = bpp / 8
 | 
						|
    row_size_unpadded = width * bytes_per_pixel
 | 
						|
    bmp_stride = (row_size_unpadded + 3) & ~3
 | 
						|
 | 
						|
    {
 | 
						|
      width:,
 | 
						|
      height:,
 | 
						|
      bpp:,
 | 
						|
      pixel_data_offset:,
 | 
						|
      bmp_stride:,
 | 
						|
      orientation:
 | 
						|
    }
 | 
						|
  end
 | 
						|
 | 
						|
  def extract_raw_pixel_data_blob(bmp_bytes, pixel_data_offset, bmp_stride, height)
 | 
						|
    expected_pixel_data_size = bmp_stride * height
 | 
						|
 | 
						|
    if pixel_data_offset + expected_pixel_data_size > bmp_bytes.bytesize
 | 
						|
      actual_available = bmp_bytes.bytesize - pixel_data_offset
 | 
						|
      actual_available = 0 if actual_available.negative?
 | 
						|
      raise ArgumentError,
 | 
						|
            "Pixel data segment (offset #{pixel_data_offset}, expected size #{expected_pixel_data_size}) " \
 | 
						|
            "exceeds BMP file size (#{bmp_bytes.bytesize}). " \
 | 
						|
            "Only #{actual_available} bytes available after offset."
 | 
						|
    end
 | 
						|
 | 
						|
    raw_pixel_data_from_file = bmp_bytes.byteslice(pixel_data_offset, expected_pixel_data_size)
 | 
						|
 | 
						|
    if raw_pixel_data_from_file.nil? || raw_pixel_data_from_file.bytesize < expected_pixel_data_size
 | 
						|
      raise ArgumentError,
 | 
						|
            "Extracted pixel data is smaller (#{raw_pixel_data_from_file&.bytesize || 0} bytes) " \
 | 
						|
            "than expected (#{expected_pixel_data_size} bytes based on stride and height)."
 | 
						|
    end
 | 
						|
 | 
						|
    raw_pixel_data_from_file
 | 
						|
  end
 | 
						|
 | 
						|
  def prepare_unpadded_pixel_data_string(raw_pixel_data_from_file, bpp, width, height, bmp_stride)
 | 
						|
    bytes_per_pixel = bpp / 8
 | 
						|
    actual_row_width_bytes = width * bytes_per_pixel
 | 
						|
 | 
						|
    unpadded_rows = Array.new(height)
 | 
						|
    current_offset_in_blob = 0
 | 
						|
 | 
						|
    height.times do |i|
 | 
						|
      if current_offset_in_blob + actual_row_width_bytes > raw_pixel_data_from_file.bytesize
 | 
						|
        raise ArgumentError,
 | 
						|
              "Not enough data in pixel blob for row #{i}. Offset #{current_offset_in_blob}, " \
 | 
						|
              "row width #{actual_row_width_bytes}, blob size #{raw_pixel_data_from_file.bytesize}"
 | 
						|
      end
 | 
						|
 | 
						|
      unpadded_row_slice = raw_pixel_data_from_file.byteslice(current_offset_in_blob, actual_row_width_bytes)
 | 
						|
 | 
						|
      if unpadded_row_slice.nil? || unpadded_row_slice.bytesize < actual_row_width_bytes
 | 
						|
        raise ArgumentError, "Failed to slice a full unpadded row from pixel data blob for row #{i}."
 | 
						|
      end
 | 
						|
 | 
						|
      unpadded_rows[i] = unpadded_row_slice
 | 
						|
      current_offset_in_blob += bmp_stride
 | 
						|
    end
 | 
						|
 | 
						|
    unpadded_rows.join
 | 
						|
  end
 | 
						|
 | 
						|
  def band3_recomb
 | 
						|
    @band3_recomb ||=
 | 
						|
      Vips::Image.new_from_array(
 | 
						|
        [
 | 
						|
          [0, 0, 1],
 | 
						|
          [0, 1, 0],
 | 
						|
          [1, 0, 0]
 | 
						|
        ]
 | 
						|
      )
 | 
						|
  end
 | 
						|
 | 
						|
  def band4_recomb
 | 
						|
    @band4_recomb ||= Vips::Image.new_from_array(
 | 
						|
      [
 | 
						|
        [0, 0, 1, 0],
 | 
						|
        [0, 1, 0, 0],
 | 
						|
        [1, 0, 0, 0]
 | 
						|
      ]
 | 
						|
    )
 | 
						|
  end
 | 
						|
  # rubocop:enable Metrics
 | 
						|
end
 |