prawn 0.4.1 → 0.5.0.1
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.
- data/lib/prawn.rb +2 -72
- metadata +33 -224
- data/COPYING +0 -340
- data/LICENSE +0 -56
- data/README +0 -40
- data/Rakefile +0 -79
- data/data/encodings/win_ansi.txt +0 -29
- data/data/fonts/Action Man.dfont +0 -0
- data/data/fonts/Activa.ttf +0 -0
- data/data/fonts/Chalkboard.ttf +0 -0
- data/data/fonts/Courier-Bold.afm +0 -342
- data/data/fonts/Courier-BoldOblique.afm +0 -342
- data/data/fonts/Courier-Oblique.afm +0 -342
- data/data/fonts/Courier.afm +0 -342
- data/data/fonts/DejaVuSans.ttf +0 -0
- data/data/fonts/Dustismo_Roman.ttf +0 -0
- data/data/fonts/Helvetica-Bold.afm +0 -2827
- data/data/fonts/Helvetica-BoldOblique.afm +0 -2827
- data/data/fonts/Helvetica-Oblique.afm +0 -3051
- data/data/fonts/Helvetica.afm +0 -3051
- data/data/fonts/MustRead.html +0 -19
- data/data/fonts/Symbol.afm +0 -213
- data/data/fonts/Times-Bold.afm +0 -2588
- data/data/fonts/Times-BoldItalic.afm +0 -2384
- data/data/fonts/Times-Italic.afm +0 -2667
- data/data/fonts/Times-Roman.afm +0 -2419
- data/data/fonts/ZapfDingbats.afm +0 -225
- data/data/fonts/comicsans.ttf +0 -0
- data/data/fonts/gkai00mp.ttf +0 -0
- data/data/images/arrow.png +0 -0
- data/data/images/arrow2.png +0 -0
- data/data/images/barcode_issue.png +0 -0
- data/data/images/dice.alpha +0 -0
- data/data/images/dice.dat +0 -0
- data/data/images/dice.png +0 -0
- data/data/images/fractal.jpg +0 -0
- data/data/images/letterhead.jpg +0 -0
- data/data/images/page_white_text.alpha +0 -0
- data/data/images/page_white_text.dat +0 -0
- data/data/images/page_white_text.png +0 -0
- data/data/images/pigs.jpg +0 -0
- data/data/images/rails.dat +0 -0
- data/data/images/rails.png +0 -0
- data/data/images/ruport.png +0 -0
- data/data/images/ruport_data.dat +0 -0
- data/data/images/ruport_transparent.png +0 -0
- data/data/images/ruport_type0.png +0 -0
- data/data/images/stef.jpg +0 -0
- data/data/images/web-links.dat +0 -1
- data/data/images/web-links.png +0 -0
- data/data/shift_jis_text.txt +0 -1
- data/examples/bounding_box/bounding_boxes.rb +0 -44
- data/examples/bounding_box/lazy_bounding_boxes.rb +0 -28
- data/examples/bounding_box/padded_box.rb +0 -24
- data/examples/bounding_box/russian_boxes.rb +0 -37
- data/examples/general/background.rb +0 -20
- data/examples/general/canvas.rb +0 -16
- data/examples/general/measurement_units.rb +0 -52
- data/examples/general/multi_page_layout.rb +0 -17
- data/examples/general/page_geometry.rb +0 -32
- data/examples/graphics/basic_images.rb +0 -27
- data/examples/graphics/cmyk.rb +0 -13
- data/examples/graphics/curves.rb +0 -12
- data/examples/graphics/hexagon.rb +0 -14
- data/examples/graphics/image_fit.rb +0 -16
- data/examples/graphics/image_flow.rb +0 -38
- data/examples/graphics/image_position.rb +0 -18
- data/examples/graphics/line.rb +0 -33
- data/examples/graphics/png_types.rb +0 -23
- data/examples/graphics/polygons.rb +0 -17
- data/examples/graphics/remote_images.rb +0 -12
- data/examples/graphics/ruport_style_helpers.rb +0 -20
- data/examples/graphics/stroke_bounds.rb +0 -23
- data/examples/m17n/chinese_text_wrapping.rb +0 -20
- data/examples/m17n/euro.rb +0 -16
- data/examples/m17n/sjis.rb +0 -29
- data/examples/m17n/utf8.rb +0 -14
- data/examples/m17n/win_ansi_charset.rb +0 -55
- data/examples/text/alignment.rb +0 -19
- data/examples/text/dfont.rb +0 -49
- data/examples/text/family_based_styling.rb +0 -25
- data/examples/text/flowing_text_with_header_and_footer.rb +0 -37
- data/examples/text/font_calculations.rb +0 -92
- data/examples/text/font_size.rb +0 -34
- data/examples/text/kerning.rb +0 -31
- data/examples/text/simple_text.rb +0 -18
- data/examples/text/simple_text_ttf.rb +0 -18
- data/examples/text/span.rb +0 -30
- data/examples/text/text_box.rb +0 -26
- data/examples/text/text_flow.rb +0 -68
- data/lib/prawn/compatibility.rb +0 -38
- data/lib/prawn/document.rb +0 -309
- data/lib/prawn/document/annotations.rb +0 -63
- data/lib/prawn/document/bounding_box.rb +0 -368
- data/lib/prawn/document/destinations.rb +0 -81
- data/lib/prawn/document/internals.rb +0 -126
- data/lib/prawn/document/page_geometry.rb +0 -79
- data/lib/prawn/document/span.rb +0 -55
- data/lib/prawn/document/text.rb +0 -185
- data/lib/prawn/document/text/box.rb +0 -76
- data/lib/prawn/document/text/wrapping.rb +0 -59
- data/lib/prawn/encoding.rb +0 -121
- data/lib/prawn/errors.rb +0 -40
- data/lib/prawn/font.rb +0 -277
- data/lib/prawn/font/afm.rb +0 -202
- data/lib/prawn/font/dfont.rb +0 -31
- data/lib/prawn/font/ttf.rb +0 -326
- data/lib/prawn/graphics.rb +0 -257
- data/lib/prawn/graphics/color.rb +0 -140
- data/lib/prawn/images.rb +0 -339
- data/lib/prawn/images/jpg.rb +0 -45
- data/lib/prawn/images/png.rb +0 -199
- data/lib/prawn/literal_string.rb +0 -14
- data/lib/prawn/measurement_extensions.rb +0 -46
- data/lib/prawn/measurements.rb +0 -71
- data/lib/prawn/name_tree.rb +0 -165
- data/lib/prawn/pdf_object.rb +0 -73
- data/lib/prawn/reference.rb +0 -59
- data/spec/annotations_spec.rb +0 -90
- data/spec/bounding_box_spec.rb +0 -141
- data/spec/destinations_spec.rb +0 -15
- data/spec/document_spec.rb +0 -193
- data/spec/font_spec.rb +0 -234
- data/spec/graphics_spec.rb +0 -209
- data/spec/images_spec.rb +0 -68
- data/spec/jpg_spec.rb +0 -25
- data/spec/measurement_units_spec.rb +0 -23
- data/spec/name_tree_spec.rb +0 -103
- data/spec/pdf_object_spec.rb +0 -112
- data/spec/png_spec.rb +0 -196
- data/spec/reference_spec.rb +0 -42
- data/spec/spec_helper.rb +0 -23
- data/spec/text_spec.rb +0 -178
- data/vendor/pdf-inspector/README +0 -18
- data/vendor/pdf-inspector/lib/pdf/inspector.rb +0 -25
- data/vendor/pdf-inspector/lib/pdf/inspector/graphics.rb +0 -80
- data/vendor/pdf-inspector/lib/pdf/inspector/page.rb +0 -16
- data/vendor/pdf-inspector/lib/pdf/inspector/text.rb +0 -31
- data/vendor/pdf-inspector/lib/pdf/inspector/xobject.rb +0 -19
- data/vendor/ttfunk/data/fonts/DejaVuSans.ttf +0 -0
- data/vendor/ttfunk/data/fonts/comicsans.ttf +0 -0
- data/vendor/ttfunk/example.rb +0 -45
- data/vendor/ttfunk/lib/ttfunk.rb +0 -102
- data/vendor/ttfunk/lib/ttfunk/directory.rb +0 -17
- data/vendor/ttfunk/lib/ttfunk/encoding/mac_roman.rb +0 -88
- data/vendor/ttfunk/lib/ttfunk/encoding/windows_1252.rb +0 -69
- data/vendor/ttfunk/lib/ttfunk/reader.rb +0 -44
- data/vendor/ttfunk/lib/ttfunk/resource_file.rb +0 -78
- data/vendor/ttfunk/lib/ttfunk/subset.rb +0 -18
- data/vendor/ttfunk/lib/ttfunk/subset/base.rb +0 -141
- data/vendor/ttfunk/lib/ttfunk/subset/mac_roman.rb +0 -46
- data/vendor/ttfunk/lib/ttfunk/subset/unicode.rb +0 -48
- data/vendor/ttfunk/lib/ttfunk/subset/unicode_8bit.rb +0 -63
- data/vendor/ttfunk/lib/ttfunk/subset/windows_1252.rb +0 -51
- data/vendor/ttfunk/lib/ttfunk/subset_collection.rb +0 -72
- data/vendor/ttfunk/lib/ttfunk/table.rb +0 -46
- data/vendor/ttfunk/lib/ttfunk/table/cmap.rb +0 -34
- data/vendor/ttfunk/lib/ttfunk/table/cmap/format00.rb +0 -54
- data/vendor/ttfunk/lib/ttfunk/table/cmap/format04.rb +0 -126
- data/vendor/ttfunk/lib/ttfunk/table/cmap/subtable.rb +0 -79
- data/vendor/ttfunk/lib/ttfunk/table/glyf.rb +0 -64
- data/vendor/ttfunk/lib/ttfunk/table/glyf/compound.rb +0 -81
- data/vendor/ttfunk/lib/ttfunk/table/glyf/simple.rb +0 -37
- data/vendor/ttfunk/lib/ttfunk/table/head.rb +0 -44
- data/vendor/ttfunk/lib/ttfunk/table/hhea.rb +0 -41
- data/vendor/ttfunk/lib/ttfunk/table/hmtx.rb +0 -47
- data/vendor/ttfunk/lib/ttfunk/table/kern.rb +0 -79
- data/vendor/ttfunk/lib/ttfunk/table/kern/format0.rb +0 -62
- data/vendor/ttfunk/lib/ttfunk/table/loca.rb +0 -43
- data/vendor/ttfunk/lib/ttfunk/table/maxp.rb +0 -40
- data/vendor/ttfunk/lib/ttfunk/table/name.rb +0 -119
- data/vendor/ttfunk/lib/ttfunk/table/os2.rb +0 -78
- data/vendor/ttfunk/lib/ttfunk/table/post.rb +0 -91
- data/vendor/ttfunk/lib/ttfunk/table/post/format10.rb +0 -43
- data/vendor/ttfunk/lib/ttfunk/table/post/format20.rb +0 -35
- data/vendor/ttfunk/lib/ttfunk/table/post/format25.rb +0 -23
- data/vendor/ttfunk/lib/ttfunk/table/post/format30.rb +0 -17
- data/vendor/ttfunk/lib/ttfunk/table/post/format40.rb +0 -17
- data/vendor/ttfunk/lib/ttfunk/table/simple.rb +0 -14
@@ -1,368 +0,0 @@
|
|
1
|
-
# encoding: utf-8
|
2
|
-
|
3
|
-
# bounding_box.rb : Implements a mechanism for shifting the coordinate space
|
4
|
-
#
|
5
|
-
# Copyright May 2008, Gregory Brown. All Rights Reserved.
|
6
|
-
#
|
7
|
-
# This is free software. Please see the LICENSE and COPYING files for details.
|
8
|
-
|
9
|
-
module Prawn
|
10
|
-
class Document
|
11
|
-
|
12
|
-
# A bounding box serves two important purposes:
|
13
|
-
# * Provide bounds for flowing text, starting at a given point
|
14
|
-
# * Translate the origin (0,0) for graphics primitives, for the purposes
|
15
|
-
# of simplifying coordinate math.
|
16
|
-
#
|
17
|
-
# When flowing text, the usage of a bounding box is simple. Text will
|
18
|
-
# begin at the point specified, flowing the width of the bounding box.
|
19
|
-
# After the block exits, the cursor position will be moved to
|
20
|
-
# the bottom of the bounding box (y - height). If flowing text exceeds
|
21
|
-
# the height of the bounding box, the text will be continued on the next
|
22
|
-
# page, starting again at the top-left corner of the bounding box.
|
23
|
-
#
|
24
|
-
# pdf.bounding_box([100,500], :width => 100, :height => 300) do
|
25
|
-
# pdf.text "This text will flow in a very narrow box starting" +
|
26
|
-
# "from [100,500]. The pointer will then be moved to [100,200]" +
|
27
|
-
# "and return to the margin_box"
|
28
|
-
# end
|
29
|
-
#
|
30
|
-
# When translating coordinates, the idea is to allow the user to draw
|
31
|
-
# relative to the origin, and then translate their drawing to a specified
|
32
|
-
# area of the document, rather than adjust all their drawing coordinates
|
33
|
-
# to match this new region.
|
34
|
-
#
|
35
|
-
# Take for example two triangles which share one point, drawn from the
|
36
|
-
# origin:
|
37
|
-
#
|
38
|
-
# pdf.polygon [0,250], [0,0], [150,100]
|
39
|
-
# pdf.polygon [100,0], [150,100], [200,0]
|
40
|
-
#
|
41
|
-
# It would be easy enough to translate these triangles to another point,
|
42
|
-
# e.g [200,200]
|
43
|
-
#
|
44
|
-
# pdf.polygon [200,450], [200,200], [350,300]
|
45
|
-
# pdf.polygon [300,200], [350,300], [400,200]
|
46
|
-
#
|
47
|
-
# However, each time you want to move the drawing, you'd need to alter
|
48
|
-
# every point in the drawing calls, which as you might imagine, can become
|
49
|
-
# tedious.
|
50
|
-
#
|
51
|
-
# If instead, we think of the drawing as being bounded by a box, we can
|
52
|
-
# see that the image is 200 points wide by 250 points tall.
|
53
|
-
#
|
54
|
-
# To translate it to a new origin, we simply select a point at (x,y+height)
|
55
|
-
#
|
56
|
-
# Using the [200,200] example:
|
57
|
-
#
|
58
|
-
# pdf.bounding_box([200,450], :width => 200, :height => 250) do
|
59
|
-
# pdf.polygon [0,250], [0,0], [150,100]
|
60
|
-
# pdf.polygon [100,0], [150,100], [200,0]
|
61
|
-
# end
|
62
|
-
#
|
63
|
-
# Notice that the drawing is still relative to the origin. If we want to
|
64
|
-
# move this drawing around the document, we simply need to recalculate the
|
65
|
-
# top-left corner of the rectangular bounding-box, and all of our graphics
|
66
|
-
# calls remain unmodified.
|
67
|
-
#
|
68
|
-
# By default, bounding boxes are specified relative to the document's
|
69
|
-
# margin_box (which is itself a bounding box). You can also nest bounding
|
70
|
-
# boxes, allowing you to build components which are relative to each other
|
71
|
-
#
|
72
|
-
# pdf.bouding_box([200,450], :width => 200, :height => 250) do
|
73
|
-
# pdf.bounding_box([50,200], :width => 50, :height => 50) do
|
74
|
-
# # a 50x50 bounding box that starts 50 pixels left and 50 pixels down
|
75
|
-
# # the parent bounding box.
|
76
|
-
# end
|
77
|
-
# end
|
78
|
-
#
|
79
|
-
# If you wish to position the bounding boxes at absolute coordinates rather
|
80
|
-
# than relative to the margins or other bounding boxes, you can use canvas()
|
81
|
-
#
|
82
|
-
# pdf.canvas do
|
83
|
-
# pdf.bounding_box([200,450], :width => 200, :height => 250) do
|
84
|
-
# # positioned at 'real' (200,450)
|
85
|
-
# end
|
86
|
-
# end
|
87
|
-
#
|
88
|
-
# Of course, if you use canvas, you will be responsible for ensuring that
|
89
|
-
# you remain within the printable area of your document.
|
90
|
-
#
|
91
|
-
def bounding_box(*args, &block)
|
92
|
-
init_bounding_box(block) do |_|
|
93
|
-
translate!(args[0])
|
94
|
-
@bounding_box = BoundingBox.new(self, *args)
|
95
|
-
end
|
96
|
-
end
|
97
|
-
|
98
|
-
|
99
|
-
# A LazyBoundingBox is simply a BoundingBox with an action tied to it to be
|
100
|
-
# executed later. The lazy_bounding_box method takes the same arguments as
|
101
|
-
# bounding_box, but returns a LazyBoundingBox object instead of executing
|
102
|
-
# the code block directly.
|
103
|
-
#
|
104
|
-
# You can then call LazyBoundingBox#draw at any time (or multiple times if
|
105
|
-
# you wish), and the contents of the block will then be run. This can be
|
106
|
-
# useful for assembling repeating page elements or reusable components.
|
107
|
-
#
|
108
|
-
# file = "lazy_bounding_boxes.pdf"
|
109
|
-
# Prawn::Document.generate(file, :skip_page_creation => true) do
|
110
|
-
# point = [bounds.right-50, bounds.bottom + 25]
|
111
|
-
# page_counter = lazy_bounding_box(point, :width => 50) do
|
112
|
-
# text "Page: #{page_count}"
|
113
|
-
# end
|
114
|
-
#
|
115
|
-
# 10.times do
|
116
|
-
# start_new_page
|
117
|
-
# text "Some text"
|
118
|
-
# page_counter.draw
|
119
|
-
# end
|
120
|
-
# end
|
121
|
-
#
|
122
|
-
def lazy_bounding_box(*args,&block)
|
123
|
-
translate!(args[0])
|
124
|
-
box = LazyBoundingBox.new(self,*args)
|
125
|
-
box.action(&block)
|
126
|
-
return box
|
127
|
-
end
|
128
|
-
|
129
|
-
# A shortcut to produce a bounding box which is mapped to the document's
|
130
|
-
# absolute coordinates, regardless of how things are nested or margin sizes.
|
131
|
-
#
|
132
|
-
# pdf.canvas do
|
133
|
-
# pdf.line pdf.bounds.bottom_left, pdf.bounds.top_right
|
134
|
-
# end
|
135
|
-
#
|
136
|
-
def canvas(&block)
|
137
|
-
init_bounding_box(block, :hold_position => true) do |_|
|
138
|
-
@bounding_box = BoundingBox.new(self, [0,page_dimensions[3]],
|
139
|
-
:width => page_dimensions[2],
|
140
|
-
:height => page_dimensions[3]
|
141
|
-
)
|
142
|
-
end
|
143
|
-
end
|
144
|
-
|
145
|
-
# A bounding box with the same dimensions of its parents, minus a margin
|
146
|
-
# on all sides
|
147
|
-
#
|
148
|
-
def padded_box(margin, &block)
|
149
|
-
bounding_box [bounds.left + margin, bounds.top - margin],
|
150
|
-
:width => bounds.width - (margin * 2),
|
151
|
-
:height => bounds.height - (margin * 2), &block
|
152
|
-
end
|
153
|
-
|
154
|
-
# A header is a LazyBoundingBox drawn relative to the margins that can be
|
155
|
-
# repeated on every page of the document.
|
156
|
-
#
|
157
|
-
# Unless <tt>:width</tt> or <tt>:height</tt> are specified, the margin_box
|
158
|
-
# width and height are used.
|
159
|
-
#
|
160
|
-
# header margin_box.top_left do
|
161
|
-
# text "Here's My Fancy Header", :size => 25, :align => :center
|
162
|
-
# stroke_horizontal_rule
|
163
|
-
# end
|
164
|
-
#
|
165
|
-
def header(top_left,options={},&block)
|
166
|
-
@header = repeating_page_element(top_left,options,&block)
|
167
|
-
end
|
168
|
-
|
169
|
-
# A footer is a LazyBoundingBox drawn relative to the margins that can be
|
170
|
-
# repeated on every page of the document.
|
171
|
-
#
|
172
|
-
# Unless <tt>:width</tt> or <tt>:height</tt> are specified, the margin_box
|
173
|
-
# width and height are used.
|
174
|
-
#
|
175
|
-
# footer [margin_box.left, margin_box.bottom + 25] do
|
176
|
-
# stroke_horizontal_rule
|
177
|
-
# text "And here's a sexy footer", :size => 16
|
178
|
-
# end
|
179
|
-
#
|
180
|
-
def footer(top_left,options={},&block)
|
181
|
-
@footer = repeating_page_element(top_left,options,&block)
|
182
|
-
end
|
183
|
-
|
184
|
-
private
|
185
|
-
|
186
|
-
def init_bounding_box(user_block, options={}, &init_block)
|
187
|
-
parent_box = @bounding_box
|
188
|
-
|
189
|
-
init_block.call(parent_box)
|
190
|
-
|
191
|
-
self.y = @bounding_box.absolute_top
|
192
|
-
user_block.call
|
193
|
-
self.y = @bounding_box.absolute_bottom unless options[:hold_position]
|
194
|
-
|
195
|
-
@bounding_box = parent_box
|
196
|
-
end
|
197
|
-
|
198
|
-
def repeating_page_element(top_left,options={},&block)
|
199
|
-
r = LazyBoundingBox.new(self, translate(top_left),
|
200
|
-
:width => options[:width] || margin_box.width,
|
201
|
-
:height => options[:height] || margin_box.height )
|
202
|
-
r.action(&block)
|
203
|
-
return r
|
204
|
-
end
|
205
|
-
|
206
|
-
class BoundingBox
|
207
|
-
|
208
|
-
def initialize(parent, point, options={}) #:nodoc:
|
209
|
-
@parent = parent
|
210
|
-
@x, @y = point
|
211
|
-
@width, @height = options[:width], options[:height]
|
212
|
-
end
|
213
|
-
|
214
|
-
# The translated origin (x,y-height) which describes the location
|
215
|
-
# of the bottom left corner of the bounding box
|
216
|
-
#
|
217
|
-
def anchor
|
218
|
-
[@x, @y - height]
|
219
|
-
end
|
220
|
-
|
221
|
-
# Relative left x-coordinate of the bounding box. (Always 0)
|
222
|
-
#
|
223
|
-
def left
|
224
|
-
0
|
225
|
-
end
|
226
|
-
|
227
|
-
# Relative right x-coordinate of the bounding box. (Equal to the box width)
|
228
|
-
#
|
229
|
-
def right
|
230
|
-
@width
|
231
|
-
end
|
232
|
-
|
233
|
-
# Relative top y-coordinate of the bounding box. (Equal to the box height)
|
234
|
-
#
|
235
|
-
def top
|
236
|
-
height
|
237
|
-
end
|
238
|
-
|
239
|
-
# Relative bottom y-coordinate of the bounding box (Always 0)
|
240
|
-
#
|
241
|
-
def bottom
|
242
|
-
0
|
243
|
-
end
|
244
|
-
|
245
|
-
# Relative top-left point of the bounding_box
|
246
|
-
#
|
247
|
-
def top_left
|
248
|
-
[left,top]
|
249
|
-
end
|
250
|
-
|
251
|
-
# Relative top-right point of the bounding box
|
252
|
-
#
|
253
|
-
def top_right
|
254
|
-
[right,top]
|
255
|
-
end
|
256
|
-
|
257
|
-
# Relative bottom-right point of the bounding box
|
258
|
-
#
|
259
|
-
def bottom_right
|
260
|
-
[right,bottom]
|
261
|
-
end
|
262
|
-
|
263
|
-
# Relative bottom-left point of the bounding box
|
264
|
-
#
|
265
|
-
def bottom_left
|
266
|
-
[left,bottom]
|
267
|
-
end
|
268
|
-
|
269
|
-
# Absolute left x-coordinate of the bounding box
|
270
|
-
#
|
271
|
-
def absolute_left
|
272
|
-
@x
|
273
|
-
end
|
274
|
-
|
275
|
-
# Absolute right x-coordinate of the bounding box
|
276
|
-
#
|
277
|
-
def absolute_right
|
278
|
-
@x + width
|
279
|
-
end
|
280
|
-
|
281
|
-
# Absolute top y-coordinate of the bounding box
|
282
|
-
#
|
283
|
-
def absolute_top
|
284
|
-
@y
|
285
|
-
end
|
286
|
-
|
287
|
-
# Absolute bottom y-coordinate of the bottom box
|
288
|
-
#
|
289
|
-
def absolute_bottom
|
290
|
-
@y - height
|
291
|
-
end
|
292
|
-
|
293
|
-
# Absolute top-left point of the bounding box
|
294
|
-
#
|
295
|
-
def absolute_top_left
|
296
|
-
[absolute_left, absolute_top]
|
297
|
-
end
|
298
|
-
|
299
|
-
# Absolute top-right point of the bounding box
|
300
|
-
#
|
301
|
-
def absolute_top_right
|
302
|
-
[absolute_right, absolute_top]
|
303
|
-
end
|
304
|
-
|
305
|
-
# Absolute bottom-left point of the bounding box
|
306
|
-
#
|
307
|
-
def absolute_bottom_left
|
308
|
-
[absolute_left, absolute_bottom]
|
309
|
-
end
|
310
|
-
|
311
|
-
# Absolute bottom-left point of the bounding box
|
312
|
-
#
|
313
|
-
def absolute_bottom_right
|
314
|
-
[absolute_right, absolute_bottom]
|
315
|
-
end
|
316
|
-
|
317
|
-
# Width of the bounding box
|
318
|
-
#
|
319
|
-
def width
|
320
|
-
@width
|
321
|
-
end
|
322
|
-
|
323
|
-
# Height of the bounding box. If the box is 'stretchy' (unspecified
|
324
|
-
# height attribute), height is calculated as the distance from the top of
|
325
|
-
# the box to the current drawing position.
|
326
|
-
#
|
327
|
-
def height
|
328
|
-
@height || absolute_top - @parent.y
|
329
|
-
end
|
330
|
-
|
331
|
-
# Returns +false+ when the box has a defined height, +true+ when the height
|
332
|
-
# is being calculated on the fly based on the current vertical position.
|
333
|
-
#
|
334
|
-
def stretchy?
|
335
|
-
!@height
|
336
|
-
end
|
337
|
-
|
338
|
-
end
|
339
|
-
|
340
|
-
class LazyBoundingBox < BoundingBox
|
341
|
-
|
342
|
-
# Defines the block to be executed by LazyBoundingBox#draw.
|
343
|
-
# Usually, this will be used via a higher level interface.
|
344
|
-
# See the documentation for Document#lazy_bounding_box, Document#header,
|
345
|
-
# and Document#footer
|
346
|
-
#
|
347
|
-
def action(&block)
|
348
|
-
@action = block
|
349
|
-
end
|
350
|
-
|
351
|
-
# Sets Document#bounds to use the LazyBoundingBox for its bounds,
|
352
|
-
# runs the block specified by LazyBoundingBox#action,
|
353
|
-
# and then restores the original bounds of the document.
|
354
|
-
#
|
355
|
-
def draw
|
356
|
-
@parent.mask(:y) do
|
357
|
-
parent_box = @parent.bounds
|
358
|
-
@parent.bounds = self
|
359
|
-
@parent.y = absolute_top
|
360
|
-
@action.call
|
361
|
-
@parent.bounds = parent_box
|
362
|
-
end
|
363
|
-
end
|
364
|
-
|
365
|
-
end
|
366
|
-
|
367
|
-
end
|
368
|
-
end
|
@@ -1,81 +0,0 @@
|
|
1
|
-
# encoding: utf-8
|
2
|
-
|
3
|
-
# annotations.rb : Implements destination support for PDF
|
4
|
-
#
|
5
|
-
# Copyright November 2008, Jamis Buck. All Rights Reserved.
|
6
|
-
#
|
7
|
-
# This is free software. Please see the LICENSE and COPYING files for details.
|
8
|
-
#
|
9
|
-
require 'prawn/name_tree'
|
10
|
-
|
11
|
-
module Prawn
|
12
|
-
class Document
|
13
|
-
module Destinations
|
14
|
-
# The maximum number of children to fit into a single node in the Dests tree.
|
15
|
-
NAME_TREE_CHILDREN_LIMIT = 20 #:nodoc:
|
16
|
-
|
17
|
-
# The Dests name tree in the Name dictionary (see Prawn::Document::Internal#names).
|
18
|
-
# This name tree is used to store named destinations (PDF spec 8.2.1).
|
19
|
-
# (For more on name trees, see section 3.8.4 in the PDF spec.)
|
20
|
-
def dests
|
21
|
-
names.data[:Dests] ||= ref(Prawn::NameTree::Node.new(self, NAME_TREE_CHILDREN_LIMIT))
|
22
|
-
end
|
23
|
-
|
24
|
-
# Adds a new destination to the dests name tree (see #dests). The
|
25
|
-
# +reference+ parameter will be converted into a Prawn::Reference if
|
26
|
-
# it is not already one.
|
27
|
-
def add_dest(name, reference)
|
28
|
-
reference = ref(reference) unless reference.is_a?(Prawn::Reference)
|
29
|
-
dests.data.add(name, reference)
|
30
|
-
end
|
31
|
-
|
32
|
-
# Return a Dest specification for a specific location (and optional zoom
|
33
|
-
# level).
|
34
|
-
def dest_xyz(left, top, zoom=nil, page=@current_page)
|
35
|
-
[page, :XYZ, left, top, zoom]
|
36
|
-
end
|
37
|
-
|
38
|
-
# Return a Dest specification that will fit the given page into the
|
39
|
-
# viewport.
|
40
|
-
def dest_fit(page=@current_page)
|
41
|
-
[page, :Fit]
|
42
|
-
end
|
43
|
-
|
44
|
-
# Return a Dest specification that will fit the given page horizontally
|
45
|
-
# into the viewport, aligned vertically at the given top coordinate.
|
46
|
-
def dest_fit_horizontally(top, page=@current_page)
|
47
|
-
[page, :FitH, top]
|
48
|
-
end
|
49
|
-
|
50
|
-
# Return a Dest specification that will fit the given page vertically
|
51
|
-
# into the viewport, aligned horizontally at the given left coordinate.
|
52
|
-
def dest_fit_vertically(left, page=@current_page)
|
53
|
-
[page, :FitV, left]
|
54
|
-
end
|
55
|
-
|
56
|
-
# Return a Dest specification that will fit the given rectangle into the
|
57
|
-
# viewport, for the given page.
|
58
|
-
def dest_fit_rect(left, bottom, right, top, page=@current_page)
|
59
|
-
[page, :FitR, left, bottom, right, top]
|
60
|
-
end
|
61
|
-
|
62
|
-
# Return a Dest specfication that will fit the given page's bounding box
|
63
|
-
# into the viewport.
|
64
|
-
def dest_fit_bounds(page=@current_page)
|
65
|
-
[page, :FitB]
|
66
|
-
end
|
67
|
-
|
68
|
-
# Same as #dest_fit_horizontally, but works on the page's bounding box
|
69
|
-
# instead of the entire page.
|
70
|
-
def dest_fit_bounds_horizontally(top, page=@current_page)
|
71
|
-
[page, :FitBH, top]
|
72
|
-
end
|
73
|
-
|
74
|
-
# Same as #dest_fit_vertically, but works on the page's bounding box
|
75
|
-
# instead of the entire page.
|
76
|
-
def dest_fit_bounds_vertically(left, page=@current_page)
|
77
|
-
[page, :FitBV, left]
|
78
|
-
end
|
79
|
-
end
|
80
|
-
end
|
81
|
-
end
|
@@ -1,126 +0,0 @@
|
|
1
|
-
# encoding: utf-8
|
2
|
-
#
|
3
|
-
# internals.rb : Implements document internals for Prawn
|
4
|
-
#
|
5
|
-
# Copyright August 2008, Gregory Brown. All Rights Reserved.
|
6
|
-
#
|
7
|
-
# This is free software. Please see the LICENSE and COPYING files for details.
|
8
|
-
|
9
|
-
module Prawn
|
10
|
-
class Document
|
11
|
-
|
12
|
-
# This module exposes a few low-level PDF features for those who want
|
13
|
-
# to extend Prawn's core functionality. If you are not comfortable with
|
14
|
-
# low level PDF functionality as defined by Adobe's specification, chances
|
15
|
-
# are you won't need anything you find here.
|
16
|
-
#
|
17
|
-
module Internals
|
18
|
-
# Creates a new Prawn::Reference and adds it to the Document's object
|
19
|
-
# list. The +data+ argument is anything that Prawn::PdfObject() can convert.
|
20
|
-
#
|
21
|
-
# If a block is given, it will be invoked just before the object is written
|
22
|
-
# out to the PDF document stream. This allows you to do deferred processing
|
23
|
-
# on some references (such as fonts, which you might know all the details
|
24
|
-
# about until the last page of the document is finished).
|
25
|
-
def ref(data, &block)
|
26
|
-
@objects.push(Prawn::Reference.new(@objects.size + 1, data, &block)).last
|
27
|
-
end
|
28
|
-
|
29
|
-
# Appends a raw string to the current page content.
|
30
|
-
#
|
31
|
-
# # Raw line drawing example:
|
32
|
-
# x1,y1,x2,y2 = 100,500,300,550
|
33
|
-
# pdf.add_content("%.3f %.3f m" % [ x1, y1 ]) # move
|
34
|
-
# pdf.add_content("%.3f %.3f l" % [ x2, y2 ]) # draw path
|
35
|
-
# pdf.add_content("S") # stroke
|
36
|
-
#
|
37
|
-
def add_content(str)
|
38
|
-
@page_content << str << "\n"
|
39
|
-
end
|
40
|
-
|
41
|
-
# Add a new type to the current pages ProcSet
|
42
|
-
#
|
43
|
-
def proc_set(*types)
|
44
|
-
@current_page.data[:ProcSet] ||= ref([])
|
45
|
-
@current_page.data[:ProcSet].data |= types
|
46
|
-
end
|
47
|
-
|
48
|
-
# The Resources dictionary for the current page
|
49
|
-
#
|
50
|
-
def page_resources
|
51
|
-
@current_page.data[:Resources] ||= {}
|
52
|
-
end
|
53
|
-
|
54
|
-
# The Font dictionary for the current page
|
55
|
-
#
|
56
|
-
def page_fonts
|
57
|
-
page_resources[:Font] ||= {}
|
58
|
-
end
|
59
|
-
|
60
|
-
# The XObject dictionary for the current page
|
61
|
-
def page_xobjects
|
62
|
-
page_resources[:XObject] ||= {}
|
63
|
-
end
|
64
|
-
|
65
|
-
# The Name dictionary (PDF spec 3.6.3) for this document. It is
|
66
|
-
# lazily initialized, so that documents that do not need a name
|
67
|
-
# dictionary do not incur the additional overhead.
|
68
|
-
def names
|
69
|
-
@root.data[:Names] ||= ref(:Type => :Names)
|
70
|
-
end
|
71
|
-
|
72
|
-
private
|
73
|
-
|
74
|
-
def finish_page_content
|
75
|
-
@header.draw if @header
|
76
|
-
@footer.draw if @footer
|
77
|
-
add_content "Q"
|
78
|
-
@page_content.compress_stream if compression_enabled?
|
79
|
-
@page_content.data[:Length] = @page_content.stream.size
|
80
|
-
end
|
81
|
-
|
82
|
-
# Write out the PDF Header, as per spec 3.4.1
|
83
|
-
def render_header(output)
|
84
|
-
# pdf version
|
85
|
-
output << "%PDF-1.3\n"
|
86
|
-
|
87
|
-
# 4 binary chars, as recommended by the spec
|
88
|
-
output << "\xFF\xFF\xFF\xFF\n"
|
89
|
-
end
|
90
|
-
|
91
|
-
# Write out the PDF Body, as per spec 3.4.2
|
92
|
-
def render_body(output)
|
93
|
-
@objects.each do |ref|
|
94
|
-
ref.offset = output.size
|
95
|
-
output << ref.object
|
96
|
-
end
|
97
|
-
end
|
98
|
-
|
99
|
-
# Write out the PDF Cross Reference Table, as per spec 3.4.3
|
100
|
-
def render_xref(output)
|
101
|
-
@xref_offset = output.size
|
102
|
-
output << "xref\n"
|
103
|
-
output << "0 #{@objects.size + 1}\n"
|
104
|
-
output << "0000000000 65535 f \n"
|
105
|
-
@objects.each do |ref|
|
106
|
-
output.printf("%010d", ref.offset)
|
107
|
-
output << " 00000 n \n"
|
108
|
-
end
|
109
|
-
end
|
110
|
-
|
111
|
-
# Write out the PDF Trailer, as per spec 3.4.4
|
112
|
-
def render_trailer(output)
|
113
|
-
trailer_hash = {:Size => @objects.size + 1,
|
114
|
-
:Root => @root,
|
115
|
-
:Info => @info}
|
116
|
-
|
117
|
-
output << "trailer\n"
|
118
|
-
output << Prawn::PdfObject(trailer_hash) << "\n"
|
119
|
-
output << "startxref\n"
|
120
|
-
output << @xref_offset << "\n"
|
121
|
-
output << "%%EOF"
|
122
|
-
end
|
123
|
-
|
124
|
-
end
|
125
|
-
end
|
126
|
-
end
|