# frozen_string_literal: true require_relative '../table' module TTFunk class Table class Hhea < Table attr_reader :version attr_reader :ascent attr_reader :descent attr_reader :line_gap attr_reader :advance_width_max attr_reader :min_left_side_bearing attr_reader :min_right_side_bearing attr_reader :x_max_extent attr_reader :carot_slope_rise attr_reader :carot_slope_run attr_reader :caret_offset attr_reader :metric_data_format attr_reader :number_of_metrics class << self def encode(hhea, hmtx, original, mapping) ''.b.tap do |table| table << [hhea.version].pack('N') table << [ hhea.ascent, hhea.descent, hhea.line_gap, *min_max_values_for(original, mapping), hhea.carot_slope_rise, hhea.carot_slope_run, hhea.caret_offset, 0, 0, 0, 0, hhea.metric_data_format, hmtx[:number_of_metrics] ].pack('n*') end end private def min_max_values_for(original, mapping) min_lsb = Min.new min_rsb = Min.new max_aw = Max.new max_extent = Max.new mapping.each do |_, old_glyph_id| horiz_metrics = original.horizontal_metrics.for(old_glyph_id) next unless horiz_metrics min_lsb << horiz_metrics.left_side_bearing max_aw << horiz_metrics.advance_width glyph = original.find_glyph(old_glyph_id) next unless glyph x_delta = glyph.x_max - glyph.x_min min_rsb << horiz_metrics.advance_width - horiz_metrics.left_side_bearing - x_delta max_extent << horiz_metrics.left_side_bearing + x_delta end [ max_aw.value_or(0), min_lsb.value_or(0), min_rsb.value_or(0), max_extent.value_or(0) ] end end private def parse! @version = read(4, 'N').first @ascent, @descent, @line_gap = read_signed(3) @advance_width_max = read(2, 'n').first @min_left_side_bearing, @min_right_side_bearing, @x_max_extent, @carot_slope_rise, @carot_slope_run, @caret_offset, _reserved, _reserved, _reserved, _reserved, @metric_data_format = read_signed(11) @number_of_metrics = read(2, 'n').first end end end end