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.
Files changed (346) hide show
  1. checksums.yaml +7 -0
  2. data/CONTRIBUTERS +3 -0
  3. data/LICENSE +26 -0
  4. data/README.md +88 -0
  5. data/Rakefile +121 -0
  6. data/VERSION +1 -0
  7. data/agpl-3.0.txt +661 -0
  8. data/bin/hexapdf +6 -0
  9. data/data/hexapdf/afm/Courier-Bold.afm +342 -0
  10. data/data/hexapdf/afm/Courier-BoldOblique.afm +342 -0
  11. data/data/hexapdf/afm/Courier-Oblique.afm +342 -0
  12. data/data/hexapdf/afm/Courier.afm +342 -0
  13. data/data/hexapdf/afm/Helvetica-Bold.afm +2827 -0
  14. data/data/hexapdf/afm/Helvetica-BoldOblique.afm +2827 -0
  15. data/data/hexapdf/afm/Helvetica-Oblique.afm +3051 -0
  16. data/data/hexapdf/afm/Helvetica.afm +3051 -0
  17. data/data/hexapdf/afm/MustRead.html +1 -0
  18. data/data/hexapdf/afm/Symbol.afm +213 -0
  19. data/data/hexapdf/afm/Times-Bold.afm +2588 -0
  20. data/data/hexapdf/afm/Times-BoldItalic.afm +2384 -0
  21. data/data/hexapdf/afm/Times-Italic.afm +2667 -0
  22. data/data/hexapdf/afm/Times-Roman.afm +2419 -0
  23. data/data/hexapdf/afm/ZapfDingbats.afm +225 -0
  24. data/data/hexapdf/encoding/glyphlist.txt +4305 -0
  25. data/data/hexapdf/encoding/zapfdingbats.txt +225 -0
  26. data/examples/arc.rb +50 -0
  27. data/examples/graphics.rb +274 -0
  28. data/examples/hello_world.rb +16 -0
  29. data/examples/machupicchu.jpg +0 -0
  30. data/examples/merging.rb +24 -0
  31. data/examples/optimizing.rb +20 -0
  32. data/examples/show_char_bboxes.rb +55 -0
  33. data/examples/standard_pdf_fonts.rb +72 -0
  34. data/examples/truetype.rb +45 -0
  35. data/lib/hexapdf/cli/extract.rb +128 -0
  36. data/lib/hexapdf/cli/info.rb +121 -0
  37. data/lib/hexapdf/cli/inspect.rb +157 -0
  38. data/lib/hexapdf/cli/modify.rb +218 -0
  39. data/lib/hexapdf/cli.rb +121 -0
  40. data/lib/hexapdf/configuration.rb +392 -0
  41. data/lib/hexapdf/content/canvas.rb +1974 -0
  42. data/lib/hexapdf/content/color_space.rb +364 -0
  43. data/lib/hexapdf/content/graphic_object/arc.rb +267 -0
  44. data/lib/hexapdf/content/graphic_object/endpoint_arc.rb +208 -0
  45. data/lib/hexapdf/content/graphic_object/solid_arc.rb +173 -0
  46. data/lib/hexapdf/content/graphic_object.rb +81 -0
  47. data/lib/hexapdf/content/graphics_state.rb +579 -0
  48. data/lib/hexapdf/content/operator.rb +1072 -0
  49. data/lib/hexapdf/content/parser.rb +204 -0
  50. data/lib/hexapdf/content/processor.rb +451 -0
  51. data/lib/hexapdf/content/transformation_matrix.rb +172 -0
  52. data/lib/hexapdf/content.rb +47 -0
  53. data/lib/hexapdf/data_dir.rb +51 -0
  54. data/lib/hexapdf/dictionary.rb +303 -0
  55. data/lib/hexapdf/dictionary_fields.rb +382 -0
  56. data/lib/hexapdf/document.rb +589 -0
  57. data/lib/hexapdf/document_utils.rb +209 -0
  58. data/lib/hexapdf/encryption/aes.rb +206 -0
  59. data/lib/hexapdf/encryption/arc4.rb +93 -0
  60. data/lib/hexapdf/encryption/fast_aes.rb +79 -0
  61. data/lib/hexapdf/encryption/fast_arc4.rb +67 -0
  62. data/lib/hexapdf/encryption/identity.rb +63 -0
  63. data/lib/hexapdf/encryption/ruby_aes.rb +447 -0
  64. data/lib/hexapdf/encryption/ruby_arc4.rb +96 -0
  65. data/lib/hexapdf/encryption/security_handler.rb +494 -0
  66. data/lib/hexapdf/encryption/standard_security_handler.rb +616 -0
  67. data/lib/hexapdf/encryption.rb +94 -0
  68. data/lib/hexapdf/error.rb +73 -0
  69. data/lib/hexapdf/filter/ascii85_decode.rb +160 -0
  70. data/lib/hexapdf/filter/ascii_hex_decode.rb +87 -0
  71. data/lib/hexapdf/filter/dct_decode.rb +57 -0
  72. data/lib/hexapdf/filter/encryption.rb +59 -0
  73. data/lib/hexapdf/filter/flate_decode.rb +93 -0
  74. data/lib/hexapdf/filter/jpx_decode.rb +56 -0
  75. data/lib/hexapdf/filter/lzw_decode.rb +191 -0
  76. data/lib/hexapdf/filter/predictor.rb +266 -0
  77. data/lib/hexapdf/filter/run_length_decode.rb +108 -0
  78. data/lib/hexapdf/filter.rb +176 -0
  79. data/lib/hexapdf/font/cmap/parser.rb +146 -0
  80. data/lib/hexapdf/font/cmap/writer.rb +176 -0
  81. data/lib/hexapdf/font/cmap.rb +90 -0
  82. data/lib/hexapdf/font/encoding/base.rb +77 -0
  83. data/lib/hexapdf/font/encoding/difference_encoding.rb +64 -0
  84. data/lib/hexapdf/font/encoding/glyph_list.rb +150 -0
  85. data/lib/hexapdf/font/encoding/mac_expert_encoding.rb +221 -0
  86. data/lib/hexapdf/font/encoding/mac_roman_encoding.rb +265 -0
  87. data/lib/hexapdf/font/encoding/standard_encoding.rb +205 -0
  88. data/lib/hexapdf/font/encoding/symbol_encoding.rb +244 -0
  89. data/lib/hexapdf/font/encoding/win_ansi_encoding.rb +280 -0
  90. data/lib/hexapdf/font/encoding/zapf_dingbats_encoding.rb +250 -0
  91. data/lib/hexapdf/font/encoding.rb +68 -0
  92. data/lib/hexapdf/font/true_type/font.rb +179 -0
  93. data/lib/hexapdf/font/true_type/table/cmap.rb +103 -0
  94. data/lib/hexapdf/font/true_type/table/cmap_subtable.rb +384 -0
  95. data/lib/hexapdf/font/true_type/table/directory.rb +92 -0
  96. data/lib/hexapdf/font/true_type/table/glyf.rb +166 -0
  97. data/lib/hexapdf/font/true_type/table/head.rb +143 -0
  98. data/lib/hexapdf/font/true_type/table/hhea.rb +109 -0
  99. data/lib/hexapdf/font/true_type/table/hmtx.rb +79 -0
  100. data/lib/hexapdf/font/true_type/table/loca.rb +79 -0
  101. data/lib/hexapdf/font/true_type/table/maxp.rb +112 -0
  102. data/lib/hexapdf/font/true_type/table/name.rb +218 -0
  103. data/lib/hexapdf/font/true_type/table/os2.rb +200 -0
  104. data/lib/hexapdf/font/true_type/table/post.rb +230 -0
  105. data/lib/hexapdf/font/true_type/table.rb +155 -0
  106. data/lib/hexapdf/font/true_type.rb +48 -0
  107. data/lib/hexapdf/font/true_type_wrapper.rb +240 -0
  108. data/lib/hexapdf/font/type1/afm_parser.rb +230 -0
  109. data/lib/hexapdf/font/type1/character_metrics.rb +67 -0
  110. data/lib/hexapdf/font/type1/font.rb +123 -0
  111. data/lib/hexapdf/font/type1/font_metrics.rb +117 -0
  112. data/lib/hexapdf/font/type1/pfb_parser.rb +71 -0
  113. data/lib/hexapdf/font/type1.rb +52 -0
  114. data/lib/hexapdf/font/type1_wrapper.rb +193 -0
  115. data/lib/hexapdf/font_loader/from_configuration.rb +70 -0
  116. data/lib/hexapdf/font_loader/standard14.rb +98 -0
  117. data/lib/hexapdf/font_loader.rb +85 -0
  118. data/lib/hexapdf/font_utils.rb +89 -0
  119. data/lib/hexapdf/image_loader/jpeg.rb +166 -0
  120. data/lib/hexapdf/image_loader/pdf.rb +89 -0
  121. data/lib/hexapdf/image_loader/png.rb +410 -0
  122. data/lib/hexapdf/image_loader.rb +68 -0
  123. data/lib/hexapdf/importer.rb +139 -0
  124. data/lib/hexapdf/name_tree_node.rb +78 -0
  125. data/lib/hexapdf/number_tree_node.rb +67 -0
  126. data/lib/hexapdf/object.rb +363 -0
  127. data/lib/hexapdf/parser.rb +349 -0
  128. data/lib/hexapdf/rectangle.rb +99 -0
  129. data/lib/hexapdf/reference.rb +98 -0
  130. data/lib/hexapdf/revision.rb +206 -0
  131. data/lib/hexapdf/revisions.rb +194 -0
  132. data/lib/hexapdf/serializer.rb +326 -0
  133. data/lib/hexapdf/stream.rb +279 -0
  134. data/lib/hexapdf/task/dereference.rb +109 -0
  135. data/lib/hexapdf/task/optimize.rb +230 -0
  136. data/lib/hexapdf/task.rb +68 -0
  137. data/lib/hexapdf/tokenizer.rb +406 -0
  138. data/lib/hexapdf/type/catalog.rb +107 -0
  139. data/lib/hexapdf/type/embedded_file.rb +87 -0
  140. data/lib/hexapdf/type/file_specification.rb +232 -0
  141. data/lib/hexapdf/type/font.rb +81 -0
  142. data/lib/hexapdf/type/font_descriptor.rb +109 -0
  143. data/lib/hexapdf/type/font_simple.rb +190 -0
  144. data/lib/hexapdf/type/font_true_type.rb +47 -0
  145. data/lib/hexapdf/type/font_type1.rb +162 -0
  146. data/lib/hexapdf/type/form.rb +103 -0
  147. data/lib/hexapdf/type/graphics_state_parameter.rb +79 -0
  148. data/lib/hexapdf/type/image.rb +73 -0
  149. data/lib/hexapdf/type/info.rb +70 -0
  150. data/lib/hexapdf/type/names.rb +69 -0
  151. data/lib/hexapdf/type/object_stream.rb +224 -0
  152. data/lib/hexapdf/type/page.rb +355 -0
  153. data/lib/hexapdf/type/page_tree_node.rb +269 -0
  154. data/lib/hexapdf/type/resources.rb +212 -0
  155. data/lib/hexapdf/type/trailer.rb +128 -0
  156. data/lib/hexapdf/type/viewer_preferences.rb +73 -0
  157. data/lib/hexapdf/type/xref_stream.rb +204 -0
  158. data/lib/hexapdf/type.rb +67 -0
  159. data/lib/hexapdf/utils/bit_field.rb +87 -0
  160. data/lib/hexapdf/utils/bit_stream.rb +148 -0
  161. data/lib/hexapdf/utils/lru_cache.rb +65 -0
  162. data/lib/hexapdf/utils/math_helpers.rb +55 -0
  163. data/lib/hexapdf/utils/object_hash.rb +130 -0
  164. data/lib/hexapdf/utils/pdf_doc_encoding.rb +93 -0
  165. data/lib/hexapdf/utils/sorted_tree_node.rb +339 -0
  166. data/lib/hexapdf/version.rb +39 -0
  167. data/lib/hexapdf/writer.rb +199 -0
  168. data/lib/hexapdf/xref_section.rb +152 -0
  169. data/lib/hexapdf.rb +34 -0
  170. data/man/man1/hexapdf.1 +249 -0
  171. data/test/data/aes-test-vectors/CBCGFSbox-128-decrypt.data.gz +0 -0
  172. data/test/data/aes-test-vectors/CBCGFSbox-128-encrypt.data.gz +0 -0
  173. data/test/data/aes-test-vectors/CBCGFSbox-192-decrypt.data.gz +0 -0
  174. data/test/data/aes-test-vectors/CBCGFSbox-192-encrypt.data.gz +0 -0
  175. data/test/data/aes-test-vectors/CBCGFSbox-256-decrypt.data.gz +0 -0
  176. data/test/data/aes-test-vectors/CBCGFSbox-256-encrypt.data.gz +0 -0
  177. data/test/data/aes-test-vectors/CBCKeySbox-128-decrypt.data.gz +0 -0
  178. data/test/data/aes-test-vectors/CBCKeySbox-128-encrypt.data.gz +0 -0
  179. data/test/data/aes-test-vectors/CBCKeySbox-192-decrypt.data.gz +0 -0
  180. data/test/data/aes-test-vectors/CBCKeySbox-192-encrypt.data.gz +0 -0
  181. data/test/data/aes-test-vectors/CBCKeySbox-256-decrypt.data.gz +0 -0
  182. data/test/data/aes-test-vectors/CBCKeySbox-256-encrypt.data.gz +0 -0
  183. data/test/data/aes-test-vectors/CBCVarKey-128-decrypt.data.gz +0 -0
  184. data/test/data/aes-test-vectors/CBCVarKey-128-encrypt.data.gz +0 -0
  185. data/test/data/aes-test-vectors/CBCVarKey-192-decrypt.data.gz +0 -0
  186. data/test/data/aes-test-vectors/CBCVarKey-192-encrypt.data.gz +0 -0
  187. data/test/data/aes-test-vectors/CBCVarKey-256-decrypt.data.gz +0 -0
  188. data/test/data/aes-test-vectors/CBCVarKey-256-encrypt.data.gz +0 -0
  189. data/test/data/aes-test-vectors/CBCVarTxt-128-decrypt.data.gz +0 -0
  190. data/test/data/aes-test-vectors/CBCVarTxt-128-encrypt.data.gz +0 -0
  191. data/test/data/aes-test-vectors/CBCVarTxt-192-decrypt.data.gz +0 -0
  192. data/test/data/aes-test-vectors/CBCVarTxt-192-encrypt.data.gz +0 -0
  193. data/test/data/aes-test-vectors/CBCVarTxt-256-decrypt.data.gz +0 -0
  194. data/test/data/aes-test-vectors/CBCVarTxt-256-encrypt.data.gz +0 -0
  195. data/test/data/fonts/Ubuntu-Title.ttf +0 -0
  196. data/test/data/images/cmyk.jpg +0 -0
  197. data/test/data/images/fillbytes.jpg +0 -0
  198. data/test/data/images/gray.jpg +0 -0
  199. data/test/data/images/greyscale-1bit.png +0 -0
  200. data/test/data/images/greyscale-2bit.png +0 -0
  201. data/test/data/images/greyscale-4bit.png +0 -0
  202. data/test/data/images/greyscale-8bit.png +0 -0
  203. data/test/data/images/greyscale-alpha-8bit.png +0 -0
  204. data/test/data/images/greyscale-trns-8bit.png +0 -0
  205. data/test/data/images/greyscale-with-gamma1.0.png +0 -0
  206. data/test/data/images/greyscale-with-gamma1.5.png +0 -0
  207. data/test/data/images/indexed-1bit.png +0 -0
  208. data/test/data/images/indexed-2bit.png +0 -0
  209. data/test/data/images/indexed-4bit.png +0 -0
  210. data/test/data/images/indexed-8bit.png +0 -0
  211. data/test/data/images/indexed-alpha-4bit.png +0 -0
  212. data/test/data/images/indexed-alpha-8bit.png +0 -0
  213. data/test/data/images/rgb.jpg +0 -0
  214. data/test/data/images/truecolour-8bit.png +0 -0
  215. data/test/data/images/truecolour-alpha-8bit.png +0 -0
  216. data/test/data/images/truecolour-gama-chrm-8bit.png +0 -0
  217. data/test/data/images/truecolour-srgb-8bit.png +0 -0
  218. data/test/data/minimal.pdf +44 -0
  219. data/test/data/standard-security-handler/README +9 -0
  220. data/test/data/standard-security-handler/bothpwd-aes-128bit-V4.pdf +44 -0
  221. data/test/data/standard-security-handler/bothpwd-aes-256bit-V5.pdf +0 -0
  222. data/test/data/standard-security-handler/bothpwd-arc4-128bit-V2.pdf +43 -0
  223. data/test/data/standard-security-handler/bothpwd-arc4-128bit-V4.pdf +43 -0
  224. data/test/data/standard-security-handler/bothpwd-arc4-40bit-V1.pdf +0 -0
  225. data/test/data/standard-security-handler/nopwd-aes-128bit-V4.pdf +43 -0
  226. data/test/data/standard-security-handler/nopwd-aes-256bit-V5.pdf +0 -0
  227. data/test/data/standard-security-handler/nopwd-arc4-128bit-V2.pdf +43 -0
  228. data/test/data/standard-security-handler/nopwd-arc4-128bit-V4.pdf +43 -0
  229. data/test/data/standard-security-handler/nopwd-arc4-40bit-V1.pdf +43 -0
  230. data/test/data/standard-security-handler/ownerpwd-aes-128bit-V4.pdf +0 -0
  231. data/test/data/standard-security-handler/ownerpwd-aes-256bit-V5.pdf +43 -0
  232. data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V2.pdf +43 -0
  233. data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V4.pdf +43 -0
  234. data/test/data/standard-security-handler/ownerpwd-arc4-40bit-V1.pdf +43 -0
  235. data/test/data/standard-security-handler/userpwd-aes-128bit-V4.pdf +43 -0
  236. data/test/data/standard-security-handler/userpwd-aes-256bit-V5.pdf +43 -0
  237. data/test/data/standard-security-handler/userpwd-arc4-128bit-V2.pdf +0 -0
  238. data/test/data/standard-security-handler/userpwd-arc4-128bit-V4.pdf +0 -0
  239. data/test/data/standard-security-handler/userpwd-arc4-40bit-V1.pdf +43 -0
  240. data/test/hexapdf/common_tokenizer_tests.rb +204 -0
  241. data/test/hexapdf/content/common.rb +31 -0
  242. data/test/hexapdf/content/graphic_object/test_arc.rb +93 -0
  243. data/test/hexapdf/content/graphic_object/test_endpoint_arc.rb +91 -0
  244. data/test/hexapdf/content/graphic_object/test_solid_arc.rb +86 -0
  245. data/test/hexapdf/content/test_canvas.rb +1113 -0
  246. data/test/hexapdf/content/test_color_space.rb +97 -0
  247. data/test/hexapdf/content/test_graphics_state.rb +138 -0
  248. data/test/hexapdf/content/test_operator.rb +619 -0
  249. data/test/hexapdf/content/test_parser.rb +66 -0
  250. data/test/hexapdf/content/test_processor.rb +156 -0
  251. data/test/hexapdf/content/test_transformation_matrix.rb +64 -0
  252. data/test/hexapdf/encryption/common.rb +87 -0
  253. data/test/hexapdf/encryption/test_aes.rb +121 -0
  254. data/test/hexapdf/encryption/test_arc4.rb +39 -0
  255. data/test/hexapdf/encryption/test_fast_aes.rb +17 -0
  256. data/test/hexapdf/encryption/test_fast_arc4.rb +12 -0
  257. data/test/hexapdf/encryption/test_identity.rb +21 -0
  258. data/test/hexapdf/encryption/test_ruby_aes.rb +23 -0
  259. data/test/hexapdf/encryption/test_ruby_arc4.rb +20 -0
  260. data/test/hexapdf/encryption/test_security_handler.rb +356 -0
  261. data/test/hexapdf/encryption/test_standard_security_handler.rb +274 -0
  262. data/test/hexapdf/filter/common.rb +53 -0
  263. data/test/hexapdf/filter/test_ascii85_decode.rb +60 -0
  264. data/test/hexapdf/filter/test_ascii_hex_decode.rb +33 -0
  265. data/test/hexapdf/filter/test_encryption.rb +24 -0
  266. data/test/hexapdf/filter/test_flate_decode.rb +35 -0
  267. data/test/hexapdf/filter/test_lzw_decode.rb +52 -0
  268. data/test/hexapdf/filter/test_predictor.rb +183 -0
  269. data/test/hexapdf/filter/test_run_length_decode.rb +32 -0
  270. data/test/hexapdf/font/cmap/test_parser.rb +67 -0
  271. data/test/hexapdf/font/cmap/test_writer.rb +58 -0
  272. data/test/hexapdf/font/encoding/test_base.rb +35 -0
  273. data/test/hexapdf/font/encoding/test_difference_encoding.rb +21 -0
  274. data/test/hexapdf/font/encoding/test_glyph_list.rb +59 -0
  275. data/test/hexapdf/font/encoding/test_zapf_dingbats_encoding.rb +16 -0
  276. data/test/hexapdf/font/test_encoding.rb +27 -0
  277. data/test/hexapdf/font/test_true_type_wrapper.rb +110 -0
  278. data/test/hexapdf/font/test_type1_wrapper.rb +66 -0
  279. data/test/hexapdf/font/true_type/common.rb +19 -0
  280. data/test/hexapdf/font/true_type/table/test_cmap.rb +59 -0
  281. data/test/hexapdf/font/true_type/table/test_cmap_subtable.rb +133 -0
  282. data/test/hexapdf/font/true_type/table/test_directory.rb +35 -0
  283. data/test/hexapdf/font/true_type/table/test_glyf.rb +58 -0
  284. data/test/hexapdf/font/true_type/table/test_head.rb +76 -0
  285. data/test/hexapdf/font/true_type/table/test_hhea.rb +40 -0
  286. data/test/hexapdf/font/true_type/table/test_hmtx.rb +38 -0
  287. data/test/hexapdf/font/true_type/table/test_loca.rb +43 -0
  288. data/test/hexapdf/font/true_type/table/test_maxp.rb +62 -0
  289. data/test/hexapdf/font/true_type/table/test_name.rb +95 -0
  290. data/test/hexapdf/font/true_type/table/test_os2.rb +65 -0
  291. data/test/hexapdf/font/true_type/table/test_post.rb +89 -0
  292. data/test/hexapdf/font/true_type/test_font.rb +120 -0
  293. data/test/hexapdf/font/true_type/test_table.rb +41 -0
  294. data/test/hexapdf/font/type1/test_afm_parser.rb +51 -0
  295. data/test/hexapdf/font/type1/test_font.rb +68 -0
  296. data/test/hexapdf/font/type1/test_pfb_parser.rb +37 -0
  297. data/test/hexapdf/font_loader/test_from_configuration.rb +28 -0
  298. data/test/hexapdf/font_loader/test_standard14.rb +22 -0
  299. data/test/hexapdf/image_loader/test_jpeg.rb +83 -0
  300. data/test/hexapdf/image_loader/test_pdf.rb +47 -0
  301. data/test/hexapdf/image_loader/test_png.rb +258 -0
  302. data/test/hexapdf/task/test_dereference.rb +46 -0
  303. data/test/hexapdf/task/test_optimize.rb +137 -0
  304. data/test/hexapdf/test_configuration.rb +82 -0
  305. data/test/hexapdf/test_data_dir.rb +32 -0
  306. data/test/hexapdf/test_dictionary.rb +284 -0
  307. data/test/hexapdf/test_dictionary_fields.rb +185 -0
  308. data/test/hexapdf/test_document.rb +574 -0
  309. data/test/hexapdf/test_document_utils.rb +144 -0
  310. data/test/hexapdf/test_filter.rb +96 -0
  311. data/test/hexapdf/test_font_utils.rb +47 -0
  312. data/test/hexapdf/test_importer.rb +78 -0
  313. data/test/hexapdf/test_object.rb +177 -0
  314. data/test/hexapdf/test_parser.rb +394 -0
  315. data/test/hexapdf/test_rectangle.rb +36 -0
  316. data/test/hexapdf/test_reference.rb +41 -0
  317. data/test/hexapdf/test_revision.rb +139 -0
  318. data/test/hexapdf/test_revisions.rb +93 -0
  319. data/test/hexapdf/test_serializer.rb +169 -0
  320. data/test/hexapdf/test_stream.rb +262 -0
  321. data/test/hexapdf/test_tokenizer.rb +30 -0
  322. data/test/hexapdf/test_writer.rb +120 -0
  323. data/test/hexapdf/test_xref_section.rb +35 -0
  324. data/test/hexapdf/type/test_catalog.rb +30 -0
  325. data/test/hexapdf/type/test_embedded_file.rb +16 -0
  326. data/test/hexapdf/type/test_file_specification.rb +148 -0
  327. data/test/hexapdf/type/test_font.rb +35 -0
  328. data/test/hexapdf/type/test_font_descriptor.rb +51 -0
  329. data/test/hexapdf/type/test_font_simple.rb +190 -0
  330. data/test/hexapdf/type/test_font_type1.rb +128 -0
  331. data/test/hexapdf/type/test_form.rb +60 -0
  332. data/test/hexapdf/type/test_info.rb +14 -0
  333. data/test/hexapdf/type/test_names.rb +9 -0
  334. data/test/hexapdf/type/test_object_stream.rb +84 -0
  335. data/test/hexapdf/type/test_page.rb +260 -0
  336. data/test/hexapdf/type/test_page_tree_node.rb +255 -0
  337. data/test/hexapdf/type/test_resources.rb +167 -0
  338. data/test/hexapdf/type/test_trailer.rb +109 -0
  339. data/test/hexapdf/type/test_xref_stream.rb +131 -0
  340. data/test/hexapdf/utils/test_bit_field.rb +47 -0
  341. data/test/hexapdf/utils/test_lru_cache.rb +22 -0
  342. data/test/hexapdf/utils/test_object_hash.rb +115 -0
  343. data/test/hexapdf/utils/test_pdf_doc_encoding.rb +18 -0
  344. data/test/hexapdf/utils/test_sorted_tree_node.rb +232 -0
  345. data/test/test_helper.rb +56 -0
  346. 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