nurettin-prawn 1.0.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (303) hide show
  1. data/COPYING +2 -0
  2. data/GPLv2 +340 -0
  3. data/GPLv3 +674 -0
  4. data/Gemfile +18 -0
  5. data/LICENSE +56 -0
  6. data/README.md +98 -0
  7. data/Rakefile +45 -0
  8. data/data/encodings/win_ansi.txt +29 -0
  9. data/data/fonts/Action Man.dfont +0 -0
  10. data/data/fonts/Activa.ttf +0 -0
  11. data/data/fonts/Chalkboard.ttf +0 -0
  12. data/data/fonts/Courier-Bold.afm +342 -0
  13. data/data/fonts/Courier-BoldOblique.afm +342 -0
  14. data/data/fonts/Courier-Oblique.afm +342 -0
  15. data/data/fonts/Courier.afm +342 -0
  16. data/data/fonts/DejaVuSans.ttf +0 -0
  17. data/data/fonts/Dustismo_Roman.ttf +0 -0
  18. data/data/fonts/Helvetica-Bold.afm +2827 -0
  19. data/data/fonts/Helvetica-BoldOblique.afm +2827 -0
  20. data/data/fonts/Helvetica-Oblique.afm +3051 -0
  21. data/data/fonts/Helvetica.afm +3051 -0
  22. data/data/fonts/MustRead.html +19 -0
  23. data/data/fonts/Symbol.afm +213 -0
  24. data/data/fonts/Times-Bold.afm +2588 -0
  25. data/data/fonts/Times-BoldItalic.afm +2384 -0
  26. data/data/fonts/Times-Italic.afm +2667 -0
  27. data/data/fonts/Times-Roman.afm +2419 -0
  28. data/data/fonts/ZapfDingbats.afm +225 -0
  29. data/data/fonts/comicsans.ttf +0 -0
  30. data/data/fonts/gkai00mp.ttf +0 -0
  31. data/data/images/16bit.alpha +0 -0
  32. data/data/images/16bit.dat +0 -0
  33. data/data/images/16bit.png +0 -0
  34. data/data/images/arrow.png +0 -0
  35. data/data/images/arrow2.png +0 -0
  36. data/data/images/barcode_issue.png +0 -0
  37. data/data/images/dice.alpha +0 -0
  38. data/data/images/dice.dat +0 -0
  39. data/data/images/dice.png +0 -0
  40. data/data/images/dice_interlaced.png +0 -0
  41. data/data/images/fractal.jpg +0 -0
  42. data/data/images/letterhead.jpg +0 -0
  43. data/data/images/page_white_text.alpha +0 -0
  44. data/data/images/page_white_text.dat +0 -0
  45. data/data/images/page_white_text.png +0 -0
  46. data/data/images/pigs.jpg +0 -0
  47. data/data/images/prawn.png +0 -0
  48. data/data/images/rails.dat +0 -0
  49. data/data/images/rails.png +0 -0
  50. data/data/images/ruport.png +0 -0
  51. data/data/images/ruport_data.dat +0 -0
  52. data/data/images/ruport_transparent.png +0 -0
  53. data/data/images/ruport_type0.png +0 -0
  54. data/data/images/stef.jpg +0 -0
  55. data/data/images/tru256.bmp +0 -0
  56. data/data/images/web-links.dat +1 -0
  57. data/data/images/web-links.png +0 -0
  58. data/data/pdfs/complex_template.pdf +0 -0
  59. data/data/pdfs/contains_ttf_font.pdf +0 -0
  60. data/data/pdfs/encrypted.pdf +0 -0
  61. data/data/pdfs/form.pdf +820 -0
  62. data/data/pdfs/hexagon.pdf +61 -0
  63. data/data/pdfs/indirect_reference.pdf +86 -0
  64. data/data/pdfs/multipage_template.pdf +127 -0
  65. data/data/pdfs/nested_pages.pdf +118 -0
  66. data/data/pdfs/page_without_mediabox.pdf +193 -0
  67. data/data/pdfs/resources_as_indirect_object.pdf +83 -0
  68. data/data/pdfs/two_hexagons.pdf +90 -0
  69. data/data/pdfs/version_1_6.pdf +61 -0
  70. data/data/shift_jis_text.txt +1 -0
  71. data/lib/prawn.rb +29 -0
  72. data/lib/prawn/compatibility.rb +87 -0
  73. data/lib/prawn/core.rb +87 -0
  74. data/lib/prawn/core/annotations.rb +61 -0
  75. data/lib/prawn/core/byte_string.rb +9 -0
  76. data/lib/prawn/core/destinations.rb +90 -0
  77. data/lib/prawn/core/document_state.rb +79 -0
  78. data/lib/prawn/core/literal_string.rb +16 -0
  79. data/lib/prawn/core/name_tree.rb +177 -0
  80. data/lib/prawn/core/object_store.rb +320 -0
  81. data/lib/prawn/core/page.rb +212 -0
  82. data/lib/prawn/core/pdf_object.rb +125 -0
  83. data/lib/prawn/core/reference.rb +119 -0
  84. data/lib/prawn/core/text.rb +268 -0
  85. data/lib/prawn/core/text/formatted/arranger.rb +294 -0
  86. data/lib/prawn/core/text/formatted/line_wrap.rb +288 -0
  87. data/lib/prawn/core/text/formatted/wrap.rb +153 -0
  88. data/lib/prawn/document.rb +707 -0
  89. data/lib/prawn/document/bounding_box.rb +510 -0
  90. data/lib/prawn/document/column_box.rb +132 -0
  91. data/lib/prawn/document/graphics_state.rb +136 -0
  92. data/lib/prawn/document/internals.rb +173 -0
  93. data/lib/prawn/document/page_geometry.rb +136 -0
  94. data/lib/prawn/document/snapshot.rb +89 -0
  95. data/lib/prawn/document/span.rb +55 -0
  96. data/lib/prawn/encoding.rb +121 -0
  97. data/lib/prawn/errors.rb +99 -0
  98. data/lib/prawn/font.rb +386 -0
  99. data/lib/prawn/font/afm.rb +203 -0
  100. data/lib/prawn/font/dfont.rb +42 -0
  101. data/lib/prawn/font/ttf.rb +343 -0
  102. data/lib/prawn/graphics.rb +523 -0
  103. data/lib/prawn/graphics/cap_style.rb +46 -0
  104. data/lib/prawn/graphics/color.rb +231 -0
  105. data/lib/prawn/graphics/dash.rb +82 -0
  106. data/lib/prawn/graphics/join_style.rb +47 -0
  107. data/lib/prawn/graphics/patterns.rb +137 -0
  108. data/lib/prawn/graphics/transformation.rb +156 -0
  109. data/lib/prawn/graphics/transparency.rb +99 -0
  110. data/lib/prawn/images.rb +196 -0
  111. data/lib/prawn/images/image.rb +65 -0
  112. data/lib/prawn/images/jpg.rb +85 -0
  113. data/lib/prawn/images/png.rb +362 -0
  114. data/lib/prawn/layout.rb +20 -0
  115. data/lib/prawn/layout/grid.rb +259 -0
  116. data/lib/prawn/measurement_extensions.rb +46 -0
  117. data/lib/prawn/measurements.rb +71 -0
  118. data/lib/prawn/outline.rb +326 -0
  119. data/lib/prawn/repeater.rb +122 -0
  120. data/lib/prawn/security.rb +269 -0
  121. data/lib/prawn/soft_mask.rb +94 -0
  122. data/lib/prawn/stamp.rb +134 -0
  123. data/lib/prawn/table.rb +609 -0
  124. data/lib/prawn/table/cell.rb +776 -0
  125. data/lib/prawn/table/cell/image.rb +70 -0
  126. data/lib/prawn/table/cell/in_table.rb +27 -0
  127. data/lib/prawn/table/cell/span_dummy.rb +88 -0
  128. data/lib/prawn/table/cell/subtable.rb +65 -0
  129. data/lib/prawn/table/cell/text.rb +152 -0
  130. data/lib/prawn/table/cells.rb +260 -0
  131. data/lib/prawn/text.rb +420 -0
  132. data/lib/prawn/text/box.rb +141 -0
  133. data/lib/prawn/text/formatted.rb +4 -0
  134. data/lib/prawn/text/formatted/box.rb +563 -0
  135. data/lib/prawn/text/formatted/fragment.rb +253 -0
  136. data/lib/prawn/text/formatted/parser.rb +217 -0
  137. data/lib/prawn/utilities.rb +44 -0
  138. data/manual/basic_concepts/adding_pages.rb +27 -0
  139. data/manual/basic_concepts/basic_concepts.rb +34 -0
  140. data/manual/basic_concepts/creation.rb +39 -0
  141. data/manual/basic_concepts/cursor.rb +33 -0
  142. data/manual/basic_concepts/measurement.rb +25 -0
  143. data/manual/basic_concepts/origin.rb +38 -0
  144. data/manual/basic_concepts/other_cursor_helpers.rb +40 -0
  145. data/manual/bounding_box/bounding_box.rb +39 -0
  146. data/manual/bounding_box/bounds.rb +49 -0
  147. data/manual/bounding_box/canvas.rb +24 -0
  148. data/manual/bounding_box/creation.rb +23 -0
  149. data/manual/bounding_box/indentation.rb +46 -0
  150. data/manual/bounding_box/nesting.rb +45 -0
  151. data/manual/bounding_box/russian_boxes.rb +40 -0
  152. data/manual/bounding_box/stretchy.rb +31 -0
  153. data/manual/document_and_page_options/background.rb +27 -0
  154. data/manual/document_and_page_options/document_and_page_options.rb +31 -0
  155. data/manual/document_and_page_options/metadata.rb +23 -0
  156. data/manual/document_and_page_options/page_margins.rb +38 -0
  157. data/manual/document_and_page_options/page_size.rb +34 -0
  158. data/manual/example_file.rb +116 -0
  159. data/manual/example_helper.rb +430 -0
  160. data/manual/example_package.rb +53 -0
  161. data/manual/example_section.rb +46 -0
  162. data/manual/graphics/circle_and_ellipse.rb +22 -0
  163. data/manual/graphics/color.rb +24 -0
  164. data/manual/graphics/common_lines.rb +28 -0
  165. data/manual/graphics/fill_and_stroke.rb +42 -0
  166. data/manual/graphics/fill_rules.rb +37 -0
  167. data/manual/graphics/gradients.rb +37 -0
  168. data/manual/graphics/graphics.rb +58 -0
  169. data/manual/graphics/helper.rb +17 -0
  170. data/manual/graphics/line_width.rb +35 -0
  171. data/manual/graphics/lines_and_curves.rb +41 -0
  172. data/manual/graphics/polygon.rb +29 -0
  173. data/manual/graphics/rectangle.rb +21 -0
  174. data/manual/graphics/rotate.rb +28 -0
  175. data/manual/graphics/scale.rb +41 -0
  176. data/manual/graphics/soft_masks.rb +46 -0
  177. data/manual/graphics/stroke_cap.rb +31 -0
  178. data/manual/graphics/stroke_dash.rb +43 -0
  179. data/manual/graphics/stroke_join.rb +30 -0
  180. data/manual/graphics/translate.rb +29 -0
  181. data/manual/graphics/transparency.rb +35 -0
  182. data/manual/images/absolute_position.rb +23 -0
  183. data/manual/images/fit.rb +21 -0
  184. data/manual/images/horizontal.rb +25 -0
  185. data/manual/images/images.rb +40 -0
  186. data/manual/images/plain_image.rb +18 -0
  187. data/manual/images/scale.rb +22 -0
  188. data/manual/images/vertical.rb +28 -0
  189. data/manual/images/width_and_height.rb +25 -0
  190. data/manual/layout/boxes.rb +27 -0
  191. data/manual/layout/content.rb +25 -0
  192. data/manual/layout/layout.rb +28 -0
  193. data/manual/layout/simple_grid.rb +23 -0
  194. data/manual/manual/cover.rb +26 -0
  195. data/manual/manual/foreword.rb +13 -0
  196. data/manual/manual/how_to_read_this_manual.rb +41 -0
  197. data/manual/manual/manual.rb +36 -0
  198. data/manual/outline/add_subsection_to.rb +61 -0
  199. data/manual/outline/insert_section_after.rb +47 -0
  200. data/manual/outline/outline.rb +32 -0
  201. data/manual/outline/sections_and_pages.rb +67 -0
  202. data/manual/repeatable_content/page_numbering.rb +54 -0
  203. data/manual/repeatable_content/repeatable_content.rb +31 -0
  204. data/manual/repeatable_content/repeater.rb +55 -0
  205. data/manual/repeatable_content/stamp.rb +41 -0
  206. data/manual/security/encryption.rb +31 -0
  207. data/manual/security/permissions.rb +38 -0
  208. data/manual/security/security.rb +28 -0
  209. data/manual/syntax_highlight.rb +52 -0
  210. data/manual/table/basic_block.rb +53 -0
  211. data/manual/table/before_rendering_page.rb +26 -0
  212. data/manual/table/cell_border_lines.rb +24 -0
  213. data/manual/table/cell_borders_and_bg.rb +31 -0
  214. data/manual/table/cell_dimensions.rb +30 -0
  215. data/manual/table/cell_text.rb +38 -0
  216. data/manual/table/column_widths.rb +30 -0
  217. data/manual/table/content_and_subtables.rb +39 -0
  218. data/manual/table/creation.rb +27 -0
  219. data/manual/table/filtering.rb +36 -0
  220. data/manual/table/flow_and_header.rb +17 -0
  221. data/manual/table/image_cells.rb +33 -0
  222. data/manual/table/position.rb +29 -0
  223. data/manual/table/row_colors.rb +20 -0
  224. data/manual/table/span.rb +30 -0
  225. data/manual/table/style.rb +22 -0
  226. data/manual/table/table.rb +52 -0
  227. data/manual/table/width.rb +27 -0
  228. data/manual/templates/full_template.rb +23 -0
  229. data/manual/templates/page_template.rb +47 -0
  230. data/manual/templates/templates.rb +26 -0
  231. data/manual/text/alignment.rb +44 -0
  232. data/manual/text/color.rb +24 -0
  233. data/manual/text/column_box.rb +32 -0
  234. data/manual/text/fallback_fonts.rb +37 -0
  235. data/manual/text/font.rb +41 -0
  236. data/manual/text/font_size.rb +45 -0
  237. data/manual/text/font_style.rb +23 -0
  238. data/manual/text/formatted_callbacks.rb +60 -0
  239. data/manual/text/formatted_text.rb +50 -0
  240. data/manual/text/free_flowing_text.rb +51 -0
  241. data/manual/text/group.rb +29 -0
  242. data/manual/text/inline.rb +43 -0
  243. data/manual/text/kerning_and_character_spacing.rb +39 -0
  244. data/manual/text/leading.rb +25 -0
  245. data/manual/text/line_wrapping.rb +41 -0
  246. data/manual/text/paragraph_indentation.rb +26 -0
  247. data/manual/text/positioned_text.rb +38 -0
  248. data/manual/text/registering_families.rb +48 -0
  249. data/manual/text/rendering_and_color.rb +37 -0
  250. data/manual/text/right_to_left_text.rb +43 -0
  251. data/manual/text/rotation.rb +43 -0
  252. data/manual/text/single_usage.rb +37 -0
  253. data/manual/text/text.rb +75 -0
  254. data/manual/text/text_box_excess.rb +32 -0
  255. data/manual/text/text_box_extensions.rb +45 -0
  256. data/manual/text/text_box_overflow.rb +44 -0
  257. data/manual/text/utf8.rb +28 -0
  258. data/manual/text/win_ansi_charset.rb +59 -0
  259. data/prawn.gemspec +46 -0
  260. data/spec/annotations_spec.rb +90 -0
  261. data/spec/bounding_box_spec.rb +493 -0
  262. data/spec/cell_spec.rb +584 -0
  263. data/spec/column_box_spec.rb +33 -0
  264. data/spec/data/curves.pdf +66 -0
  265. data/spec/destinations_spec.rb +15 -0
  266. data/spec/document_spec.rb +736 -0
  267. data/spec/extensions/encoding_helpers.rb +6 -0
  268. data/spec/extensions/mocha.rb +44 -0
  269. data/spec/font_spec.rb +433 -0
  270. data/spec/formatted_text_arranger_spec.rb +421 -0
  271. data/spec/formatted_text_box_spec.rb +640 -0
  272. data/spec/formatted_text_fragment_spec.rb +298 -0
  273. data/spec/graphics_spec.rb +651 -0
  274. data/spec/grid_spec.rb +85 -0
  275. data/spec/images_spec.rb +140 -0
  276. data/spec/inline_formatted_text_parser_spec.rb +515 -0
  277. data/spec/jpg_spec.rb +25 -0
  278. data/spec/line_wrap_spec.rb +333 -0
  279. data/spec/measurement_units_spec.rb +23 -0
  280. data/spec/name_tree_spec.rb +112 -0
  281. data/spec/object_store_spec.rb +170 -0
  282. data/spec/outline_spec.rb +426 -0
  283. data/spec/pdf_object_spec.rb +172 -0
  284. data/spec/png_spec.rb +240 -0
  285. data/spec/reference_spec.rb +105 -0
  286. data/spec/repeater_spec.rb +158 -0
  287. data/spec/security_spec.rb +126 -0
  288. data/spec/snapshot_spec.rb +186 -0
  289. data/spec/soft_mask_spec.rb +117 -0
  290. data/spec/span_spec.rb +49 -0
  291. data/spec/spec_helper.rb +36 -0
  292. data/spec/stamp_spec.rb +159 -0
  293. data/spec/stroke_styles_spec.rb +193 -0
  294. data/spec/table_spec.rb +1209 -0
  295. data/spec/template_spec.rb +351 -0
  296. data/spec/text_at_spec.rb +129 -0
  297. data/spec/text_box_spec.rb +1029 -0
  298. data/spec/text_rendering_mode_spec.rb +45 -0
  299. data/spec/text_spacing_spec.rb +93 -0
  300. data/spec/text_spec.rb +421 -0
  301. data/spec/text_with_inline_formatting_spec.rb +35 -0
  302. data/spec/transparency_spec.rb +89 -0
  303. metadata +544 -0
