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,364 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2016 Thomas Leitner
|
8
|
+
#
|
9
|
+
# HexaPDF is free software: you can redistribute it and/or modify it
|
10
|
+
# under the terms of the GNU Affero General Public License version 3 as
|
11
|
+
# published by the Free Software Foundation with the addition of the
|
12
|
+
# following permission added to Section 15 as permitted in Section 7(a):
|
13
|
+
# FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
|
14
|
+
# THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
|
15
|
+
# INFRINGEMENT OF THIRD PARTY RIGHTS.
|
16
|
+
#
|
17
|
+
# HexaPDF is distributed in the hope that it will be useful, but WITHOUT
|
18
|
+
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
20
|
+
# License for more details.
|
21
|
+
#
|
22
|
+
# You should have received a copy of the GNU Affero General Public License
|
23
|
+
# along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
|
24
|
+
#
|
25
|
+
# The interactive user interfaces in modified source and object code
|
26
|
+
# versions of HexaPDF must display Appropriate Legal Notices, as required
|
27
|
+
# under Section 5 of the GNU Affero General Public License version 3.
|
28
|
+
#
|
29
|
+
# In accordance with Section 7(b) of the GNU Affero General Public
|
30
|
+
# License, a covered work must retain the producer line in every PDF that
|
31
|
+
# is created or manipulated using HexaPDF.
|
32
|
+
#++
|
33
|
+
|
34
|
+
require 'hexapdf/error'
|
35
|
+
|
36
|
+
module HexaPDF
|
37
|
+
module Content
|
38
|
+
|
39
|
+
# This module contains the color space implementations.
|
40
|
+
#
|
41
|
+
# == General Information
|
42
|
+
#
|
43
|
+
# The PDF specification defines several color spaces. Probably the most used ones are the
|
44
|
+
# device color spaces DeviceRGB, DeviceCMYK and DeviceGray. However, there are several others.
|
45
|
+
# For example, patterns are also implemented via color spaces.
|
46
|
+
#
|
47
|
+
# HexaPDF provides implementations for the most common color spaces. Additional ones can
|
48
|
+
# easily be added. After implementing one it just has to be registered on the global
|
49
|
+
# configuration object under the 'color_space.map' key.
|
50
|
+
#
|
51
|
+
# Color space implementations are currently used so that different colors can be
|
52
|
+
# distinguished and to provide better error handling.
|
53
|
+
#
|
54
|
+
#
|
55
|
+
# == Color Space Implementations
|
56
|
+
#
|
57
|
+
# A color space implementation consists of two classes: one for the color space and one for
|
58
|
+
# its colors.
|
59
|
+
#
|
60
|
+
# The class for the color space needs to respond to the following methods:
|
61
|
+
#
|
62
|
+
# #initialize(definition)::
|
63
|
+
# Creates the color space using the given array with the color space definition. The first
|
64
|
+
# item in the array is always the color space family, the other items are color space
|
65
|
+
# specific.
|
66
|
+
#
|
67
|
+
# #family::
|
68
|
+
# Returns the PDF name of the color space family this color space belongs to.
|
69
|
+
#
|
70
|
+
# #definition::
|
71
|
+
# Returns the color space definition as array.
|
72
|
+
#
|
73
|
+
# #default_color::
|
74
|
+
# Returns the default color for this color space.
|
75
|
+
#
|
76
|
+
# #color(*args)::
|
77
|
+
# Returns the color corresponding to the given arguments. The number and types of the
|
78
|
+
# arguments differ from one color space to another.
|
79
|
+
#
|
80
|
+
# The class representing a color in the color space needs to respond to the following methods:
|
81
|
+
#
|
82
|
+
# #color_space::
|
83
|
+
# Returns the associated color space object.
|
84
|
+
#
|
85
|
+
# #components::
|
86
|
+
# Returns an array of components that uniquely identifies this color within the color space.
|
87
|
+
#
|
88
|
+
# See: PDF1.7 s8.6
|
89
|
+
module ColorSpace
|
90
|
+
|
91
|
+
# This module includes utility functions that are useful for all color classes.
|
92
|
+
module ColorUtils
|
93
|
+
|
94
|
+
# Normalizes the given color value so that it is in the range from 0.0 to 1.0.
|
95
|
+
#
|
96
|
+
# The conversion is done in the following way:
|
97
|
+
#
|
98
|
+
# * If the color value is an Integer, it is converted to a float and divided by +upper+.
|
99
|
+
# * If the color value is greater than 1.0, it is set to 1.0.
|
100
|
+
# * If the color value is less than 0.0, it is set to 0.0.
|
101
|
+
def normalize_value(value, upper)
|
102
|
+
value = value.to_f / upper if value.kind_of?(Integer)
|
103
|
+
(value < 0 ? 0.0 : (value > 1 ? 1.0 : value))
|
104
|
+
end
|
105
|
+
private :normalize_value
|
106
|
+
|
107
|
+
# Compares this color to another one by looking at their associated color spaces and their
|
108
|
+
# components.
|
109
|
+
def ==(other)
|
110
|
+
other.respond_to?(:components) && other.respond_to?(:color_space) &&
|
111
|
+
components == other.components && color_space == other.color_space
|
112
|
+
end
|
113
|
+
|
114
|
+
end
|
115
|
+
|
116
|
+
# This class represents a "universal" color space that is used for all color spaces that
|
117
|
+
# aren't implemented yet.
|
118
|
+
class Universal
|
119
|
+
|
120
|
+
# The color space definition used for creating this universal color space.
|
121
|
+
attr_reader :definition
|
122
|
+
|
123
|
+
# Creates the universal color space for the given color space definition.
|
124
|
+
def initialize(definition)
|
125
|
+
@definition = definition
|
126
|
+
end
|
127
|
+
|
128
|
+
# The default universal color.
|
129
|
+
def default_color
|
130
|
+
Color.new(self)
|
131
|
+
end
|
132
|
+
|
133
|
+
# Creates a new universal color object. The number of arguments isn't restricted.
|
134
|
+
def color(*args)
|
135
|
+
Color.new(self, *args)
|
136
|
+
end
|
137
|
+
|
138
|
+
# Returns the PDF color space family this color space belongs to.
|
139
|
+
def family
|
140
|
+
@definition[0]
|
141
|
+
end
|
142
|
+
|
143
|
+
# Compares this universal color space to another one by looking at their definitions.
|
144
|
+
def ==(other)
|
145
|
+
other.kind_of?(self.class) && definition == other.definition
|
146
|
+
end
|
147
|
+
|
148
|
+
# A single color in the universal color space.
|
149
|
+
#
|
150
|
+
# This doesn't represent a real color but is a place holder for a color in a color space
|
151
|
+
# that isn't implemented yet.
|
152
|
+
class Color
|
153
|
+
|
154
|
+
include ColorUtils
|
155
|
+
|
156
|
+
# Returns the specific Universal color space used for this color.
|
157
|
+
attr_reader :color_space
|
158
|
+
|
159
|
+
# Creates a new universal color with the given components.
|
160
|
+
def initialize(color_space, *components)
|
161
|
+
@color_space = color_space
|
162
|
+
@components = components
|
163
|
+
end
|
164
|
+
|
165
|
+
# Returns the componets of the universal color, i.e. all arguments provided on
|
166
|
+
# initialization.
|
167
|
+
def components
|
168
|
+
@components
|
169
|
+
end
|
170
|
+
|
171
|
+
end
|
172
|
+
|
173
|
+
end
|
174
|
+
|
175
|
+
|
176
|
+
# The DeviceRGB color space.
|
177
|
+
class DeviceRGB
|
178
|
+
|
179
|
+
# The one (and only) DeviceRGB color space.
|
180
|
+
DEFAULT = new
|
181
|
+
|
182
|
+
# Returns the DeviceRGB color space object.
|
183
|
+
def self.new
|
184
|
+
DEFAULT
|
185
|
+
end
|
186
|
+
|
187
|
+
# Returns the default color for the DeviceRGB color space.
|
188
|
+
def default_color
|
189
|
+
Color.new(0.0, 0.0, 0.0)
|
190
|
+
end
|
191
|
+
|
192
|
+
# Returns the color object for the given red, green and blue components.
|
193
|
+
def color(r, g, b)
|
194
|
+
Color.new(r, g, b)
|
195
|
+
end
|
196
|
+
|
197
|
+
# Returns +:DeviceRGB+.
|
198
|
+
def family
|
199
|
+
:DeviceRGB
|
200
|
+
end
|
201
|
+
|
202
|
+
# A color in the DeviceRGB color space.
|
203
|
+
#
|
204
|
+
# Color values can either be integers in the range from 0 to 255 or floating point numbers
|
205
|
+
# between 0.0 and 1.0. The integer color values are automatically normalized to the
|
206
|
+
# DeviceRGB color value range of 0.0 to 1.0.
|
207
|
+
#
|
208
|
+
# See: PDF1.7 s8.6.4.3
|
209
|
+
class Color
|
210
|
+
|
211
|
+
include ColorUtils
|
212
|
+
|
213
|
+
# Initializes the color with the +r+ (red), +g+ (green) and +b+ (blue) components.
|
214
|
+
#
|
215
|
+
# Each argument has to be either an integer between 0 and 255 or a float between 0.0 and
|
216
|
+
# 1.0.
|
217
|
+
def initialize(r, g, b)
|
218
|
+
@r = normalize_value(r, 255)
|
219
|
+
@g = normalize_value(g, 255)
|
220
|
+
@b = normalize_value(b, 255)
|
221
|
+
end
|
222
|
+
|
223
|
+
# Returns the DeviceRGB color space module.
|
224
|
+
def color_space
|
225
|
+
DeviceRGB::DEFAULT
|
226
|
+
end
|
227
|
+
|
228
|
+
# Returns the RGB color as an array of normalized color values.
|
229
|
+
def components
|
230
|
+
[@r, @g, @b]
|
231
|
+
end
|
232
|
+
|
233
|
+
end
|
234
|
+
|
235
|
+
end
|
236
|
+
|
237
|
+
|
238
|
+
# The DeviceCMYK color space.
|
239
|
+
class DeviceCMYK
|
240
|
+
|
241
|
+
# The one (and only) DeviceCMYK color space.
|
242
|
+
DEFAULT = new
|
243
|
+
|
244
|
+
# Returns the DeviceCMYK color space object.
|
245
|
+
def self.new
|
246
|
+
DEFAULT
|
247
|
+
end
|
248
|
+
|
249
|
+
# Returns the default color for the DeviceCMYK color space.
|
250
|
+
def default_color
|
251
|
+
Color.new(0.0, 0.0, 0.0, 1.0)
|
252
|
+
end
|
253
|
+
|
254
|
+
# Returns the color object for the given cyan, magenta, yellow and black components.
|
255
|
+
def color(c, m, y, k)
|
256
|
+
Color.new(c, m, y, k)
|
257
|
+
end
|
258
|
+
|
259
|
+
# Returns +:DeviceCMYK+.
|
260
|
+
def family
|
261
|
+
:DeviceCMYK
|
262
|
+
end
|
263
|
+
|
264
|
+
# A color in the DeviceCMYK color space.
|
265
|
+
#
|
266
|
+
# Color values can either be integers in the range from 0 to 100 or floating point numbers
|
267
|
+
# between 0.0 and 1.0. The integer color values are automatically normalized to the
|
268
|
+
# DeviceCMYK color value range of 0.0 to 1.0.
|
269
|
+
#
|
270
|
+
# See: PDF1.7 s8.6.4.4
|
271
|
+
class Color
|
272
|
+
|
273
|
+
include ColorUtils
|
274
|
+
|
275
|
+
# Initializes the color with the +c+ (cyan), +m+ (magenta), +y+ (yellow) and +k+ (black)
|
276
|
+
# components.
|
277
|
+
#
|
278
|
+
# Each argument has to be either an integer between 0 and 100 or a float between 0.0 and
|
279
|
+
# 1.0.
|
280
|
+
def initialize(c, m, y, k)
|
281
|
+
@c = normalize_value(c, 100)
|
282
|
+
@m = normalize_value(m, 100)
|
283
|
+
@y = normalize_value(y, 100)
|
284
|
+
@k = normalize_value(k, 100)
|
285
|
+
end
|
286
|
+
|
287
|
+
# Returns the DeviceCMYK color space module.
|
288
|
+
def color_space
|
289
|
+
DeviceCMYK::DEFAULT
|
290
|
+
end
|
291
|
+
|
292
|
+
# Returns the CMYK color as an array of normalized color values.
|
293
|
+
def components
|
294
|
+
[@c, @m, @y, @k]
|
295
|
+
end
|
296
|
+
|
297
|
+
end
|
298
|
+
|
299
|
+
end
|
300
|
+
|
301
|
+
|
302
|
+
# The DeviceGray color space.
|
303
|
+
class DeviceGray
|
304
|
+
|
305
|
+
# The one (and only) DeviceGray color space.
|
306
|
+
DEFAULT = new
|
307
|
+
|
308
|
+
# Returns the DeviceGray color space object.
|
309
|
+
def self.new
|
310
|
+
DEFAULT
|
311
|
+
end
|
312
|
+
|
313
|
+
# Returns the default color for the DeviceGray color space.
|
314
|
+
def default_color
|
315
|
+
Color.new(0.0)
|
316
|
+
end
|
317
|
+
|
318
|
+
# Returns the color object for the given gray component.
|
319
|
+
def color(gray)
|
320
|
+
Color.new(gray)
|
321
|
+
end
|
322
|
+
|
323
|
+
# Returns +:DeviceGray+.
|
324
|
+
def family
|
325
|
+
:DeviceGray
|
326
|
+
end
|
327
|
+
|
328
|
+
# A color in the DeviceGray color space.
|
329
|
+
#
|
330
|
+
# Color values can either be integers in the range from 0 to 255 or floating point numbers
|
331
|
+
# between 0.0 and 1.0. The integer color values are automatically normalized to the
|
332
|
+
# DeviceGray color value range of 0.0 to 1.0.
|
333
|
+
#
|
334
|
+
# See: PDF1.7 s8.6.4.2
|
335
|
+
class Color
|
336
|
+
|
337
|
+
include ColorUtils
|
338
|
+
|
339
|
+
# Initializes the color with the +gray+ component.
|
340
|
+
#
|
341
|
+
# The argument +gray+ has to be either an integer between 0 and 255 or a float between
|
342
|
+
# 0.0 and 1.0.
|
343
|
+
def initialize(gray)
|
344
|
+
@gray = normalize_value(gray, 255)
|
345
|
+
end
|
346
|
+
|
347
|
+
# Returns the DeviceGray color space module.
|
348
|
+
def color_space
|
349
|
+
DeviceGray::DEFAULT
|
350
|
+
end
|
351
|
+
|
352
|
+
# Returns the normalized gray value as an array.
|
353
|
+
def components
|
354
|
+
[@gray]
|
355
|
+
end
|
356
|
+
|
357
|
+
end
|
358
|
+
|
359
|
+
end
|
360
|
+
|
361
|
+
end
|
362
|
+
|
363
|
+
end
|
364
|
+
end
|
@@ -0,0 +1,267 @@
|
|
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 in center parameterization arc that is approximated using
|
41
|
+
# Bezier curves. It can be used to draw circles, circular arcs, ellipses and elliptical arcs,
|
42
|
+
# all either in clockwise or counterclockwise direction and optionally inclined in respect to
|
43
|
+
# the x-axis.
|
44
|
+
#
|
45
|
+
# See: ELL - https://www.spaceroots.org/documents/ellipse/elliptical-arc.pdf
|
46
|
+
class Arc
|
47
|
+
|
48
|
+
include HexaPDF::Utils::MathHelpers
|
49
|
+
|
50
|
+
# Creates and configures a new elliptical arc object.
|
51
|
+
#
|
52
|
+
# See #configure for the allowed keyword arguments.
|
53
|
+
def self.configure(**kwargs)
|
54
|
+
new.configure(kwargs)
|
55
|
+
end
|
56
|
+
|
57
|
+
# The maximal number of curves used for approximating a complete ellipse.
|
58
|
+
#
|
59
|
+
# The higher the value the better the approximation will be but it will also take longer
|
60
|
+
# to compute. The value should not be lower than 4. Default value is 6 which already
|
61
|
+
# provides a good approximation.
|
62
|
+
attr_accessor :max_curves
|
63
|
+
|
64
|
+
# x-coordinate of center point
|
65
|
+
attr_reader :cx
|
66
|
+
|
67
|
+
# y-coordinate of center point
|
68
|
+
attr_reader :cy
|
69
|
+
|
70
|
+
# Length of semi-major axis
|
71
|
+
attr_reader :a
|
72
|
+
|
73
|
+
# Length of semi-minor axis
|
74
|
+
attr_reader :b
|
75
|
+
|
76
|
+
# Start angle in degrees
|
77
|
+
attr_reader :start_angle
|
78
|
+
|
79
|
+
# End angle in degrees
|
80
|
+
attr_reader :end_angle
|
81
|
+
|
82
|
+
# Inclination in degrees of semi-major axis in respect to x-axis
|
83
|
+
attr_reader :inclination
|
84
|
+
|
85
|
+
# Direction of arc - if +true+ in clockwise direction, else in counterclockwise direction
|
86
|
+
attr_reader :clockwise
|
87
|
+
|
88
|
+
# Creates an elliptical arc with default values (a counterclockwise unit circle at the
|
89
|
+
# origin).
|
90
|
+
def initialize
|
91
|
+
@max_curves = 6
|
92
|
+
@cx = @cy = 0
|
93
|
+
@a = @b = 1
|
94
|
+
@start_angle = 0
|
95
|
+
@end_angle = 360
|
96
|
+
@inclination = 0
|
97
|
+
@clockwise = false
|
98
|
+
calculate_cached_values
|
99
|
+
end
|
100
|
+
|
101
|
+
# Configures the arc with
|
102
|
+
#
|
103
|
+
# * center point (+cx+, +cy+),
|
104
|
+
# * semi-major axis +a+,
|
105
|
+
# * semi-minor axis +b+,
|
106
|
+
# * start angle of +start_angle+ degrees,
|
107
|
+
# * end angle of +end_angle+ degrees and
|
108
|
+
# * an inclination in respect to the x-axis of +inclination+ degrees.
|
109
|
+
#
|
110
|
+
# The +clockwise+ argument determines if the arc is drawn in the counterclockwise direction
|
111
|
+
# (+false+) or in the clockwise direction (+true+).
|
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, a: nil, b: nil, start_angle: nil, end_angle: nil,
|
118
|
+
inclination: nil, clockwise: nil)
|
119
|
+
@cx = cx if cx
|
120
|
+
@cy = cy if cy
|
121
|
+
@a = a.abs if a
|
122
|
+
@b = b.abs if b
|
123
|
+
if @a == 0 || @b == 0
|
124
|
+
raise HexaPDF::Error, "Semi-major and semi-minor axes must be greater than zero"
|
125
|
+
end
|
126
|
+
@start_angle = start_angle if start_angle
|
127
|
+
@end_angle = end_angle if end_angle
|
128
|
+
@inclination = inclination if inclination
|
129
|
+
@clockwise = clockwise unless clockwise.nil?
|
130
|
+
calculate_cached_values
|
131
|
+
self
|
132
|
+
end
|
133
|
+
|
134
|
+
# Returns the start point of the elliptical arc.
|
135
|
+
def start_point
|
136
|
+
evaluate(@start_eta)
|
137
|
+
end
|
138
|
+
|
139
|
+
# Returns the end point of the elliptical arc.
|
140
|
+
def end_point
|
141
|
+
evaluate(@end_eta)
|
142
|
+
end
|
143
|
+
|
144
|
+
# Returns the point at +angle+ degrees on the ellipse.
|
145
|
+
#
|
146
|
+
# Note that the point may not lie on the arc itself!
|
147
|
+
def point_at(angle)
|
148
|
+
evaluate(angle_to_param(angle))
|
149
|
+
end
|
150
|
+
|
151
|
+
# Draws the arc on the given Canvas.
|
152
|
+
#
|
153
|
+
# If the argument +move_to_start+ is +true+, a Canvas#move_to operation is executed to
|
154
|
+
# move the current point to the start point of the arc. Otherwise it is assumed that the
|
155
|
+
# current point already coincides with the start point
|
156
|
+
#
|
157
|
+
# The #max_curves value is set to the value of the configuration option
|
158
|
+
# 'graphic_object.arc.max_curves' before drawing.
|
159
|
+
def draw(canvas, move_to_start: true)
|
160
|
+
@max_curves = canvas.context.document.config['graphic_object.arc.max_curves']
|
161
|
+
canvas.move_to(*start_point) if move_to_start
|
162
|
+
curves.each {|curve| canvas.curve_to(*curve)}
|
163
|
+
end
|
164
|
+
|
165
|
+
# Returns an array of arrays that contain the points for the Bezier curves which are used
|
166
|
+
# for approximating the elliptical arc between #start_point and #end_point.
|
167
|
+
#
|
168
|
+
# One subarray consists of
|
169
|
+
#
|
170
|
+
# [end_point_x, end_point_y, p1: control_point_1, p2: control_point_2]
|
171
|
+
#
|
172
|
+
# The first start point is the one returned by #start_point, the other start points are
|
173
|
+
# the end points of the curve before.
|
174
|
+
#
|
175
|
+
# The format of the subarray is chosen so that it can be fed to the Canvas#curve_to
|
176
|
+
# method by using array splatting.
|
177
|
+
#
|
178
|
+
# See: ELL s3.4.1 (especially the last box on page 18)
|
179
|
+
def curves
|
180
|
+
result = []
|
181
|
+
|
182
|
+
# Number of curves to use, maximal segment angle is 2*PI/max_curves
|
183
|
+
n = [@max_curves, ((@end_eta - @start_eta).abs / (2 * Math::PI / @max_curves)).ceil].min
|
184
|
+
d_eta = (@end_eta - @start_eta) / n
|
185
|
+
|
186
|
+
alpha = Math.sin(d_eta) * (Math.sqrt(4 + 3 * Math.tan(d_eta / 2)**2) - 1) / 3
|
187
|
+
|
188
|
+
eta2 = @start_eta
|
189
|
+
p2x, p2y = evaluate(eta2)
|
190
|
+
p2x_prime, p2y_prime = derivative_evaluate(eta2)
|
191
|
+
1.upto(n) do
|
192
|
+
p1x = p2x
|
193
|
+
p1y = p2y
|
194
|
+
p1x_prime = p2x_prime
|
195
|
+
p1y_prime = p2y_prime
|
196
|
+
|
197
|
+
eta2 += d_eta
|
198
|
+
p2x, p2y = evaluate(eta2)
|
199
|
+
p2x_prime, p2y_prime = derivative_evaluate(eta2)
|
200
|
+
|
201
|
+
result << [p2x, p2y,
|
202
|
+
p1: [p1x + alpha * p1x_prime, p1y + alpha * p1y_prime],
|
203
|
+
p2: [p2x - alpha * p2x_prime, p2y - alpha * p2y_prime]]
|
204
|
+
end
|
205
|
+
|
206
|
+
result
|
207
|
+
end
|
208
|
+
|
209
|
+
private
|
210
|
+
|
211
|
+
# Calculates the values that are derived from the input values and needed for the
|
212
|
+
# calculations
|
213
|
+
def calculate_cached_values
|
214
|
+
theta = deg_to_rad(@inclination)
|
215
|
+
@cos_theta = Math.cos(theta)
|
216
|
+
@sin_theta = Math.sin(theta)
|
217
|
+
|
218
|
+
# (see ELL s2.2.1) Calculating start_eta and end_eta so that
|
219
|
+
# start_eta < end_eta <= start_eta + 2*PI if counterclockwise
|
220
|
+
# end_eta < start_eta <= end_eta + 2*PI if clockwise
|
221
|
+
@start_eta = angle_to_param(@start_angle)
|
222
|
+
@end_eta = angle_to_param(@end_angle)
|
223
|
+
if !@clockwise && @end_eta <= @start_eta
|
224
|
+
@end_eta += 2 * Math::PI
|
225
|
+
elsif @clockwise && @end_eta >= @start_eta
|
226
|
+
@start_eta += 2 * Math::PI
|
227
|
+
end
|
228
|
+
end
|
229
|
+
|
230
|
+
# Converts the +angle+ in degrees to the parameter used for the parametric function
|
231
|
+
# defining the ellipse.
|
232
|
+
#
|
233
|
+
# The return value is between 0 and 2*PI.
|
234
|
+
def angle_to_param(angle)
|
235
|
+
angle = deg_to_rad(angle % 360)
|
236
|
+
eta = Math.atan2(Math.sin(angle) / @b, Math.cos(angle) / @a)
|
237
|
+
eta += 2 * Math::PI if eta < 0
|
238
|
+
eta
|
239
|
+
end
|
240
|
+
|
241
|
+
# Returns an array containing the x and y coordinates of the point on the elliptical arc
|
242
|
+
# specified by the parameter +eta+.
|
243
|
+
#
|
244
|
+
# See: ELL s2.2.1 (3)
|
245
|
+
def evaluate(eta)
|
246
|
+
a_cos_eta = @a * Math.cos(eta)
|
247
|
+
b_sin_eta = @b * Math.sin(eta)
|
248
|
+
[@cx + a_cos_eta * @cos_theta - b_sin_eta * @sin_theta,
|
249
|
+
@cy + a_cos_eta * @sin_theta + b_sin_eta * @cos_theta]
|
250
|
+
end
|
251
|
+
|
252
|
+
# Returns an array containing the derivative of the parametric function defining the
|
253
|
+
# ellipse evaluated at +eta+.
|
254
|
+
#
|
255
|
+
# See: ELL s2.2.1 (4)
|
256
|
+
def derivative_evaluate(eta)
|
257
|
+
a_sin_eta = @a * Math.sin(eta)
|
258
|
+
b_cos_eta = @b * Math.cos(eta)
|
259
|
+
[- a_sin_eta * @cos_theta - b_cos_eta * @sin_theta,
|
260
|
+
- a_sin_eta * @sin_theta + b_cos_eta * @cos_theta]
|
261
|
+
end
|
262
|
+
|
263
|
+
end
|
264
|
+
|
265
|
+
end
|
266
|
+
end
|
267
|
+
end
|