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,208 @@
|
|
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/utils/math_helpers'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Content
|
38
|
+
module GraphicObject
|
39
|
+
|
40
|
+
# This class describes an elliptical arc in endpoint parameterization. It allows one to
|
41
|
+
# generate an arc from the current point to a given point, similar to Content::Canvas#line_to.
|
42
|
+
#
|
43
|
+
# See: GraphicObject::Arc, ARC - https://www.w3.org/TR/SVG/implnote.html#ArcImplementationNotes
|
44
|
+
class EndpointArc
|
45
|
+
|
46
|
+
EPSILON = 1e-10
|
47
|
+
|
48
|
+
include Utils::MathHelpers
|
49
|
+
|
50
|
+
# Creates and configures a new endpoint arc object.
|
51
|
+
#
|
52
|
+
# See #configure for the allowed keyword arguments.
|
53
|
+
def self.configure(**kwargs)
|
54
|
+
new.configure(kwargs)
|
55
|
+
end
|
56
|
+
|
57
|
+
# x-coordinate of endpoint
|
58
|
+
attr_reader :x
|
59
|
+
|
60
|
+
# y-coordinate of endpoint
|
61
|
+
attr_reader :y
|
62
|
+
|
63
|
+
# Length of semi-major axis
|
64
|
+
attr_reader :a
|
65
|
+
|
66
|
+
# Length of semi-minor axis
|
67
|
+
attr_reader :b
|
68
|
+
|
69
|
+
# Inclination in degrees of semi-major axis in respect to x-axis
|
70
|
+
attr_reader :inclination
|
71
|
+
|
72
|
+
# Large arc choice - if +true+ use the large arc (i.e. the one spanning more than 180
|
73
|
+
# degrees), else the small arc
|
74
|
+
attr_reader :large_arc
|
75
|
+
|
76
|
+
# Direction of arc - if +true+ in clockwise direction, else in counterclockwise direction
|
77
|
+
attr_reader :clockwise
|
78
|
+
|
79
|
+
# Creates an endpoint arc with default values x=0, y=0, a=0, b=0, inclination=0,
|
80
|
+
# large_arc=true, clockwise=false (a line to the origin).
|
81
|
+
def initialize
|
82
|
+
@x = @y = 0
|
83
|
+
@a = @b = 0
|
84
|
+
@inclination = 0
|
85
|
+
@large_arc = true
|
86
|
+
@clockwise = false
|
87
|
+
end
|
88
|
+
|
89
|
+
# Configures the endpoint arc with
|
90
|
+
#
|
91
|
+
# * endpoint (+x+, +y+),
|
92
|
+
# * semi-major axis +a+,
|
93
|
+
# * semi-minor axis +b+,
|
94
|
+
# * an inclination in respect to the x-axis of +inclination+ degrees,
|
95
|
+
# * the given large_arc flag and
|
96
|
+
# * the given clockwise flag.
|
97
|
+
#
|
98
|
+
# The +large_arc+ option determines whether the large arc, i.e. the one spanning more than
|
99
|
+
# 180 degrees, is used (+true+) or the small arc (+false+).
|
100
|
+
#
|
101
|
+
# The +clockwise+ option determines if the arc is drawn in the counterclockwise direction
|
102
|
+
# (+false+) or in the clockwise direction (+true+).
|
103
|
+
#
|
104
|
+
# Any arguments not specified are not modified and retain their old value, see #initialize
|
105
|
+
# for the inital values.
|
106
|
+
#
|
107
|
+
# Returns self.
|
108
|
+
def configure(x: nil, y: nil, a: nil, b: nil, inclination: nil, large_arc: nil,
|
109
|
+
clockwise: nil)
|
110
|
+
@x = x if x
|
111
|
+
@y = y if y
|
112
|
+
@a = a.abs if a
|
113
|
+
@b = b.abs if b
|
114
|
+
@inclination = inclination % 360 if inclination
|
115
|
+
@large_arc = large_arc unless large_arc.nil?
|
116
|
+
@clockwise = clockwise unless clockwise.nil?
|
117
|
+
|
118
|
+
self
|
119
|
+
end
|
120
|
+
|
121
|
+
# Draws the arc on the given Canvas.
|
122
|
+
def draw(canvas)
|
123
|
+
x1, y1 = *canvas.current_point
|
124
|
+
|
125
|
+
# ARC F.6.2 - nothing to do if endpoint is equal to current point
|
126
|
+
return if float_equal(x1, @x) && float_equal(y1, @y)
|
127
|
+
|
128
|
+
if @a == 0 || @b == 0
|
129
|
+
# ARC F.6.2, F.6.6 - just use a line if it is not really an arc
|
130
|
+
canvas.line_to(@x, @y)
|
131
|
+
else
|
132
|
+
values = compute_arc_values(x1, y1)
|
133
|
+
arc = canvas.graphic_object(:arc, values)
|
134
|
+
arc.draw(canvas, move_to_start: false)
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|
138
|
+
private
|
139
|
+
|
140
|
+
# Compute the center parameterization from the endpoint parameterization.
|
141
|
+
#
|
142
|
+
# The argument (x1, y1) is the starting point.
|
143
|
+
#
|
144
|
+
# See: ARC F.6.5, F.6.6
|
145
|
+
def compute_arc_values(x1, y1)
|
146
|
+
x2 = @x
|
147
|
+
y2 = @y
|
148
|
+
rx = @a
|
149
|
+
ry = @b
|
150
|
+
theta = deg_to_rad(@inclination)
|
151
|
+
cos_theta = Math.cos(theta)
|
152
|
+
sin_theta = Math.sin(theta)
|
153
|
+
|
154
|
+
# F.6.5.1
|
155
|
+
x1p = (x1 - x2) / 2.0 * cos_theta + (y1 - y2) / 2.0 * sin_theta
|
156
|
+
y1p = (x1 - x2) / 2.0 * -sin_theta + (y1 - y2) / 2.0 * cos_theta
|
157
|
+
|
158
|
+
x1ps = x1p**2
|
159
|
+
y1ps = y1p**2
|
160
|
+
rxs = rx**2
|
161
|
+
rys = ry**2
|
162
|
+
|
163
|
+
# F.6.6.2
|
164
|
+
l = x1ps / rxs + y1ps / rys
|
165
|
+
if l > 1
|
166
|
+
rx *= Math.sqrt(l)
|
167
|
+
ry *= Math.sqrt(l)
|
168
|
+
rxs = rx**2
|
169
|
+
rys = ry**2
|
170
|
+
end
|
171
|
+
|
172
|
+
# F.6.5.2
|
173
|
+
sqrt = (rxs * rys - rxs * y1ps - rys * x1ps) / (rxs * y1ps + rys * x1ps)
|
174
|
+
sqrt = 0 if sqrt.abs < EPSILON
|
175
|
+
sqrt = Math.sqrt(sqrt)
|
176
|
+
sqrt *= -1 unless @large_arc == @clockwise
|
177
|
+
cxp = sqrt * rx * y1p / ry
|
178
|
+
cyp = - sqrt * ry * x1p / rx
|
179
|
+
|
180
|
+
# F.6.5.3
|
181
|
+
cx = cos_theta * cxp - sin_theta * cyp + (x1 + x2) / 2.0
|
182
|
+
cy = sin_theta * cxp + cos_theta * cyp + (y1 + y2) / 2.0
|
183
|
+
|
184
|
+
# F.6.5.5
|
185
|
+
start_angle = compute_angle_to_x_axis((x1p - cxp) / rx, (y1p - cyp) / ry)
|
186
|
+
|
187
|
+
# F.6.5.6 (modified bc we just need the end angle)
|
188
|
+
end_angle = compute_angle_to_x_axis((-x1p - cxp) / rx, (-y1p - cyp) / ry)
|
189
|
+
|
190
|
+
{cx: cx, cy: cy, a: rx, b: ry, start_angle: start_angle, end_angle: end_angle,
|
191
|
+
inclination: @inclination, clockwise: @clockwise}
|
192
|
+
end
|
193
|
+
|
194
|
+
# Compares two float numbers if they are within a certain delta.
|
195
|
+
def float_equal(a, b)
|
196
|
+
(a - b).abs < EPSILON
|
197
|
+
end
|
198
|
+
|
199
|
+
# Computes the angle in degrees between the x-axis and the vector.
|
200
|
+
def compute_angle_to_x_axis(vx, vy)
|
201
|
+
(vy < 0 ? -1 : 1) * rad_to_deg(Math.acos(vx / Math.sqrt(vx**2 + vy**2)))
|
202
|
+
end
|
203
|
+
|
204
|
+
end
|
205
|
+
|
206
|
+
end
|
207
|
+
end
|
208
|
+
end
|
@@ -0,0 +1,173 @@
|
|
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
|
+
module Content
|
36
|
+
module GraphicObject
|
37
|
+
|
38
|
+
# This graphic object represents a solid elliptical arc, i.e. an arc that has an inner and
|
39
|
+
# an outer set of a/b values.
|
40
|
+
#
|
41
|
+
# Thus it can be used to create
|
42
|
+
#
|
43
|
+
# * an (elliptical) disk (when the inner a/b are zero and the difference between start and
|
44
|
+
# end angles is greater than or equal to 360),
|
45
|
+
#
|
46
|
+
# * an (elliptical) sector (when the inner a/b are zero and the difference between start
|
47
|
+
# and end angles is less than 360),
|
48
|
+
#
|
49
|
+
# * an (elliptical) annulus (when the inner a/b are nonzero and the difference between
|
50
|
+
# start and end angles is greater than or equal to 360), and
|
51
|
+
#
|
52
|
+
# * an (elliptical) annular sector (when the inner a/b are nonzero and the difference
|
53
|
+
# between start and end angles is less than 360)
|
54
|
+
#
|
55
|
+
# See: Arc
|
56
|
+
class SolidArc
|
57
|
+
|
58
|
+
# Creates and configures a new solid arc object.
|
59
|
+
#
|
60
|
+
# See #configure for the allowed keyword arguments.
|
61
|
+
def self.configure(**kwargs)
|
62
|
+
new.configure(kwargs)
|
63
|
+
end
|
64
|
+
|
65
|
+
# x-coordinate of center point
|
66
|
+
attr_reader :cx
|
67
|
+
|
68
|
+
# y-coordinate of center point
|
69
|
+
attr_reader :cy
|
70
|
+
|
71
|
+
# Length of inner semi-major axis
|
72
|
+
attr_reader :inner_a
|
73
|
+
|
74
|
+
# Length of inner semi-minor axis
|
75
|
+
attr_reader :inner_b
|
76
|
+
|
77
|
+
# Length of outer semi-major axis
|
78
|
+
attr_reader :outer_a
|
79
|
+
|
80
|
+
# Length of outer semi-minor axis
|
81
|
+
attr_reader :outer_b
|
82
|
+
|
83
|
+
# Start angle in degrees
|
84
|
+
attr_reader :start_angle
|
85
|
+
|
86
|
+
# End angle in degrees
|
87
|
+
attr_reader :end_angle
|
88
|
+
|
89
|
+
# Inclination in degrees of semi-major axis in respect to x-axis
|
90
|
+
attr_reader :inclination
|
91
|
+
|
92
|
+
# Creates a solid arc with default values (a unit disk at the origin).
|
93
|
+
def initialize
|
94
|
+
@cx = @cy = 0
|
95
|
+
@inner_a = @inner_b = 0
|
96
|
+
@outer_a = @outer_b = 1
|
97
|
+
@start_angle = 0
|
98
|
+
@end_angle = 0
|
99
|
+
@inclination = 0
|
100
|
+
end
|
101
|
+
|
102
|
+
# Configures the solid arc with
|
103
|
+
#
|
104
|
+
# * center point (+cx+, +cy+),
|
105
|
+
# * inner semi-major axis +inner_a+,
|
106
|
+
# * inner semi-minor axis +inner_b+,
|
107
|
+
# * outer semi-major axis +outer_a+,
|
108
|
+
# * outer semi-minor axis +outer_b+,
|
109
|
+
# * start angle of +start_angle+ degrees,
|
110
|
+
# * end angle of +end_angle+ degrees and
|
111
|
+
# * an inclination in respect to the x-axis of +inclination+ degrees.
|
112
|
+
#
|
113
|
+
# Any arguments not specified are not modified and retain their old value, see #initialize
|
114
|
+
# for the inital values.
|
115
|
+
#
|
116
|
+
# Returns self.
|
117
|
+
def configure(cx: nil, cy: nil, inner_a: nil, inner_b: nil, outer_a: nil, outer_b: nil,
|
118
|
+
start_angle: nil, end_angle: nil, inclination: nil)
|
119
|
+
@cx = cx if cx
|
120
|
+
@cy = cy if cy
|
121
|
+
@inner_a = inner_a.abs if inner_a
|
122
|
+
@inner_b = inner_b.abs if inner_b
|
123
|
+
@outer_a = outer_a.abs if outer_a
|
124
|
+
@outer_b = outer_b.abs if outer_b
|
125
|
+
@start_angle = start_angle % 360 if start_angle
|
126
|
+
@end_angle = end_angle % 360 if end_angle
|
127
|
+
@inclination = inclination if inclination
|
128
|
+
|
129
|
+
self
|
130
|
+
end
|
131
|
+
|
132
|
+
# Draws the solid arc on the given Canvas.
|
133
|
+
def draw(canvas)
|
134
|
+
angle_difference = (@end_angle - @start_angle).abs
|
135
|
+
if @inner_a == 0 && @inner_b == 0
|
136
|
+
arc = canvas.graphic_object(:arc, cx: @cx, cy: @cy, a: @outer_a, b: @outer_b,
|
137
|
+
start_angle: @start_angle, end_angle: @end_angle,
|
138
|
+
inclination: @inclination, clockwise: false)
|
139
|
+
if angle_difference == 0
|
140
|
+
arc.draw(canvas)
|
141
|
+
canvas.close_subpath
|
142
|
+
else
|
143
|
+
canvas.move_to(@cx, @cy)
|
144
|
+
canvas.line_to(*arc.start_point)
|
145
|
+
arc.draw(canvas, move_to_start: false)
|
146
|
+
canvas.close_subpath
|
147
|
+
end
|
148
|
+
else
|
149
|
+
inner = canvas.graphic_object(:arc, cx: @cx, cy: @cy, a: @inner_a, b: @inner_b,
|
150
|
+
start_angle: @end_angle, end_angle: @start_angle,
|
151
|
+
inclination: @inclination, clockwise: true)
|
152
|
+
outer = canvas.graphic_object(:arc, cx: @cx, cy: @cy, a: @outer_a, b: @outer_b,
|
153
|
+
start_angle: @start_angle, end_angle: @end_angle,
|
154
|
+
inclination: @inclination, clockwise: false)
|
155
|
+
if angle_difference == 0
|
156
|
+
outer.draw(canvas)
|
157
|
+
canvas.close_subpath
|
158
|
+
inner.draw(canvas)
|
159
|
+
canvas.close_subpath
|
160
|
+
else
|
161
|
+
outer.draw(canvas)
|
162
|
+
canvas.line_to(*inner.start_point)
|
163
|
+
inner.draw(canvas, move_to_start: false)
|
164
|
+
canvas.close_subpath
|
165
|
+
end
|
166
|
+
end
|
167
|
+
end
|
168
|
+
|
169
|
+
end
|
170
|
+
|
171
|
+
end
|
172
|
+
end
|
173
|
+
end
|
@@ -0,0 +1,81 @@
|
|
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
|
+
module Content
|
36
|
+
|
37
|
+
# == Overview
|
38
|
+
#
|
39
|
+
# This module contains classes describing graphic objects that can be drawn on a Canvas.
|
40
|
+
#
|
41
|
+
# Since the PDF specification only provides the most common path creation operators, more
|
42
|
+
# complex graphic objects need more than one operator for their creation. By defining this
|
43
|
+
# graphic object interface (see below) such complex objects can be drawn in a consistent
|
44
|
+
# manner on a Canvas.
|
45
|
+
#
|
46
|
+
# A graphic object should only use the path creation methods or other graphic objects when it
|
47
|
+
# is drawn. Stroking and filling, or optionally clipping, is left to the user.
|
48
|
+
#
|
49
|
+
# The Content::Canvas class provides a Content::Canvas#draw method that can be used to draw
|
50
|
+
# complex graphic objects as well as a Content::Canvas#graphic_object method to retrieve an
|
51
|
+
# instance of a graphic object for custom use. The latter method uses graphic object factories
|
52
|
+
# that can be registered via a name using the document specific 'graphic_object.map'
|
53
|
+
# configuration option.
|
54
|
+
#
|
55
|
+
# == Implementation of a Graphic Object
|
56
|
+
#
|
57
|
+
# Graphic objects are normally implemented as classes since this automatically allows using
|
58
|
+
# the class itself as the graphic object's factory.
|
59
|
+
#
|
60
|
+
# A graphic object factory is an object that responds to #configure(**kwargs) and returns a
|
61
|
+
# configured graphic object. When the factory is implemented as a class, the #configure method
|
62
|
+
# should be a class method returning properly configured instances of the class.
|
63
|
+
#
|
64
|
+
# A graphic object itself has to respond to two methods:
|
65
|
+
#
|
66
|
+
# #configure(**kwargs)::
|
67
|
+
# This method is used for re-configuring the graphic object and it should return the
|
68
|
+
# graphic object itself, not a new object.
|
69
|
+
#
|
70
|
+
# #draw(canvas)::
|
71
|
+
# This method is used for drawing the graphic object on the given Canvas.
|
72
|
+
module GraphicObject
|
73
|
+
|
74
|
+
autoload(:Arc, 'hexapdf/content/graphic_object/arc')
|
75
|
+
autoload(:EndpointArc, 'hexapdf/content/graphic_object/endpoint_arc')
|
76
|
+
autoload(:SolidArc, 'hexapdf/content/graphic_object/solid_arc')
|
77
|
+
|
78
|
+
end
|
79
|
+
|
80
|
+
end
|
81
|
+
end
|