# -*- encoding: utf-8; frozen_string_literal: true -*- # #-- # This file is part of HexaPDF. # # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby # Copyright (C) 2014-2024 Thomas Leitner # # HexaPDF is free software: you can redistribute it and/or modify it # under the terms of the GNU Affero General Public License version 3 as # published by the Free Software Foundation with the addition of the # following permission added to Section 15 as permitted in Section 7(a): # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON # INFRINGEMENT OF THIRD PARTY RIGHTS. # # HexaPDF is distributed in the hope that it will be useful, but WITHOUT # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public # License for more details. # # You should have received a copy of the GNU Affero General Public License # along with HexaPDF. If not, see . # # The interactive user interfaces in modified source and object code # versions of HexaPDF must display Appropriate Legal Notices, as required # under Section 5 of the GNU Affero General Public License version 3. # # In accordance with Section 7(b) of the GNU Affero General Public # License, a covered work must retain the producer line in every PDF that # is created or manipulated using HexaPDF. # # If the GNU Affero General Public License doesn't fit your need, # commercial licenses are available at . #++ require 'hexapdf/type/font' require 'hexapdf/stream' require 'hexapdf/font/cmap' module HexaPDF module Type # Represents a composite PDF font. # # Composites fonts wrap a descendant CIDFont and use CIDs to identify glyphs. A CID can be # encoded in one or more bytes and an associated CMap specifies how this encoding is done. # Composite fonts also allow for vertical writing mode and support TrueType as well as OpenType # fonts. # # See: PDF2.0 s9.7 class FontType0 < Font define_field :Subtype, type: Symbol, required: true, default: :Type0 define_field :BaseFont, type: Symbol, required: true define_field :Encoding, type: [Stream, Symbol], required: true define_field :DescendantFonts, type: PDFArray, required: true # Returns the CID font of this type 0 font. def descendant_font cache(:descendant_font) do document.wrap(self[:DescendantFonts][0]) end end # Returns the font descriptor of the descendant font. def font_descriptor descendant_font[:FontDescriptor] end # Returns the writing mode which is either :horizontal or :vertical. def writing_mode cmap.wmode == 0 ? :horizontal : :vertical end # Decodes the given string into an array of CIDs. def decode(string) cmap.read_codes(string) end # Returns the UTF-8 string for the given code, or calls the configuration option # 'font.on_missing_unicode_mapping' if no mapping was found. def to_utf8(code) to_unicode_cmap&.to_unicode(code) || ucs2_cmap&.to_unicode(code) || missing_unicode_mapping(code) end # Returns the unscaled width of the given CID in glyph units, or 0 if the width for the code # point is missing. def width(code) descendant_font.width(cmap.to_cid(code)) end # Returns the bounding box of the font or +nil+ if it is not found. def bounding_box descendant_font.bounding_box end # Returns +true+ if the font is embedded. def embedded? descendant_font.embedded? end # Returns the embeeded font file object or +nil+ if the font is not embedded. def font_file descendant_font.font_file end # Returns whether word spacing is applicable when using this font. # # Note that the return value is cached when accessed the first time. # # See: PDF2.0 s9.3.3 def word_spacing_applicable? @word_spacing_applicable ||= (cmap.read_codes("\x20") && true rescue false) end private # Returns the CMap used for decoding strings for this font. # # Note that the CMap is cached internally when accessed the first time. def cmap cache(:cmap) do val = self[:Encoding] if val.kind_of?(Symbol) HexaPDF::Font::CMap.for_name(val.to_s) elsif val.kind_of?(HexaPDF::Stream) HexaPDF::Font::CMap.parse(val.stream) else raise HexaPDF::Error, "Unknown value for font's encoding: #{self[:Encoding]}" end end end # Returns the UCS-2 CMap used for extracting text when no ToUnicode CMap is available, or # +nil+ if the UCS-2 CMap could not be determined. # # Note that the CMap is cached internally when accessed the first time. # # See: PDF2.0 s9.10.2 def ucs2_cmap cache(:ucs2_cmap) do encoding = self[:Encoding] system_info = descendant_font[:CIDSystemInfo] registry = system_info[:Registry] ordering = system_info[:Ordering] if (encoding.kind_of?(Symbol) && HexaPDF::Font::CMap.predefined?(encoding.to_s) && encoding != :'Identity-H' && encoding != :'Identity-V') || (registry == "Adobe" && ['GB1', 'CNS1', 'Japan1', 'Korea1'].include?(ordering)) HexaPDF::Font::CMap.for_name("#{registry}-#{ordering}-UCS2") end end end end end end