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,349 @@
|
|
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/error'
|
35
|
+
require 'hexapdf/tokenizer'
|
36
|
+
require 'hexapdf/stream'
|
37
|
+
require 'hexapdf/xref_section'
|
38
|
+
|
39
|
+
module HexaPDF
|
40
|
+
|
41
|
+
# Parses an IO stream according to PDF1.7 to get at the contained objects.
|
42
|
+
#
|
43
|
+
# This class also contains higher-level methods for getting indirect objects and revisions.
|
44
|
+
#
|
45
|
+
# See: PDF1.7 s7
|
46
|
+
class Parser
|
47
|
+
|
48
|
+
# Creates a new parser for the given IO object.
|
49
|
+
#
|
50
|
+
# PDF references are resolved using the associated Document object.
|
51
|
+
def initialize(io, document)
|
52
|
+
@io = io
|
53
|
+
@tokenizer = Tokenizer.new(io)
|
54
|
+
@document = document
|
55
|
+
@object_stream_data = {}
|
56
|
+
retrieve_pdf_header_offset_and_version
|
57
|
+
end
|
58
|
+
|
59
|
+
# Loads the indirect (potentially compressed) object specified by the given cross-reference
|
60
|
+
# entry.
|
61
|
+
#
|
62
|
+
# For information about the +xref_entry+ argument, have a look at HexaPDF::XRefSection and
|
63
|
+
# HexaPDF::XRefSection::Entry.
|
64
|
+
def load_object(xref_entry)
|
65
|
+
obj, oid, gen, stream =
|
66
|
+
case xref_entry.type
|
67
|
+
when :in_use
|
68
|
+
parse_indirect_object(xref_entry.pos)
|
69
|
+
when :free
|
70
|
+
[nil, xref_entry.oid, xref_entry.gen, nil]
|
71
|
+
when :compressed
|
72
|
+
load_compressed_object(xref_entry)
|
73
|
+
else
|
74
|
+
raise_malformed("Invalid cross-reference type '#{xref_entry.type}' encountered")
|
75
|
+
end
|
76
|
+
|
77
|
+
if xref_entry.oid != 0 && (oid != xref_entry.oid || gen != xref_entry.gen)
|
78
|
+
raise_malformed("The oid,gen (#{oid},#{gen}) values of the indirect object don't match " \
|
79
|
+
"the values (#{xref_entry.oid},#{xref_entry.gen}) from the xref")
|
80
|
+
end
|
81
|
+
|
82
|
+
@document.wrap(obj, oid: oid, gen: gen, stream: stream)
|
83
|
+
end
|
84
|
+
|
85
|
+
# Parses the indirect object at the specified offset.
|
86
|
+
#
|
87
|
+
# This method is used by a PDF Document to load objects. It should **not** be used by any
|
88
|
+
# other object because invalid object positions lead to errors.
|
89
|
+
#
|
90
|
+
# Returns an array containing [object, oid, gen, stream].
|
91
|
+
#
|
92
|
+
# See: PDF1.7 s7.3.10, s7.3.8
|
93
|
+
def parse_indirect_object(offset = nil)
|
94
|
+
@tokenizer.pos = offset + @header_offset if offset
|
95
|
+
oid = @tokenizer.next_token
|
96
|
+
gen = @tokenizer.next_token
|
97
|
+
tok = @tokenizer.next_token
|
98
|
+
unless oid.kind_of?(Integer) && gen.kind_of?(Integer) &&
|
99
|
+
tok.kind_of?(Tokenizer::Token) && tok == 'obj'.freeze
|
100
|
+
raise_malformed("No valid object found", pos: offset)
|
101
|
+
end
|
102
|
+
|
103
|
+
if (tok = @tokenizer.peek_token) && tok.kind_of?(Tokenizer::Token) && tok == 'endobj'.freeze
|
104
|
+
maybe_raise("No indirect object value between 'obj' and 'endobj'", pos: @tokenizer.pos)
|
105
|
+
object = nil
|
106
|
+
else
|
107
|
+
object = @tokenizer.next_object
|
108
|
+
end
|
109
|
+
|
110
|
+
tok = @tokenizer.next_token
|
111
|
+
|
112
|
+
if tok.kind_of?(Tokenizer::Token) && tok == 'stream'.freeze
|
113
|
+
unless object.kind_of?(Hash)
|
114
|
+
raise_malformed("A stream needs a dictionary, not a(n) #{object.class}", pos: offset)
|
115
|
+
end
|
116
|
+
tok1 = @tokenizer.next_byte
|
117
|
+
tok2 = @tokenizer.next_byte if tok1 == 13 # 13=CR, 10=LF
|
118
|
+
if tok1 != 10 && tok1 != 13
|
119
|
+
raise_malformed("Keyword stream must be followed by LF or CR/LF", pos: @tokenizer.pos)
|
120
|
+
elsif tok1 == 13 && tok2 != 10
|
121
|
+
maybe_raise("Keyword stream must be followed by LF or CR/LF, not CR alone",
|
122
|
+
pos: @tokenizer.pos)
|
123
|
+
@tokenizer.pos -= 1
|
124
|
+
end
|
125
|
+
|
126
|
+
# Note that getting :Length might move the IO pointer (when resolving references)
|
127
|
+
pos = @tokenizer.pos
|
128
|
+
length = if object[:Length].kind_of?(Integer)
|
129
|
+
object[:Length]
|
130
|
+
elsif object[:Length].kind_of?(Reference)
|
131
|
+
@document.deref(object[:Length]).value
|
132
|
+
else
|
133
|
+
0
|
134
|
+
end
|
135
|
+
@tokenizer.pos = pos + length
|
136
|
+
|
137
|
+
tok = @tokenizer.next_token
|
138
|
+
unless tok.kind_of?(Tokenizer::Token) && tok == 'endstream'.freeze
|
139
|
+
maybe_raise("Invalid stream length, keyword endstream not found", pos: @tokenizer.pos)
|
140
|
+
@tokenizer.pos = pos
|
141
|
+
if @tokenizer.scan_until(/(?=\n?endstream)/)
|
142
|
+
length = @tokenizer.pos - pos
|
143
|
+
tok = @tokenizer.next_token
|
144
|
+
else
|
145
|
+
raise_malformed("Stream content must be followed by keyword endstream",
|
146
|
+
pos: @tokenizer.pos)
|
147
|
+
end
|
148
|
+
end
|
149
|
+
tok = @tokenizer.next_token
|
150
|
+
|
151
|
+
object[:Length] = length
|
152
|
+
stream = StreamData.new(@tokenizer.io, offset: pos, length: length,
|
153
|
+
filter: @document.unwrap(object[:Filter]),
|
154
|
+
decode_parms: @document.unwrap(object[:DecodeParms]))
|
155
|
+
end
|
156
|
+
|
157
|
+
unless tok.kind_of?(Tokenizer::Token) && tok == 'endobj'.freeze
|
158
|
+
maybe_raise("Indirect object must be followed by keyword endobj", pos: @tokenizer.pos)
|
159
|
+
end
|
160
|
+
|
161
|
+
[object, oid, gen, stream]
|
162
|
+
end
|
163
|
+
|
164
|
+
# Loads the compressed object identified by the cross-reference entry.
|
165
|
+
def load_compressed_object(xref_entry)
|
166
|
+
unless @object_stream_data.key?(xref_entry.objstm)
|
167
|
+
obj = @document.object(xref_entry.objstm)
|
168
|
+
unless obj.respond_to?(:parse_stream)
|
169
|
+
raise_malformed("Object with oid=#{xref_entry.objstm} is not an object stream")
|
170
|
+
end
|
171
|
+
@object_stream_data[xref_entry.objstm] = obj.parse_stream
|
172
|
+
end
|
173
|
+
|
174
|
+
[*@object_stream_data[xref_entry.objstm].object_by_index(xref_entry.pos), xref_entry.gen, nil]
|
175
|
+
end
|
176
|
+
|
177
|
+
# Loads a single revision whose cross-reference section/stream is located at the given
|
178
|
+
# position.
|
179
|
+
#
|
180
|
+
# Returns an HexaPDF::XRefSection object and the accompanying trailer dictionary.
|
181
|
+
def load_revision(pos)
|
182
|
+
if xref_section?(pos)
|
183
|
+
xref_section, trailer = parse_xref_section_and_trailer(pos)
|
184
|
+
else
|
185
|
+
obj = load_object(XRefSection.in_use_entry(0, 0, pos))
|
186
|
+
unless obj.respond_to?(:xref_section)
|
187
|
+
raise_malformed("Object is not a cross-reference stream", pos: pos)
|
188
|
+
end
|
189
|
+
xref_section = obj.xref_section
|
190
|
+
trailer = obj.trailer
|
191
|
+
unless xref_section.entry?(obj.oid, obj.gen)
|
192
|
+
maybe_raise("Cross-reference stream doesn't contain entry for itself", pos: pos)
|
193
|
+
xref_section.add_in_use_entry(obj.oid, obj.gen, pos)
|
194
|
+
end
|
195
|
+
end
|
196
|
+
xref_section.delete(0)
|
197
|
+
[xref_section, trailer]
|
198
|
+
end
|
199
|
+
|
200
|
+
# Looks at the given offset and returns +true+ if there is a cross-reference section at that
|
201
|
+
# position.
|
202
|
+
def xref_section?(offset)
|
203
|
+
@tokenizer.pos = offset + @header_offset
|
204
|
+
token = @tokenizer.peek_token
|
205
|
+
token.kind_of?(Tokenizer::Token) && token == 'xref'
|
206
|
+
end
|
207
|
+
|
208
|
+
# Parses the cross-reference section at the given position and the following trailer and
|
209
|
+
# returns them as an array consisting of an HexaPDF::XRefSection instance and a hash.
|
210
|
+
#
|
211
|
+
# This method can only parse cross-reference sections, not cross-reference streams!
|
212
|
+
#
|
213
|
+
# See: PDF1.7 s7.5.4, s7.5.5; ADB1.7 sH.3-3.4.3
|
214
|
+
def parse_xref_section_and_trailer(offset)
|
215
|
+
@tokenizer.pos = offset + @header_offset
|
216
|
+
token = @tokenizer.next_token
|
217
|
+
unless token.kind_of?(Tokenizer::Token) && token == 'xref'
|
218
|
+
raise_malformed("Xref section doesn't start with keyword xref", pos: @tokenizer.pos)
|
219
|
+
end
|
220
|
+
|
221
|
+
xref = XRefSection.new
|
222
|
+
start = @tokenizer.next_token
|
223
|
+
while start.kind_of?(Integer)
|
224
|
+
number_of_entries = @tokenizer.next_token
|
225
|
+
unless number_of_entries.kind_of?(Integer)
|
226
|
+
raise_malformed("Invalid cross-reference subsection start", pos: @tokenizer.pos)
|
227
|
+
end
|
228
|
+
|
229
|
+
@tokenizer.skip_whitespace
|
230
|
+
start.upto(start + number_of_entries - 1) do |oid|
|
231
|
+
pos, gen, type = @tokenizer.next_xref_entry
|
232
|
+
if xref.entry?(oid)
|
233
|
+
next
|
234
|
+
elsif type == 'n'.freeze
|
235
|
+
if pos == 0 || gen > 65535
|
236
|
+
maybe_raise("Invalid in use cross-reference entry in cross-reference section",
|
237
|
+
pos: @tokenizer.pos)
|
238
|
+
xref.add_free_entry(oid, gen)
|
239
|
+
else
|
240
|
+
xref.add_in_use_entry(oid, gen, pos)
|
241
|
+
end
|
242
|
+
else
|
243
|
+
xref.add_free_entry(oid, gen)
|
244
|
+
end
|
245
|
+
end
|
246
|
+
start = @tokenizer.next_token
|
247
|
+
end
|
248
|
+
|
249
|
+
unless start.kind_of?(Tokenizer::Token) && start == 'trailer'
|
250
|
+
raise_malformed("Trailer doesn't start with keyword trailer", pos: @tokenizer.pos)
|
251
|
+
end
|
252
|
+
|
253
|
+
trailer = @tokenizer.next_object
|
254
|
+
unless trailer.kind_of?(Hash)
|
255
|
+
raise_malformed("Trailer is a #{trailer.class} instead of a dictionary ", pos: @tokenizer.pos)
|
256
|
+
end
|
257
|
+
|
258
|
+
[xref, trailer]
|
259
|
+
end
|
260
|
+
|
261
|
+
# Returns the offset of the main cross-reference section/stream.
|
262
|
+
#
|
263
|
+
# Implementation note: Normally, the %%EOF marker has to be on the last line, however, Adobe
|
264
|
+
# viewers relax this restriction and so do we.
|
265
|
+
#
|
266
|
+
# If strict parsing is disabled, the whole file is searched for the offset.
|
267
|
+
#
|
268
|
+
# See: PDF1.7 s7.5.5, ADB1.7 sH.3-3.4.4
|
269
|
+
def startxref_offset
|
270
|
+
@io.seek(0, IO::SEEK_END)
|
271
|
+
step_size = 1024
|
272
|
+
pos = @io.pos
|
273
|
+
eof_not_found = startxref_missing = false
|
274
|
+
|
275
|
+
while pos != 0
|
276
|
+
@io.pos = [pos - step_size, 0].max
|
277
|
+
pos = @io.pos
|
278
|
+
lines = @io.read(step_size + 40).split(/[\r\n]+/)
|
279
|
+
|
280
|
+
eof_index = lines.rindex {|l| l.strip == '%%EOF' }
|
281
|
+
unless eof_index
|
282
|
+
eof_not_found = true
|
283
|
+
next
|
284
|
+
end
|
285
|
+
unless eof_index >= 2 && lines[eof_index - 2].strip == "startxref"
|
286
|
+
startxref_missing = true
|
287
|
+
next
|
288
|
+
end
|
289
|
+
|
290
|
+
break # we found the startxref offset
|
291
|
+
end
|
292
|
+
|
293
|
+
if eof_not_found
|
294
|
+
maybe_raise("PDF file trailer with end-of-file marker not found", pos: pos,
|
295
|
+
force: !eof_index)
|
296
|
+
elsif startxref_missing
|
297
|
+
maybe_raise("PDF file trailer is missing startxref keyword", pos: pos,
|
298
|
+
force: eof_index < 2 || lines[eof_index - 2].strip != "startxref")
|
299
|
+
end
|
300
|
+
|
301
|
+
lines[eof_index - 1].to_i
|
302
|
+
end
|
303
|
+
|
304
|
+
# Returns the PDF version number that is stored in the file header.
|
305
|
+
#
|
306
|
+
# See: PDF1.7 s7.5.2
|
307
|
+
def file_header_version
|
308
|
+
unless @header_version
|
309
|
+
raise_malformed("PDF file header is missing or corrupt", pos: 0)
|
310
|
+
end
|
311
|
+
@header_version
|
312
|
+
end
|
313
|
+
|
314
|
+
private
|
315
|
+
|
316
|
+
# Retrieves the offset of the PDF header and the PDF version number in it.
|
317
|
+
#
|
318
|
+
# The PDF header should normally appear on the first line. However, Adobe relaxes this
|
319
|
+
# restriction so that the header may appear in the first 1024 bytes. We follow the Adobe
|
320
|
+
# convention.
|
321
|
+
#
|
322
|
+
# See: PDF1.7 s7.5.2, ADB1.7 sH.3-3.4.1
|
323
|
+
def retrieve_pdf_header_offset_and_version
|
324
|
+
@io.seek(0)
|
325
|
+
@header_offset = @io.read(1024).index(/%PDF-(\d\.\d)/) || 0
|
326
|
+
@header_version = $1
|
327
|
+
end
|
328
|
+
|
329
|
+
# Raises a HexaPDF::MalformedPDFError with the given message and source position.
|
330
|
+
def raise_malformed(msg, pos: nil)
|
331
|
+
raise HexaPDF::MalformedPDFError.new(msg, pos: pos)
|
332
|
+
end
|
333
|
+
|
334
|
+
# Calls the block stored in the config option +parser.on_correctable_error+ with the document,
|
335
|
+
# the given message and the position. If the returned value is +true+, raises a
|
336
|
+
# HexaPDF::MalformedPDFError. Otherwise the error is corrected and parsing continues.
|
337
|
+
#
|
338
|
+
# If the option +force+ is used, the block is not called and the error is raised immediately.
|
339
|
+
def maybe_raise(msg, pos: nil, force: false)
|
340
|
+
if force || @document.config['parser.on_correctable_error'].call(@document, msg, pos)
|
341
|
+
error = HexaPDF::MalformedPDFError.new(msg, pos: pos)
|
342
|
+
error.set_backtrace(caller(1))
|
343
|
+
raise error
|
344
|
+
end
|
345
|
+
end
|
346
|
+
|
347
|
+
end
|
348
|
+
|
349
|
+
end
|
@@ -0,0 +1,99 @@
|
|
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/object'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
|
38
|
+
# Implementation of the PDF rectangle data structure.
|
39
|
+
#
|
40
|
+
# Rectangles are used for describing page and bounding boxes. They are represented by arrays of
|
41
|
+
# four numbers specifying the (x,y) coordinates of *any* diagonally opposite corners.
|
42
|
+
#
|
43
|
+
# This class simplifies the usage of rectangles by automatically normalizing the coordinates so
|
44
|
+
# that they are in the order:
|
45
|
+
#
|
46
|
+
# [llx, lly, urx, ury]
|
47
|
+
#
|
48
|
+
# where +llx+ is the lower-left x-coordinate, +lly+ is the lower-left y-coordinate, +urx+ is the
|
49
|
+
# upper-right x-coordinate and +ury+ is the upper-right y-coordinate.
|
50
|
+
#
|
51
|
+
# See: PDF1.7 s7.9.5
|
52
|
+
class Rectangle < HexaPDF::Object
|
53
|
+
|
54
|
+
# Returns the x-coordinate of the lower-left corner.
|
55
|
+
def left
|
56
|
+
value[0]
|
57
|
+
end
|
58
|
+
|
59
|
+
# Returns the x-coordinate of the upper-right corner.
|
60
|
+
def right
|
61
|
+
value[2]
|
62
|
+
end
|
63
|
+
|
64
|
+
# Returns the y-coordinate of the lower-left corner.
|
65
|
+
def bottom
|
66
|
+
value[1]
|
67
|
+
end
|
68
|
+
|
69
|
+
# Returns the y-coordinate of the upper-right corner.
|
70
|
+
def top
|
71
|
+
value[3]
|
72
|
+
end
|
73
|
+
|
74
|
+
# Returns the width of the rectangle.
|
75
|
+
def width
|
76
|
+
value[2] - value[0]
|
77
|
+
end
|
78
|
+
|
79
|
+
# Returns the height of the rectangle.
|
80
|
+
def height
|
81
|
+
value[3] - value[1]
|
82
|
+
end
|
83
|
+
|
84
|
+
private
|
85
|
+
|
86
|
+
# Ensures that the value is an array containing four numbers that specify the lower-left and
|
87
|
+
# upper-right corner.
|
88
|
+
def after_data_change
|
89
|
+
super
|
90
|
+
unless value.kind_of?(Array) && value.size == 4
|
91
|
+
raise ArgumentError, "A PDF rectangle structure must contain an array of four numbers"
|
92
|
+
end
|
93
|
+
value[0], value[2] = value[2], value[0] if value[0] > value[2]
|
94
|
+
value[1], value[3] = value[3], value[1] if value[1] > value[3]
|
95
|
+
end
|
96
|
+
|
97
|
+
end
|
98
|
+
|
99
|
+
end
|
@@ -0,0 +1,98 @@
|
|
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/error'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
|
38
|
+
# A reference to an indirect object.
|
39
|
+
#
|
40
|
+
# The PDF syntax allows for references to existing and non-existing indirect objects. Such
|
41
|
+
# references are represented with objects of this class.
|
42
|
+
#
|
43
|
+
# Note that after initialization changing the object or generation numbers is not possible
|
44
|
+
# anymore!
|
45
|
+
#
|
46
|
+
# The methods #hash and #eql? are implemented so that objects of this class can be used as hash
|
47
|
+
# keys. Furthermore the implementation is compatible to the one of Object, i.e. the hash of a
|
48
|
+
# Reference object is the same as the hash of an indirect Object.
|
49
|
+
#
|
50
|
+
# See: PDF1.7 s7.3.10, Object
|
51
|
+
class Reference
|
52
|
+
|
53
|
+
include Comparable
|
54
|
+
|
55
|
+
# Returns the object number of the referenced indirect object.
|
56
|
+
attr_reader :oid
|
57
|
+
|
58
|
+
# Returns the generation number of the referenced indirect object.
|
59
|
+
attr_reader :gen
|
60
|
+
|
61
|
+
# Creates a new Reference with the given object number and, optionally, generation number.
|
62
|
+
def initialize(oid, gen = 0)
|
63
|
+
@oid = Integer(oid)
|
64
|
+
@gen = Integer(gen)
|
65
|
+
end
|
66
|
+
|
67
|
+
# Compares this object to another object.
|
68
|
+
#
|
69
|
+
# If the other object does not respond to +oid+ or +gen+, +nil+ is returned. Otherwise objects
|
70
|
+
# are ordered first by object number and then by generation number.
|
71
|
+
def <=>(other)
|
72
|
+
return nil unless other.respond_to?(:oid) && other.respond_to?(:gen)
|
73
|
+
(oid == other.oid ? gen <=> other.gen : oid <=> other.oid)
|
74
|
+
end
|
75
|
+
|
76
|
+
# Returns +true+ if the other object is a Reference and has the same object and generation
|
77
|
+
# numbers.
|
78
|
+
def ==(other)
|
79
|
+
other.kind_of?(Reference) && oid == other.oid && gen == other.gen
|
80
|
+
end
|
81
|
+
|
82
|
+
# Returns +true+ if the other object references the same PDF object as this reference object.
|
83
|
+
def eql?(other)
|
84
|
+
other.respond_to?(:oid) && oid == other.oid && other.respond_to?(:gen) && gen == other.gen
|
85
|
+
end
|
86
|
+
|
87
|
+
# Computes the hash value based on the object and generation numbers.
|
88
|
+
def hash
|
89
|
+
oid.hash ^ gen.hash
|
90
|
+
end
|
91
|
+
|
92
|
+
def inspect #:nodoc:
|
93
|
+
"#<#{self.class.name} [#{oid}, #{gen}]>"
|
94
|
+
end
|
95
|
+
|
96
|
+
end
|
97
|
+
|
98
|
+
end
|