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,269 @@
|
|
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/dictionary'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Type
|
38
|
+
|
39
|
+
# Represents a node in the page tree of the PDF's document.
|
40
|
+
#
|
41
|
+
# The page tree is a tree structure containing page tree nodes for the root and intermediate
|
42
|
+
# nodes and page objects for the leaf nodes (see Page). The root node of the page tree is
|
43
|
+
# linked via the /Pages entry in the Catalog.
|
44
|
+
#
|
45
|
+
# All operations except #add_page on the page tree are rather expensive because page tree
|
46
|
+
# nodes and page objects can be mixed. This means that for finding a page at a specific index
|
47
|
+
# we have to go through all objects that come before it.
|
48
|
+
#
|
49
|
+
# Page indices are zero-based, not one-based. Therefore the first page has an index of 0!
|
50
|
+
#
|
51
|
+
# Since the page tree needs a certain structure it is not advised to directly modify page tree
|
52
|
+
# nodes. The validation feature can correct most problems but until the page tree is in order
|
53
|
+
# the methods may not work correctly!
|
54
|
+
#
|
55
|
+
# Newly created pages use the 'page.default_media_box' configuration option for the /MediaBox
|
56
|
+
# value. If an inherited /Resources dictionary does *not* exist, an empty one is created for
|
57
|
+
# the page.
|
58
|
+
#
|
59
|
+
# See: PDF1.7 s7.7.3.2, Page
|
60
|
+
class PageTreeNode < Dictionary
|
61
|
+
|
62
|
+
define_field :Type, type: Symbol, required: true, default: :Pages
|
63
|
+
define_field :Parent, type: Dictionary, indirect: true
|
64
|
+
define_field :Kids, type: Array, required: true, default: []
|
65
|
+
define_field :Count, type: Integer, required: true, default: 0
|
66
|
+
|
67
|
+
# Inheritable page fields
|
68
|
+
define_field :Resources, type: :XXResources
|
69
|
+
define_field :MediaBox, type: Rectangle
|
70
|
+
define_field :CropBox, type: Rectangle
|
71
|
+
define_field :Rotate, type: Integer
|
72
|
+
|
73
|
+
|
74
|
+
# Returns +true+ since page tree objects must always be indirect.
|
75
|
+
def must_be_indirect?
|
76
|
+
true
|
77
|
+
end
|
78
|
+
|
79
|
+
# Returns the number of pages under this page tree.
|
80
|
+
#
|
81
|
+
# *Note*: If this methods is not called on the root object of the page tree, the returned
|
82
|
+
# number is not the total number of pages in the document!
|
83
|
+
def page_count
|
84
|
+
self[:Count]
|
85
|
+
end
|
86
|
+
|
87
|
+
# Returns the page for the zero-based index or +nil+ if no such page exists.
|
88
|
+
#
|
89
|
+
# Negative indices count backwards from the end, i.e. -1 is the last page.
|
90
|
+
def page(index)
|
91
|
+
index = self[:Count] + index if index < 0
|
92
|
+
return nil if index < 0 || index >= self[:Count]
|
93
|
+
|
94
|
+
self[:Kids].each do |kid|
|
95
|
+
kid = document.deref(kid)
|
96
|
+
if kid.type == :Page
|
97
|
+
if index == 0
|
98
|
+
return kid
|
99
|
+
else
|
100
|
+
index -= 1
|
101
|
+
end
|
102
|
+
elsif index < kid[:Count]
|
103
|
+
return kid.page(index)
|
104
|
+
else
|
105
|
+
index -= kid[:Count]
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
# Inserts the page or a new empty page at the zero-based index and returns it.
|
111
|
+
#
|
112
|
+
# Negative indices count backwards from the end, i.e. -1 is the last page. When using
|
113
|
+
# negative indices, the page will be inserted after that element. So using an index of -1
|
114
|
+
# will insert the page after the last page.
|
115
|
+
#
|
116
|
+
# Must be called on the root of the page tree, otherwise the /Count entries are not
|
117
|
+
# correctly updated!
|
118
|
+
def insert_page(index, page = nil)
|
119
|
+
page ||= new_page
|
120
|
+
index = self[:Count] + index + 1 if index < 0
|
121
|
+
|
122
|
+
if index >= self[:Count]
|
123
|
+
self[:Kids] << page
|
124
|
+
page[:Parent] = self
|
125
|
+
page[:Resources] ||= {}
|
126
|
+
else
|
127
|
+
self[:Kids].each_with_index do |kid, kid_index|
|
128
|
+
kid = document.deref(kid)
|
129
|
+
if index == 0
|
130
|
+
self[:Kids].insert(kid_index, page)
|
131
|
+
page[:Parent] = self
|
132
|
+
break
|
133
|
+
elsif kid.type == :Page
|
134
|
+
index -= 1
|
135
|
+
elsif index <= kid[:Count]
|
136
|
+
kid.insert_page(index, page)
|
137
|
+
break
|
138
|
+
else
|
139
|
+
index -= kid[:Count]
|
140
|
+
end
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
self[:Count] += 1
|
145
|
+
|
146
|
+
page
|
147
|
+
end
|
148
|
+
|
149
|
+
# Adds the page or a new empty page at the end and returns it.
|
150
|
+
def add_page(page = nil)
|
151
|
+
insert_page(-1, page)
|
152
|
+
end
|
153
|
+
|
154
|
+
# Deletes the page at the position specified by the zero-based index and returns it. If an
|
155
|
+
# invalid index is specified, +nil+ is returned.
|
156
|
+
#
|
157
|
+
# Negative indices count backwards from the end, i.e. -1 is the last page.
|
158
|
+
#
|
159
|
+
# Must be called on the root of the page tree, otherwise the /Count entries are not
|
160
|
+
# correctly updated!
|
161
|
+
def delete_page(index)
|
162
|
+
index = self[:Count] + index if index < 0
|
163
|
+
return nil if index < 0 || index >= self[:Count]
|
164
|
+
|
165
|
+
page = nil
|
166
|
+
self[:Count] -= 1
|
167
|
+
self[:Kids].each_with_index do |kid, kid_index|
|
168
|
+
kid = document.deref(kid)
|
169
|
+
if kid.type == :Page && index == 0
|
170
|
+
page = self[:Kids].delete_at(kid_index)
|
171
|
+
document.delete(page)
|
172
|
+
break
|
173
|
+
elsif kid.type == :Page
|
174
|
+
index -= 1
|
175
|
+
elsif index < kid[:Count]
|
176
|
+
page = kid.delete_page(index)
|
177
|
+
if kid[:Count] == 0
|
178
|
+
self[:Kids].delete_at(kid_index)
|
179
|
+
document.delete(kid)
|
180
|
+
elsif kid[:Count] == 1
|
181
|
+
self[:Kids][kid_index] = kid[:Kids][0]
|
182
|
+
kid[:Kids][0][:Parent] = self
|
183
|
+
document.delete(kid)
|
184
|
+
end
|
185
|
+
break
|
186
|
+
else
|
187
|
+
index -= kid[:Count]
|
188
|
+
end
|
189
|
+
end
|
190
|
+
|
191
|
+
page
|
192
|
+
end
|
193
|
+
|
194
|
+
# :call-seq:
|
195
|
+
# pages.each_page {|page| block } -> pages
|
196
|
+
# pages.each_page -> Enumerator
|
197
|
+
#
|
198
|
+
# Iterates over all pages that are beneath this page tree node, from the first to the last
|
199
|
+
# page.
|
200
|
+
def each_page(&block)
|
201
|
+
return to_enum(__method__) unless block_given?
|
202
|
+
|
203
|
+
self[:Kids].each do |kid|
|
204
|
+
kid = document.deref(kid)
|
205
|
+
if kid.type == :Page
|
206
|
+
yield(kid)
|
207
|
+
else
|
208
|
+
kid.each_page(&block)
|
209
|
+
end
|
210
|
+
end
|
211
|
+
|
212
|
+
self
|
213
|
+
end
|
214
|
+
|
215
|
+
private
|
216
|
+
|
217
|
+
# Returns a new page object, correctly initialized using the document's configuration
|
218
|
+
# options.
|
219
|
+
def new_page
|
220
|
+
media_box = config['page.default_media_box']
|
221
|
+
media_box = Page::PAPER_SIZE[media_box].dup if Page::PAPER_SIZE.key?(media_box)
|
222
|
+
document.add(Type: :Page, MediaBox: media_box)
|
223
|
+
end
|
224
|
+
|
225
|
+
# Ensures that the /Count and /Parent fields of the whole page tree are set up correctly and
|
226
|
+
# that there is at least one page node. This is therefore only done for the root node of the
|
227
|
+
# page tree!
|
228
|
+
def perform_validation
|
229
|
+
super
|
230
|
+
return if key?(:Parent)
|
231
|
+
|
232
|
+
validate_node = lambda do |node|
|
233
|
+
count = 0
|
234
|
+
node[:Kids].reject! do |kid|
|
235
|
+
kid = document.deref(kid)
|
236
|
+
if !kid.kind_of?(HexaPDF::Object) || kid.null? ||
|
237
|
+
(kid.type != :Page && kid.type != :Pages)
|
238
|
+
yield("Invalid object in page tree node", true)
|
239
|
+
next true
|
240
|
+
elsif kid.type == :Page
|
241
|
+
count += 1
|
242
|
+
else
|
243
|
+
count += validate_node.call(kid)
|
244
|
+
end
|
245
|
+
if kid[:Parent] != node
|
246
|
+
yield("Field Parent of page tree node (#{kid.oid},#{kid.gen}) is invalid", true)
|
247
|
+
kid[:Parent] = node
|
248
|
+
end
|
249
|
+
false
|
250
|
+
end
|
251
|
+
if node[:Count] != count
|
252
|
+
yield("Field Count of page tree node (#{node.oid},#{node.gen}) is invalid", true)
|
253
|
+
node[:Count] = count
|
254
|
+
end
|
255
|
+
count
|
256
|
+
end
|
257
|
+
|
258
|
+
validate_node.call(self)
|
259
|
+
|
260
|
+
if self[:Count] == 0
|
261
|
+
yield("A PDF document needs at least one page", true)
|
262
|
+
add_page.validate {|msg, correctable| yield(msg, correctable)}
|
263
|
+
end
|
264
|
+
end
|
265
|
+
|
266
|
+
end
|
267
|
+
|
268
|
+
end
|
269
|
+
end
|
@@ -0,0 +1,212 @@
|
|
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/configuration'
|
36
|
+
require 'hexapdf/dictionary'
|
37
|
+
require 'hexapdf/content/color_space'
|
38
|
+
|
39
|
+
module HexaPDF
|
40
|
+
module Type
|
41
|
+
|
42
|
+
# Represents the resources needed by a content stream.
|
43
|
+
#
|
44
|
+
# See: PDF1.7 s7.8.3
|
45
|
+
class Resources < Dictionary
|
46
|
+
|
47
|
+
define_field :ExtGState, type: Dictionary
|
48
|
+
define_field :ColorSpace, type: Dictionary
|
49
|
+
define_field :Pattern, type: Dictionary
|
50
|
+
define_field :Shading, type: Dictionary, version: '1.3'
|
51
|
+
define_field :XObject, type: Dictionary
|
52
|
+
define_field :Font, type: Dictionary
|
53
|
+
define_field :ProcSet, type: Array
|
54
|
+
define_field :Properties, type: Dictionary, version: '1.2'
|
55
|
+
|
56
|
+
# Returns +:XXResources+.
|
57
|
+
def type
|
58
|
+
:XXResources
|
59
|
+
end
|
60
|
+
|
61
|
+
# Returns the color space stored under the given name.
|
62
|
+
#
|
63
|
+
# If the color space is not found, an error is raised.
|
64
|
+
#
|
65
|
+
# Note: The color spaces :DeviceGray, :DeviceRGB and :DeviceCMYK are returned without a
|
66
|
+
# lookup since they are fixed.
|
67
|
+
def color_space(name)
|
68
|
+
case name
|
69
|
+
when :DeviceRGB, :DeviceGray, :DeviceCMYK
|
70
|
+
GlobalConfiguration.constantize('color_space.map'.freeze, name).new
|
71
|
+
else
|
72
|
+
space_definition = self[:ColorSpace] && self[:ColorSpace][name]
|
73
|
+
if space_definition.nil?
|
74
|
+
raise HexaPDF::Error, "Color space '#{name}' not found in the resources"
|
75
|
+
elsif space_definition.kind_of?(Array)
|
76
|
+
space_definition.map! {|item| document.deref(item)}
|
77
|
+
space_family = space_definition[0]
|
78
|
+
else
|
79
|
+
space_family = space_definition
|
80
|
+
space_definition = [space_definition]
|
81
|
+
end
|
82
|
+
|
83
|
+
GlobalConfiguration.constantize('color_space.map'.freeze, space_family) do
|
84
|
+
HexaPDF::Content::ColorSpace::Universal
|
85
|
+
end.new(space_definition)
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
# Adds the color space to the resources and returns the name under which it is stored.
|
90
|
+
#
|
91
|
+
# If there already exists a color space with the same definition, it is reused. The device
|
92
|
+
# color spaces +:DeviceGray+, +:DeviceRGB+ and +:DeviceCMYK+ are never stored, their
|
93
|
+
# respective name is just returned.
|
94
|
+
def add_color_space(color_space)
|
95
|
+
family = color_space.family
|
96
|
+
return family if family == :DeviceRGB || family == :DeviceGray || family == :DeviceCMYK
|
97
|
+
|
98
|
+
definition = color_space.definition
|
99
|
+
self[:ColorSpace] = {} unless key?(:ColorSpace)
|
100
|
+
color_space_dict = self[:ColorSpace]
|
101
|
+
|
102
|
+
name, _value = color_space_dict.value.find do |_k, v|
|
103
|
+
v.map! {|item| document.deref(item)}
|
104
|
+
v == definition
|
105
|
+
end
|
106
|
+
unless name
|
107
|
+
name = create_resource_name(color_space_dict.value, 'CS')
|
108
|
+
color_space_dict[name] = definition
|
109
|
+
end
|
110
|
+
name
|
111
|
+
end
|
112
|
+
|
113
|
+
# Returns the XObject stored under the given name.
|
114
|
+
#
|
115
|
+
# If the XObject is not found, an error is raised.
|
116
|
+
def xobject(name)
|
117
|
+
object_getter(:XObject, name)
|
118
|
+
end
|
119
|
+
|
120
|
+
# Adds the XObject to the resources and returns the name under which it is stored.
|
121
|
+
#
|
122
|
+
# If there already exists a name for the given XObject, it is just returned.
|
123
|
+
def add_xobject(object)
|
124
|
+
object_setter(:XObject, 'XO'.freeze, object)
|
125
|
+
end
|
126
|
+
|
127
|
+
# Returns the graphics state parameter dictionary (see Type::GraphicsStateParameter) stored
|
128
|
+
# under the given name.
|
129
|
+
#
|
130
|
+
# If the dictionary is not found, an error is raised.
|
131
|
+
def ext_gstate(name)
|
132
|
+
object_getter(:ExtGState, name)
|
133
|
+
end
|
134
|
+
|
135
|
+
# Adds the graphics state parameter dictionary to the resources and returns the name under
|
136
|
+
# which it is stored.
|
137
|
+
#
|
138
|
+
# If there already exists a name for the given dictionary, it is just returned.
|
139
|
+
def add_ext_gstate(object)
|
140
|
+
object_setter(:ExtGState, 'GS'.freeze, object)
|
141
|
+
end
|
142
|
+
|
143
|
+
# Returns the font dictionary stored under the given name.
|
144
|
+
#
|
145
|
+
# If the dictionary is not found, an error is raised.
|
146
|
+
def font(name)
|
147
|
+
object_getter(:Font, name)
|
148
|
+
end
|
149
|
+
|
150
|
+
# Adds the font dictionary to the resources and returns the name under which it is stored.
|
151
|
+
#
|
152
|
+
# If there already exists a name for the given dictionary, it is just returned.
|
153
|
+
def add_font(object)
|
154
|
+
object_setter(:Font, 'F'.freeze, object)
|
155
|
+
end
|
156
|
+
|
157
|
+
private
|
158
|
+
|
159
|
+
# Helper method for returning an entry of a subdictionary.
|
160
|
+
def object_getter(dict_name, name)
|
161
|
+
obj = self[dict_name] && self[dict_name][name]
|
162
|
+
if obj.nil?
|
163
|
+
raise HexaPDF::Error, "No object called '#{name}' stored under /#{dict_name}"
|
164
|
+
end
|
165
|
+
obj
|
166
|
+
end
|
167
|
+
|
168
|
+
# Helper method for setting an entry of a subdictionary.
|
169
|
+
def object_setter(dict_name, prefix, object)
|
170
|
+
self[dict_name] = {} unless key?(dict_name)
|
171
|
+
dict = self[dict_name]
|
172
|
+
name, _value = dict.each.find {|_, dict_obj| dict_obj == object}
|
173
|
+
unless name
|
174
|
+
name = create_resource_name(dict.value, prefix)
|
175
|
+
dict[name] = object
|
176
|
+
end
|
177
|
+
name
|
178
|
+
end
|
179
|
+
|
180
|
+
# Returns a unique name that can be used to store a resource in the given hash.
|
181
|
+
def create_resource_name(hash, prefix)
|
182
|
+
n = hash.size + 1
|
183
|
+
while true
|
184
|
+
name = :"#{prefix}#{n}"
|
185
|
+
return name unless hash.key?(name)
|
186
|
+
n += 1
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
# Ensures that a valid procedure set is available.
|
191
|
+
def perform_validation
|
192
|
+
super
|
193
|
+
val = self[:ProcSet]
|
194
|
+
if !val
|
195
|
+
self[:ProcSet] = [:PDF, :Text, :ImageB, :ImageC, :ImageI]
|
196
|
+
else
|
197
|
+
val.reject! do |name|
|
198
|
+
case name
|
199
|
+
when :PDF, :Text, :ImageB, :ImageC, :ImageI
|
200
|
+
false
|
201
|
+
else
|
202
|
+
yield("Invalid page procedure set name /#{name}", true)
|
203
|
+
true
|
204
|
+
end
|
205
|
+
end
|
206
|
+
end
|
207
|
+
end
|
208
|
+
|
209
|
+
end
|
210
|
+
|
211
|
+
end
|
212
|
+
end
|
@@ -0,0 +1,128 @@
|
|
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/dictionary'
|
35
|
+
require 'digest/md5'
|
36
|
+
|
37
|
+
module HexaPDF
|
38
|
+
module Type
|
39
|
+
|
40
|
+
# Represents the PDF file trailer.
|
41
|
+
#
|
42
|
+
# The file trailer is the starting point for the PDF's object tree. It links to the Catalog
|
43
|
+
# (the main PDF document structure) and the Info dictionary and holds the information
|
44
|
+
# necessary for encrypting the PDF document.
|
45
|
+
#
|
46
|
+
# Since a PDF document can contain multiple revisions, each revision needs to have its own
|
47
|
+
# file trailer (see HexaPDF::Revision#trailer).
|
48
|
+
#
|
49
|
+
# When cross-reference streams are used the information that is normally stored in the file
|
50
|
+
# trailer is stored directly in the cross-reference stream dictionary. However, a
|
51
|
+
# HexaPDF::Revision object's trailer dictionary is always of this type. Only when a
|
52
|
+
# cross-reference stream is written is the trailer integrated into the stream's dictionary.
|
53
|
+
#
|
54
|
+
# See: PDF1.7 s7.5.5, s14.4
|
55
|
+
# XRefStream
|
56
|
+
class Trailer < Dictionary
|
57
|
+
|
58
|
+
define_field :Size, type: Integer, indirect: false # will be auto-set when written
|
59
|
+
define_field :Prev, type: Integer
|
60
|
+
define_field :Root, type: :Catalog, indirect: true
|
61
|
+
define_field :Encrypt, type: Dictionary
|
62
|
+
define_field :Info, type: :XXInfo, indirect: true
|
63
|
+
define_field :ID, type: Array
|
64
|
+
define_field :XRefStm, type: Integer, version: '1.5'
|
65
|
+
|
66
|
+
# Returns +:XXTrailer+.
|
67
|
+
def type
|
68
|
+
:XXTrailer
|
69
|
+
end
|
70
|
+
|
71
|
+
# Returns the document's Catalog (see Type::Catalog), creating it if needed.
|
72
|
+
def catalog
|
73
|
+
self[:Root] ||= document.add(Type: :Catalog)
|
74
|
+
end
|
75
|
+
|
76
|
+
# Returns the document's information dictionary (see Type::Info), creating it if needed.
|
77
|
+
def info
|
78
|
+
self[:Info] ||= document.add({}, type: :XXInfo)
|
79
|
+
end
|
80
|
+
|
81
|
+
# Sets the /ID field to an array of two copies of a random string and returns this array.
|
82
|
+
#
|
83
|
+
# See: PDF1.7 14.4
|
84
|
+
def set_random_id
|
85
|
+
value[:ID] = [Digest::MD5.digest(rand.to_s)] * 2
|
86
|
+
end
|
87
|
+
|
88
|
+
# Updates the second part of the /ID field (the first part should always be the same for a
|
89
|
+
# PDF file, the second part should change with each write).
|
90
|
+
def update_id
|
91
|
+
if !value[:ID]
|
92
|
+
set_random_id
|
93
|
+
else
|
94
|
+
value[:ID][1] = Digest::MD5.digest(rand.to_s)
|
95
|
+
end
|
96
|
+
end
|
97
|
+
|
98
|
+
private
|
99
|
+
|
100
|
+
# Validates the trailer.
|
101
|
+
def perform_validation
|
102
|
+
super
|
103
|
+
unless value[:ID]
|
104
|
+
msg = if value[:Encrypt]
|
105
|
+
"ID field is required when an Encrypt dictionary is present"
|
106
|
+
else
|
107
|
+
"ID field should always be set"
|
108
|
+
end
|
109
|
+
yield(msg, true)
|
110
|
+
set_random_id
|
111
|
+
end
|
112
|
+
|
113
|
+
unless value[:Root]
|
114
|
+
yield("A PDF document must have a Catalog dictionary", true)
|
115
|
+
value[:Root] = document.add(Type: :Catalog)
|
116
|
+
value[:Root].validate {|message, correctable| yield(message, correctable)}
|
117
|
+
end
|
118
|
+
|
119
|
+
if value[:Encrypt] && (!document.security_handler ||
|
120
|
+
!document.security_handler.encryption_key_valid?)
|
121
|
+
yield("Encryption key doesn't match encryption dictionary", false)
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
end
|
126
|
+
|
127
|
+
end
|
128
|
+
end
|
@@ -0,0 +1,73 @@
|
|
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/dictionary'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Type
|
38
|
+
|
39
|
+
# Represents the PDF's viewer preferences dictionary which defines how a document should be
|
40
|
+
# presented on screen or in print.
|
41
|
+
#
|
42
|
+
# This dictionary is linked via the /ViewerPreferences entry from the Type::Catalog.
|
43
|
+
#
|
44
|
+
# See: PDF1.7 s12.2, Catalog
|
45
|
+
class ViewerPreferences < Dictionary
|
46
|
+
|
47
|
+
define_field :HideToolbar, type: Boolean, default: false
|
48
|
+
define_field :HideMenubar, type: Boolean, default: false
|
49
|
+
define_field :HideWindowUI, type: Boolean, default: false
|
50
|
+
define_field :FitWindow, type: Boolean, default: false
|
51
|
+
define_field :CenterWindow, type: Boolean, default: false
|
52
|
+
define_field :DisplayDocTitle, type: Boolean, default: false, version: '1.4'
|
53
|
+
define_field :NonFullScreenPageMode, type: Symbol, default: :UseNone
|
54
|
+
define_field :Direction, type: Symbol, default: :L2R, version: '1.3'
|
55
|
+
define_field :ViewArea, type: Symbol, default: :CropBox, version: '1.4'
|
56
|
+
define_field :ViewClip, type: Symbol, default: :CropBox, version: '1.4'
|
57
|
+
define_field :PrintArea, type: Symbol, default: :CropBox, version: '1.4'
|
58
|
+
define_field :PrintClip, type: Symbol, default: :CropBox, version: '1.4'
|
59
|
+
define_field :PrintScaling, type: Symbol, default: :AppDefault, version: '1.6'
|
60
|
+
define_field :Duplex, type: Symbol, version: '1.7'
|
61
|
+
define_field :PickTrayByPDFSize, type: Boolean, version: '1.7'
|
62
|
+
define_field :PrintPageRange, type: Array, version: '1.7'
|
63
|
+
define_field :NumCopies, type: Integer, version: '1.7'
|
64
|
+
|
65
|
+
# Returns +:XXViewerPreferences+.
|
66
|
+
def type
|
67
|
+
:XXViewerPreferences
|
68
|
+
end
|
69
|
+
|
70
|
+
end
|
71
|
+
|
72
|
+
end
|
73
|
+
end
|