sugar_png 0.0.1 → 0.4.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (273) hide show
  1. data/Gemfile +2 -1
  2. data/Gemfile.lock +4 -4
  3. data/README.md +105 -0
  4. data/README.md.tpl +99 -0
  5. data/Rakefile +41 -6
  6. data/VERSION +1 -1
  7. data/data/1-dump-hex.rb +22 -0
  8. data/data/2-marshal-test.rb +22 -0
  9. data/data/3-hex2marshal.rb +71 -0
  10. data/data/font/00 +0 -0
  11. data/data/font/01 +0 -0
  12. data/data/font/02 +0 -0
  13. data/data/font/03 +0 -0
  14. data/data/font/04 +0 -0
  15. data/data/font/05 +0 -0
  16. data/data/font/06 +0 -0
  17. data/data/font/07 +0 -0
  18. data/data/font/09 +0 -0
  19. data/data/font/0a +0 -0
  20. data/data/font/0b +0 -0
  21. data/data/font/0c +0 -0
  22. data/data/font/0d +0 -0
  23. data/data/font/0e +0 -0
  24. data/data/font/0f +0 -0
  25. data/data/font/10 +0 -0
  26. data/data/font/11 +0 -0
  27. data/data/font/12 +0 -0
  28. data/data/font/13 +0 -0
  29. data/data/font/14 +0 -0
  30. data/data/font/15 +0 -0
  31. data/data/font/16 +0 -0
  32. data/data/font/17 +0 -0
  33. data/data/font/18 +0 -0
  34. data/data/font/19 +0 -0
  35. data/data/font/1a +0 -0
  36. data/data/font/1b +0 -0
  37. data/data/font/1c +0 -0
  38. data/data/font/1d +0 -0
  39. data/data/font/1e +0 -0
  40. data/data/font/1f +0 -0
  41. data/data/font/20 +0 -0
  42. data/data/font/21 +0 -0
  43. data/data/font/22 +0 -0
  44. data/data/font/23 +0 -0
  45. data/data/font/24 +0 -0
  46. data/data/font/25 +0 -0
  47. data/data/font/26 +0 -0
  48. data/data/font/27 +0 -0
  49. data/data/font/28 +0 -0
  50. data/data/font/29 +0 -0
  51. data/data/font/2a +0 -0
  52. data/data/font/2b +0 -0
  53. data/data/font/2c +0 -0
  54. data/data/font/2d +0 -0
  55. data/data/font/2e +0 -0
  56. data/data/font/2f +0 -0
  57. data/data/font/30 +0 -0
  58. data/data/font/31 +0 -0
  59. data/data/font/32 +0 -0
  60. data/data/font/33 +0 -0
  61. data/data/font/34 +0 -0
  62. data/data/font/35 +0 -0
  63. data/data/font/36 +0 -0
  64. data/data/font/37 +0 -0
  65. data/data/font/38 +0 -0
  66. data/data/font/39 +0 -0
  67. data/data/font/3a +0 -0
  68. data/data/font/3b +0 -0
  69. data/data/font/3c +0 -0
  70. data/data/font/3d +0 -0
  71. data/data/font/3e +0 -0
  72. data/data/font/3f +0 -0
  73. data/data/font/40 +0 -0
  74. data/data/font/41 +0 -0
  75. data/data/font/42 +0 -0
  76. data/data/font/43 +0 -0
  77. data/data/font/44 +0 -0
  78. data/data/font/45 +0 -0
  79. data/data/font/46 +0 -0
  80. data/data/font/47 +0 -0
  81. data/data/font/48 +0 -0
  82. data/data/font/49 +0 -0
  83. data/data/font/4a +0 -0
  84. data/data/font/4b +0 -0
  85. data/data/font/4c +0 -0
  86. data/data/font/4d +0 -0
  87. data/data/font/4e +0 -0
  88. data/data/font/4f +0 -0
  89. data/data/font/50 +0 -0
  90. data/data/font/51 +0 -0
  91. data/data/font/52 +0 -0
  92. data/data/font/53 +0 -0
  93. data/data/font/54 +0 -0
  94. data/data/font/55 +0 -0
  95. data/data/font/56 +0 -0
  96. data/data/font/57 +0 -0
  97. data/data/font/58 +0 -0
  98. data/data/font/59 +0 -0
  99. data/data/font/5a +0 -0
  100. data/data/font/5b +0 -0
  101. data/data/font/5c +0 -0
  102. data/data/font/5d +0 -0
  103. data/data/font/5e +0 -0
  104. data/data/font/5f +0 -0
  105. data/data/font/60 +0 -0
  106. data/data/font/61 +0 -0
  107. data/data/font/62 +0 -0
  108. data/data/font/63 +0 -0
  109. data/data/font/64 +0 -0
  110. data/data/font/65 +0 -0
  111. data/data/font/66 +0 -0
  112. data/data/font/67 +0 -0
  113. data/data/font/68 +0 -0
  114. data/data/font/69 +0 -0
  115. data/data/font/6a +0 -0
  116. data/data/font/6b +0 -0
  117. data/data/font/6c +0 -0
  118. data/data/font/6d +0 -0
  119. data/data/font/6e +0 -0
  120. data/data/font/6f +0 -0
  121. data/data/font/70 +0 -0
  122. data/data/font/71 +0 -0
  123. data/data/font/72 +0 -0
  124. data/data/font/73 +0 -0
  125. data/data/font/74 +0 -0
  126. data/data/font/75 +0 -0
  127. data/data/font/76 +0 -0
  128. data/data/font/77 +0 -0
  129. data/data/font/78 +0 -0
  130. data/data/font/79 +0 -0
  131. data/data/font/7a +0 -0
  132. data/data/font/7b +0 -0
  133. data/data/font/7c +0 -0
  134. data/data/font/7d +0 -0
  135. data/data/font/7e +0 -0
  136. data/data/font/7f +0 -0
  137. data/data/font/80 +0 -0
  138. data/data/font/81 +0 -0
  139. data/data/font/82 +0 -0
  140. data/data/font/83 +0 -0
  141. data/data/font/84 +0 -0
  142. data/data/font/85 +0 -0
  143. data/data/font/86 +0 -0
  144. data/data/font/87 +0 -0
  145. data/data/font/88 +0 -0
  146. data/data/font/89 +0 -0
  147. data/data/font/8a +0 -0
  148. data/data/font/8b +0 -0
  149. data/data/font/8c +0 -0
  150. data/data/font/8d +0 -0
  151. data/data/font/8e +0 -0
  152. data/data/font/8f +0 -0
  153. data/data/font/90 +0 -0
  154. data/data/font/91 +0 -0
  155. data/data/font/92 +0 -0
  156. data/data/font/93 +0 -0
  157. data/data/font/94 +0 -0
  158. data/data/font/95 +0 -0
  159. data/data/font/96 +0 -0
  160. data/data/font/97 +0 -0
  161. data/data/font/98 +0 -0
  162. data/data/font/99 +0 -0
  163. data/data/font/9a +0 -0
  164. data/data/font/9b +0 -0
  165. data/data/font/9c +0 -0
  166. data/data/font/9d +0 -0
  167. data/data/font/9e +0 -0
  168. data/data/font/9f +0 -0
  169. data/data/font/a0 +0 -0
  170. data/data/font/a1 +0 -0
  171. data/data/font/a2 +0 -0
  172. data/data/font/a3 +0 -0
  173. data/data/font/a4 +0 -0
  174. data/data/font/a5 +0 -0
  175. data/data/font/a6 +0 -0
  176. data/data/font/a7 +0 -0
  177. data/data/font/a8 +0 -0
  178. data/data/font/a9 +0 -0
  179. data/data/font/aa +0 -0
  180. data/data/font/ac +0 -0
  181. data/data/font/ad +0 -0
  182. data/data/font/ae +0 -0
  183. data/data/font/af +0 -0
  184. data/data/font/b0 +0 -0
  185. data/data/font/b1 +0 -0
  186. data/data/font/b2 +0 -0
  187. data/data/font/b3 +0 -0
  188. data/data/font/b4 +0 -0
  189. data/data/font/b5 +0 -0
  190. data/data/font/b6 +0 -0
  191. data/data/font/b7 +0 -0
  192. data/data/font/b8 +0 -0
  193. data/data/font/b9 +0 -0
  194. data/data/font/ba +0 -0
  195. data/data/font/bb +0 -0
  196. data/data/font/bc +0 -0
  197. data/data/font/bd +0 -0
  198. data/data/font/be +0 -0
  199. data/data/font/bf +0 -0
  200. data/data/font/c0 +0 -0
  201. data/data/font/c1 +0 -0
  202. data/data/font/c2 +0 -0
  203. data/data/font/c3 +0 -0
  204. data/data/font/c4 +0 -0
  205. data/data/font/c5 +0 -0
  206. data/data/font/c6 +0 -0
  207. data/data/font/c7 +0 -0
  208. data/data/font/c8 +0 -0
  209. data/data/font/c9 +0 -0
  210. data/data/font/ca +0 -0
  211. data/data/font/cb +0 -0
  212. data/data/font/cc +0 -0
  213. data/data/font/cd +0 -0
  214. data/data/font/ce +0 -0
  215. data/data/font/cf +0 -0
  216. data/data/font/d0 +0 -0
  217. data/data/font/d1 +0 -0
  218. data/data/font/d2 +0 -0
  219. data/data/font/d3 +0 -0
  220. data/data/font/d4 +0 -0
  221. data/data/font/d5 +0 -0
  222. data/data/font/d6 +0 -0
  223. data/data/font/d7 +0 -0
  224. data/data/font/f9 +0 -0
  225. data/data/font/fa +0 -0
  226. data/data/font/fb +0 -0
  227. data/data/font/fc +0 -0
  228. data/data/font/fd +0 -0
  229. data/data/font/fe +0 -0
  230. data/data/font/ff +0 -0
  231. data/data/unifont-5.1.20080820.7z +0 -0
  232. data/data/unused-4-read-tar.rb +49 -0
  233. data/lib/sugar_png.rb +164 -2
  234. data/lib/sugar_png/border.rb +20 -0
  235. data/lib/sugar_png/color.rb +118 -0
  236. data/lib/sugar_png/datastream.rb +2 -17
  237. data/lib/sugar_png/dyn_accessor.rb +28 -0
  238. data/lib/sugar_png/font.rb +44 -0
  239. data/lib/sugar_png/glyph.rb +36 -0
  240. data/lib/sugar_png/image.rb +104 -0
  241. data/samples/damaged_chunk.png +0 -0
  242. data/samples/png_suite/f00n0g08_reference.png +0 -0
  243. data/samples/png_suite/f00n2c08_reference.png +0 -0
  244. data/samples/png_suite/f01n0g08_reference.png +0 -0
  245. data/samples/png_suite/f01n2c08_reference.png +0 -0
  246. data/samples/png_suite/f02n0g08_reference.png +0 -0
  247. data/samples/png_suite/f02n2c08_reference.png +0 -0
  248. data/samples/png_suite/f03n0g08_reference.png +0 -0
  249. data/samples/png_suite/f03n2c08_reference.png +0 -0
  250. data/samples/png_suite/f04n0g08_reference.png +0 -0
  251. data/samples/png_suite/f04n2c08_reference.png +0 -0
  252. data/samples/readme/explicit_image_dimensions_bg_color.png +0 -0
  253. data/samples/readme/hello_world.png +0 -0
  254. data/samples/readme/japanese_text_with_rainbow_borders_zoomed_4x.png +0 -0
  255. data/samples/readme/pixels_can_be_set_using_ranges_enumerators_arrays.png +0 -0
  256. data/samples/readme/playing_with_transparency_16_bit_color_depth.png +0 -0
  257. data/samples/readme/white_noise.png +0 -0
  258. data/samples/rgba.tar.bz2 +0 -0
  259. data/samples/text_chunk.png +0 -0
  260. data/samples/ztxt_chunk.png +0 -0
  261. data/spec/color_spec.rb +81 -0
  262. data/spec/font_spec.rb +82 -0
  263. data/spec/magic/border_spec.rb +49 -0
  264. data/spec/magic/create_spec.rb +42 -0
  265. data/spec/magic/pixels_spec.rb +73 -0
  266. data/spec/magic/text_spec.rb +113 -0
  267. data/spec/magic/zoom_spec.rb +38 -0
  268. data/spec/png_suite_spec.rb +145 -0
  269. data/spec/spec_helper.rb +43 -1
  270. data/spec/sugar_png/datastream_spec.rb +32 -0
  271. data/spec/support/png_suite.rb +43 -0
  272. data/sugar_png.gemspec +275 -6
  273. metadata +287 -5
