pdf-labels 1.0.0
Sign up to get free protection for your applications and to get access to all the features.
- data/History.txt +8 -0
- data/LICENCE +38 -0
- data/Manifest.txt +141 -0
- data/README.txt +72 -0
- data/Rakefile +30 -0
- data/lib/alias.rb +8 -0
- data/lib/glabel_template.rb +36 -0
- data/lib/label.rb +52 -0
- data/lib/layout.rb +13 -0
- data/lib/length_node.rb +47 -0
- data/lib/markup.rb +25 -0
- data/lib/pdf_label_page.rb +171 -0
- data/lib/pdf_labels.rb +6 -0
- data/lib/template.rb +37 -0
- data/templates/avery-iso-templates.xml +222 -0
- data/templates/avery-other-templates.xml +21 -0
- data/templates/avery-us-templates.xml +599 -0
- data/templates/glabels-2.0.dtd +329 -0
- data/templates/misc-iso-templates.xml +434 -0
- data/templates/misc-other-templates.xml +21 -0
- data/templates/misc-us-templates.xml +183 -0
- data/templates/paper-sizes.xml +37 -0
- data/templates/zweckform-iso-templates.xml +197 -0
- data/test/test_pdf_label_page.rb +91 -0
- data/vendor/color.rb +87 -0
- data/vendor/color/cmyk.rb +182 -0
- data/vendor/color/css.rb +27 -0
- data/vendor/color/grayscale.rb +135 -0
- data/vendor/color/hsl.rb +130 -0
- data/vendor/color/palette.rb +15 -0
- data/vendor/color/palette/gimp.rb +107 -0
- data/vendor/color/palette/monocontrast.rb +180 -0
- data/vendor/color/rgb-colors.rb +189 -0
- data/vendor/color/rgb.rb +311 -0
- data/vendor/color/rgb/metallic.rb +28 -0
- data/vendor/color/yiq.rb +78 -0
- data/vendor/pdf/charts.rb +13 -0
- data/vendor/pdf/charts/stddev.rb +433 -0
- data/vendor/pdf/grid.rb +135 -0
- data/vendor/pdf/math.rb +108 -0
- data/vendor/pdf/pagenumbers.rb +288 -0
- data/vendor/pdf/quickref.rb +331 -0
- data/vendor/pdf/simpletable.rb +947 -0
- data/vendor/pdf/techbook.rb +901 -0
- data/vendor/pdf/writer.rb +2801 -0
- data/vendor/pdf/writer/arc4.rb +63 -0
- data/vendor/pdf/writer/fontmetrics.rb +202 -0
- data/vendor/pdf/writer/fonts/Courier-Bold.afm +342 -0
- data/vendor/pdf/writer/fonts/Courier-BoldOblique.afm +342 -0
- data/vendor/pdf/writer/fonts/Courier-Oblique.afm +342 -0
- data/vendor/pdf/writer/fonts/Courier.afm +342 -0
- data/vendor/pdf/writer/fonts/Helvetica-Bold.afm +2827 -0
- data/vendor/pdf/writer/fonts/Helvetica-BoldOblique.afm +2827 -0
- data/vendor/pdf/writer/fonts/Helvetica-Oblique.afm +3051 -0
- data/vendor/pdf/writer/fonts/Helvetica.afm +3051 -0
- data/vendor/pdf/writer/fonts/Symbol.afm +213 -0
- data/vendor/pdf/writer/fonts/Times-Bold.afm +2588 -0
- data/vendor/pdf/writer/fonts/Times-BoldItalic.afm +2384 -0
- data/vendor/pdf/writer/fonts/Times-Italic.afm +2667 -0
- data/vendor/pdf/writer/fonts/Times-Roman.afm +2419 -0
- data/vendor/pdf/writer/fonts/ZapfDingbats.afm +225 -0
- data/vendor/pdf/writer/graphics.rb +813 -0
- data/vendor/pdf/writer/graphics/imageinfo.rb +365 -0
- data/vendor/pdf/writer/lang.rb +44 -0
- data/vendor/pdf/writer/lang/en.rb +104 -0
- data/vendor/pdf/writer/object.rb +23 -0
- data/vendor/pdf/writer/object/action.rb +40 -0
- data/vendor/pdf/writer/object/annotation.rb +42 -0
- data/vendor/pdf/writer/object/catalog.rb +39 -0
- data/vendor/pdf/writer/object/contents.rb +69 -0
- data/vendor/pdf/writer/object/destination.rb +40 -0
- data/vendor/pdf/writer/object/encryption.rb +53 -0
- data/vendor/pdf/writer/object/font.rb +68 -0
- data/vendor/pdf/writer/object/fontdescriptor.rb +34 -0
- data/vendor/pdf/writer/object/fontencoding.rb +40 -0
- data/vendor/pdf/writer/object/image.rb +308 -0
- data/vendor/pdf/writer/object/info.rb +79 -0
- data/vendor/pdf/writer/object/outline.rb +30 -0
- data/vendor/pdf/writer/object/outlines.rb +30 -0
- data/vendor/pdf/writer/object/page.rb +195 -0
- data/vendor/pdf/writer/object/pages.rb +115 -0
- data/vendor/pdf/writer/object/procset.rb +46 -0
- data/vendor/pdf/writer/object/viewerpreferences.rb +74 -0
- data/vendor/pdf/writer/ohash.rb +58 -0
- data/vendor/pdf/writer/oreader.rb +25 -0
- data/vendor/pdf/writer/state.rb +48 -0
- data/vendor/pdf/writer/strokestyle.rb +140 -0
- data/vendor/transaction/simple.rb +693 -0
- data/vendor/transaction/simple/group.rb +133 -0
- data/vendor/transaction/simple/threadsafe.rb +52 -0
- data/vendor/transaction/simple/threadsafe/group.rb +23 -0
- data/vendor/xml-mapping/ChangeLog +128 -0
- data/vendor/xml-mapping/LICENSE +56 -0
- data/vendor/xml-mapping/README +386 -0
- data/vendor/xml-mapping/README_XPATH +175 -0
- data/vendor/xml-mapping/Rakefile +214 -0
- data/vendor/xml-mapping/TODO.txt +32 -0
- data/vendor/xml-mapping/doc/xpath_impl_notes.txt +119 -0
- data/vendor/xml-mapping/examples/company.rb +34 -0
- data/vendor/xml-mapping/examples/company.xml +26 -0
- data/vendor/xml-mapping/examples/company_usage.intin.rb +19 -0
- data/vendor/xml-mapping/examples/company_usage.intout +39 -0
- data/vendor/xml-mapping/examples/order.rb +61 -0
- data/vendor/xml-mapping/examples/order.xml +54 -0
- data/vendor/xml-mapping/examples/order_signature_enhanced.rb +7 -0
- data/vendor/xml-mapping/examples/order_signature_enhanced.xml +9 -0
- data/vendor/xml-mapping/examples/order_signature_enhanced_usage.intin.rb +12 -0
- data/vendor/xml-mapping/examples/order_signature_enhanced_usage.intout +16 -0
- data/vendor/xml-mapping/examples/order_usage.intin.rb +73 -0
- data/vendor/xml-mapping/examples/order_usage.intout +147 -0
- data/vendor/xml-mapping/examples/time_augm.intin.rb +19 -0
- data/vendor/xml-mapping/examples/time_augm.intout +23 -0
- data/vendor/xml-mapping/examples/time_node.rb +27 -0
- data/vendor/xml-mapping/examples/xpath_create_new.intin.rb +85 -0
- data/vendor/xml-mapping/examples/xpath_create_new.intout +181 -0
- data/vendor/xml-mapping/examples/xpath_docvsroot.intin.rb +30 -0
- data/vendor/xml-mapping/examples/xpath_docvsroot.intout +34 -0
- data/vendor/xml-mapping/examples/xpath_ensure_created.intin.rb +62 -0
- data/vendor/xml-mapping/examples/xpath_ensure_created.intout +114 -0
- data/vendor/xml-mapping/examples/xpath_pathological.intin.rb +42 -0
- data/vendor/xml-mapping/examples/xpath_pathological.intout +56 -0
- data/vendor/xml-mapping/examples/xpath_usage.intin.rb +51 -0
- data/vendor/xml-mapping/examples/xpath_usage.intout +57 -0
- data/vendor/xml-mapping/install.rb +40 -0
- data/vendor/xml-mapping/lib/xml/mapping.rb +14 -0
- data/vendor/xml-mapping/lib/xml/mapping/base.rb +571 -0
- data/vendor/xml-mapping/lib/xml/mapping/standard_nodes.rb +343 -0
- data/vendor/xml-mapping/lib/xml/mapping/version.rb +8 -0
- data/vendor/xml-mapping/lib/xml/xxpath.rb +354 -0
- data/vendor/xml-mapping/test/all_tests.rb +6 -0
- data/vendor/xml-mapping/test/company.rb +56 -0
- data/vendor/xml-mapping/test/documents_folders.rb +33 -0
- data/vendor/xml-mapping/test/fixtures/bookmarks1.xml +24 -0
- data/vendor/xml-mapping/test/fixtures/company1.xml +85 -0
- data/vendor/xml-mapping/test/fixtures/documents_folders.xml +71 -0
- data/vendor/xml-mapping/test/fixtures/documents_folders2.xml +30 -0
- data/vendor/xml-mapping/test/multiple_mappings.rb +80 -0
- data/vendor/xml-mapping/test/tests_init.rb +2 -0
- data/vendor/xml-mapping/test/xml_mapping_adv_test.rb +84 -0
- data/vendor/xml-mapping/test/xml_mapping_test.rb +201 -0
- data/vendor/xml-mapping/test/xpath_test.rb +273 -0
- metadata +191 -0
data/vendor/color/rgb.rb
ADDED
@@ -0,0 +1,311 @@
|
|
1
|
+
#--
|
2
|
+
# Colour management with Ruby.
|
3
|
+
#
|
4
|
+
# Copyright 2005 Austin Ziegler
|
5
|
+
# http://rubyforge.org/ruby-pdf/
|
6
|
+
#
|
7
|
+
# Licensed under a MIT-style licence.
|
8
|
+
#
|
9
|
+
# $Id: rgb.rb,v 1.6 2005/08/08 02:44:17 austin Exp $
|
10
|
+
#++
|
11
|
+
|
12
|
+
# An RGB colour object.
|
13
|
+
class Color::RGB
|
14
|
+
# The format of a DeviceRGB colour for PDF. In color-tools 2.0 this will
|
15
|
+
# be removed from this package and added back as a modification by the
|
16
|
+
# PDF::Writer package.
|
17
|
+
PDF_FORMAT_STR = "%.3f %.3f %.3f %s"
|
18
|
+
|
19
|
+
class << self
|
20
|
+
# Creates an RGB colour object from percentages 0..100.
|
21
|
+
#
|
22
|
+
# Color::RGB.from_percentage(10, 20 30)
|
23
|
+
def from_percentage(r = 0, g = 0, b = 0)
|
24
|
+
from_fraction(r / 100.0, g / 100.0, b / 100.0)
|
25
|
+
end
|
26
|
+
|
27
|
+
# Creates an RGB colour object from fractional values 0..1.
|
28
|
+
#
|
29
|
+
# Color::RGB.from_fraction(.3, .2, .1)
|
30
|
+
def from_fraction(r = 0.0, g = 0.0, b = 0.0)
|
31
|
+
colour = Color::RGB.new
|
32
|
+
colour.r = r
|
33
|
+
colour.g = g
|
34
|
+
colour.b = b
|
35
|
+
colour
|
36
|
+
end
|
37
|
+
|
38
|
+
# Creates an RGB colour object from an HTML colour descriptor (e.g.,
|
39
|
+
# <tt>"fed"</tt> or <tt>"#cabbed;"</tt>.
|
40
|
+
#
|
41
|
+
# Color::RGB.from_html("fed")
|
42
|
+
# Color::RGB.from_html("#fed")
|
43
|
+
# Color::RGB.from_html("#cabbed")
|
44
|
+
# Color::RGB.from_html("cabbed")
|
45
|
+
def from_html(html_colour)
|
46
|
+
html_colour = html_colour.gsub(%r{[#;]}, '')
|
47
|
+
case html_colour.size
|
48
|
+
when 3
|
49
|
+
colours = html_colour.scan(%r{[0-9A-Fa-f]}).map { |el| (el * 2).to_i(16) }
|
50
|
+
when 6
|
51
|
+
colours = html_colour.scan(%r<[0-9A-Fa-f]{2}>).map { |el| el.to_i(16) }
|
52
|
+
else
|
53
|
+
raise ArgumentError
|
54
|
+
end
|
55
|
+
|
56
|
+
Color::RGB.new(*colours)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
# Compares the other colour to this one. The other colour will be
|
61
|
+
# converted to RGB before comparison, so the comparison between a RGB
|
62
|
+
# colour and a non-RGB colour will be approximate and based on the other
|
63
|
+
# colour's default #to_rgb conversion. If there is no #to_rgb
|
64
|
+
# conversion, this will raise an exception. This will report that two
|
65
|
+
# RGB colours are equivalent if all component values are within 1e-4
|
66
|
+
# (0.0001) of each other.
|
67
|
+
def ==(other)
|
68
|
+
other = other.to_rgb
|
69
|
+
other.kind_of?(Color::RGB) and
|
70
|
+
((@r - other.r).abs <= 1e-4) and
|
71
|
+
((@g - other.g).abs <= 1e-4) and
|
72
|
+
((@b - other.b).abs <= 1e-4)
|
73
|
+
end
|
74
|
+
|
75
|
+
# Creates an RGB colour object from the standard range 0..255.
|
76
|
+
#
|
77
|
+
# Color::RGB.new(32, 64, 128)
|
78
|
+
# Color::RGB.new(0x20, 0x40, 0x80)
|
79
|
+
def initialize(r = 0, g = 0, b = 0)
|
80
|
+
@r = r / 255.0
|
81
|
+
@g = g / 255.0
|
82
|
+
@b = b / 255.0
|
83
|
+
end
|
84
|
+
|
85
|
+
# Present the colour as a DeviceRGB fill colour string for PDF. This
|
86
|
+
# will be removed from the default package in color-tools 2.0.
|
87
|
+
def pdf_fill
|
88
|
+
PDF_FORMAT_STR % [ @r, @g, @b, "rg" ]
|
89
|
+
end
|
90
|
+
|
91
|
+
# Present the colour as a DeviceRGB stroke colour string for PDF. This
|
92
|
+
# will be removed from the default package in color-tools 2.0.
|
93
|
+
def pdf_stroke
|
94
|
+
PDF_FORMAT_STR % [ @r, @g, @b, "RG" ]
|
95
|
+
end
|
96
|
+
|
97
|
+
# Present the colour as an HTML/CSS colour string.
|
98
|
+
def html
|
99
|
+
r = (@r * 255).round
|
100
|
+
r = 255 if r > 255
|
101
|
+
|
102
|
+
g = (@g * 255).round
|
103
|
+
g = 255 if g > 255
|
104
|
+
|
105
|
+
b = (@b * 255).round
|
106
|
+
b = 255 if b > 255
|
107
|
+
|
108
|
+
"#%02x%02x%02x" % [ r, g, b ]
|
109
|
+
end
|
110
|
+
|
111
|
+
# Converts the RGB colour to CMYK. Most colour experts strongly suggest
|
112
|
+
# that this is not a good idea (some even suggesting that it's a very
|
113
|
+
# bad idea). CMYK represents additive percentages of inks on white
|
114
|
+
# paper, whereas RGB represents mixed colour intensities on a black
|
115
|
+
# screen.
|
116
|
+
#
|
117
|
+
# However, the colour conversion can be done. The basic method is
|
118
|
+
# multi-step:
|
119
|
+
#
|
120
|
+
# 1. Convert the R, G, and B components to C, M, and Y components.
|
121
|
+
# c = 1.0 � r
|
122
|
+
# m = 1.0 � g
|
123
|
+
# y = 1.0 � b
|
124
|
+
# 2. Compute the minimum amount of black (K) required to smooth the
|
125
|
+
# colour in inks.
|
126
|
+
# k = min(c, m, y)
|
127
|
+
# 3. Perform undercolour removal on the C, M, and Y components of the
|
128
|
+
# colours because less of each colour is needed for each bit of
|
129
|
+
# black. Also, regenerate the black (K) based on the undercolour
|
130
|
+
# removal so that the colour is more accurately represented in ink.
|
131
|
+
# c = min(1.0, max(0.0, c � UCR(k)))
|
132
|
+
# m = min(1.0, max(0.0, m � UCR(k)))
|
133
|
+
# y = min(1.0, max(0.0, y � UCR(k)))
|
134
|
+
# k = min(1.0, max(0.0, BG(k)))
|
135
|
+
#
|
136
|
+
# The undercolour removal function and the black generation functions
|
137
|
+
# return a value based on the brightness of the RGB colour.
|
138
|
+
def to_cmyk
|
139
|
+
c = 1.0 - @r.to_f
|
140
|
+
m = 1.0 - @g.to_f
|
141
|
+
y = 1.0 - @b.to_f
|
142
|
+
|
143
|
+
k = [c, m, y].min
|
144
|
+
k = k - (k * brightness)
|
145
|
+
|
146
|
+
c = [1.0, [0.0, c - k].max].min
|
147
|
+
m = [1.0, [0.0, m - k].max].min
|
148
|
+
y = [1.0, [0.0, y - k].max].min
|
149
|
+
k = [1.0, [0.0, k].max].min
|
150
|
+
|
151
|
+
Color::CMYK.from_fraction(c, m, y, k)
|
152
|
+
end
|
153
|
+
|
154
|
+
def to_rgb(ignored = nil)
|
155
|
+
self
|
156
|
+
end
|
157
|
+
|
158
|
+
# Returns the YIQ (NTSC) colour encoding of the RGB value.
|
159
|
+
def to_yiq
|
160
|
+
y = (@r * 0.299) + (@g * 0.587) + (@b * 0.114)
|
161
|
+
i = (@r * 0.596) + (@g * -0.275) + (@b * -0.321)
|
162
|
+
q = (@r * 0.212) + (@g * -0.523) + (@b * 0.311)
|
163
|
+
Color::YIQ.from_fraction(y, i, q)
|
164
|
+
end
|
165
|
+
|
166
|
+
# Returns the HSL colour encoding of the RGB value.
|
167
|
+
def to_hsl
|
168
|
+
min = [ @r, @g, @b ].min
|
169
|
+
max = [ @r, @g, @b ].max
|
170
|
+
delta = (max - min).to_f
|
171
|
+
|
172
|
+
lum = (max + min) / 2.0
|
173
|
+
|
174
|
+
if delta <= 1e-5 # close to 0.0, so it's a grey
|
175
|
+
hue = 0
|
176
|
+
sat = 0
|
177
|
+
else
|
178
|
+
if (lum - 0.5) <= 1e-5
|
179
|
+
sat = delta / (max + min).to_f
|
180
|
+
else
|
181
|
+
sat = delta / (2 - max - min).to_f
|
182
|
+
end
|
183
|
+
|
184
|
+
if @r == max
|
185
|
+
hue = (@g - @b) / delta.to_f
|
186
|
+
elsif @g == max
|
187
|
+
hue = (2.0 + @b - @r) / delta.to_f
|
188
|
+
elsif (@b - max) <= 1e-5
|
189
|
+
hue = (4.0 + @r - @g) / delta.to_f
|
190
|
+
end
|
191
|
+
hue /= 6.0
|
192
|
+
|
193
|
+
hue += 1 if hue < 0
|
194
|
+
hue -= 1 if hue > 1
|
195
|
+
end
|
196
|
+
Color::HSL.from_fraction(hue, sat, lum)
|
197
|
+
end
|
198
|
+
|
199
|
+
# Mix the RGB hue with White so that the RGB hue is the specified
|
200
|
+
# percentage of the resulting colour. Strictly speaking, this isn't a
|
201
|
+
# darken_by operation.
|
202
|
+
def lighten_by(percent)
|
203
|
+
mix_with(White, percent)
|
204
|
+
end
|
205
|
+
|
206
|
+
# Mix the RGB hue with Black so that the RGB hue is the specified
|
207
|
+
# percentage of the resulting colour. Strictly speaking, this isn't a
|
208
|
+
# darken_by operation.
|
209
|
+
def darken_by(percent)
|
210
|
+
mix_with(Black, percent)
|
211
|
+
end
|
212
|
+
|
213
|
+
# Mix the mask colour (which must be an RGB object) with the current
|
214
|
+
# colour at the stated opacity percentage (0..100).
|
215
|
+
def mix_with(mask, opacity)
|
216
|
+
opacity /= 100.0
|
217
|
+
rgb = self.dup
|
218
|
+
|
219
|
+
rgb.r = (@r * opacity) + (mask.r * (1 - opacity))
|
220
|
+
rgb.g = (@g * opacity) + (mask.g * (1 - opacity))
|
221
|
+
rgb.b = (@b * opacity) + (mask.b * (1 - opacity))
|
222
|
+
|
223
|
+
rgb
|
224
|
+
end
|
225
|
+
|
226
|
+
# Returns the brightness value for a colour, a number between 0..1.
|
227
|
+
# Based on the Y value of YIQ encoding, representing luminosity, or
|
228
|
+
# perceived brightness.
|
229
|
+
#
|
230
|
+
# This may be modified in a future version of color-tools to use the
|
231
|
+
# luminosity value of HSL.
|
232
|
+
def brightness
|
233
|
+
to_yiq.y
|
234
|
+
end
|
235
|
+
def to_grayscale
|
236
|
+
Color::GrayScale.from_fraction(to_hsl.l)
|
237
|
+
end
|
238
|
+
|
239
|
+
alias to_greyscale to_grayscale
|
240
|
+
|
241
|
+
# Returns a new colour with the brightness adjusted by the specified
|
242
|
+
# percentage. Negative percentages will darken the colour; positive
|
243
|
+
# percentages will brighten the colour.
|
244
|
+
#
|
245
|
+
# Color::RGB::DarkBlue.adjust_brightness(10)
|
246
|
+
# Color::RGB::DarkBlue.adjust_brightness(-10)
|
247
|
+
def adjust_brightness(percent)
|
248
|
+
percent /= 100.0
|
249
|
+
percent += 1.0
|
250
|
+
percent = [ percent, 2.0 ].min
|
251
|
+
percent = [ 0.0, percent ].max
|
252
|
+
|
253
|
+
hsl = to_hsl
|
254
|
+
hsl.l *= percent
|
255
|
+
hsl.to_rgb
|
256
|
+
end
|
257
|
+
|
258
|
+
# Returns a new colour with the saturation adjusted by the specified
|
259
|
+
# percentage. Negative percentages will reduce the saturation; positive
|
260
|
+
# percentages will increase the saturation.
|
261
|
+
#
|
262
|
+
# Color::RGB::DarkBlue.adjust_saturation(10)
|
263
|
+
# Color::RGB::DarkBlue.adjust_saturation(-10)
|
264
|
+
def adjust_saturation(percent)
|
265
|
+
percent /= 100.0
|
266
|
+
percent += 1.0
|
267
|
+
percent = [ percent, 2.0 ].min
|
268
|
+
percent = [ 0.0, percent ].max
|
269
|
+
|
270
|
+
hsl = to_hsl
|
271
|
+
hsl.s *= percent
|
272
|
+
hsl.to_rgb
|
273
|
+
end
|
274
|
+
|
275
|
+
# Returns a new colour with the hue adjusted by the specified
|
276
|
+
# percentage. Negative percentages will reduce the hue; positive
|
277
|
+
# percentages will increase the hue.
|
278
|
+
#
|
279
|
+
# Color::RGB::DarkBlue.adjust_hue(10)
|
280
|
+
# Color::RGB::DarkBlue.adjust_hue(-10)
|
281
|
+
def adjust_hue(percent)
|
282
|
+
percent /= 100.0
|
283
|
+
percent += 1.0
|
284
|
+
percent = [ percent, 2.0 ].min
|
285
|
+
percent = [ 0.0, percent ].max
|
286
|
+
|
287
|
+
hsl = to_hsl
|
288
|
+
hsl.h *= percent
|
289
|
+
hsl.to_rgb
|
290
|
+
end
|
291
|
+
|
292
|
+
attr_accessor :r, :g, :b
|
293
|
+
remove_method :r=, :g=, :b= ;
|
294
|
+
def r=(rr) #:nodoc:
|
295
|
+
rr = 1.0 if rr > 1
|
296
|
+
rr = 0.0 if rr < 0
|
297
|
+
@r = rr
|
298
|
+
end
|
299
|
+
def g=(gg) #:nodoc:
|
300
|
+
gg = 1.0 if gg > 1
|
301
|
+
gg = 0.0 if gg < 0
|
302
|
+
@g = gg
|
303
|
+
end
|
304
|
+
def b=(bb) #:nodoc:
|
305
|
+
bb = 1.0 if bb > 1
|
306
|
+
bb = 0.0 if bb < 0
|
307
|
+
@b = bb
|
308
|
+
end
|
309
|
+
end
|
310
|
+
|
311
|
+
require 'color/rgb-colors'
|
@@ -0,0 +1,28 @@
|
|
1
|
+
#--
|
2
|
+
# Colour management with Ruby.
|
3
|
+
#
|
4
|
+
# Copyright 2005 Austin Ziegler
|
5
|
+
# http://rubyforge.org/ruby-pdf/
|
6
|
+
#
|
7
|
+
# Licensed under a MIT-style licence.
|
8
|
+
#
|
9
|
+
# $Id: metallic.rb,v 1.1 2005/08/05 23:07:20 austin Exp $
|
10
|
+
#++
|
11
|
+
|
12
|
+
# This namespace contains some RGB metallic colours suggested by Jim Freeze.
|
13
|
+
module Color::RGB::Metallic
|
14
|
+
Aluminum = Color::RGB.new(0x99, 0x99, 0x99)
|
15
|
+
CoolCopper = Color::RGB.new(0xd9, 0x87, 0x19)
|
16
|
+
Copper = Color::RGB.new(0xb8, 0x73, 0x33)
|
17
|
+
Iron = Color::RGB.new(0x4c, 0x4c, 0x4c)
|
18
|
+
Lead = Color::RGB.new(0x19, 0x19, 0x19)
|
19
|
+
Magnesium = Color::RGB.new(0xb3, 0xb3, 0xb3)
|
20
|
+
Mercury = Color::RGB.new(0xe6, 0xe6, 0xe6)
|
21
|
+
Nickel = Color::RGB.new(0x80, 0x80, 0x80)
|
22
|
+
PolySilicon = Color::RGB.new(0x60, 0x00, 0x00)
|
23
|
+
Poly = PolySilicon
|
24
|
+
Silver = Color::RGB.new(0xcc, 0xcc, 0xcc)
|
25
|
+
Steel = Color::RGB.new(0x66, 0x66, 0x66)
|
26
|
+
Tin = Color::RGB.new(0x7f, 0x7f, 0x7f)
|
27
|
+
Tungsten = Color::RGB.new(0x33, 0x33, 0x33)
|
28
|
+
end
|
data/vendor/color/yiq.rb
ADDED
@@ -0,0 +1,78 @@
|
|
1
|
+
#--
|
2
|
+
# Colour management with Ruby.
|
3
|
+
#
|
4
|
+
# Copyright 2005 Austin Ziegler
|
5
|
+
# http://rubyforge.org/ruby-pdf/
|
6
|
+
#
|
7
|
+
# Licensed under a MIT-style licence.
|
8
|
+
#
|
9
|
+
# $Id: yiq.rb,v 1.3 2005/08/08 02:44:17 austin Exp $
|
10
|
+
#++
|
11
|
+
|
12
|
+
# A colour object representing YIQ (NTSC) colour encoding.
|
13
|
+
class Color::YIQ
|
14
|
+
# Creates a YIQ colour object from fractional values 0 .. 1.
|
15
|
+
#
|
16
|
+
# Color::YIQ.new(0.3, 0.2, 0.1)
|
17
|
+
def self.from_fraction(y = 0, i = 0, q = 0)
|
18
|
+
color = Color::YIQ.new
|
19
|
+
color.y = y
|
20
|
+
color.i = i
|
21
|
+
color.q = q
|
22
|
+
color
|
23
|
+
end
|
24
|
+
|
25
|
+
# Creates a YIQ colour object from percentages 0 .. 100.
|
26
|
+
#
|
27
|
+
# Color::YIQ.new(10, 20, 30)
|
28
|
+
def initialize(y = 0, i = 0, q = 0)
|
29
|
+
@y = y / 100.0
|
30
|
+
@i = i / 100.0
|
31
|
+
@q = q / 100.0
|
32
|
+
end
|
33
|
+
|
34
|
+
# Compares the other colour to this one. The other colour will be
|
35
|
+
# converted to YIQ before comparison, so the comparison between a YIQ
|
36
|
+
# colour and a non-YIQ colour will be approximate and based on the other
|
37
|
+
# colour's #to_yiq conversion. If there is no #to_yiq conversion, this
|
38
|
+
# will raise an exception. This will report that two YIQ values are
|
39
|
+
# equivalent if all component colours are within 1e-4 (0.0001) of each
|
40
|
+
# other.
|
41
|
+
def ==(other)
|
42
|
+
other = other.to_yiq
|
43
|
+
other.kind_of?(Color::YIQ) and
|
44
|
+
((@y - other.y).abs <= 1e-4) and
|
45
|
+
((@i - other.i).abs <= 1e-4) and
|
46
|
+
((@q - other.q).abs <= 1e-4)
|
47
|
+
end
|
48
|
+
|
49
|
+
def to_yiq
|
50
|
+
self
|
51
|
+
end
|
52
|
+
|
53
|
+
def brightness
|
54
|
+
@y
|
55
|
+
end
|
56
|
+
def to_grayscale
|
57
|
+
Color::GrayScale.new(@y)
|
58
|
+
end
|
59
|
+
alias to_greyscale to_grayscale
|
60
|
+
|
61
|
+
attr_accessor :y, :i, :q
|
62
|
+
remove_method :y=, :i=, :q=
|
63
|
+
def y=(yy) #:nodoc:
|
64
|
+
yy = 1.0 if yy > 1
|
65
|
+
yy = 0.0 if yy < 0
|
66
|
+
@y = yy
|
67
|
+
end
|
68
|
+
def i=(ii) #:nodoc:
|
69
|
+
ii = 1.0 if ii > 1
|
70
|
+
ii = 0.0 if ii < 0
|
71
|
+
@i = ii
|
72
|
+
end
|
73
|
+
def q=(qq) #:nodoc:
|
74
|
+
qq = 1.0 if qq > 1
|
75
|
+
qq = 0.0 if qq < 0
|
76
|
+
@q = qq
|
77
|
+
end
|
78
|
+
end
|
@@ -0,0 +1,13 @@
|
|
1
|
+
#--
|
2
|
+
# PDF::Writer for Ruby.
|
3
|
+
# http://rubyforge.org/projects/ruby-pdf/
|
4
|
+
# Copyright 2003 - 2005 Austin Ziegler.
|
5
|
+
#
|
6
|
+
# Licensed under a MIT-style licence. See LICENCE in the main distribution
|
7
|
+
# for full licensing information.
|
8
|
+
#
|
9
|
+
# $Id: charts.rb,v 1.2 2005/05/16 03:59:21 austin Exp $
|
10
|
+
#++
|
11
|
+
# A namespace for charts that can be drawn on PDF::Writer canvases.
|
12
|
+
module PDF::Charts
|
13
|
+
end
|
@@ -0,0 +1,433 @@
|
|
1
|
+
#--
|
2
|
+
# PDF::Writer for Ruby.
|
3
|
+
# http://rubyforge.org/projects/ruby-pdf/
|
4
|
+
# Copyright 2003 - 2005 Austin Ziegler.
|
5
|
+
#
|
6
|
+
# Licensed under a MIT-style licence. See LICENCE in the main distribution
|
7
|
+
# for full licensing information.
|
8
|
+
#
|
9
|
+
# $Id: stddev.rb,v 1.12 2005/10/12 14:41:40 austin Exp $
|
10
|
+
#++
|
11
|
+
require 'pdf/writer'
|
12
|
+
require 'pdf/charts'
|
13
|
+
require 'ostruct'
|
14
|
+
|
15
|
+
# Creates a standard deviation chart. This is a type of chart that is
|
16
|
+
# effective for the display of survey results or other data that can
|
17
|
+
# easily be measured in terms of the average and the standard deviation
|
18
|
+
# from that average.
|
19
|
+
#
|
20
|
+
# The scale of responses is the vertical scale; the average data points
|
21
|
+
# and standard deviation values are the horizontal scale.
|
22
|
+
class PDF::Charts::StdDev
|
23
|
+
VERSION = '1.1.4'
|
24
|
+
|
25
|
+
# A data element.
|
26
|
+
DataPoint = Struct.new(:label, :average, :stddev)
|
27
|
+
|
28
|
+
# A label for displaying the scale (vertical) of data in the dataset or
|
29
|
+
# the data set identifiers.
|
30
|
+
class Label
|
31
|
+
def initialize
|
32
|
+
yield self if block_given?
|
33
|
+
end
|
34
|
+
|
35
|
+
# The height of the label, in PDF user units. Ignored for scale
|
36
|
+
# labels.
|
37
|
+
attr_accessor :height
|
38
|
+
# The background color of the label. Ignored for scale labels.
|
39
|
+
attr_accessor :background_color
|
40
|
+
# The text color of the label.
|
41
|
+
attr_accessor :text_color
|
42
|
+
# The text size, in points, of the label.
|
43
|
+
attr_accessor :text_size
|
44
|
+
# The padding of the label. Only used for scale labels.
|
45
|
+
attr_accessor :pad
|
46
|
+
# The decimal precision of the label. Only used for scale labels.
|
47
|
+
attr_accessor :decimal_precision
|
48
|
+
end
|
49
|
+
|
50
|
+
# The scale of the dataset.
|
51
|
+
class Scale
|
52
|
+
def initialize(args = { })
|
53
|
+
@range = args[:range]
|
54
|
+
@step = args[:step]
|
55
|
+
@style = args[:style]
|
56
|
+
@show_labels = false
|
57
|
+
|
58
|
+
yield self if block_given?
|
59
|
+
|
60
|
+
raise TypeError, PDF::Lange[:charts_stddev_scale_norange] if @range.nil?
|
61
|
+
raise TypeError, PDF::Lange[:charts_stddev_scale_nostep] if @step.nil?
|
62
|
+
end
|
63
|
+
|
64
|
+
# Range of the scale. This should be a Range object.
|
65
|
+
attr_accessor :range
|
66
|
+
# The lower end of the range of the scale. The scale range may be
|
67
|
+
# modified by changing this value.
|
68
|
+
attr_accessor :first
|
69
|
+
remove_method :first, :first= ;
|
70
|
+
def first #:nodoc:
|
71
|
+
@range.first
|
72
|
+
end
|
73
|
+
def first=(ff) #:nodoc:
|
74
|
+
@range = (ff..@range.last)
|
75
|
+
end
|
76
|
+
# The upper end of the range of the scale. The scale range may be
|
77
|
+
# modified by changing this value.
|
78
|
+
attr_accessor :last
|
79
|
+
remove_method :last, :last= ;
|
80
|
+
def last #:nodoc:
|
81
|
+
@range.last
|
82
|
+
end
|
83
|
+
def last=(ll) #:nodoc:
|
84
|
+
@range = (@range.first..ll)
|
85
|
+
end
|
86
|
+
# Defines the step of the scale. Each step represents a vertical
|
87
|
+
# position on the chart.
|
88
|
+
attr_accessor :step
|
89
|
+
# Defines the line style for the scale on the chart. If this is unset
|
90
|
+
# (+nil+), there will be no horizontal marks across the chart for the
|
91
|
+
# steps of the scale.
|
92
|
+
attr_accessor :style
|
93
|
+
# Shows the scale labels if +true+.
|
94
|
+
attr_accessor :show_labels
|
95
|
+
# Defines the label options.
|
96
|
+
attr_accessor :label
|
97
|
+
end
|
98
|
+
|
99
|
+
# This is any line that will be drawn; this is a combination of the line
|
100
|
+
# style (which must be a PDF::Writer::StrokeStyle object) and a color.
|
101
|
+
class Marker
|
102
|
+
def initialize
|
103
|
+
yield self if block_given?
|
104
|
+
end
|
105
|
+
|
106
|
+
# The stroke style of the marker.
|
107
|
+
attr_accessor :style
|
108
|
+
# The stroke color of the marker.
|
109
|
+
attr_accessor :color
|
110
|
+
end
|
111
|
+
|
112
|
+
def initialize
|
113
|
+
@data = []
|
114
|
+
|
115
|
+
@scale = Scale.new do |scale|
|
116
|
+
scale.range = 0..6
|
117
|
+
scale.step = 1
|
118
|
+
scale.style = PDF::Writer::StrokeStyle.new(0.25)
|
119
|
+
scale.show_labels = false
|
120
|
+
scale.label = Label.new do |label|
|
121
|
+
label.text_size = 8
|
122
|
+
label.text_color = Color::RGB::Black
|
123
|
+
label.pad = 2
|
124
|
+
label.decimal_precision = 1
|
125
|
+
end
|
126
|
+
end
|
127
|
+
@leading_gap = 10
|
128
|
+
@show_labels = true
|
129
|
+
@label = Label.new do |label|
|
130
|
+
label.height = 25
|
131
|
+
label.background_color = Color::RGB::Black
|
132
|
+
label.text_color = Color::RGB::White
|
133
|
+
label.text_size = 12
|
134
|
+
end
|
135
|
+
|
136
|
+
@outer_borders = Marker.new do |marker|
|
137
|
+
marker.style = PDF::Writer::StrokeStyle.new(1.5)
|
138
|
+
marker.color = Color::RGB::Black
|
139
|
+
end
|
140
|
+
@inner_borders = nil
|
141
|
+
|
142
|
+
@dot = Marker.new do |marker|
|
143
|
+
marker.style = PDF::Writer::StrokeStyle.new(5)
|
144
|
+
marker.color = Color::RGB::Black
|
145
|
+
end
|
146
|
+
@bar = Marker.new do |marker|
|
147
|
+
marker.style = PDF::Writer::StrokeStyle.new(0.5)
|
148
|
+
marker.color = Color::RGB::Black
|
149
|
+
end
|
150
|
+
@upper_crossbar = Marker.new do |marker|
|
151
|
+
marker.style = PDF::Writer::StrokeStyle.new(1)
|
152
|
+
marker.color = Color::RGB::Black
|
153
|
+
end
|
154
|
+
@lower_crossbar = Marker.new do |marker|
|
155
|
+
marker.style = PDF::Writer::StrokeStyle.new(1)
|
156
|
+
marker.color = Color::RGB::Black
|
157
|
+
end
|
158
|
+
|
159
|
+
@height = 200
|
160
|
+
@maximum_width = 500
|
161
|
+
@datapoint_width = 35
|
162
|
+
|
163
|
+
yield self if block_given?
|
164
|
+
end
|
165
|
+
|
166
|
+
# The data used to generate the standard deviation chart. This is an
|
167
|
+
# array of DataPoint objects, each containing a +label+, an +average+,
|
168
|
+
# and the +stddev+ (standard deviation) from that average.
|
169
|
+
attr_reader :data
|
170
|
+
# The scale of the chart. All values must be within this range. This
|
171
|
+
# will be a Scale object. It defaults to a scale of 0..6 with a step of
|
172
|
+
# 1.
|
173
|
+
attr_accessor :scale
|
174
|
+
|
175
|
+
# The minimum gap between the chart and the bottom of the page, in
|
176
|
+
# PDF user units.
|
177
|
+
attr_accessor :leading_gap
|
178
|
+
|
179
|
+
# This will be +true+ if labels are to be displayed.
|
180
|
+
attr_accessor :show_labels
|
181
|
+
# The label style of the labels if they are displayed. This must be a
|
182
|
+
# PDF::Charts::StdDev::Label object.
|
183
|
+
attr_accessor :label
|
184
|
+
|
185
|
+
# The inner border style. If +nil+, no inner borders are drawn. This is
|
186
|
+
# a PDF::Charts::StdDev::Marker object.
|
187
|
+
attr_accessor :inner_borders
|
188
|
+
# The outer border style. If +nil+, no inner borders are drawn. This is
|
189
|
+
# a PDF::Charts::StdDev::Marker object.
|
190
|
+
attr_accessor :outer_borders
|
191
|
+
|
192
|
+
# The dot marker. A filled circle will be drawn with this information.
|
193
|
+
# If +nil+, the dot will not be drawn. This is a
|
194
|
+
# PDF::Charts::StdDev::Marker object.
|
195
|
+
attr_accessor :dot
|
196
|
+
# The standard deviation bar. A line will be drawn through the dot
|
197
|
+
# marker (if drawn) from the upper to lower standard deviation.
|
198
|
+
# If +nil+, the line will not be drawn. This is a
|
199
|
+
# PDF::Charts::StdDev::Marker object.
|
200
|
+
attr_accessor :bar
|
201
|
+
# The upper crossbar. A line will be drawn across the top of the
|
202
|
+
# standard deviation bar to the width of the dot marker. If #dot is
|
203
|
+
# +nil+, then the line will be twice as wide as it is thick. If +nil+,
|
204
|
+
# the upper crossbar will not be drawn. This is a
|
205
|
+
# PDF::Charts::StdDev::Marker object.
|
206
|
+
attr_accessor :upper_crossbar
|
207
|
+
# The lower crossbar. A line will be drawn across the bottom of the
|
208
|
+
# standard deviation bar to the width of the dot marker. If #dot is
|
209
|
+
# +nil+, then the line will be twice as wide as it is thick. If +nil+,
|
210
|
+
# the lower crossbar will not be drawn. This is a
|
211
|
+
# PDF::Charts::StdDev::Marker object.
|
212
|
+
attr_accessor :lower_crossbar
|
213
|
+
|
214
|
+
# The height of the chart in PDF user units. Default 200 units.
|
215
|
+
attr_accessor :height
|
216
|
+
# The maximum width of the chart in PDF user units. Default 500 units.
|
217
|
+
attr_accessor :maximum_width
|
218
|
+
# The width of a single datapoint.
|
219
|
+
attr_accessor :datapoint_width
|
220
|
+
|
221
|
+
# Draw the standard deviation chart on the supplied PDF document.
|
222
|
+
def render_on(pdf)
|
223
|
+
raise TypeError, PDF::Writer::Lang[:charts_stddev_data_empty] if @data.empty?
|
224
|
+
data = @data.dup
|
225
|
+
leftover_data = nil
|
226
|
+
|
227
|
+
loop do
|
228
|
+
# Set up the scale information.
|
229
|
+
scale = []
|
230
|
+
|
231
|
+
(@scale.first + @scale.step).step(@scale.last, @scale.step) do |ii|
|
232
|
+
scale << "%01.#{@scale.label.decimal_precision}f" % ii
|
233
|
+
end
|
234
|
+
|
235
|
+
scales = PDF::Writer::OHash.new
|
236
|
+
scale.each_with_index do |gg, ii|
|
237
|
+
scales[ii] = OpenStruct.new
|
238
|
+
scales[ii].value = gg
|
239
|
+
end
|
240
|
+
|
241
|
+
# Add information about the scales' locations to the scales
|
242
|
+
# hash. Note that the count is one smaller than it should be, so we're
|
243
|
+
# increasing it. The first scale is the bottom of the chart.
|
244
|
+
scale_count = scale.size + 1
|
245
|
+
|
246
|
+
label_height_adjuster = 0
|
247
|
+
label_height_adjuster = @label.height if @show_labels
|
248
|
+
|
249
|
+
chart_area_height = @height - label_height_adjuster
|
250
|
+
scale_height = chart_area_height / scale_count.to_f
|
251
|
+
|
252
|
+
scales.each_key do |index|
|
253
|
+
this_height = scale_height * (index + 1) + @label.height
|
254
|
+
scales[index].line_height = this_height
|
255
|
+
if @scale.show_labels
|
256
|
+
scales[index].label_height = this_height -
|
257
|
+
(@scale.label.text_size / 3.0)
|
258
|
+
end
|
259
|
+
end
|
260
|
+
|
261
|
+
# How many sections do we need in this chart, and how wide will it
|
262
|
+
# need to be?
|
263
|
+
chunk_width = @datapoint_width
|
264
|
+
num_chunks = data.size
|
265
|
+
widest_scale_label = 0
|
266
|
+
|
267
|
+
if @scale.show_labels
|
268
|
+
scales.each_value do |scale|
|
269
|
+
this_width = pdf.text_width(scale.value, @scale.label.text_size)
|
270
|
+
widest_scale_label = this_width if this_width > widest_scale_label
|
271
|
+
end
|
272
|
+
end
|
273
|
+
|
274
|
+
chart_width = chunk_width * num_chunks
|
275
|
+
total_width = chart_width + widest_scale_label + @scale.label.pad
|
276
|
+
|
277
|
+
# What happens if the projected width of the chart is too big?
|
278
|
+
# Figure out how to break the chart in pieces.
|
279
|
+
if total_width > @maximum_width
|
280
|
+
max_column_count = 0
|
281
|
+
base_width = widest_scale_label + @scale.label.pad
|
282
|
+
(1..(num_chunks + 1)).each do |ii|
|
283
|
+
if (base_width + (ii * chunk_width)) > @maximum_width
|
284
|
+
break
|
285
|
+
else
|
286
|
+
max_column_count += 1
|
287
|
+
end
|
288
|
+
end
|
289
|
+
|
290
|
+
leftover_data = data.slice!(max_column_count, -1)
|
291
|
+
|
292
|
+
num_chunks = data.size
|
293
|
+
chart_width = chunk_width * num_chunks
|
294
|
+
total_width = chart_width + widest_scale_label + @scale.label.pad
|
295
|
+
end
|
296
|
+
|
297
|
+
chart_y = pdf.y - @height + @leading_gap
|
298
|
+
chart_y += (@outer_borders.style.width * 2.0) if @outer_borders
|
299
|
+
|
300
|
+
if chart_y < pdf.bottom_margin
|
301
|
+
pdf.start_new_page
|
302
|
+
chart_y = pdf.y - @height
|
303
|
+
chart_y += (@outer_borders.style.width * 2.0) if @outer_borders
|
304
|
+
end
|
305
|
+
|
306
|
+
chart_x = pdf.absolute_x_middle - (total_width / 2.0) + widest_scale_label
|
307
|
+
|
308
|
+
# Add labels, if needed.
|
309
|
+
if @show_labels
|
310
|
+
pdf.save_state
|
311
|
+
pdf.fill_color! @label.background_color
|
312
|
+
# Draw a rectangle for each label
|
313
|
+
num_chunks.times do |ii|
|
314
|
+
this_x = chart_x + ii * chunk_width
|
315
|
+
pdf.rectangle(this_x, chart_y, chunk_width, @label.height).fill
|
316
|
+
end
|
317
|
+
|
318
|
+
# Add a border above the label rectangle.
|
319
|
+
if @outer_borders
|
320
|
+
pdf.stroke_style! @outer_borders.style
|
321
|
+
pdf.line(chart_x, chart_y + @label.height, chart_x + chart_width, chart_y + @label.height).stroke
|
322
|
+
end
|
323
|
+
pdf.fill_color! @label.text_color
|
324
|
+
|
325
|
+
data.each_with_index do |datum, ii|
|
326
|
+
label = datum.label.to_s
|
327
|
+
label_width = pdf.text_width(label, @label.text_size)
|
328
|
+
this_x = chart_x + (ii * chunk_width) + (chunk_width / 2.0) - (label_width / 2.0)
|
329
|
+
this_y = chart_y + (@label.height / 2.0) - (@label.text_size / 3.0)
|
330
|
+
pdf.add_text(this_x, this_y, label, @label.text_size)
|
331
|
+
end
|
332
|
+
pdf.restore_state
|
333
|
+
end
|
334
|
+
|
335
|
+
if @inner_borders
|
336
|
+
pdf.save_state
|
337
|
+
pdf.stroke_color! @inner_borders.color
|
338
|
+
pdf.stroke_style! @inner_borders.style
|
339
|
+
(num_chunks - 1).times do |ii|
|
340
|
+
this_x = chart_x + (ii * chunk_width) + chunk_width
|
341
|
+
pdf.line(this_x, chart_y, this_x, chart_y + @height).stroke
|
342
|
+
end
|
343
|
+
pdf.restore_state
|
344
|
+
end
|
345
|
+
|
346
|
+
pdf.save_state
|
347
|
+
if @outer_borders
|
348
|
+
pdf.stroke_color! @outer_borders.color
|
349
|
+
pdf.stroke_style! @outer_borders.style
|
350
|
+
pdf.rectangle(chart_x, chart_y, chart_width, @height).stroke
|
351
|
+
end
|
352
|
+
|
353
|
+
if @scale.style
|
354
|
+
pdf.save_state
|
355
|
+
pdf.stroke_style! @scale.style
|
356
|
+
scales.each_value do |scale|
|
357
|
+
this_y = chart_y + scale.line_height
|
358
|
+
pdf.line(chart_x, this_y, chart_x + chart_width, this_y).stroke
|
359
|
+
end
|
360
|
+
pdf.restore_state
|
361
|
+
end
|
362
|
+
|
363
|
+
if @scale.show_labels
|
364
|
+
pdf.save_state
|
365
|
+
scales.each_value do |scale|
|
366
|
+
this_y = chart_y + scale.label_height
|
367
|
+
label_width = pdf.text_width(scale.value, @scale.label.text_size)
|
368
|
+
this_x = chart_x - label_width - @scale.label.pad
|
369
|
+
pdf.fill_color! @scale.label.text_color
|
370
|
+
pdf.add_text(this_x, this_y, scale.value, @scale.label.text_size)
|
371
|
+
end
|
372
|
+
pdf.restore_state
|
373
|
+
end
|
374
|
+
|
375
|
+
data.each_with_index do |datum, ii|
|
376
|
+
avg_height = datum.average * scale_height
|
377
|
+
stddev_height = datum.stddev * scale_height
|
378
|
+
this_y = chart_y + label_height_adjuster + avg_height
|
379
|
+
this_x = chart_x + (ii * chunk_width) + (chunk_width / 2.0)
|
380
|
+
line_top_y = this_y + (stddev_height / 2.0)
|
381
|
+
line_bot_y = this_y - (stddev_height / 2.0)
|
382
|
+
|
383
|
+
# Plot the dot
|
384
|
+
if @dot
|
385
|
+
pdf.stroke_color! @dot.color
|
386
|
+
pdf.stroke_style! @dot.style
|
387
|
+
pdf.circle_at(this_x, this_y, (@dot.style.width / 2.0)).fill
|
388
|
+
end
|
389
|
+
|
390
|
+
# Plot the bar
|
391
|
+
if @bar
|
392
|
+
pdf.stroke_color! @bar.color
|
393
|
+
pdf.stroke_style! @bar.style
|
394
|
+
pdf.line(this_x, line_top_y, this_x, line_bot_y).stroke
|
395
|
+
end
|
396
|
+
|
397
|
+
# Plot the crossbars
|
398
|
+
if @upper_crossbar
|
399
|
+
if @dot
|
400
|
+
cb_width = @dot.style.width
|
401
|
+
else
|
402
|
+
cb_width = @upper_crossbar.style.width
|
403
|
+
end
|
404
|
+
pdf.stroke_color! @upper_crossbar.color
|
405
|
+
pdf.stroke_style! @upper_crossbar.style
|
406
|
+
pdf.line(this_x - cb_width, line_top_y, this_x + cb_width, line_top_y).stroke
|
407
|
+
end
|
408
|
+
if @lower_crossbar
|
409
|
+
if @dot
|
410
|
+
cb_width = @dot.style.width
|
411
|
+
else
|
412
|
+
cb_width = @lower_crossbar.style.width
|
413
|
+
end
|
414
|
+
pdf.stroke_color! @lower_crossbar.color
|
415
|
+
pdf.stroke_style! @lower_crossbar.style
|
416
|
+
|
417
|
+
pdf.line(this_x - cb_width, line_bot_y, this_x + cb_width, line_bot_y).stroke
|
418
|
+
end
|
419
|
+
end
|
420
|
+
|
421
|
+
pdf.restore_state
|
422
|
+
|
423
|
+
pdf.y = chart_y
|
424
|
+
|
425
|
+
break if leftover_data.nil?
|
426
|
+
|
427
|
+
data = leftover_data
|
428
|
+
leftover_data = nil
|
429
|
+
end
|
430
|
+
|
431
|
+
pdf.y
|
432
|
+
end
|
433
|
+
end
|