@@ -0,0 +1,298 @@
1
+ # encoding: utf-8
2
+
3
+ require File.join(File.expand_path(File.dirname(__FILE__)), "spec_helper")
4
+
5
+ describe "Text::Formatted::Fragment#space_count" do
6
+ it "should return the number of spaces in the fragment" do
7
+ create_pdf
8
+ format_state = { }
9
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world ",
10
+ format_state,
11
+ @pdf)
12
+ fragment.space_count.should == 2
13
+ end
14
+ it "should exclude trailing spaces from the count when " +
15
+ ":exclude_trailing_white_space => true" do
16
+ create_pdf
17
+ format_state = { :exclude_trailing_white_space => true }
18
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world ",
19
+ format_state,
20
+ @pdf)
21
+ fragment.space_count.should == 1
22
+ end
23
+ end
24
+
25
+ describe "Text::Formatted::Fragment#include_trailing_white_space!" do
26
+ it "should make the fragment include trailing white space" do
27
+ create_pdf
28
+ format_state = { :exclude_trailing_white_space => true }
29
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world ",
30
+ format_state,
31
+ @pdf)
32
+ fragment.space_count.should == 1
33
+ fragment.include_trailing_white_space!
34
+ fragment.space_count.should == 2
35
+ end
36
+ end
37
+
38
+ describe "Text::Formatted::Fragment#text" do
39
+ it "should return the fragment text" do
40
+ create_pdf
41
+ format_state = { }
42
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world ",
43
+ format_state,
44
+ @pdf)
45
+ fragment.text.should == "hello world "
46
+ end
47
+ it "should return the fragment text without trailing spaces when " +
48
+ ":exclude_trailing_white_space => true" do
49
+ create_pdf
50
+ format_state = { :exclude_trailing_white_space => true }
51
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world ",
52
+ format_state,
53
+ @pdf)
54
+ fragment.text.should == "hello world"
55
+ end
56
+ end
57
+
58
+ describe "Text::Formatted::Fragment#word_spacing=" do
59
+ before(:each) do
60
+ create_pdf
61
+ format_state = { :styles => [:bold, :italic],
62
+ :color => nil,
63
+ :link => nil,
64
+ :anchor => nil,
65
+ :font => nil,
66
+ :size => nil }
67
+ @fragment = Prawn::Text::Formatted::Fragment.new("hello world",
68
+ format_state,
69
+ @pdf)
70
+ @fragment.width = 100
71
+ @fragment.left = 50
72
+ @fragment.baseline = 200
73
+ @fragment.line_height = 27
74
+ @fragment.descender = 7
75
+ @fragment.ascender = 17
76
+ @fragment.word_spacing = 10
77
+ end
78
+
79
+ it "should account for word_spacing in #width" do
80
+ @fragment.width.should == 110
81
+ end
82
+ it "should account for word_spacing in #bounding_box" do
83
+ target_box = [50, 193, 160, 217]
84
+ @fragment.bounding_box.should == target_box
85
+ end
86
+ it "should account for word_spacing in #absolute_bounding_box" do
87
+ target_box = [50, 193, 160, 217]
88
+ target_box[0] += @pdf.bounds.absolute_left
89
+ target_box[1] += @pdf.bounds.absolute_bottom
90
+ target_box[2] += @pdf.bounds.absolute_left
91
+ target_box[3] += @pdf.bounds.absolute_bottom
92
+ @fragment.absolute_bounding_box.should == target_box
93
+ end
94
+ it "should account for word_spacing in #underline_points" do
95
+ y = 198.75
96
+ target_points = [[50, y], [160, y]]
97
+ @fragment.underline_points.should == target_points
98
+ end
99
+ it "should account for word_spacing in #strikethrough_points" do
100
+ y = 200 + @fragment.ascender * 0.3
101
+ target_points = [[50, y], [160, y]]
102
+ @fragment.strikethrough_points.should == target_points
103
+ end
104
+ end
105
+
106
+ describe "Text::Formatted::Fragment" do
107
+ before(:each) do
108
+ create_pdf
109
+ format_state = { :styles => [:bold, :italic],
110
+ :color => nil,
111
+ :link => nil,
112
+ :anchor => nil,
113
+ :font => nil,
114
+ :size => nil }
115
+ @fragment = Prawn::Text::Formatted::Fragment.new("hello world",
116
+ format_state,
117
+ @pdf)
118
+ @fragment.width = 100
119
+ @fragment.left = 50
120
+ @fragment.baseline = 200
121
+ @fragment.line_height = 27
122
+ @fragment.descender = 7
123
+ @fragment.ascender = 17
124
+ end
125
+
126
+ describe "#width" do
127
+ it "should return the width" do
128
+ @fragment.width.should == 100
129
+ end
130
+ end
131
+
132
+ describe "#styles" do
133
+ it "should return the styles array" do
134
+ @fragment.styles.should == [:bold, :italic]
135
+ end
136
+ it "should never return nil" do
137
+ format_state = { :styles => nil,
138
+ :color => nil,
139
+ :link => nil,
140
+ :anchor => nil,
141
+ :font => nil,
142
+ :size => nil }
143
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world",
144
+ format_state,
145
+ @pdf)
146
+ fragment.styles.should == []
147
+ end
148
+ end
149
+
150
+ describe "#line_height" do
151
+ it "should return the line_height" do
152
+ @fragment.line_height.should == 27
153
+ end
154
+ end
155
+
156
+ describe "#ascender" do
157
+ it "should return the ascender" do
158
+ @fragment.ascender.should == 17
159
+ end
160
+ end
161
+
162
+ describe "#descender" do
163
+ it "should return the descender" do
164
+ @fragment.descender.should == 7
165
+ end
166
+ end
167
+
168
+ describe "#y_offset" do
169
+ it "should be zero" do
170
+ @fragment.y_offset.should == 0
171
+ end
172
+ end
173
+
174
+ describe "#bounding_box" do
175
+ it "should return the bounding box surrounding the fragment" do
176
+ target_box = [50, 193, 150, 217]
177
+ @fragment.bounding_box.should == target_box
178
+ end
179
+ end
180
+
181
+ describe "#absolute_bounding_box" do
182
+ it "should return the bounding box surrounding the fragment" +
183
+ " in absolute coordinates" do
184
+ target_box = [50, 193, 150, 217]
185
+ target_box[0] += @pdf.bounds.absolute_left
186
+ target_box[1] += @pdf.bounds.absolute_bottom
187
+ target_box[2] += @pdf.bounds.absolute_left
188
+ target_box[3] += @pdf.bounds.absolute_bottom
189
+ @fragment.absolute_bounding_box.should == target_box
190
+ end
191
+ end
192
+
193
+ describe "#underline_points" do
194
+ it "should define a line under the fragment" do
195
+ y = 198.75
196
+ target_points = [[50, y], [150, y]]
197
+ @fragment.underline_points.should == target_points
198
+ end
199
+ end
200
+
201
+ describe "#strikethrough_points" do
202
+ it "should define a line through the fragment" do
203
+ y = 200 + @fragment.ascender * 0.3
204
+ target_points = [[50, y], [150, y]]
205
+ @fragment.strikethrough_points.should == target_points
206
+ end
207
+ end
208
+ end
209
+
210
+ describe "Text::Formatted::Fragment that is a subscript" do
211
+ before(:each) do
212
+ create_pdf
213
+ format_state = { :styles => [:subscript],
214
+ :color => nil,
215
+ :link => nil,
216
+ :anchor => nil,
217
+ :font => nil,
218
+ :size => nil }
219
+ @fragment = Prawn::Text::Formatted::Fragment.new("hello world",
220
+ format_state,
221
+ @pdf)
222
+ @fragment.line_height = 27
223
+ @fragment.descender = 7
224
+ @fragment.ascender = 17
225
+ end
226
+ describe "#subscript?" do
227
+ it "should be_true" do
228
+ @fragment.should be_subscript
229
+ end
230
+ end
231
+ describe "#y_offset" do
232
+ it "should return a negative value" do
233
+ @fragment.y_offset.should be < 0
234
+ end
235
+ end
236
+ end
237
+
238
+ describe "Text::Formatted::Fragment that is a superscript" do
239
+ before(:each) do
240
+ create_pdf
241
+ format_state = { :styles => [:superscript],
242
+ :color => nil,
243
+ :link => nil,
244
+ :anchor => nil,
245
+ :font => nil,
246
+ :size => nil }
247
+ @fragment = Prawn::Text::Formatted::Fragment.new("hello world",
248
+ format_state,
249
+ @pdf)
250
+ @fragment.line_height = 27
251
+ @fragment.descender = 7
252
+ @fragment.ascender = 17
253
+ end
254
+ describe "#superscript?" do
255
+ it "should be_true" do
256
+ @fragment.should be_superscript
257
+ end
258
+ end
259
+ describe "#y_offset" do
260
+ it "should return a positive value" do
261
+ @fragment.y_offset.should be > 0
262
+ end
263
+ end
264
+ end
265
+
266
+ describe "Text::Formatted::Fragment with :direction => :rtl" do
267
+ it "#text should be reversed" do
268
+ create_pdf
269
+ format_state = { :direction => :rtl }
270
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world",
271
+ format_state,
272
+ @pdf)
273
+ fragment.text.should == "dlrow olleh"
274
+ end
275
+ end
276
+
277
+ describe "Text::Formatted::Fragment default_direction=" do
278
+ it "should set the direction if there is no fragment level direction " +
279
+ "specification" do
280
+ create_pdf
281
+ format_state = { }
282
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world",
283
+ format_state,
284
+ @pdf)
285
+ fragment.default_direction = :rtl
286
+ fragment.direction.should == :rtl
287
+ end
288
+ it "should not set the direction if there is a fragment level direction " +
289
+ "specification" do
290
+ create_pdf
291
+ format_state = { :direction => :rtl }
292
+ fragment = Prawn::Text::Formatted::Fragment.new("hello world",
293
+ format_state,
294
+ @pdf)
295
+ fragment.default_direction = :ltr
296
+ fragment.direction.should == :rtl
297
+ end
298
+ end
@@ -0,0 +1,651 @@
1
+ # encoding: utf-8
2
+
3
+ require File.join(File.expand_path(File.dirname(__FILE__)), "spec_helper")
4
+
5
+ describe "When drawing a line" do
6
+
7
+ before(:each) { create_pdf }
8
+
9
+ it "should draw a line from (100,600) to (100,500)" do
10
+ @pdf.line([100,600],[100,500])
11
+
12
+ line_drawing = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
13
+
14
+ line_drawing.points.should == [[100,600],[100,500]]
15
+ end
16
+
17
+ it "should draw two lines at (100,600) to (100,500) " +
18
+ "and (75,100) to (50,125)" do
19
+ @pdf.line(100,600,100,500)
20
+ @pdf.line(75,100,50,125)
21
+
22
+ line_drawing = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
23
+
24
+ line_drawing.points.should ==
25
+ [[100.0, 600.0], [100.0, 500.0], [75.0, 100.0], [50.0, 125.0]]
26
+ end
27
+
28
+ it "should properly set line width via line_width=" do
29
+ @pdf.line_width = 10
30
+ line = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
31
+ line.widths.first.should == 10
32
+ end
33
+
34
+ it "should properly set line width via line_width(width)" do
35
+ @pdf.line_width(10)
36
+ line = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
37
+ line.widths.first.should == 10
38
+ end
39
+
40
+ it "should carry the current line width settings over to new pages" do
41
+ @pdf.line_width(10)
42
+ @pdf.start_new_page
43
+ line = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
44
+ line.widths.length.should == 2
45
+ line.widths[1].should == 10
46
+ end
47
+
48
+ describe "(Horizontally)" do
49
+ it "should draw from [x1,pdf.y],[x2,pdf.y]" do
50
+ @pdf.horizontal_line(100,150)
51
+ @line = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
52
+ @line.points.should == [[100.0 + @pdf.bounds.absolute_left, @pdf.y],
53
+ [150.0 + @pdf.bounds.absolute_left, @pdf.y]]
54
+ end
55
+
56
+ it "should draw a line from (200, 250) to (300, 250)" do
57
+ @pdf.horizontal_line(200,300,:at => 250)
58
+ line_drawing = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
59
+ line_drawing.points.should == [[200,250],[300,250]]
60
+ end
61
+ end
62
+
63
+ describe "(Vertically)" do
64
+ it "should draw a line from (350, 300) to (350, 400)" do
65
+ @pdf.vertical_line(300,400,:at => 350)
66
+ line_drawing = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
67
+ line_drawing.points.should == [[350,300],[350,400]]
68
+ end
69
+ it "should require a y coordinate" do
70
+ lambda { @pdf.vertical_line(400,500) }.
71
+ should raise_error(ArgumentError)
72
+ end
73
+ end
74
+
75
+ end
76
+
77
+ describe "When drawing a polygon" do
78
+
79
+ before(:each) { create_pdf }
80
+
81
+ it "should draw each line passed to polygon()" do
82
+ @pdf.polygon([100,500],[100,400],[200,400])
83
+
84
+ line_drawing = PDF::Inspector::Graphics::Line.analyze(@pdf.render)
85
+ line_drawing.points.should == [[100,500],[100,400],[200,400],[100,500]]
86
+ end
87
+
88
+ end
89
+
90
+ describe "When drawing a rectangle" do
91
+
92
+ before(:each) { create_pdf }
93
+
94
+ it "should use a point, width, and height for coords" do
95
+ @pdf.rectangle [200,200], 50, 100
96
+
97
+ rectangles = PDF::Inspector::Graphics::Rectangle.
98
+ analyze(@pdf.render).rectangles
99
+ # PDF uses bottom left corner
100
+ rectangles[0][:point].should == [200,100]
101
+ rectangles[0][:width].should == 50
102
+ rectangles[0][:height].should == 100
103
+
104
+ end
105
+
106
+ end
107
+
108
+ describe "When drawing a curve" do
109
+
110
+ before(:each) { create_pdf }
111
+
112
+ it "should draw a bezier curve from 50,50 to 100,100" do
113
+ @pdf.move_to [50,50]
114
+ @pdf.curve_to [100,100],:bounds => [[20,90], [90,70]]
115
+ curve = PDF::Inspector::Graphics::Curve.analyze(@pdf.render)
116
+ curve.coords.should == [50.0, 50.0, 20.0, 90.0, 90.0, 70.0, 100.0, 100.0]
117
+ end
118
+
119
+ it "should draw a bezier curve from 100,100 to 50,50" do
120
+ @pdf.curve [100,100], [50,50], :bounds => [[20,90], [90,75]]
121
+ curve = PDF::Inspector::Graphics::Curve.analyze(@pdf.render)
122
+ curve.coords.should == [100.0, 100.0, 20.0, 90.0, 90.0, 75.0, 50.0, 50.0]
123
+ end
124
+
125
+
126
+ end
127
+
128
+ describe "When drawing a rounded rectangle" do
129
+ before(:each) do
130
+ create_pdf
131
+ @pdf.rounded_rectangle([50, 550], 50, 100, 10)
132
+ curve = PDF::Inspector::Graphics::Curve.analyze(@pdf.render)
133
+ curve_points = []
134
+ curve.coords.each_slice(2) {|p| curve_points << p}
135
+ @original_point = curve_points.shift
136
+ curves = []
137
+ curve_points.each_slice(3) {|c| curves << c}
138
+ line_points = PDF::Inspector::Graphics::Line.analyze(@pdf.render).points
139
+ line_points.shift
140
+ @all_coords = []
141
+ line_points.zip(curves).flatten.each_slice(2) {|p| @all_coords << p }
142
+ @all_coords.unshift @original_point
143
+ end
144
+
145
+ it "should draw a rectangle by connecting lines with rounded bezier curves" do
146
+ @all_coords.should == [[60.0, 550.0],[90.0, 550.0], [95.523, 550.0], [100.0, 545.523], [100.0, 540.0],
147
+ [100.0, 460.0], [100.0, 454.477], [95.523, 450.0], [90.0, 450.0],
148
+ [60.0, 450.0], [54.477, 450.0], [50.0, 454.477], [50.0, 460.0],
149
+ [50.0, 540.0], [50.0, 545.523], [54.477, 550.0], [60.0, 550.0]]
150
+ end
151
+
152
+ it "should start and end with the same point" do
153
+ @original_point.should == @all_coords.last
154
+ end
155
+
156
+
157
+ end
158
+
159
+ describe "When drawing an ellipse" do
160
+ before(:each) do
161
+ create_pdf
162
+ @pdf.ellipse [100,100], 25, 50
163
+ @curve = PDF::Inspector::Graphics::Curve.analyze(@pdf.render)
164
+ end
165
+
166
+ it "should use a Bézier approximation" do
167
+ @curve.coords.should ==
168
+ [125.0, 100.0,
169
+
170
+ 125.0, 127.614,
171
+ 113.807, 150,
172
+ 100.0, 150.0,
173
+
174
+ 86.193, 150.0,
175
+ 75.0, 127.614,
176
+ 75.0, 100.0,
177
+
178
+ 75.0, 72.386,
179
+ 86.193, 50.0,
180
+ 100.0, 50.0,
181
+
182
+ 113.807, 50.0,
183
+ 125.0, 72.386,
184
+ 125.0, 100.0,
185
+
186
+ 100.0, 100.0]
187
+ end
188
+
189
+ it "should move the pointer to the center of the ellipse after drawing" do
190
+ @curve.coords[-2..-1].should == [100,100]
191
+ end
192
+
193
+ end
194
+
195
+ describe "When drawing a circle" do
196
+ before(:each) do
197
+ create_pdf
198
+ @pdf.circle [100,100], 25
199
+ @pdf.ellipse [100,100], 25, 25
200
+ @curve = PDF::Inspector::Graphics::Curve.analyze(@pdf.render)
201
+ end
202
+
203
+ it "should stroke the same path as the equivalent ellipse" do
204
+ middle = @curve.coords.length / 2
205
+ @curve.coords[0...middle].should == @curve.coords[middle..-1]
206
+ end
207
+ end
208
+
209
+ describe "When filling" do
210
+ before(:each) { create_pdf }
211
+
212
+ it "should default to the f operator (nonzero winding number rule)" do
213
+ @pdf.expects(:add_content).with("f")
214
+ @pdf.fill
215
+ end
216
+
217
+ it "should use f* for :fill_rule => :even_odd" do
218
+ @pdf.expects(:add_content).with("f*")
219
+ @pdf.fill(:fill_rule => :even_odd)
220
+ end
221
+
222
+ it "should use b by default for fill_and_stroke (nonzero winding number)" do
223
+ @pdf.expects(:add_content).with("b")
224
+ @pdf.fill_and_stroke
225
+ end
226
+
227
+ it "should use b* for fill_and_stroke(:fill_rule => :even_odd)" do
228
+ @pdf.expects(:add_content).with("b*")
229
+ @pdf.fill_and_stroke(:fill_rule => :even_odd)
230
+ end
231
+ end
232
+
233
+ describe "When setting colors" do
234
+
235
+ before(:each) { create_pdf }
236
+
237
+ it "should set stroke colors" do
238
+ @pdf.stroke_color "ffcccc"
239
+ colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
240
+ # 100% red, 80% green, 80% blue
241
+ colors.stroke_color.should == [1.0, 0.8, 0.8]
242
+ end
243
+
244
+ it "should set fill colors" do
245
+ @pdf.fill_color "ccff00"
246
+ colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
247
+ # 80% red, 100% green, 0% blue
248
+ colors.fill_color.should == [0.8,1.0,0]
249
+ end
250
+
251
+ it "should reset the colors on each new page if they have been defined" do
252
+ @pdf.fill_color "ccff00"
253
+ #colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
254
+
255
+ # colors.fill_color_count.should == 2
256
+ # colors.stroke_color_count.should == 1
257
+ @pdf.start_new_page
258
+ @pdf.stroke_color "ff00cc"
259
+
260
+ #colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
261
+ # colors.fill_color_count.should == 3
262
+
263
+ @pdf.start_new_page
264
+ colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
265
+ colors.fill_color_count.should == 3
266
+ colors.stroke_color_count.should == 2
267
+
268
+ colors.fill_color.should == [0.8,1.0,0.0]
269
+ colors.stroke_color.should == [1.0,0.0,0.8]
270
+ end
271
+
272
+ it "should set the color space when setting colors on new pages to please fussy readers" do
273
+ @pdf.stroke_color "000000"
274
+ @pdf.stroke { @pdf.rectangle([10, 10], 10, 10) }
275
+ @pdf.start_new_page
276
+ @pdf.stroke_color "000000"
277
+ @pdf.stroke { @pdf.rectangle([10, 10], 10, 10) }
278
+ colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
279
+ colors.stroke_color_space_count[:DeviceRGB].should == 2
280
+ end
281
+ end
282
+
283
+ describe "Patterns" do
284
+ before(:each) { create_pdf }
285
+
286
+ describe 'linear gradients' do
287
+ it "should create a /Pattern resource" do
288
+ @pdf.fill_gradient [0, @pdf.bounds.height],
289
+ [@pdf.bounds.width, @pdf.bounds.height], 'FF0000', '0000FF'
290
+
291
+ grad = PDF::Inspector::Graphics::Pattern.analyze(@pdf.render)
292
+ pattern = grad.patterns.values.first
293
+
294
+ pattern.should_not be_nil
295
+ pattern[:Shading][:ShadingType].should == 2
296
+ pattern[:Shading][:Coords].should == [0, 0, @pdf.bounds.width, 0]
297
+ pattern[:Shading][:Function][:C0].zip([1, 0, 0]).all?{ |x1, x2|
298
+ (x1-x2).abs < 0.01
299
+ }.should be_true
300
+ pattern[:Shading][:Function][:C1].zip([0, 0, 1]).all?{ |x1, x2|
301
+ (x1-x2).abs < 0.01
302
+ }.should be_true
303
+ end
304
+
305
+ it "fill_gradient should set fill color to the pattern" do
306
+ @pdf.fill_gradient [0, @pdf.bounds.height],
307
+ [@pdf.bounds.width, @pdf.bounds.height], 'FF0000', '0000FF'
308
+
309
+ str = @pdf.render
310
+ str.should =~ %r{/Pattern\s+cs\s*/SP-?\d+\s+scn}
311
+ end
312
+
313
+ it "stroke_gradient should set stroke color to the pattern" do
314
+ @pdf.stroke_gradient [0, @pdf.bounds.height],
315
+ [@pdf.bounds.width, @pdf.bounds.height], 'FF0000', '0000FF'
316
+
317
+ str = @pdf.render
318
+ str.should =~ %r{/Pattern\s+CS\s*/SP-?\d+\s+SCN}
319
+ end
320
+ end
321
+
322
+ describe 'radial gradients' do
323
+ it "should create a /Pattern resource" do
324
+ @pdf.fill_gradient [0, @pdf.bounds.height], 10,
325
+ [@pdf.bounds.width, @pdf.bounds.height], 20, 'FF0000', '0000FF'
326
+
327
+ grad = PDF::Inspector::Graphics::Pattern.analyze(@pdf.render)
328
+ pattern = grad.patterns.values.first
329
+
330
+ pattern.should_not be_nil
331
+ pattern[:Shading][:ShadingType].should == 3
332
+ pattern[:Shading][:Coords].should == [0, 0, 10, @pdf.bounds.width, 0, 20]
333
+ pattern[:Shading][:Function][:C0].zip([1, 0, 0]).all?{ |x1, x2|
334
+ (x1-x2).abs < 0.01
335
+ }.should be_true
336
+ pattern[:Shading][:Function][:C1].zip([0, 0, 1]).all?{ |x1, x2|
337
+ (x1-x2).abs < 0.01
338
+ }.should be_true
339
+ end
340
+
341
+ it "fill_gradient should set fill color to the pattern" do
342
+ @pdf.fill_gradient [0, @pdf.bounds.height], 10,
343
+ [@pdf.bounds.width, @pdf.bounds.height], 20, 'FF0000', '0000FF'
344
+
345
+ str = @pdf.render
346
+ str.should =~ %r{/Pattern\s+cs\s*/SP-?\d+\s+scn}
347
+ end
348
+
349
+ it "stroke_gradient should set stroke color to the pattern" do
350
+ @pdf.stroke_gradient [0, @pdf.bounds.height], 10,
351
+ [@pdf.bounds.width, @pdf.bounds.height], 20, 'FF0000', '0000FF'
352
+
353
+ str = @pdf.render
354
+ str.should =~ %r{/Pattern\s+CS\s*/SP-?\d+\s+SCN}
355
+ end
356
+ end
357
+ end
358
+
359
+ describe "When using painting shortcuts" do
360
+ before(:each) { create_pdf }
361
+
362
+ it "should convert stroke_some_method(args) into some_method(args); stroke" do
363
+ @pdf.expects(:line_to).with([100,100])
364
+ @pdf.expects(:stroke)
365
+
366
+ @pdf.stroke_line_to [100,100]
367
+ end
368
+
369
+ it "should convert fill_some_method(args) into some_method(args); fill" do
370
+ @pdf.expects(:line_to).with([100,100])
371
+ @pdf.expects(:fill)
372
+
373
+ @pdf.fill_line_to [100,100]
374
+ end
375
+
376
+ it "should not break method_missing" do
377
+ lambda { @pdf.i_have_a_pretty_girlfriend_named_jia }.
378
+ should raise_error(NoMethodError)
379
+ end
380
+ end
381
+
382
+ describe "When using graphics states" do
383
+ before(:each) { create_pdf }
384
+
385
+ it "should add the right content on save_graphics_state" do
386
+ @pdf.expects(:add_content).with('q')
387
+
388
+ @pdf.save_graphics_state
389
+ end
390
+
391
+ it "should add the right content on restore_graphics_state" do
392
+ @pdf.expects(:add_content).with('Q')
393
+
394
+ @pdf.restore_graphics_state
395
+ end
396
+
397
+ it "should save and restore when save_graphics_state is used with a block" do
398
+ state = sequence "state"
399
+ @pdf.expects(:add_content).with('q').in_sequence(state)
400
+ @pdf.expects(:foo).in_sequence(state)
401
+ @pdf.expects(:add_content).with('Q').in_sequence(state)
402
+
403
+ @pdf.save_graphics_state do
404
+ @pdf.foo
405
+ end
406
+ end
407
+
408
+ it "should add the previous color space when restoring to a graphic state with different color space" do
409
+ @pdf.stroke_color '000000'
410
+ @pdf.save_graphics_state
411
+ @pdf.stroke_color 0, 0, 0, 0
412
+ @pdf.restore_graphics_state
413
+ @pdf.stroke_color 0, 0, 100, 0
414
+ @pdf.graphic_state.color_space.should == {:stroke=>:DeviceCMYK}
415
+ colors = PDF::Inspector::Graphics::Color.analyze(@pdf.render)
416
+ colors.color_space.should == :DeviceCMYK
417
+ colors.stroke_color_space_count[:DeviceCMYK].should == 2
418
+ end
419
+
420
+ it "should use the correct dash setting after restoring and starting new page" do
421
+ @pdf.dash 5
422
+ @pdf.save_graphics_state
423
+ @pdf.dash 10
424
+ @pdf.graphic_state.dash[:dash].should == 10
425
+ @pdf.restore_graphics_state
426
+ @pdf.start_new_page
427
+ @pdf.graphic_state.dash[:dash].should == 5
428
+ end
429
+
430
+ it "the current graphic state should keep track of previous unchanged settings" do
431
+ @pdf.stroke_color '000000'
432
+ @pdf.save_graphics_state
433
+ @pdf.dash 5
434
+ @pdf.save_graphics_state
435
+ @pdf.cap_style :round
436
+ @pdf.save_graphics_state
437
+ @pdf.fill_color 0, 0, 100, 0
438
+ @pdf.save_graphics_state
439
+
440
+ @pdf.graphic_state.stroke_color.should == "000000"
441
+ @pdf.graphic_state.join_style.should == :miter
442
+ @pdf.graphic_state.fill_color.should == [0, 0, 100, 0]
443
+ @pdf.graphic_state.cap_style.should == :round
444
+ @pdf.graphic_state.color_space.should == {:fill=>:DeviceCMYK, :stroke=>:DeviceRGB}
445
+ @pdf.graphic_state.dash.should == {:space=>5, :phase=>0, :dash=>5}
446
+ @pdf.graphic_state.line_width.should == 1
447
+ end
448
+
449
+
450
+
451
+ it "should not add extra graphic space closings when rendering multiple times" do
452
+ @pdf.render
453
+ state = PDF::Inspector::Graphics::State.analyze(@pdf.render)
454
+ state.save_graphics_state_count.should == 1
455
+ state.restore_graphics_state_count.should == 1
456
+ end
457
+
458
+ it "should add extra graphic state enclosings when content is added on multiple renderings" do
459
+ @pdf.render
460
+ @pdf.text "Adding a bit more content"
461
+ state = PDF::Inspector::Graphics::State.analyze(@pdf.render)
462
+ state.save_graphics_state_count.should == 2
463
+ state.restore_graphics_state_count.should == 2
464
+ end
465
+
466
+ it "adds extra graphic state enclosings when new settings are applied on multiple renderings" do
467
+ @pdf.render
468
+ @pdf.stroke_color 0, 0, 0, 0
469
+ state = PDF::Inspector::Graphics::State.analyze(@pdf.render)
470
+ state.save_graphics_state_count.should == 2
471
+ state.restore_graphics_state_count.should == 2
472
+ end
473
+
474
+
475
+ it "should raise_error error if closing an empty graphic stack" do
476
+ lambda {
477
+ @pdf.render
478
+ @pdf.restore_graphics_state
479
+ }.should raise_error(Prawn::Errors::EmptyGraphicStateStack)
480
+ end
481
+ end
482
+
483
+ describe "When using transformation matrix" do
484
+ before(:each) { create_pdf }
485
+
486
+ # Note: The (approximate) number of significant decimal digits of precision in fractional
487
+ # part is 5 (PDF Reference, Third Edition, p. 706)
488
+
489
+ it "should send the right content on transformation_matrix" do
490
+ @pdf.expects(:add_content).with('1.00000 0.00000 0.12346 -1.00000 5.50000 20.00000 cm')
491
+ @pdf.transformation_matrix 1, 0, 0.123456789, -1.0, 5.5, 20
492
+ end
493
+
494
+ it "should use fixed digits with very small number" do
495
+ values = Array.new(6, 0.000000000001)
496
+ string = Array.new(6, "0.00000").join " "
497
+ @pdf.expects(:add_content).with("#{string} cm")
498
+ @pdf.transformation_matrix *values
499
+ end
500
+
501
+ it "should be received by the inspector" do
502
+ @pdf.transformation_matrix 1, 0, 0, -1, 5.5, 20
503
+ matrices = PDF::Inspector::Graphics::Matrix.analyze(@pdf.render)
504
+ matrices.matrices.should == [[1, 0, 0, -1, 5.5, 20]]
505
+ end
506
+
507
+ it "should save the graphics state inside the given block" do
508
+ values = Array.new(6, 0.000000000001)
509
+ string = Array.new(6, "0.00000").join " "
510
+ process = sequence "process"
511
+
512
+ @pdf.expects(:save_graphics_state).with().in_sequence(process)
513
+ @pdf.expects(:add_content).with("#{string} cm").in_sequence(process)
514
+ @pdf.expects(:do_something).with().in_sequence(process)
515
+ @pdf.expects(:restore_graphics_state).with().in_sequence(process)
516
+ @pdf.transformation_matrix(*values) do
517
+ @pdf.do_something
518
+ end
519
+ end
520
+
521
+ end
522
+
523
+ describe "When using transformations shortcuts" do
524
+ before(:each) do
525
+ create_pdf
526
+ @x, @y = 12, 54.32
527
+ @angle = 12.32
528
+ @cos = Math.cos(@angle * Math::PI / 180)
529
+ @sin = Math.sin(@angle * Math::PI / 180)
530
+ @factor = 0.12
531
+ end
532
+
533
+ describe "#rotate" do
534
+ it "should rotate" do
535
+ @pdf.expects(:transformation_matrix).with(@cos, @sin, -@sin, @cos, 0, 0)
536
+ @pdf.rotate(@angle)
537
+ end
538
+ end
539
+
540
+ describe "#rotate with :origin option" do
541
+ it "should rotate around the origin" do
542
+ x_prime = @x * @cos - @y * @sin
543
+ y_prime = @x * @sin + @y * @cos
544
+
545
+ @pdf.rotate(@angle, :origin => [@x, @y]) { @pdf.text('hello world') }
546
+
547
+ matrices = PDF::Inspector::Graphics::Matrix.analyze(@pdf.render)
548
+ matrices.matrices[0].should == [1, 0, 0, 1,
549
+ reduce_precision(@x - x_prime),
550
+ reduce_precision(@y - y_prime)]
551
+ matrices.matrices[1].should == [reduce_precision(@cos),
552
+ reduce_precision(@sin),
553
+ reduce_precision(-@sin),
554
+ reduce_precision(@cos), 0, 0]
555
+ end
556
+
557
+ it "should rotate around the origin in a document with a margin" do
558
+ @pdf = Prawn::Document.new
559
+
560
+ @pdf.rotate(@angle, :origin => [@x, @y]) { @pdf.text('hello world') }
561
+
562
+ x = @x + @pdf.bounds.absolute_left
563
+ y = @y + @pdf.bounds.absolute_bottom
564
+ x_prime = x * @cos - y * @sin
565
+ y_prime = x * @sin + y * @cos
566
+
567
+ matrices = PDF::Inspector::Graphics::Matrix.analyze(@pdf.render)
568
+ matrices.matrices[0].should == [1, 0, 0, 1,
569
+ reduce_precision(x - x_prime),
570
+ reduce_precision(y - y_prime)]
571
+ matrices.matrices[1].should == [reduce_precision(@cos),
572
+ reduce_precision(@sin),
573
+ reduce_precision(-@sin),
574
+ reduce_precision(@cos), 0, 0]
575
+ end
576
+
577
+ it "should raise_error BlockRequired if no block is given" do
578
+ lambda {
579
+ @pdf.rotate(@angle, :origin => [@x, @y])
580
+ }.should raise_error(Prawn::Errors::BlockRequired)
581
+ end
582
+
583
+ def reduce_precision(float)
584
+ ("%.5f" % float).to_f
585
+ end
586
+ end
587
+
588
+ describe "#translate" do
589
+ it "should translate" do
590
+ x, y = 12, 54.32
591
+ @pdf.expects(:transformation_matrix).with(1, 0, 0, 1, x, y)
592
+ @pdf.translate(x, y)
593
+ end
594
+ end
595
+
596
+ describe "#scale" do
597
+ it "should scale" do
598
+ @pdf.expects(:transformation_matrix).with(@factor, 0, 0, @factor, 0, 0)
599
+ @pdf.scale(@factor)
600
+ end
601
+ end
602
+
603
+ describe "#scale with :origin option" do
604
+ it "should scale from the origin" do
605
+ x_prime = @factor * @x
606
+ y_prime = @factor * @y
607
+
608
+ @pdf.scale(@factor, :origin => [@x, @y]) { @pdf.text('hello world') }
609
+
610
+ matrices = PDF::Inspector::Graphics::Matrix.analyze(@pdf.render)
611
+ matrices.matrices[0].should == [1, 0, 0, 1,
612
+ reduce_precision(@x - x_prime),
613
+ reduce_precision(@y - y_prime)]
614
+ matrices.matrices[1].should == [@factor, 0, 0, @factor, 0, 0]
615
+ end
616
+
617
+ it "should scale from the origin in a document with a margin" do
618
+ @pdf = Prawn::Document.new
619
+ x = @x + @pdf.bounds.absolute_left
620
+ y = @y + @pdf.bounds.absolute_bottom
621
+ x_prime = @factor * x
622
+ y_prime = @factor * y
623
+
624
+ @pdf.scale(@factor, :origin => [@x, @y]) { @pdf.text('hello world') }
625
+
626
+ matrices = PDF::Inspector::Graphics::Matrix.analyze(@pdf.render)
627
+ matrices.matrices[0].should == [1, 0, 0, 1,
628
+ reduce_precision(x - x_prime),
629
+ reduce_precision(y - y_prime)]
630
+ matrices.matrices[1].should == [@factor, 0, 0, @factor, 0, 0]
631
+ end
632
+
633
+ it "should raise_error BlockRequired if no block is given" do
634
+ lambda {
635
+ @pdf.scale(@factor, :origin => [@x, @y])
636
+ }.should raise_error(Prawn::Errors::BlockRequired)
637
+ end
638
+
639
+ def reduce_precision(float)
640
+ ("%.5f" % float).to_f
641
+ end
642
+ end
643
+
644
+ # describe "skew" do
645
+ # it "should skew" do
646
+ # a, b = 30, 50.2
647
+ # @pdf.expects(:transformation_matrix).with(1, Math.tan(a * Math::PI / 180), Math.tan(b * Math::PI / 180), 1, 0, 0)
648
+ # @pdf.skew(a, b)
649
+ # end
650
+ # end
651
+ end