@@ -1,5 +1,167 @@
1
1
  require 'zpng'
2
2
 
3
- SugarPNG = ZPNG
3
+ require 'sugar_png/dyn_accessor'
4
+ require 'sugar_png/color'
5
+ require 'sugar_png/border'
6
+ require 'sugar_png/image'
7
+ require 'sugar_png/font'
8
+ require 'sugar_png/glyph'
4
9
 
5
- require 'sugar_png/datastream'
10
+ class SugarPNG
11
+ DEFAULT_BG = :transparent
12
+ DEFAULT_FG = :black
13
+
14
+ Canvas = Datastream = Image
15
+
16
+ class Exception < ::Exception; end
17
+ class ArgumentError < Exception; end
18
+
19
+ extend DynAccessor
20
+ dyn_accessor :width, :height, :zoom, :depth
21
+ dyn_accessor :bg => %w'background bg_color background_color'
22
+ dyn_accessor :fg => %w'foreground fg_color foreground_color color'
23
+
24
+ def initialize h={}, &block
25
+ @bg = DEFAULT_BG
26
+ @fg = DEFAULT_FG
27
+ @zoom = 1
28
+ clear
29
+
30
+ if block_given?
31
+ if block.arity == 1
32
+ yield self
33
+ else
34
+ instance_eval &block
35
+ end
36
+ end
37
+
38
+ # boring non-magic hash
39
+ h.each do |k,v|
40
+ self.send "#{k}=", v
41
+ end
42
+ end
43
+
44
+ # reset all drawings
45
+ def clear
46
+ @pixels = Hash.new{ |k,v| k[v] = {} }
47
+ @borders = []
48
+ end
49
+
50
+ # set pixels
51
+ # accepted coordinate values:
52
+ # a) boring Integers
53
+ # b) neat Arrays
54
+ # c) long Ranges
55
+ # d) super Enumerators
56
+ #
57
+ # accepted color values: see SugarPNG::Color
58
+ def []= ax, ay, color
59
+ Array(ay).each do |y|
60
+ Array(ax).each do |x|
61
+ @pixels[y][x] = color
62
+ end
63
+ end
64
+ end
65
+
66
+ # same as above, but color argument can be optional
67
+ def pixel ax, ay, color = @fg
68
+ Array(ay).each do |y|
69
+ Array(ax).each do |x|
70
+ @pixels[y][x] = color
71
+ end
72
+ end
73
+ end
74
+ alias :pixels :pixel
75
+
76
+ %w'put_pixel set_pixel point dot'.each do |x|
77
+ # plural & singular aliases to increase entropy & prevent global singularity
78
+ class_eval "alias :#{x} :pixel; alias :#{x}s :pixels"
79
+ end
80
+
81
+ # draw image border with specified color
82
+ def border size, color = nil
83
+ color ||= @fg || @bg
84
+ @borders << Border.new( size.is_a?(Hash) ? size : {:size => size, :color => color} )
85
+ end
86
+
87
+ # same as border, but default color is background
88
+ def padding size, color = @bg
89
+ border size, color
90
+ end
91
+
92
+ # draw a single glyph, used from within text()
93
+ def draw_glyph glyph, x0, y, color
94
+ #TODO: optimize?
95
+ glyph.to_a.each do |row|
96
+ x = x0
97
+ row.each do |bit|
98
+ self[x,y] = color if bit == 1
99
+ x += 1
100
+ end
101
+ y += 1
102
+ end
103
+ end
104
+
105
+ # draws text, optional arguments are :color, :x, :y
106
+ def text text, h = {}
107
+ font = @font ||= Font.new
108
+ color = h[:color] || @fg
109
+ y = h[:y] || 0
110
+ text.split(/[\r\n]+/).each do |line|
111
+ x = h[:x] || 0
112
+ line.each_char do |c|
113
+ glyph = font[c]
114
+ draw_glyph glyph, x, y, color
115
+ x += glyph.width
116
+ end
117
+ y += font.height
118
+ end
119
+ end
120
+
121
+ # export PNG to file
122
+ def save fname
123
+ File.open(fname, "wb"){ |f| f<<to_s }
124
+ end
125
+
126
+ # get PNG as bytestream, for saving it to file manually, or for sending via HTTP
127
+ def to_s
128
+ height = @height || ((t=@pixels.keys.max) && t+1 ) || 0
129
+ width = @width || ((t=@pixels.values.map(&:keys).map(&:max).max) && t+1 ) || 0
130
+
131
+ xofs = yofs = 0
132
+ xmax = width-1
133
+ ymax = height-1
134
+
135
+ if @borders.any?
136
+ width += @borders.map(&:width).inject(&:+)
137
+ height += @borders.map(&:height).inject(&:+)
138
+ xofs += @borders.map(&:left).inject(&:+)
139
+ yofs += @borders.map(&:top).inject(&:+)
140
+ end
141
+
142
+ raise(Exception.new("invalid image height #{height}")) if height <= 0
143
+ raise(Exception.new("invalid image width #{width}")) if width <= 0
144
+
145
+ img = Image.new :width => width, :height => height, :depth => @depth
146
+ img.clear(_color(@bg)) if @bg
147
+ img.draw_borders(@borders.each{ |b| b.color = _color(b.color)} )
148
+
149
+ @pixels.each do |y, xh|
150
+ next if y>ymax
151
+ xh.each do |x, c|
152
+ next if x>xmax
153
+ img[x+xofs,y+yofs] = _color(c)
154
+ end
155
+ end
156
+
157
+ img.zoom(@zoom).export
158
+ end
159
+ alias :export :to_s
160
+
161
+ private
162
+
163
+ # create color from any of the supported color representations
164
+ def _color c
165
+ c = c.is_a?(ZPNG::Color) ? c : Color.new(c, :depth => @depth)
166
+ end
167
+ end
@@ -0,0 +1,20 @@
1
+ class SugarPNG
2
+ class Border
3
+ attr_accessor :left, :right, :top, :bottom, :color
4
+ def initialize h
5
+ @color = h[:color] || raise(ArgumentError.new("border color must be set"))
6
+ @left = (h[:left] || h[:size]).to_i
7
+ @right = (h[:right] || h[:size]).to_i
8
+ @top = (h[:top] || h[:size]).to_i
9
+ @bottom= (h[:bottom]|| h[:size]).to_i
10
+ end
11
+
12
+ def width
13
+ @left + @right
14
+ end
15
+
16
+ def height
17
+ @top + @bottom
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,118 @@
1
+ class SugarPNG
2
+ class Color < ZPNG::Color
3
+ class << self
4
+ def a color
5
+ color.a || (2**color.depth-1)
6
+ end
7
+
8
+ def r color; color.r; end
9
+ def g color; color.g; end
10
+ def b color; color.b; end
11
+ end
12
+
13
+ # accepted color values:
14
+ # a) Strings: "blue", "RED", ...
15
+ # b) Symbols: :blue, :red, ...
16
+ # c) HTML notation: #cc3344, #ccc, ...
17
+ # d) 3..4 int args: (10, 20, 30) - RGB; (0x20,0x30,0x40,0xff) - RGBA
18
+ # e) Hexadecimals 0xaabbcc - RGB; 0x80aabbcc - RGBA
19
+ # f) Hashes (long): { :red => 10, :green => 30, :blue => 40, :alpha => 0x80 }
20
+ # g) Hashes (shrt): { r:10, g:20, b:30, a:50 } (alpha is optional)
21
+ #
22
+ # all notations also accept one optional last argument - hash:
23
+ # :depth => 1..16 - color depth of each channel, including alpha
24
+ # :alpha => 0..2^depth - explicit definition of alpha value
25
+ def initialize *args
26
+ h = args.last.is_a?(Hash) ? args.pop : {}
27
+
28
+ @depth = h.delete(:depth) || 8
29
+ raise ArgumentError.new "invalid depth: #@depth" unless (1..16).include?(@depth)
30
+
31
+ max = 2**@depth-1
32
+
33
+ case args.size
34
+ when 0 # single Hash that already in h
35
+ # init colors to default values, we'll assign them later
36
+ @r = @g = @b = 0
37
+ when 1 # String(name or html), Symbol, Array, Integer, Hash
38
+ case arg = args.first
39
+ when String
40
+ _from_string(arg)
41
+
42
+ when Symbol
43
+ _from_string(arg.to_s)
44
+
45
+ when Array
46
+ case arg.size
47
+ when 3
48
+ @r,@g,@b = arg
49
+ when 4
50
+ @r,@g,@b,@a = arg
51
+ else
52
+ raise ArgumentError.new "invalid array size: #{arg.size}, must be 3 or 4"
53
+ end
54
+ when Integer
55
+ @r = (arg >> (@depth*2)) & max
56
+ @g = (arg >> (@depth)) & max
57
+ @b = arg & max
58
+ when Hash
59
+ # init colors to default values, we'll assign them later
60
+ @r = @g = @b = 0
61
+ h.merge! args.first
62
+ else
63
+ raise ArgumentError.new "invalid argument type: #{args.first.class}"
64
+ end
65
+ when 3 # r, g, b
66
+ @r, @g, @b = args
67
+ when 4 # r, g, b, a
68
+ @r, @g, @b, @a = args
69
+ else
70
+ raise ArgumentError.new "invalid number of arguments: #{args.size}"
71
+ end
72
+
73
+ @a ||= max # opaque by default
74
+
75
+ h.each do |k,v|
76
+ case k
77
+ when :r, :red
78
+ @r = v
79
+ when :g, :green
80
+ @g = v
81
+ when :b, :blue
82
+ @b = v
83
+ when :a, :alpha
84
+ @a = v
85
+ else
86
+ raise ArgumentError.new "invalid key: #{k}"
87
+ end
88
+ end
89
+
90
+ [:r, :g, :b, :a].each do |x|
91
+ v = self.send(x).to_i
92
+ raise ArgumentError.new "invalid channel value: #{v}" if v<0 || v>max
93
+ end
94
+ end
95
+
96
+ private
97
+
98
+ # initialize self from string, either HTML color or simple color name
99
+ # @depth must already be set!
100
+ def _from_string s
101
+ if s[0,1] == "#"
102
+ # html colors #aabbcc or #abc
103
+ case s.size
104
+ when 4
105
+ @r,@g,@b = s[1..-1].split('').map{ |x| x.to_i(16)*17 }
106
+ when 7
107
+ @r,@g,@b = s[1..-1].scan(/../).map{ |x| x.to_i(16) }
108
+ else
109
+ raise ArgumentError.new "invalid HTML color #{s}"
110
+ end
111
+ else
112
+ @r, @g, @b, @a = self.class.const_get(s.strip.upcase).to_depth(@depth).to_a
113
+ end
114
+ rescue
115
+ raise ArgumentError.new "invalid color name: #{s}"
116
+ end
117
+ end
118
+ end
@@ -1,19 +1,4 @@
1
- module SugarPNG
2
- class Datastream
3
- attr_accessor :img
4
-
5
- def initialize h={}
6
- @img = h[:img]
7
- end
8
-
9
- def metadata
10
- {}
11
- end
12
-
13
- class << self
14
- def from_file fname
15
- self.new :img => ZPNG::Image.load(fname)
16
- end
17
- end
1
+ class SugarPNG
2
+ class Datastream < ZPNG::Image
18
3
  end
