hexapdf 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/CONTRIBUTERS +3 -0
- data/LICENSE +26 -0
- data/README.md +88 -0
- data/Rakefile +121 -0
- data/VERSION +1 -0
- data/agpl-3.0.txt +661 -0
- data/bin/hexapdf +6 -0
- data/data/hexapdf/afm/Courier-Bold.afm +342 -0
- data/data/hexapdf/afm/Courier-BoldOblique.afm +342 -0
- data/data/hexapdf/afm/Courier-Oblique.afm +342 -0
- data/data/hexapdf/afm/Courier.afm +342 -0
- data/data/hexapdf/afm/Helvetica-Bold.afm +2827 -0
- data/data/hexapdf/afm/Helvetica-BoldOblique.afm +2827 -0
- data/data/hexapdf/afm/Helvetica-Oblique.afm +3051 -0
- data/data/hexapdf/afm/Helvetica.afm +3051 -0
- data/data/hexapdf/afm/MustRead.html +1 -0
- data/data/hexapdf/afm/Symbol.afm +213 -0
- data/data/hexapdf/afm/Times-Bold.afm +2588 -0
- data/data/hexapdf/afm/Times-BoldItalic.afm +2384 -0
- data/data/hexapdf/afm/Times-Italic.afm +2667 -0
- data/data/hexapdf/afm/Times-Roman.afm +2419 -0
- data/data/hexapdf/afm/ZapfDingbats.afm +225 -0
- data/data/hexapdf/encoding/glyphlist.txt +4305 -0
- data/data/hexapdf/encoding/zapfdingbats.txt +225 -0
- data/examples/arc.rb +50 -0
- data/examples/graphics.rb +274 -0
- data/examples/hello_world.rb +16 -0
- data/examples/machupicchu.jpg +0 -0
- data/examples/merging.rb +24 -0
- data/examples/optimizing.rb +20 -0
- data/examples/show_char_bboxes.rb +55 -0
- data/examples/standard_pdf_fonts.rb +72 -0
- data/examples/truetype.rb +45 -0
- data/lib/hexapdf/cli/extract.rb +128 -0
- data/lib/hexapdf/cli/info.rb +121 -0
- data/lib/hexapdf/cli/inspect.rb +157 -0
- data/lib/hexapdf/cli/modify.rb +218 -0
- data/lib/hexapdf/cli.rb +121 -0
- data/lib/hexapdf/configuration.rb +392 -0
- data/lib/hexapdf/content/canvas.rb +1974 -0
- data/lib/hexapdf/content/color_space.rb +364 -0
- data/lib/hexapdf/content/graphic_object/arc.rb +267 -0
- data/lib/hexapdf/content/graphic_object/endpoint_arc.rb +208 -0
- data/lib/hexapdf/content/graphic_object/solid_arc.rb +173 -0
- data/lib/hexapdf/content/graphic_object.rb +81 -0
- data/lib/hexapdf/content/graphics_state.rb +579 -0
- data/lib/hexapdf/content/operator.rb +1072 -0
- data/lib/hexapdf/content/parser.rb +204 -0
- data/lib/hexapdf/content/processor.rb +451 -0
- data/lib/hexapdf/content/transformation_matrix.rb +172 -0
- data/lib/hexapdf/content.rb +47 -0
- data/lib/hexapdf/data_dir.rb +51 -0
- data/lib/hexapdf/dictionary.rb +303 -0
- data/lib/hexapdf/dictionary_fields.rb +382 -0
- data/lib/hexapdf/document.rb +589 -0
- data/lib/hexapdf/document_utils.rb +209 -0
- data/lib/hexapdf/encryption/aes.rb +206 -0
- data/lib/hexapdf/encryption/arc4.rb +93 -0
- data/lib/hexapdf/encryption/fast_aes.rb +79 -0
- data/lib/hexapdf/encryption/fast_arc4.rb +67 -0
- data/lib/hexapdf/encryption/identity.rb +63 -0
- data/lib/hexapdf/encryption/ruby_aes.rb +447 -0
- data/lib/hexapdf/encryption/ruby_arc4.rb +96 -0
- data/lib/hexapdf/encryption/security_handler.rb +494 -0
- data/lib/hexapdf/encryption/standard_security_handler.rb +616 -0
- data/lib/hexapdf/encryption.rb +94 -0
- data/lib/hexapdf/error.rb +73 -0
- data/lib/hexapdf/filter/ascii85_decode.rb +160 -0
- data/lib/hexapdf/filter/ascii_hex_decode.rb +87 -0
- data/lib/hexapdf/filter/dct_decode.rb +57 -0
- data/lib/hexapdf/filter/encryption.rb +59 -0
- data/lib/hexapdf/filter/flate_decode.rb +93 -0
- data/lib/hexapdf/filter/jpx_decode.rb +56 -0
- data/lib/hexapdf/filter/lzw_decode.rb +191 -0
- data/lib/hexapdf/filter/predictor.rb +266 -0
- data/lib/hexapdf/filter/run_length_decode.rb +108 -0
- data/lib/hexapdf/filter.rb +176 -0
- data/lib/hexapdf/font/cmap/parser.rb +146 -0
- data/lib/hexapdf/font/cmap/writer.rb +176 -0
- data/lib/hexapdf/font/cmap.rb +90 -0
- data/lib/hexapdf/font/encoding/base.rb +77 -0
- data/lib/hexapdf/font/encoding/difference_encoding.rb +64 -0
- data/lib/hexapdf/font/encoding/glyph_list.rb +150 -0
- data/lib/hexapdf/font/encoding/mac_expert_encoding.rb +221 -0
- data/lib/hexapdf/font/encoding/mac_roman_encoding.rb +265 -0
- data/lib/hexapdf/font/encoding/standard_encoding.rb +205 -0
- data/lib/hexapdf/font/encoding/symbol_encoding.rb +244 -0
- data/lib/hexapdf/font/encoding/win_ansi_encoding.rb +280 -0
- data/lib/hexapdf/font/encoding/zapf_dingbats_encoding.rb +250 -0
- data/lib/hexapdf/font/encoding.rb +68 -0
- data/lib/hexapdf/font/true_type/font.rb +179 -0
- data/lib/hexapdf/font/true_type/table/cmap.rb +103 -0
- data/lib/hexapdf/font/true_type/table/cmap_subtable.rb +384 -0
- data/lib/hexapdf/font/true_type/table/directory.rb +92 -0
- data/lib/hexapdf/font/true_type/table/glyf.rb +166 -0
- data/lib/hexapdf/font/true_type/table/head.rb +143 -0
- data/lib/hexapdf/font/true_type/table/hhea.rb +109 -0
- data/lib/hexapdf/font/true_type/table/hmtx.rb +79 -0
- data/lib/hexapdf/font/true_type/table/loca.rb +79 -0
- data/lib/hexapdf/font/true_type/table/maxp.rb +112 -0
- data/lib/hexapdf/font/true_type/table/name.rb +218 -0
- data/lib/hexapdf/font/true_type/table/os2.rb +200 -0
- data/lib/hexapdf/font/true_type/table/post.rb +230 -0
- data/lib/hexapdf/font/true_type/table.rb +155 -0
- data/lib/hexapdf/font/true_type.rb +48 -0
- data/lib/hexapdf/font/true_type_wrapper.rb +240 -0
- data/lib/hexapdf/font/type1/afm_parser.rb +230 -0
- data/lib/hexapdf/font/type1/character_metrics.rb +67 -0
- data/lib/hexapdf/font/type1/font.rb +123 -0
- data/lib/hexapdf/font/type1/font_metrics.rb +117 -0
- data/lib/hexapdf/font/type1/pfb_parser.rb +71 -0
- data/lib/hexapdf/font/type1.rb +52 -0
- data/lib/hexapdf/font/type1_wrapper.rb +193 -0
- data/lib/hexapdf/font_loader/from_configuration.rb +70 -0
- data/lib/hexapdf/font_loader/standard14.rb +98 -0
- data/lib/hexapdf/font_loader.rb +85 -0
- data/lib/hexapdf/font_utils.rb +89 -0
- data/lib/hexapdf/image_loader/jpeg.rb +166 -0
- data/lib/hexapdf/image_loader/pdf.rb +89 -0
- data/lib/hexapdf/image_loader/png.rb +410 -0
- data/lib/hexapdf/image_loader.rb +68 -0
- data/lib/hexapdf/importer.rb +139 -0
- data/lib/hexapdf/name_tree_node.rb +78 -0
- data/lib/hexapdf/number_tree_node.rb +67 -0
- data/lib/hexapdf/object.rb +363 -0
- data/lib/hexapdf/parser.rb +349 -0
- data/lib/hexapdf/rectangle.rb +99 -0
- data/lib/hexapdf/reference.rb +98 -0
- data/lib/hexapdf/revision.rb +206 -0
- data/lib/hexapdf/revisions.rb +194 -0
- data/lib/hexapdf/serializer.rb +326 -0
- data/lib/hexapdf/stream.rb +279 -0
- data/lib/hexapdf/task/dereference.rb +109 -0
- data/lib/hexapdf/task/optimize.rb +230 -0
- data/lib/hexapdf/task.rb +68 -0
- data/lib/hexapdf/tokenizer.rb +406 -0
- data/lib/hexapdf/type/catalog.rb +107 -0
- data/lib/hexapdf/type/embedded_file.rb +87 -0
- data/lib/hexapdf/type/file_specification.rb +232 -0
- data/lib/hexapdf/type/font.rb +81 -0
- data/lib/hexapdf/type/font_descriptor.rb +109 -0
- data/lib/hexapdf/type/font_simple.rb +190 -0
- data/lib/hexapdf/type/font_true_type.rb +47 -0
- data/lib/hexapdf/type/font_type1.rb +162 -0
- data/lib/hexapdf/type/form.rb +103 -0
- data/lib/hexapdf/type/graphics_state_parameter.rb +79 -0
- data/lib/hexapdf/type/image.rb +73 -0
- data/lib/hexapdf/type/info.rb +70 -0
- data/lib/hexapdf/type/names.rb +69 -0
- data/lib/hexapdf/type/object_stream.rb +224 -0
- data/lib/hexapdf/type/page.rb +355 -0
- data/lib/hexapdf/type/page_tree_node.rb +269 -0
- data/lib/hexapdf/type/resources.rb +212 -0
- data/lib/hexapdf/type/trailer.rb +128 -0
- data/lib/hexapdf/type/viewer_preferences.rb +73 -0
- data/lib/hexapdf/type/xref_stream.rb +204 -0
- data/lib/hexapdf/type.rb +67 -0
- data/lib/hexapdf/utils/bit_field.rb +87 -0
- data/lib/hexapdf/utils/bit_stream.rb +148 -0
- data/lib/hexapdf/utils/lru_cache.rb +65 -0
- data/lib/hexapdf/utils/math_helpers.rb +55 -0
- data/lib/hexapdf/utils/object_hash.rb +130 -0
- data/lib/hexapdf/utils/pdf_doc_encoding.rb +93 -0
- data/lib/hexapdf/utils/sorted_tree_node.rb +339 -0
- data/lib/hexapdf/version.rb +39 -0
- data/lib/hexapdf/writer.rb +199 -0
- data/lib/hexapdf/xref_section.rb +152 -0
- data/lib/hexapdf.rb +34 -0
- data/man/man1/hexapdf.1 +249 -0
- data/test/data/aes-test-vectors/CBCGFSbox-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-256-encrypt.data.gz +0 -0
- data/test/data/fonts/Ubuntu-Title.ttf +0 -0
- data/test/data/images/cmyk.jpg +0 -0
- data/test/data/images/fillbytes.jpg +0 -0
- data/test/data/images/gray.jpg +0 -0
- data/test/data/images/greyscale-1bit.png +0 -0
- data/test/data/images/greyscale-2bit.png +0 -0
- data/test/data/images/greyscale-4bit.png +0 -0
- data/test/data/images/greyscale-8bit.png +0 -0
- data/test/data/images/greyscale-alpha-8bit.png +0 -0
- data/test/data/images/greyscale-trns-8bit.png +0 -0
- data/test/data/images/greyscale-with-gamma1.0.png +0 -0
- data/test/data/images/greyscale-with-gamma1.5.png +0 -0
- data/test/data/images/indexed-1bit.png +0 -0
- data/test/data/images/indexed-2bit.png +0 -0
- data/test/data/images/indexed-4bit.png +0 -0
- data/test/data/images/indexed-8bit.png +0 -0
- data/test/data/images/indexed-alpha-4bit.png +0 -0
- data/test/data/images/indexed-alpha-8bit.png +0 -0
- data/test/data/images/rgb.jpg +0 -0
- data/test/data/images/truecolour-8bit.png +0 -0
- data/test/data/images/truecolour-alpha-8bit.png +0 -0
- data/test/data/images/truecolour-gama-chrm-8bit.png +0 -0
- data/test/data/images/truecolour-srgb-8bit.png +0 -0
- data/test/data/minimal.pdf +44 -0
- data/test/data/standard-security-handler/README +9 -0
- data/test/data/standard-security-handler/bothpwd-aes-128bit-V4.pdf +44 -0
- data/test/data/standard-security-handler/bothpwd-aes-256bit-V5.pdf +0 -0
- data/test/data/standard-security-handler/bothpwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/bothpwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/bothpwd-arc4-40bit-V1.pdf +0 -0
- data/test/data/standard-security-handler/nopwd-aes-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-aes-256bit-V5.pdf +0 -0
- data/test/data/standard-security-handler/nopwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-arc4-40bit-V1.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-aes-128bit-V4.pdf +0 -0
- data/test/data/standard-security-handler/ownerpwd-aes-256bit-V5.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-40bit-V1.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-aes-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-aes-256bit-V5.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-arc4-128bit-V2.pdf +0 -0
- data/test/data/standard-security-handler/userpwd-arc4-128bit-V4.pdf +0 -0
- data/test/data/standard-security-handler/userpwd-arc4-40bit-V1.pdf +43 -0
- data/test/hexapdf/common_tokenizer_tests.rb +204 -0
- data/test/hexapdf/content/common.rb +31 -0
- data/test/hexapdf/content/graphic_object/test_arc.rb +93 -0
- data/test/hexapdf/content/graphic_object/test_endpoint_arc.rb +91 -0
- data/test/hexapdf/content/graphic_object/test_solid_arc.rb +86 -0
- data/test/hexapdf/content/test_canvas.rb +1113 -0
- data/test/hexapdf/content/test_color_space.rb +97 -0
- data/test/hexapdf/content/test_graphics_state.rb +138 -0
- data/test/hexapdf/content/test_operator.rb +619 -0
- data/test/hexapdf/content/test_parser.rb +66 -0
- data/test/hexapdf/content/test_processor.rb +156 -0
- data/test/hexapdf/content/test_transformation_matrix.rb +64 -0
- data/test/hexapdf/encryption/common.rb +87 -0
- data/test/hexapdf/encryption/test_aes.rb +121 -0
- data/test/hexapdf/encryption/test_arc4.rb +39 -0
- data/test/hexapdf/encryption/test_fast_aes.rb +17 -0
- data/test/hexapdf/encryption/test_fast_arc4.rb +12 -0
- data/test/hexapdf/encryption/test_identity.rb +21 -0
- data/test/hexapdf/encryption/test_ruby_aes.rb +23 -0
- data/test/hexapdf/encryption/test_ruby_arc4.rb +20 -0
- data/test/hexapdf/encryption/test_security_handler.rb +356 -0
- data/test/hexapdf/encryption/test_standard_security_handler.rb +274 -0
- data/test/hexapdf/filter/common.rb +53 -0
- data/test/hexapdf/filter/test_ascii85_decode.rb +60 -0
- data/test/hexapdf/filter/test_ascii_hex_decode.rb +33 -0
- data/test/hexapdf/filter/test_encryption.rb +24 -0
- data/test/hexapdf/filter/test_flate_decode.rb +35 -0
- data/test/hexapdf/filter/test_lzw_decode.rb +52 -0
- data/test/hexapdf/filter/test_predictor.rb +183 -0
- data/test/hexapdf/filter/test_run_length_decode.rb +32 -0
- data/test/hexapdf/font/cmap/test_parser.rb +67 -0
- data/test/hexapdf/font/cmap/test_writer.rb +58 -0
- data/test/hexapdf/font/encoding/test_base.rb +35 -0
- data/test/hexapdf/font/encoding/test_difference_encoding.rb +21 -0
- data/test/hexapdf/font/encoding/test_glyph_list.rb +59 -0
- data/test/hexapdf/font/encoding/test_zapf_dingbats_encoding.rb +16 -0
- data/test/hexapdf/font/test_encoding.rb +27 -0
- data/test/hexapdf/font/test_true_type_wrapper.rb +110 -0
- data/test/hexapdf/font/test_type1_wrapper.rb +66 -0
- data/test/hexapdf/font/true_type/common.rb +19 -0
- data/test/hexapdf/font/true_type/table/test_cmap.rb +59 -0
- data/test/hexapdf/font/true_type/table/test_cmap_subtable.rb +133 -0
- data/test/hexapdf/font/true_type/table/test_directory.rb +35 -0
- data/test/hexapdf/font/true_type/table/test_glyf.rb +58 -0
- data/test/hexapdf/font/true_type/table/test_head.rb +76 -0
- data/test/hexapdf/font/true_type/table/test_hhea.rb +40 -0
- data/test/hexapdf/font/true_type/table/test_hmtx.rb +38 -0
- data/test/hexapdf/font/true_type/table/test_loca.rb +43 -0
- data/test/hexapdf/font/true_type/table/test_maxp.rb +62 -0
- data/test/hexapdf/font/true_type/table/test_name.rb +95 -0
- data/test/hexapdf/font/true_type/table/test_os2.rb +65 -0
- data/test/hexapdf/font/true_type/table/test_post.rb +89 -0
- data/test/hexapdf/font/true_type/test_font.rb +120 -0
- data/test/hexapdf/font/true_type/test_table.rb +41 -0
- data/test/hexapdf/font/type1/test_afm_parser.rb +51 -0
- data/test/hexapdf/font/type1/test_font.rb +68 -0
- data/test/hexapdf/font/type1/test_pfb_parser.rb +37 -0
- data/test/hexapdf/font_loader/test_from_configuration.rb +28 -0
- data/test/hexapdf/font_loader/test_standard14.rb +22 -0
- data/test/hexapdf/image_loader/test_jpeg.rb +83 -0
- data/test/hexapdf/image_loader/test_pdf.rb +47 -0
- data/test/hexapdf/image_loader/test_png.rb +258 -0
- data/test/hexapdf/task/test_dereference.rb +46 -0
- data/test/hexapdf/task/test_optimize.rb +137 -0
- data/test/hexapdf/test_configuration.rb +82 -0
- data/test/hexapdf/test_data_dir.rb +32 -0
- data/test/hexapdf/test_dictionary.rb +284 -0
- data/test/hexapdf/test_dictionary_fields.rb +185 -0
- data/test/hexapdf/test_document.rb +574 -0
- data/test/hexapdf/test_document_utils.rb +144 -0
- data/test/hexapdf/test_filter.rb +96 -0
- data/test/hexapdf/test_font_utils.rb +47 -0
- data/test/hexapdf/test_importer.rb +78 -0
- data/test/hexapdf/test_object.rb +177 -0
- data/test/hexapdf/test_parser.rb +394 -0
- data/test/hexapdf/test_rectangle.rb +36 -0
- data/test/hexapdf/test_reference.rb +41 -0
- data/test/hexapdf/test_revision.rb +139 -0
- data/test/hexapdf/test_revisions.rb +93 -0
- data/test/hexapdf/test_serializer.rb +169 -0
- data/test/hexapdf/test_stream.rb +262 -0
- data/test/hexapdf/test_tokenizer.rb +30 -0
- data/test/hexapdf/test_writer.rb +120 -0
- data/test/hexapdf/test_xref_section.rb +35 -0
- data/test/hexapdf/type/test_catalog.rb +30 -0
- data/test/hexapdf/type/test_embedded_file.rb +16 -0
- data/test/hexapdf/type/test_file_specification.rb +148 -0
- data/test/hexapdf/type/test_font.rb +35 -0
- data/test/hexapdf/type/test_font_descriptor.rb +51 -0
- data/test/hexapdf/type/test_font_simple.rb +190 -0
- data/test/hexapdf/type/test_font_type1.rb +128 -0
- data/test/hexapdf/type/test_form.rb +60 -0
- data/test/hexapdf/type/test_info.rb +14 -0
- data/test/hexapdf/type/test_names.rb +9 -0
- data/test/hexapdf/type/test_object_stream.rb +84 -0
- data/test/hexapdf/type/test_page.rb +260 -0
- data/test/hexapdf/type/test_page_tree_node.rb +255 -0
- data/test/hexapdf/type/test_resources.rb +167 -0
- data/test/hexapdf/type/test_trailer.rb +109 -0
- data/test/hexapdf/type/test_xref_stream.rb +131 -0
- data/test/hexapdf/utils/test_bit_field.rb +47 -0
- data/test/hexapdf/utils/test_lru_cache.rb +22 -0
- data/test/hexapdf/utils/test_object_hash.rb +115 -0
- data/test/hexapdf/utils/test_pdf_doc_encoding.rb +18 -0
- data/test/hexapdf/utils/test_sorted_tree_node.rb +232 -0
- data/test/test_helper.rb +56 -0
- metadata +427 -0
@@ -0,0 +1,179 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2016 Thomas Leitner
|
8
|
+
#
|
9
|
+
# HexaPDF is free software: you can redistribute it and/or modify it
|
10
|
+
# under the terms of the GNU Affero General Public License version 3 as
|
11
|
+
# published by the Free Software Foundation with the addition of the
|
12
|
+
# following permission added to Section 15 as permitted in Section 7(a):
|
13
|
+
# FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
|
14
|
+
# THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
|
15
|
+
# INFRINGEMENT OF THIRD PARTY RIGHTS.
|
16
|
+
#
|
17
|
+
# HexaPDF is distributed in the hope that it will be useful, but WITHOUT
|
18
|
+
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
20
|
+
# License for more details.
|
21
|
+
#
|
22
|
+
# You should have received a copy of the GNU Affero General Public License
|
23
|
+
# along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
|
24
|
+
#
|
25
|
+
# The interactive user interfaces in modified source and object code
|
26
|
+
# versions of HexaPDF must display Appropriate Legal Notices, as required
|
27
|
+
# under Section 5 of the GNU Affero General Public License version 3.
|
28
|
+
#
|
29
|
+
# In accordance with Section 7(b) of the GNU Affero General Public
|
30
|
+
# License, a covered work must retain the producer line in every PDF that
|
31
|
+
# is created or manipulated using HexaPDF.
|
32
|
+
#++
|
33
|
+
|
34
|
+
require 'hexapdf/font/true_type/table'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Font
|
38
|
+
module TrueType
|
39
|
+
|
40
|
+
# Represents a font in the TrueType font file format.
|
41
|
+
class Font
|
42
|
+
|
43
|
+
# The default configuration:
|
44
|
+
#
|
45
|
+
# font.ttf.table_mapping::
|
46
|
+
# The default mapping from table tag as symbol to table class name.
|
47
|
+
#
|
48
|
+
# font.ttf.cmap.unknown_format::
|
49
|
+
# Action to take when encountering unknown 'cmap' subtables. Can either be :ignore
|
50
|
+
# which ignores them or :raise which raises an error.
|
51
|
+
DEFAULT_CONFIG = {
|
52
|
+
'font.true_type.table_mapping' => {
|
53
|
+
head: 'HexaPDF::Font::TrueType::Table::Head',
|
54
|
+
cmap: 'HexaPDF::Font::TrueType::Table::Cmap',
|
55
|
+
hhea: 'HexaPDF::Font::TrueType::Table::Hhea',
|
56
|
+
hmtx: 'HexaPDF::Font::TrueType::Table::Hmtx',
|
57
|
+
loca: 'HexaPDF::Font::TrueType::Table::Loca',
|
58
|
+
maxp: 'HexaPDF::Font::TrueType::Table::Maxp',
|
59
|
+
name: 'HexaPDF::Font::TrueType::Table::Name',
|
60
|
+
post: 'HexaPDF::Font::TrueType::Table::Post',
|
61
|
+
glyf: 'HexaPDF::Font::TrueType::Table::Glyf',
|
62
|
+
'OS/2': 'HexaPDF::Font::TrueType::Table::OS2',
|
63
|
+
},
|
64
|
+
'font.true_type.cmap.unknown_format' => :ignore,
|
65
|
+
}
|
66
|
+
|
67
|
+
|
68
|
+
# The IO stream associated with this file. If this is +nil+ then the TrueType font wasn't
|
69
|
+
# originally read from an IO stream.
|
70
|
+
attr_reader :io
|
71
|
+
|
72
|
+
# The configuration for the TrueType font.
|
73
|
+
attr_reader :config
|
74
|
+
|
75
|
+
# Creates a new TrueType font file object. If an IO object is given, the TrueType font data
|
76
|
+
# is read from it.
|
77
|
+
#
|
78
|
+
# The +config+ hash can contain configuration options.
|
79
|
+
def initialize(io: nil, config: {})
|
80
|
+
@io = io
|
81
|
+
@config = DEFAULT_CONFIG.merge(config)
|
82
|
+
@tables = {}
|
83
|
+
end
|
84
|
+
|
85
|
+
# Returns the table instance for the given tag (a symbol), or +nil+ if no such table exists.
|
86
|
+
def [](tag)
|
87
|
+
return @tables[tag] if @tables.key?(tag)
|
88
|
+
|
89
|
+
entry = directory.entry(tag.to_s.b)
|
90
|
+
entry ? @tables[tag] = table_class(tag).new(self, entry) : nil
|
91
|
+
end
|
92
|
+
|
93
|
+
# Adds a new table instance for the given tag (a symbol) to the font if such a table
|
94
|
+
# instance doesn't already exist. Returns the table instance for the tag.
|
95
|
+
def add_table(tag)
|
96
|
+
@tables[tag] ||= table_class(tag).new(self)
|
97
|
+
end
|
98
|
+
|
99
|
+
# Returns the font directory.
|
100
|
+
def directory
|
101
|
+
@directory ||= Table::Directory.new(self, io ? Table::Directory::SELF_ENTRY : nil)
|
102
|
+
end
|
103
|
+
|
104
|
+
# Returns the PostScript font name.
|
105
|
+
def font_name
|
106
|
+
self[:name][:postscript_name].preferred_record
|
107
|
+
end
|
108
|
+
|
109
|
+
# Returns the full name of the font.
|
110
|
+
def full_name
|
111
|
+
self[:name][:font_name].preferred_record
|
112
|
+
end
|
113
|
+
|
114
|
+
# Returns the family name of the font.
|
115
|
+
def family_name
|
116
|
+
self[:name][:font_family].preferred_record
|
117
|
+
end
|
118
|
+
|
119
|
+
# Returns the weight of the font.
|
120
|
+
def weight
|
121
|
+
self[:"OS/2"]&.weight_class || 0
|
122
|
+
end
|
123
|
+
|
124
|
+
# Returns the bounding of the font.
|
125
|
+
def bounding_box
|
126
|
+
self[:head].bbox
|
127
|
+
end
|
128
|
+
|
129
|
+
# Returns the cap height of the font.
|
130
|
+
def cap_height
|
131
|
+
self[:"OS/2"]&.cap_height
|
132
|
+
end
|
133
|
+
|
134
|
+
# Returns the x-height of the font.
|
135
|
+
def x_height
|
136
|
+
self[:"OS/2"]&.x_height
|
137
|
+
end
|
138
|
+
|
139
|
+
# Returns the ascender of the font.
|
140
|
+
def ascender
|
141
|
+
self[:"OS/2"]&.typo_ascender || self[:hhea].ascent
|
142
|
+
end
|
143
|
+
|
144
|
+
# Returns the descender of the font.
|
145
|
+
def descender
|
146
|
+
self[:"OS/2"]&.typo_descender || self[:hhea].descent
|
147
|
+
end
|
148
|
+
|
149
|
+
# Returns the italic angle of the font, in degrees counter-clockwise from the vertical.
|
150
|
+
def italic_angle
|
151
|
+
self[:post].italic_angle.to_f
|
152
|
+
end
|
153
|
+
|
154
|
+
# Returns the dominant width of vertical stems.
|
155
|
+
#
|
156
|
+
# Note: This attribute does not actually exist in TrueType fonts, so it is estimated based
|
157
|
+
# on the #weight.
|
158
|
+
def dominant_vertical_stem_width
|
159
|
+
weight / 5
|
160
|
+
end
|
161
|
+
|
162
|
+
# Returns th glyph ID of the missing glyph, i.e. 0.
|
163
|
+
def missing_glyph_id
|
164
|
+
0
|
165
|
+
end
|
166
|
+
|
167
|
+
private
|
168
|
+
|
169
|
+
# Returns the class that is used for handling tables of the given tag.
|
170
|
+
def table_class(tag)
|
171
|
+
k = config['font.true_type.table_mapping'].fetch(tag, 'HexaPDF::Font::TrueType::Table')
|
172
|
+
::Object.const_get(k)
|
173
|
+
end
|
174
|
+
|
175
|
+
end
|
176
|
+
|
177
|
+
end
|
178
|
+
end
|
179
|
+
end
|
@@ -0,0 +1,103 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2016 Thomas Leitner
|
8
|
+
#
|
9
|
+
# HexaPDF is free software: you can redistribute it and/or modify it
|
10
|
+
# under the terms of the GNU Affero General Public License version 3 as
|
11
|
+
# published by the Free Software Foundation with the addition of the
|
12
|
+
# following permission added to Section 15 as permitted in Section 7(a):
|
13
|
+
# FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
|
14
|
+
# THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
|
15
|
+
# INFRINGEMENT OF THIRD PARTY RIGHTS.
|
16
|
+
#
|
17
|
+
# HexaPDF is distributed in the hope that it will be useful, but WITHOUT
|
18
|
+
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
20
|
+
# License for more details.
|
21
|
+
#
|
22
|
+
# You should have received a copy of the GNU Affero General Public License
|
23
|
+
# along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
|
24
|
+
#
|
25
|
+
# The interactive user interfaces in modified source and object code
|
26
|
+
# versions of HexaPDF must display Appropriate Legal Notices, as required
|
27
|
+
# under Section 5 of the GNU Affero General Public License version 3.
|
28
|
+
#
|
29
|
+
# In accordance with Section 7(b) of the GNU Affero General Public
|
30
|
+
# License, a covered work must retain the producer line in every PDF that
|
31
|
+
# is created or manipulated using HexaPDF.
|
32
|
+
#++
|
33
|
+
|
34
|
+
require 'hexapdf/font/true_type/table'
|
35
|
+
require 'hexapdf/font/true_type/table/cmap_subtable'
|
36
|
+
|
37
|
+
module HexaPDF
|
38
|
+
module Font
|
39
|
+
module TrueType
|
40
|
+
class Table
|
41
|
+
|
42
|
+
# The 'cmap' table contains subtables for mapping character codes to glyph indices.
|
43
|
+
#
|
44
|
+
# See:
|
45
|
+
# * CmapSubtable
|
46
|
+
# * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6cmap.html
|
47
|
+
class Cmap < Table
|
48
|
+
|
49
|
+
# The version of the cmap table.
|
50
|
+
attr_accessor :version
|
51
|
+
|
52
|
+
# The available cmap subtables.
|
53
|
+
attr_accessor :tables
|
54
|
+
|
55
|
+
# Returns the preferred of the available cmap subtables.
|
56
|
+
#
|
57
|
+
# A preferred table is always a table mapping Unicode characters.
|
58
|
+
def preferred_table
|
59
|
+
tables.select(&:unicode?).sort {|a, b| a.format <=> b.format}.last
|
60
|
+
end
|
61
|
+
|
62
|
+
private
|
63
|
+
|
64
|
+
def parse_table #:nodoc:
|
65
|
+
@version, num_tables = read_formatted(4, 'n2')
|
66
|
+
@tables = []
|
67
|
+
handle_unknown = font.config['font.true_type.cmap.unknown_format']
|
68
|
+
|
69
|
+
num_tables.times { @tables << read_formatted(8, 'n2N') }
|
70
|
+
offset_map = {}
|
71
|
+
@tables.map! do |platform_id, encoding_id, offset|
|
72
|
+
offset += directory_entry.offset
|
73
|
+
if offset_map.key?(offset)
|
74
|
+
subtable = offset_map[offset].dup
|
75
|
+
subtable.platform_id = platform_id
|
76
|
+
subtable.encoding_id = encoding_id
|
77
|
+
next subtable
|
78
|
+
end
|
79
|
+
|
80
|
+
subtable = CmapSubtable.new(platform_id, encoding_id)
|
81
|
+
supported = subtable.parse(io, offset)
|
82
|
+
if supported
|
83
|
+
offset_map[offset] = subtable
|
84
|
+
subtable
|
85
|
+
elsif handle_unknown == :raise
|
86
|
+
raise HexaPDF::Error, "Unknown cmap subtable format #{subtable.format}"
|
87
|
+
else
|
88
|
+
nil
|
89
|
+
end
|
90
|
+
end.compact!
|
91
|
+
end
|
92
|
+
|
93
|
+
def load_default #:nodoc:
|
94
|
+
@version = 0
|
95
|
+
@tables = []
|
96
|
+
end
|
97
|
+
|
98
|
+
end
|
99
|
+
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
@@ -0,0 +1,384 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2016 Thomas Leitner
|
8
|
+
#
|
9
|
+
# HexaPDF is free software: you can redistribute it and/or modify it
|
10
|
+
# under the terms of the GNU Affero General Public License version 3 as
|
11
|
+
# published by the Free Software Foundation with the addition of the
|
12
|
+
# following permission added to Section 15 as permitted in Section 7(a):
|
13
|
+
# FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
|
14
|
+
# THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
|
15
|
+
# INFRINGEMENT OF THIRD PARTY RIGHTS.
|
16
|
+
#
|
17
|
+
# HexaPDF is distributed in the hope that it will be useful, but WITHOUT
|
18
|
+
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
20
|
+
# License for more details.
|
21
|
+
#
|
22
|
+
# You should have received a copy of the GNU Affero General Public License
|
23
|
+
# along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
|
24
|
+
#
|
25
|
+
# The interactive user interfaces in modified source and object code
|
26
|
+
# versions of HexaPDF must display Appropriate Legal Notices, as required
|
27
|
+
# under Section 5 of the GNU Affero General Public License version 3.
|
28
|
+
#
|
29
|
+
# In accordance with Section 7(b) of the GNU Affero General Public
|
30
|
+
# License, a covered work must retain the producer line in every PDF that
|
31
|
+
# is created or manipulated using HexaPDF.
|
32
|
+
#++
|
33
|
+
|
34
|
+
require 'hexapdf/font/true_type/table'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Font
|
38
|
+
module TrueType
|
39
|
+
class Table
|
40
|
+
|
41
|
+
# Generic base class for all cmap subtables.
|
42
|
+
#
|
43
|
+
# cmap format 8.0 is currently not implemented because use of the format is discouraged in
|
44
|
+
# the specification and no font with a format 8.0 cmap subtable was available for testing.
|
45
|
+
#
|
46
|
+
# The preferred cmap format is 12.0 because it supports all of Unicode and allows for fast
|
47
|
+
# and memory efficient code-to-gid as well as gid-to-code mappings.
|
48
|
+
#
|
49
|
+
# See:
|
50
|
+
# * Cmap
|
51
|
+
# * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6cmap.html
|
52
|
+
class CmapSubtable
|
53
|
+
|
54
|
+
# The platform identifier for Unicode.
|
55
|
+
PLATFORM_UNICODE = 0
|
56
|
+
|
57
|
+
# The platform identifier for Microsoft.
|
58
|
+
PLATFORM_MICROSOFT = 3
|
59
|
+
|
60
|
+
# The platform identifier.
|
61
|
+
attr_accessor :platform_id
|
62
|
+
|
63
|
+
# The platform-specific encoding identifier.
|
64
|
+
attr_accessor :encoding_id
|
65
|
+
|
66
|
+
# The cmap format or +nil+ if the subtable wasn't read from a file.
|
67
|
+
attr_reader :format
|
68
|
+
|
69
|
+
# The language code.
|
70
|
+
attr_accessor :language
|
71
|
+
|
72
|
+
# The complete code map.
|
73
|
+
attr_accessor :code_map
|
74
|
+
|
75
|
+
# The complete gid map.
|
76
|
+
attr_accessor :gid_map
|
77
|
+
|
78
|
+
# Creates a new subtable.
|
79
|
+
def initialize(platform_id, encoding_id)
|
80
|
+
@platform_id = platform_id
|
81
|
+
@encoding_id = encoding_id
|
82
|
+
@supported = true
|
83
|
+
@code_map = {}
|
84
|
+
@gid_map = {}
|
85
|
+
@format = nil
|
86
|
+
@language = 0
|
87
|
+
end
|
88
|
+
|
89
|
+
# Returns +true+ if this subtable contains a Unicode cmap.
|
90
|
+
def unicode?
|
91
|
+
(platform_id == PLATFORM_MICROSOFT && (encoding_id == 1 || encoding_id == 10)) ||
|
92
|
+
platform_id == PLATFORM_UNICODE
|
93
|
+
end
|
94
|
+
|
95
|
+
# Returns the glyph index for the given character code or +nil+ if the character code is
|
96
|
+
# not mapped.
|
97
|
+
def [](code)
|
98
|
+
@code_map[code]
|
99
|
+
end
|
100
|
+
|
101
|
+
# Returns a character code for the given glyph index or +nil+ if the given glyph index
|
102
|
+
# does not exist or is not mapped to a character code.
|
103
|
+
#
|
104
|
+
# Note that some fonts map multiple character codes to the same glyph (e.g. hyphen and
|
105
|
+
# minus), i.e. the code-to-glyph mapping is surjective but not injective! In such a case
|
106
|
+
# one of the available character codes is returned.
|
107
|
+
def gid_to_code(gid)
|
108
|
+
@gid_map[gid]
|
109
|
+
end
|
110
|
+
|
111
|
+
# :call-seq:
|
112
|
+
# subtable.parse!(io, offset) => true or false
|
113
|
+
#
|
114
|
+
# Parses the cmap subtable from the IO at the given offset.
|
115
|
+
#
|
116
|
+
# If the subtable format is supported, the information is used to populate this object and
|
117
|
+
# +true+ is returned. Otherwise nothing is done and +false+ is returned.
|
118
|
+
def parse(io, offset)
|
119
|
+
io.pos = offset
|
120
|
+
@format = io.read(2).unpack('n').first
|
121
|
+
if [8, 10, 12].include?(@format)
|
122
|
+
io.pos += 2
|
123
|
+
length, @language = io.read(8).unpack('N2')
|
124
|
+
elsif [0, 2, 4, 6].include?(@format)
|
125
|
+
length, @language = io.read(4).unpack('n2')
|
126
|
+
end
|
127
|
+
supported = true
|
128
|
+
@code_map, @gid_map = case @format
|
129
|
+
when 0 then Format0.parse(io, length)
|
130
|
+
when 2 then Format2.parse(io, length)
|
131
|
+
when 4 then Format4.parse(io, length)
|
132
|
+
when 6 then Format6.parse(io, length)
|
133
|
+
when 10 then Format10.parse(io, length)
|
134
|
+
when 12 then Format12.parse(io, length)
|
135
|
+
else
|
136
|
+
supported = false
|
137
|
+
[{}, {}]
|
138
|
+
end
|
139
|
+
supported
|
140
|
+
end
|
141
|
+
|
142
|
+
def inspect #:nodoc:
|
143
|
+
"#<#{self.class.name} (#{platform_id}, #{encoding_id}, #{language}, " \
|
144
|
+
"#{format.inspect})>"
|
145
|
+
end
|
146
|
+
|
147
|
+
|
148
|
+
# Cmap format 0
|
149
|
+
module Format0
|
150
|
+
|
151
|
+
# :call-seq:
|
152
|
+
# Format0.parse(io, length) -> code_map
|
153
|
+
#
|
154
|
+
# Parses the format 0 cmap subtable from the given IO at the current position and
|
155
|
+
# returns the contained code map.
|
156
|
+
#
|
157
|
+
# It is assumed that the first six bytes of the subtable have already been consumed.
|
158
|
+
def self.parse(io, length)
|
159
|
+
raise HexaPDF::Error, "Invalid length #{length} for cmap format 0" if length != 262
|
160
|
+
code_map = io.read(256).unpack('C*')
|
161
|
+
gid_map = {}
|
162
|
+
code_map.each_with_index {|glyph, index| gid_map[glyph] = index}
|
163
|
+
[code_map, gid_map]
|
164
|
+
end
|
165
|
+
|
166
|
+
end
|
167
|
+
|
168
|
+
|
169
|
+
# Cmap format 2
|
170
|
+
module Format2
|
171
|
+
|
172
|
+
SubHeader = Struct.new(:first_code, :entry_count, :id_delta, :first_glyph_index)
|
173
|
+
|
174
|
+
# :call-seq:
|
175
|
+
# Format2.parse(io, length) -> code_map
|
176
|
+
#
|
177
|
+
# Parses the format 2 cmap subtable from the given IO at the current position and
|
178
|
+
# returns the contained code map.
|
179
|
+
#
|
180
|
+
# It is assumed that the first six bytes of the subtable have already been consumed.
|
181
|
+
def self.parse(io, length)
|
182
|
+
sub_header_keys = io.read(512).unpack('n*')
|
183
|
+
nr_sub_headers = 0
|
184
|
+
sub_header_keys.map! do |key|
|
185
|
+
nr_sub_headers = key if key > nr_sub_headers
|
186
|
+
key / 8
|
187
|
+
end
|
188
|
+
nr_sub_headers = 1 + nr_sub_headers / 8
|
189
|
+
|
190
|
+
sub_headers = []
|
191
|
+
nr_sub_headers.times do |i|
|
192
|
+
h = SubHeader.new(*io.read(8).unpack('n2s>n'))
|
193
|
+
# Map the currently stored id_range_offset to the corresponding glyph index by first
|
194
|
+
# changing the offset to begin from the position of the first glyph index and then
|
195
|
+
# halfing the value since each glyph is a UInt16.
|
196
|
+
h.first_glyph_index = (h.first_glyph_index - 2 - 8 * (nr_sub_headers - i - 1)) / 2
|
197
|
+
sub_headers << h
|
198
|
+
end
|
199
|
+
glyph_indexes = io.read(length - 6 - 512 - 8 * nr_sub_headers).unpack('n*')
|
200
|
+
|
201
|
+
gid_map = {}
|
202
|
+
sub_headers.each_with_index do |sub_header, i|
|
203
|
+
sub_header.entry_count.times do |j|
|
204
|
+
glyph_id = glyph_indexes[sub_header.first_glyph_index + j]
|
205
|
+
glyph_id = (glyph_id + sub_header.id_delta) % 65536 if glyph_id != 0
|
206
|
+
gid_map[glyph_id] = (sub_header_keys.index(i) << 8) + j + sub_header.first_code
|
207
|
+
end
|
208
|
+
end
|
209
|
+
|
210
|
+
[mapper(sub_header_keys, sub_headers, glyph_indexes), gid_map]
|
211
|
+
end
|
212
|
+
|
213
|
+
def self.mapper(sub_header_keys, sub_headers, glyph_indexes) #:nodoc:
|
214
|
+
Hash.new do |h, code|
|
215
|
+
i = code
|
216
|
+
i, j = i.divmod(256) if code > 255
|
217
|
+
k = sub_header_keys[i]
|
218
|
+
if !k
|
219
|
+
glyph_id = 0
|
220
|
+
elsif k > 0
|
221
|
+
sub_header = sub_headers[k]
|
222
|
+
raise HexaPDF::Error, "Second byte of character code missing" if j.nil?
|
223
|
+
j -= sub_header.first_code
|
224
|
+
if 0 <= j && j < sub_header.entry_count
|
225
|
+
glyph_id = glyph_indexes[sub_header.first_glyph_index + j]
|
226
|
+
glyph_id = (glyph_id + sub_header.id_delta) % 65536 if glyph_id != 0
|
227
|
+
else
|
228
|
+
glyph_id = 0
|
229
|
+
end
|
230
|
+
else
|
231
|
+
glyph_id = glyph_indexes[i]
|
232
|
+
end
|
233
|
+
h[code] = glyph_id unless glyph_id == 0
|
234
|
+
end
|
235
|
+
end
|
236
|
+
|
237
|
+
|
238
|
+
end
|
239
|
+
|
240
|
+
|
241
|
+
# Cmap format 4
|
242
|
+
module Format4
|
243
|
+
|
244
|
+
# :call-seq:
|
245
|
+
# Format4.parse(io, length) -> code_map
|
246
|
+
#
|
247
|
+
# Parses the format 4 cmap subtable from the given IO at the current position and
|
248
|
+
# returns the contained code map.
|
249
|
+
#
|
250
|
+
# It is assumed that the first six bytes of the subtable have already been consumed.
|
251
|
+
def self.parse(io, length)
|
252
|
+
seg_count_x2 = io.read(8).unpack('n').first
|
253
|
+
end_codes = io.read(seg_count_x2).unpack('n*')
|
254
|
+
io.pos += 2
|
255
|
+
start_codes = io.read(seg_count_x2).unpack('n*')
|
256
|
+
id_deltas = io.read(seg_count_x2).unpack('n*')
|
257
|
+
id_range_offsets = io.read(seg_count_x2).unpack('n*').map!.with_index do |offset, idx|
|
258
|
+
# Change offsets to indexes, starting from the id_range_offsets array
|
259
|
+
offset == 0 ? offset : offset / 2 + idx
|
260
|
+
end
|
261
|
+
glyph_indexes = io.read(length - 16 - seg_count_x2 * 4).unpack('n*')
|
262
|
+
mapper(end_codes, start_codes, id_deltas, id_range_offsets, glyph_indexes)
|
263
|
+
end
|
264
|
+
|
265
|
+
def self.mapper(end_codes, start_codes, id_deltas, id_range_offsets, glyph_indexes) #:nodoc:
|
266
|
+
compute_glyph_id = lambda do |index, code|
|
267
|
+
offset = id_range_offsets[index]
|
268
|
+
if offset != 0
|
269
|
+
glyph_id = glyph_indexes[offset - end_codes.length + (code - start_codes[index])]
|
270
|
+
glyph_id = (glyph_id + id_deltas[index]) % 65536 if glyph_id != 0
|
271
|
+
else
|
272
|
+
glyph_id = (code + id_deltas[index]) % 65536
|
273
|
+
end
|
274
|
+
glyph_id
|
275
|
+
end
|
276
|
+
|
277
|
+
code_map = Hash.new do |h, code|
|
278
|
+
i = end_codes.bsearch_index {|c| c >= code}
|
279
|
+
if i && start_codes[i] <= code
|
280
|
+
glyph_id = compute_glyph_id.call(i, code)
|
281
|
+
else
|
282
|
+
glyph_id = 0
|
283
|
+
end
|
284
|
+
h[code] = glyph_id unless glyph_id == 0
|
285
|
+
end
|
286
|
+
|
287
|
+
gid_map = {}
|
288
|
+
end_codes.length.times do |i|
|
289
|
+
start_codes[i].upto(end_codes[i]) do |code|
|
290
|
+
gid_map[compute_glyph_id.call(i, code)] = code
|
291
|
+
end
|
292
|
+
end
|
293
|
+
[code_map, gid_map]
|
294
|
+
end
|
295
|
+
|
296
|
+
end
|
297
|
+
|
298
|
+
|
299
|
+
# Cmap format 6
|
300
|
+
module Format6
|
301
|
+
|
302
|
+
# :call-seq:
|
303
|
+
# Format6.parse(io, length) -> code_map
|
304
|
+
#
|
305
|
+
# Parses the format 6 cmap subtable from the given IO at the current position and
|
306
|
+
# returns the contained code map.
|
307
|
+
#
|
308
|
+
# It is assumed that the first six bytes of the subtable have already been consumed.
|
309
|
+
def self.parse(io, _length)
|
310
|
+
first_code, entry_count = io.read(4).unpack('n2')
|
311
|
+
code_map = io.read(2 * entry_count).unpack('n*')
|
312
|
+
gid_map = {}
|
313
|
+
code_map = code_map.each_with_index.with_object({}) do |(g, i), hash|
|
314
|
+
hash[first_code + i] = g
|
315
|
+
gid_map[g] = first_code + i
|
316
|
+
end
|
317
|
+
[code_map, gid_map]
|
318
|
+
end
|
319
|
+
|
320
|
+
end
|
321
|
+
|
322
|
+
|
323
|
+
# Cmap format 10
|
324
|
+
module Format10
|
325
|
+
|
326
|
+
# :call-seq:
|
327
|
+
# Format10.parse(io, length) -> code_map
|
328
|
+
#
|
329
|
+
# Parses the format 10 cmap subtable from the given IO at the current position and
|
330
|
+
# returns the contained code map.
|
331
|
+
#
|
332
|
+
# It is assumed that the first twelve bytes of the subtable have already been consumed.
|
333
|
+
def self.parse(io, _length)
|
334
|
+
first_code, entry_count = io.read(8).unpack('N2')
|
335
|
+
code_map = io.read(2 * entry_count).unpack('n*')
|
336
|
+
gid_map = {}
|
337
|
+
code_map = code_map.each_with_index.with_object({}) do |(g, i), hash|
|
338
|
+
hash[first_code + i] = g
|
339
|
+
gid_map[g] = first_code + i
|
340
|
+
end
|
341
|
+
[code_map, gid_map]
|
342
|
+
end
|
343
|
+
|
344
|
+
end
|
345
|
+
|
346
|
+
|
347
|
+
# Cmap format 12
|
348
|
+
module Format12
|
349
|
+
|
350
|
+
# :call-seq:
|
351
|
+
# Format12.parse(io, length) -> code_map
|
352
|
+
#
|
353
|
+
# Parses the format 12 cmap subtable from the given IO at the current position and
|
354
|
+
# returns the contained code map.
|
355
|
+
#
|
356
|
+
# It is assumed that the first twelve bytes of the subtable have already been consumed.
|
357
|
+
def self.parse(io, _length)
|
358
|
+
mapper(io.read(4).unpack('N').first.times.map { io.read(12).unpack('N3') })
|
359
|
+
end
|
360
|
+
|
361
|
+
# The parameter +groups+ is an array containing [start_code, end_code, start_glyph_id]
|
362
|
+
# arrays.
|
363
|
+
def self.mapper(groups) #:nodoc:
|
364
|
+
code_map = Hash.new do |h, code|
|
365
|
+
group = groups.bsearch {|g| g[1] >= code}
|
366
|
+
h[code] = group[2] + (code - group[0]) if group && group[0] <= code
|
367
|
+
end
|
368
|
+
groups_by_gid = groups.sort_by {|g| g[2]}
|
369
|
+
gid_map = Hash.new do |h, gid|
|
370
|
+
group = groups_by_gid.bsearch {|g| g[2] + g[1] - g[0] >= gid}
|
371
|
+
h[gid] = group[0] + (gid - group[2]) if group && group[2] <= gid
|
372
|
+
end
|
373
|
+
[code_map, gid_map]
|
374
|
+
end
|
375
|
+
|
376
|
+
end
|
377
|
+
|
378
|
+
|
379
|
+
end
|
380
|
+
|
381
|
+
end
|
382
|
+
end
|
383
|
+
end
|
384
|
+
end
|