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,89 @@
|
|
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/document'
|
36
|
+
|
37
|
+
module HexaPDF
|
38
|
+
module ImageLoader
|
39
|
+
|
40
|
+
# This module is used for loading the first page of a PDF file.
|
41
|
+
#
|
42
|
+
# Loaded PDF graphics are represented by form XObjects instead of image XObjects. However, the
|
43
|
+
# image/xobject drawing methods of HexaPDF::Content::Canvas know how to handle them correctly so
|
44
|
+
# that this doesn't matter from a user's point of view.
|
45
|
+
#
|
46
|
+
# See: PDF1.7 s8.10
|
47
|
+
module PDF
|
48
|
+
|
49
|
+
# The magic marker that tells us if the file/IO contains an PDF file.
|
50
|
+
MAGIC_FILE_MARKER = "%PDF-".b
|
51
|
+
|
52
|
+
# :call-seq:
|
53
|
+
# PDF.handles?(filename) -> true or false
|
54
|
+
# PDF.handles?(io) -> true or false
|
55
|
+
#
|
56
|
+
# Returns +true+ if the given file or IO stream can be handled, ie. if it contains an image
|
57
|
+
# in JPEG format.
|
58
|
+
def self.handles?(file_or_io)
|
59
|
+
if file_or_io.kind_of?(String)
|
60
|
+
File.read(file_or_io, 5, mode: 'rb') == MAGIC_FILE_MARKER
|
61
|
+
else
|
62
|
+
file_or_io.rewind
|
63
|
+
file_or_io.read(5) == MAGIC_FILE_MARKER
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
# :call-seq:
|
68
|
+
# PDF.load(document, filename) -> form_obj
|
69
|
+
# PDF.load(document, io) -> form_obj
|
70
|
+
#
|
71
|
+
# Creates a PDF form XObject from the PDF file or IO stream.
|
72
|
+
#
|
73
|
+
# See: DefaultConfiguration for the meaning of 'image_loader.pdf.use_stringio'.
|
74
|
+
def self.load(document, file_or_io)
|
75
|
+
idoc = if file_or_io.kind_of?(String) && document.config['image_loader.pdf.use_stringio']
|
76
|
+
HexaPDF::Document.open(file_or_io)
|
77
|
+
elsif file_or_io.kind_of?(String)
|
78
|
+
HexaPDF::Document.new(io: File.open(file_or_io, 'rb'))
|
79
|
+
else
|
80
|
+
HexaPDF::Document.new(io: file_or_io)
|
81
|
+
end
|
82
|
+
form = idoc.pages.page(0).to_form_xobject
|
83
|
+
document.add(document.import(form))
|
84
|
+
end
|
85
|
+
|
86
|
+
end
|
87
|
+
|
88
|
+
end
|
89
|
+
end
|
@@ -0,0 +1,410 @@
|
|
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/content/graphics_state'
|
36
|
+
require 'hexapdf/utils/bit_stream'
|
37
|
+
|
38
|
+
module HexaPDF
|
39
|
+
module ImageLoader
|
40
|
+
|
41
|
+
# This class is used for loading images in the PNG format from files or IO streams.
|
42
|
+
#
|
43
|
+
# It can handle all five types of PNG images: greyscale w/wo alpha, truecolor w/wo alpha and
|
44
|
+
# indexed-color. Furthermore, it recognizes the gAMA, cHRM, sRGB and tRNS chunks and handles
|
45
|
+
# them appropriately. However, Adam7 interlaced images are not supported!
|
46
|
+
#
|
47
|
+
# Note that greyscale, truecolor and indexed-color images with alpha need to be decoded to get
|
48
|
+
# the alpha channel which takes time.
|
49
|
+
#
|
50
|
+
# All PNG specification section references are in reference to http://www.w3.org/TR/PNG/.
|
51
|
+
#
|
52
|
+
# See: PDF1.7 s7.4.4., s8.9
|
53
|
+
class PNG
|
54
|
+
|
55
|
+
# The magic marker that tells us if the file/IO contains an image in PNG format.
|
56
|
+
#
|
57
|
+
# See: PNG s5.2
|
58
|
+
MAGIC_FILE_MARKER = "\x89PNG\r\n\x1A\n".b
|
59
|
+
|
60
|
+
# The color type for PNG greyscale images without alpha, see PNG s11.2.2
|
61
|
+
GREYSCALE = 0
|
62
|
+
|
63
|
+
# The color type for PNG truecolor images without alpha, see PNG s11.2.2
|
64
|
+
TRUECOLOR = 2
|
65
|
+
|
66
|
+
# The color type for PNG indexed images with/without alpha, see PNG s11.2.2
|
67
|
+
INDEXED = 3
|
68
|
+
|
69
|
+
# The color type for PNG greyscale images with alpha, see PNG s11.2.2
|
70
|
+
GREYSCALE_ALPHA = 4
|
71
|
+
|
72
|
+
# The color type for PNG truecolor images with alpha, see PNG s11.2.2
|
73
|
+
TRUECOLOR_ALPHA = 6
|
74
|
+
|
75
|
+
# Mapping from sRGB chunk rendering intent byte to PDF rendering intent name.
|
76
|
+
RENDERING_INTENT_MAP = {
|
77
|
+
0 => Content::RenderingIntent::PERCEPTUAL,
|
78
|
+
1 => Content::RenderingIntent::RELATIVE_COLORIMETRIC,
|
79
|
+
2 => Content::RenderingIntent::SATURATION,
|
80
|
+
3 => Content::RenderingIntent::ABSOLUTE_COLORIMETRIC,
|
81
|
+
}
|
82
|
+
|
83
|
+
# The primary chromaticities and white point used by the sRGB specification.
|
84
|
+
SRGB_CHRM = [0.3127, 0.329, 0.64, 0.33, 0.3, 0.6, 0.15, 0.06]
|
85
|
+
|
86
|
+
# :call-seq:
|
87
|
+
# PNG.handles?(filename) -> true or false
|
88
|
+
# PNG.handles?(io) -> true or false
|
89
|
+
#
|
90
|
+
# Returns +true+ if the given file or IO stream can be handled, ie. if it contains an image
|
91
|
+
# in PNG format.
|
92
|
+
def self.handles?(file_or_io)
|
93
|
+
if file_or_io.kind_of?(String)
|
94
|
+
File.read(file_or_io, 8, mode: 'rb') == MAGIC_FILE_MARKER
|
95
|
+
else
|
96
|
+
file_or_io.rewind
|
97
|
+
file_or_io.read(8) == MAGIC_FILE_MARKER
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
# :call-seq:
|
102
|
+
# PNG.load(document, filename) -> image_obj
|
103
|
+
# PNG.load(document, io) -> image_obj
|
104
|
+
#
|
105
|
+
# Creates a PDF image object from the PNG file or IO stream.
|
106
|
+
def self.load(document, file_or_io)
|
107
|
+
new(document, file_or_io).load
|
108
|
+
end
|
109
|
+
|
110
|
+
def initialize(document, io) #:nodoc:
|
111
|
+
@document = document
|
112
|
+
@io = io
|
113
|
+
|
114
|
+
@color_type = nil
|
115
|
+
@intent = nil
|
116
|
+
@chrm = nil
|
117
|
+
@gamma = nil
|
118
|
+
end
|
119
|
+
|
120
|
+
def load #:nodoc:
|
121
|
+
with_io do |io|
|
122
|
+
io.seek(8, IO::SEEK_SET)
|
123
|
+
|
124
|
+
dict = {
|
125
|
+
Type: :XObject,
|
126
|
+
Subtype: :Image,
|
127
|
+
}
|
128
|
+
|
129
|
+
while true
|
130
|
+
length, type = io.read(8).unpack('Na4') # PNG s5.3
|
131
|
+
|
132
|
+
case type
|
133
|
+
when 'IDAT' # PNG s11.2.4
|
134
|
+
idat_offset = io.pos - 8
|
135
|
+
break
|
136
|
+
when 'IHDR' # PNG s11.2.2
|
137
|
+
values = io.read(length).unpack('NNC5')
|
138
|
+
dict[:Width] = values[0]
|
139
|
+
dict[:Height] = values[1]
|
140
|
+
dict[:BitsPerComponent] = values[2]
|
141
|
+
@color_type = values[3]
|
142
|
+
|
143
|
+
if values[4] != 0
|
144
|
+
raise HexaPDF::Error, "Unsupported PNG compression method"
|
145
|
+
elsif values[5] != 0
|
146
|
+
raise HexaPDF::Error, "Unsupported PNG filter method"
|
147
|
+
elsif values[6] != 0
|
148
|
+
raise HexaPDF::Error, "Unsupported PNG interlace method"
|
149
|
+
end
|
150
|
+
when 'PLTE' # PNG s11.2.3
|
151
|
+
if @color_type == INDEXED
|
152
|
+
palette = io.read(length)
|
153
|
+
hival = (palette.size / 3) - 1
|
154
|
+
if dict[:BitsPerComponent] == 8
|
155
|
+
palette = @document.add({Filter: :FlateDecode}, stream: palette)
|
156
|
+
end
|
157
|
+
dict[:ColorSpace] = [:Indexed, color_space, hival, palette]
|
158
|
+
else
|
159
|
+
io.seek(length, IO::SEEK_CUR)
|
160
|
+
end
|
161
|
+
when 'tRNS' # PNG s11.3.2
|
162
|
+
if @color_type == INDEXED
|
163
|
+
trns = io.read(length).unpack('C*')
|
164
|
+
elsif @color_type == TRUECOLOR || @color_type == GREYSCALE
|
165
|
+
dict[:Mask] = io.read(length).unpack('n*').map {|val| [val, val]}.flatten
|
166
|
+
else
|
167
|
+
io.seek(length, IO::SEEK_CUR)
|
168
|
+
end
|
169
|
+
when 'sRGB' # PNG s11.3.3.5
|
170
|
+
@intent = io.read(length).unpack('C').first
|
171
|
+
dict[:Intent] = RENDERING_INTENT_MAP[@intent]
|
172
|
+
@chrm = SRGB_CHRM
|
173
|
+
@gamma = 2.2
|
174
|
+
when 'gAMA' # PNG s11.3.3.2
|
175
|
+
gamma = 100_000.0 / io.read(length).unpack('N').first
|
176
|
+
unless @intent || gamma == 1.0 # sRGB trumps gAMA
|
177
|
+
@gamma = gamma
|
178
|
+
@chrm ||= SRGB_CHRM # don't overwrite data from a cHRM chunk
|
179
|
+
end
|
180
|
+
when 'cHRM' # PNG s11.3.3.1
|
181
|
+
chrm = io.read(length)
|
182
|
+
@chrm = chrm.unpack('N8').map {|v| v / 100_000.0} unless @intent # sRGB trumps cHRM
|
183
|
+
else
|
184
|
+
io.seek(length, IO::SEEK_CUR)
|
185
|
+
end
|
186
|
+
|
187
|
+
io.seek(4, IO::SEEK_CUR) # don't check the CRC
|
188
|
+
end
|
189
|
+
|
190
|
+
dict[:ColorSpace] ||= color_space
|
191
|
+
|
192
|
+
decode_parms = {
|
193
|
+
Predictor: 15,
|
194
|
+
Colors: @color_type == TRUECOLOR || @color_type == TRUECOLOR_ALPHA ? 3 : 1,
|
195
|
+
BitsPerComponent: dict[:BitsPerComponent],
|
196
|
+
Columns: dict[:Width],
|
197
|
+
}
|
198
|
+
|
199
|
+
if @color_type == TRUECOLOR_ALPHA || @color_type == GREYSCALE_ALPHA
|
200
|
+
image_data, mask_data = separate_alpha_channel(idat_offset, decode_parms)
|
201
|
+
add_smask_image(dict, mask_data)
|
202
|
+
stream = HexaPDF::StreamData.new(lambda { image_data },
|
203
|
+
filter: :FlateDecode,
|
204
|
+
decode_parms: decode_parms)
|
205
|
+
else
|
206
|
+
if @color_type == INDEXED && trns
|
207
|
+
mask_data = alpha_mask_for_indexed_image(idat_offset, decode_parms, trns)
|
208
|
+
add_smask_image(dict, mask_data, from_indexed: true)
|
209
|
+
end
|
210
|
+
stream = HexaPDF::StreamData.new(image_data_proc(idat_offset),
|
211
|
+
filter: :FlateDecode,
|
212
|
+
decode_parms: decode_parms)
|
213
|
+
end
|
214
|
+
|
215
|
+
obj = @document.add(dict, stream: stream)
|
216
|
+
obj.set_filter(:FlateDecode, decode_parms)
|
217
|
+
obj
|
218
|
+
end
|
219
|
+
end
|
220
|
+
|
221
|
+
private
|
222
|
+
|
223
|
+
# Yields the IO object for reading the PNG image.
|
224
|
+
#
|
225
|
+
# Automatically handles files and IO streams.
|
226
|
+
def with_io
|
227
|
+
io = (@io.kind_of?(String) ? File.new(@io, 'rb') : @io)
|
228
|
+
yield(io)
|
229
|
+
ensure
|
230
|
+
io.close if @io.kind_of?(String)
|
231
|
+
end
|
232
|
+
|
233
|
+
# Returns the PDF color space definition that should be used with the PDF image of the PNG
|
234
|
+
# file.
|
235
|
+
#
|
236
|
+
# In the case of an indexed PNG image, this returns the definition for the color space
|
237
|
+
# underlying the palette.
|
238
|
+
def color_space
|
239
|
+
if @color_type == GREYSCALE || @color_type == GREYSCALE_ALPHA
|
240
|
+
if @gamma
|
241
|
+
[:CalGray, {WhitePoint: [1.0, 1.0, 1.0], Gamma: @gamma}]
|
242
|
+
else
|
243
|
+
:DeviceGray
|
244
|
+
end
|
245
|
+
elsif @gamma || @chrm
|
246
|
+
dict = @chrm ? calrgb_definition_from_chrm(*@chrm) : {}
|
247
|
+
if @gamma
|
248
|
+
dict[:Gamma] = [@gamma, @gamma, @gamma]
|
249
|
+
dict[:WhitePoint] ||= [1.0, 1.0, 1.0]
|
250
|
+
end
|
251
|
+
[:CalRGB, dict]
|
252
|
+
else
|
253
|
+
:DeviceRGB
|
254
|
+
end
|
255
|
+
end
|
256
|
+
|
257
|
+
# Returns a hash for a CalRGB color space definition using the x,y chromaticity coordinates
|
258
|
+
# of the white point and the red, green and blue primaries.
|
259
|
+
#
|
260
|
+
# See: PDF1.7 s8.6.5.3
|
261
|
+
def calrgb_definition_from_chrm(xw, yw, xr, yr, xg, yg, xb, yb)
|
262
|
+
z = yw * ((xg - xb) * yr - (xr - xb) * yg + (xr - xg) * yb)
|
263
|
+
|
264
|
+
mya = yr * ((xg - xb) * yw - (xw - xb) * yg + (xw - xg) * yb) / z
|
265
|
+
mxa = mya * xr / yr
|
266
|
+
mza = mya * ((1 - xr) / yr - 1)
|
267
|
+
|
268
|
+
myb = - (yg * ((xr - xb) * yw - (xw - xb) * yr + (xw - xr) * yb)) / z
|
269
|
+
mxb = myb * xg / yg
|
270
|
+
mzb = myb * ((1 - xg) / yg - 1)
|
271
|
+
myc = yb * ((xr - xg) * yw - (xw - xg) * yr + (xw - xr) * yg) / z
|
272
|
+
mxc = myc * xb / yb
|
273
|
+
mzc = myc * ((1 - xb) / yb - 1)
|
274
|
+
|
275
|
+
mxw = mxa + mxb + mxc
|
276
|
+
myw = 1.0 # mya + myb + myc
|
277
|
+
mzw = mza + mzb + mzc
|
278
|
+
|
279
|
+
{WhitePoint: [mxw, myw, mzw], Matrix: [mxa, mya, mza, mxb, myb, mzb, mxc, myc, mzc]}
|
280
|
+
end
|
281
|
+
|
282
|
+
# Adds a source mask image to the image described by +dict+ using +mask_data+ as the source
|
283
|
+
# data.
|
284
|
+
#
|
285
|
+
# If the optional argument +from_indexed+ is +true+, it is assumed that the +mask_data+ was
|
286
|
+
# created from an indexed PNG and is not deflate encoded.
|
287
|
+
def add_smask_image(dict, mask_data, from_indexed: false)
|
288
|
+
decode_parms = {
|
289
|
+
Predictor: 15,
|
290
|
+
Colors: 1,
|
291
|
+
BitsPerComponent: (from_indexed ? 8 : dict[:BitsPerComponent]),
|
292
|
+
Columns: dict[:Width],
|
293
|
+
}
|
294
|
+
stream_opts = (from_indexed ? {} : {filter: :FlateDecode, decode_parms: decode_parms})
|
295
|
+
stream = HexaPDF::StreamData.new(lambda { mask_data }, stream_opts)
|
296
|
+
|
297
|
+
smask_dict = {
|
298
|
+
Type: :XObject,
|
299
|
+
Subtype: :Image,
|
300
|
+
Width: dict[:Width],
|
301
|
+
Height: dict[:Height],
|
302
|
+
ColorSpace: :DeviceGray,
|
303
|
+
BitsPerComponent: (from_indexed ? 8 : dict[:BitsPerComponent]),
|
304
|
+
}
|
305
|
+
smask = @document.add(smask_dict, stream: stream)
|
306
|
+
smask.set_filter(:FlateDecode, decode_parms)
|
307
|
+
dict[:SMask] = smask
|
308
|
+
end
|
309
|
+
|
310
|
+
# Returns a Proc object that can be used with a StreamData object to read the image data.
|
311
|
+
#
|
312
|
+
# This method is efficient because it doesn't need to uncompress or filter the image data
|
313
|
+
# but it only works for PNG images without embedded alpha channel data.
|
314
|
+
def image_data_proc(offset)
|
315
|
+
lambda do
|
316
|
+
with_io do |io|
|
317
|
+
io.seek(offset, IO::SEEK_SET)
|
318
|
+
|
319
|
+
while true
|
320
|
+
length, type = io.read(8).unpack('Na4') # PNG s5.3
|
321
|
+
break if type != 'IDAT'
|
322
|
+
|
323
|
+
chunk_size = @document.config['io.chunk_size'.freeze]
|
324
|
+
while length > 0
|
325
|
+
chunk_size = length if chunk_size > length
|
326
|
+
Fiber.yield(io.read(chunk_size))
|
327
|
+
length -= chunk_size
|
328
|
+
end
|
329
|
+
io.seek(4, IO::SEEK_CUR)
|
330
|
+
end
|
331
|
+
end
|
332
|
+
|
333
|
+
nil
|
334
|
+
end
|
335
|
+
end
|
336
|
+
|
337
|
+
# Separates the color data from the alpha data and returns an array containing the image and
|
338
|
+
# alpha data, both deflate encoded with predictor.
|
339
|
+
#
|
340
|
+
# Since we need to decompress the PNG chunks and extract the color/alpha bytes this method
|
341
|
+
# is not very fast but gets the job done as fast as possible in plain Ruby.
|
342
|
+
def separate_alpha_channel(offset, decode_parms)
|
343
|
+
bytes_per_colors = (decode_parms[:BitsPerComponent] * decode_parms[:Colors] + 7) / 8
|
344
|
+
bytes_per_alpha = (decode_parms[:BitsPerComponent] + 7) / 8
|
345
|
+
bytes_per_row = (decode_parms[:Columns] * decode_parms[:BitsPerComponent] *
|
346
|
+
(decode_parms[:Colors] + 1) + 7) / 8 + 1
|
347
|
+
image_data = ''.b
|
348
|
+
mask_data = ''.b
|
349
|
+
|
350
|
+
flate_decode = GlobalConfiguration.constantize('filter.map', :FlateDecode)
|
351
|
+
source = flate_decode.decoder(Fiber.new(&image_data_proc(offset)))
|
352
|
+
|
353
|
+
data = ''.b
|
354
|
+
while source.alive? && (new_data = source.resume)
|
355
|
+
data << new_data
|
356
|
+
while data.length >= bytes_per_row
|
357
|
+
i = 1
|
358
|
+
image_data << data.getbyte(0)
|
359
|
+
mask_data << data.getbyte(0)
|
360
|
+
while i < bytes_per_row
|
361
|
+
image_data << data.byteslice(i, bytes_per_colors)
|
362
|
+
i += bytes_per_colors
|
363
|
+
mask_data << data.byteslice(i, bytes_per_alpha)
|
364
|
+
i += bytes_per_alpha
|
365
|
+
end
|
366
|
+
data[0, bytes_per_row] = ''.freeze
|
367
|
+
end
|
368
|
+
end
|
369
|
+
|
370
|
+
image_data = Filter.string_from_source(flate_decode.encoder(Fiber.new { image_data }))
|
371
|
+
mask_data = Filter.string_from_source(flate_decode.encoder(Fiber.new { mask_data }))
|
372
|
+
|
373
|
+
[image_data, mask_data]
|
374
|
+
end
|
375
|
+
|
376
|
+
# Creates the alpha mask source data for an indexed PNG with alpha values.
|
377
|
+
#
|
378
|
+
# The returned data is *not* deflate encoded!
|
379
|
+
def alpha_mask_for_indexed_image(offset, decode_parms, trns)
|
380
|
+
width = decode_parms[:Columns]
|
381
|
+
bpc = decode_parms[:BitsPerComponent]
|
382
|
+
bytes_per_row = (width * bpc + 7) / 8 + 1
|
383
|
+
|
384
|
+
flate_decode = GlobalConfiguration.constantize('filter.map', :FlateDecode)
|
385
|
+
source = flate_decode.decoder(Fiber.new(&image_data_proc(offset)))
|
386
|
+
|
387
|
+
mask_data = ''.b
|
388
|
+
stream = HexaPDF::Utils::BitStreamReader.new
|
389
|
+
while source.alive? && (data = source.resume)
|
390
|
+
stream.append_data(data)
|
391
|
+
|
392
|
+
while stream.remaining_bits / 8 >= bytes_per_row
|
393
|
+
stream.read(8) # read filter byte
|
394
|
+
i = 0
|
395
|
+
while i < width
|
396
|
+
index = stream.read(bpc)
|
397
|
+
mask_data << (trns[index] || 255)
|
398
|
+
i += 1
|
399
|
+
end
|
400
|
+
stream.read(8 - ((width * bpc) % 8)) if bpc != 8 # read remaining fill bits
|
401
|
+
end
|
402
|
+
end
|
403
|
+
|
404
|
+
mask_data
|
405
|
+
end
|
406
|
+
|
407
|
+
end
|
408
|
+
|
409
|
+
end
|
410
|
+
end
|
@@ -0,0 +1,68 @@
|
|
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
|
+
module HexaPDF
|
35
|
+
|
36
|
+
# == Overview
|
37
|
+
#
|
38
|
+
# An *image loader* is used for loading an image and creating a suitable PDF object. Since some
|
39
|
+
# image information needs to be present in the PDF object itself (like height and width) the
|
40
|
+
# loader needs to parse the image to get the needed data.
|
41
|
+
#
|
42
|
+
#
|
43
|
+
# == Implementation of an Image Loader
|
44
|
+
#
|
45
|
+
# Each image loader is a (stateless) object (normally a module) that responds to two methods:
|
46
|
+
#
|
47
|
+
# handles?(file_or_io)::
|
48
|
+
# Should return +true+ if the given file or IO stream can be handled by the loader, i.e. if
|
49
|
+
# the content contains a suitable image.
|
50
|
+
#
|
51
|
+
# load(document, file_or_io)::
|
52
|
+
# Should add a new image XObject to the document that uses the file or IO stream as source
|
53
|
+
# and return this newly created object. This method is only invoked if #handles? has
|
54
|
+
# returned +true+ for the same +file_or_io+ object.
|
55
|
+
#
|
56
|
+
# The image XObject may use any implemented filter. For example, an image loader for JPEG files
|
57
|
+
# would typically use the DCTDecode filter instead of decoding the image itself.
|
58
|
+
#
|
59
|
+
# See: PDF1.7 s8.9
|
60
|
+
module ImageLoader
|
61
|
+
|
62
|
+
autoload(:JPEG, 'hexapdf/image_loader/jpeg')
|
63
|
+
autoload(:PNG, 'hexapdf/image_loader/png')
|
64
|
+
autoload(:PDF, 'hexapdf/image_loader/pdf')
|
65
|
+
|
66
|
+
end
|
67
|
+
|
68
|
+
end
|