19
4
  end
@@ -0,0 +1,28 @@
1
+ class SugarPNG
2
+ module DynAccessor
3
+ def dyn_accessor *names
4
+ names.each do |name|
5
+ if name.is_a?(Hash)
6
+ # dynamic accessor with alias(es)
7
+ name.each do |main, aliases|
8
+ dyn_accessor main
9
+ Array(aliases).each do |aliased|
10
+ class_eval <<-EOF
11
+ alias :#{aliased} :#{main}
12
+ alias :#{aliased}= :#{main}=
13
+ EOF
14
+ end
15
+ end
16
+ else
17
+ attr_writer name
18
+ # dynamic getter or setter based on argument given or not
19
+ class_eval <<-EOF
20
+ def #{name} arg=nil
21
+ arg ? @#{name} = arg : @#{name}
22
+ end
23
+ EOF
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,44 @@
1
+ class SugarPNG
2
+ class Font
3
+ DEFAULT_DIR = File.expand_path("../../data/font", File.dirname(__FILE__))
4
+ HEIGHT = 16
5
+
6
+ def initialize dir = DEFAULT_DIR
7
+ @dir = dir
8
+ @pages = {}
9
+ end
10
+
11
+ def height
12
+ HEIGHT
13
+ end
14
+
15
+ # get glyph by index
16
+ def [] idx
17
+ idx = idx.ord if !idx.is_a?(Integer) && idx.respond_to?(:ord)
18
+ raise ArgumentError.new("invalid idx type: #{idx.class}") unless idx.is_a?(Integer)
19
+ raise ArgumentError.new("invalid idx: #{idx.inspect}") if idx<0 || idx>0xffff
20
+
21
+ pageno = idx >> 8
22
+ @pages[pageno] ||= Page.new(File.join(@dir, "%02x" % pageno))
23
+ @pages[pageno][idx]
24
+ end
25
+
26
+ class Page
27
+ def initialize fname
28
+ @data = Marshal.load(File.binread(fname))
29
+ @glyphs = {}
30
+ end
31
+
32
+ # get glyph by index
33
+ def [] ord
34
+ idx = ord&0xff
35
+ @glyphs[idx] ||= Glyph.new(
36
+ :height => HEIGHT,
37
+ :width => @data[idx].size/2,
38
+ :data => @data[idx],
39
+ :ord => ord
40
+ )
41
+ end
42
+ end # class Page
43
+ end # class Font
44
+ end
@@ -0,0 +1,36 @@
1
+ class SugarPNG
2
+ class Glyph
3
+ attr_accessor :width, :height, :data, :ord
4
+
5
+ def initialize h = {}
6
+ @ord = h[:ord]
7
+ @data = h[:data]
8
+ @width = h[:width]
9
+ @height = h[:height]
10
+ end
11
+
12
+ def blank?
13
+ @data.tr("\x00","").empty?
14
+ end
15
+
16
+ def to_s repl=" #"
17
+ bytes_in_row = (@width/8.0).ceil
18
+ r = ''; ptr = 0
19
+ @height.times.each do
20
+ r += @data[ptr,bytes_in_row].unpack("B#@width")[0].tr("01",repl) + "\n"
21
+ ptr += bytes_in_row
22
+ end
23
+ r.chomp
24
+ end
25
+
26
+ def to_a
27
+ bytes_in_row = (@width/8.0).ceil
28
+ r = []; ptr = 0
29
+ @height.times.each do
30
+ r << @data[ptr,bytes_in_row].unpack("B#@width")[0].split('').map(&:to_i)
31
+ ptr += bytes_in_row
32
+ end
33
+ r
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,104 @@
1
+ class SugarPNG
2
+ class Image < ZPNG::Image
3
+ def to_rgba_stream
4
+ pixels.map do |color|
5
+ color.to_depth(8).to_a.pack('C*')
6
+ end.join
7
+ end
8
+
9
+ def clear color
10
+ width.times do |x|
11
+ self[x,0] = color
12
+ end
13
+ sl0 = scanlines[0]
14
+ scanlines[1..-1].each do |sl|
15
+ sl.decoded_bytes = sl0.decoded_bytes.dup
16
+ end
17
+ end
18
+
19
+ def draw_borders borders
20
+ xmin = 0
21
+ xmax = self.width - 1
22
+ ytop = 0
23
+ ybtm = self.height - 1
24
+
25
+ sumtop = borders.map(&:top).inject(&:+)
26
+ sumbtm = borders.map(&:bottom).inject(&:+)
27
+
28
+ borders.each do |b|
29
+ b.top.times do
30
+ xmin.upto(xmax){ |x| self[x,ytop] = b.color }
31
+ ytop += 1
32
+ end
33
+ b.bottom.times do
34
+ xmin.upto(xmax){ |x| self[x,ybtm] = b.color }
35
+ ybtm -= 1
36
+ end
37
+ b.left.times do
38
+ ytop.upto(sumtop){ |y| self[xmin, y] = b.color }
39
+ ybtm.downto(height-sumbtm){ |y| self[xmin, y] = b.color }
40
+ xmin += 1
41
+ end
42
+ b.right.times do
43
+ ytop.upto(sumtop){ |y| self[xmax, y] = b.color }
44
+ ybtm.downto(height-sumbtm){ |y| self[xmax, y] = b.color }
45
+ xmax -= 1
46
+ end
47
+ end
48
+
49
+ # copy remaining identical scanlines
50
+ sl0 = scanlines[ytop]
51
+ (ytop+1).upto(ybtm) do |y|
52
+ scanlines[y].decoded_bytes = sl0.decoded_bytes.dup
53
+ end
54
+ end # draw_borders
55
+
56
+ # zooms image by specified *integer* factor,
57
+ # returns self if zoom == 1, new image if zoom > 1
58
+ def zoom factor
59
+ factor = factor.to_i
60
+ return self if factor == 1 # no zoom required
61
+ raise ArgumentError.new("Invalid zoom factor #{factor}") if factor < 1
62
+
63
+ new_img = Image.new(
64
+ :width => self.width*factor,
65
+ :height => self.height*factor,
66
+ :color => self.hdr.color,
67
+ :depth => self.hdr.depth
68
+ )
69
+
70
+ if self.bpp % 8 == 0
71
+ nbytes = self.bpp / 8
72
+ # fast-zoom is possible
73
+ scanlines.each_with_index do |sl,idx|
74
+ new_sl = new_img.scanlines[idx*factor]
75
+ self.width.times do |x|
76
+ new_sl.decoded_bytes[x*factor*nbytes, factor*nbytes] =
77
+ sl.decoded_bytes[x*nbytes,nbytes]*factor
78
+ end
79
+ # copy scanlines
80
+ (factor-1).times do |i|
81
+ new_img.scanlines[idx*factor+i+1].decoded_bytes = new_sl.decoded_bytes
82
+ end
83
+ end
84
+ else
85
+ # slow-zoom
86
+ scanlines.each_with_index do |sl,idx|
87
+ new_sl = new_img.scanlines[idx*factor]
88
+ self.width.times do |x|
89
+ c = sl[x]
90
+ factor.times do |zx|
91
+ new_sl[x*factor + zx] = c
92
+ end
93
+ end
94
+ # copy scanlines
95
+ (factor-1).times do |i|
96
+ new_img.scanlines[idx*factor+i+1].decoded_bytes = new_sl.decoded_bytes
97
+ end
98
+ end
99
+ end
100
+
101
+ new_img
102
+ end
103
+ end
104
+ end