contrib/inkscape_tools: refactor main code into 2 new classes
[nit.git] / contrib / inkscape_tools / src / svg_to_png_and_nit.nit
1 # This file is part of NIT ( http://www.nitlanguage.org ).
2 #
3 # Copyright 2012-2015 Alexis Laferrière <alexis.laf@xymus.net>
4 #
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
8 #
9 # http://www.apache.org/licenses/LICENSE-2.0
10 #
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
16
17 # Extract images of objects from an SVG file using Inkscape
18 module svg_to_png_and_nit
19
20 import opts
21 import template
22
23 # Image information extracted from the SVG file
24 class Image
25 # Name extracted from the object ID minus the `0` prefix and Nit safe
26 var name: String
27
28 # Left border
29 var x: Int
30
31 # Top border
32 var y: Int
33
34 # Image width
35 var w: Int
36
37 # Image height
38 var h: Int
39
40 # Right border
41 fun right: Int do return x+w
42
43 # Bottom border
44 fun bottom: Int do return y+h
45
46 redef fun to_s do return name
47 end
48
49 # Document being processed, concerns both the source and the target
50 class Document
51 # Name of the source file
52 var drawing_name: String
53
54 # Name of the class to generate
55 var nit_class_name: String = drawing_name.capitalized + "Images" is lazy
56
57 # Scaling to apply to the exported image
58 var scale: Float
59
60 # Source minimum X
61 var min_x: Int
62
63 # Source maximum X
64 var max_x: Int
65
66 # Source minimum Y
67 var min_y: Int
68
69 # Source maximum Y
70 var max_y: Int
71
72 # Get the coordinates for `image` as `"x, y, w, h"`
73 fun coordinates(image: Image): String
74 do
75 var x = image.x.adapt(min_x, scale)
76 var y = image.y.adapt(min_y, scale)
77 var w = (image.w.to_f*scale).to_i
78 var h = (image.h.to_f*scale).to_i
79
80 return "{x}, {y}, {w}, {h}"
81 end
82 end
83
84 # Nit module with a single class to retrieve to access the extracted images
85 abstract class ImageSetSrc
86 super Template
87
88 # Target document
89 var document: Document
90
91 # Images found in the source document
92 var images: Array[Image]
93 end
94
95 # Nit module targeting the MNit framework
96 class MnitImageSetSrc
97 super ImageSetSrc
98
99 redef fun rendering
100 do
101 # Known array of images
102 var arrays_of_images = new Array[String]
103
104 # Attributes of the generated class
105 var attributes = new Array[String]
106
107 # Statements for the generated `load_all` method
108 var load_exprs = new Array[String]
109
110 # Add images to Nit source file
111 for image in images do
112 # Adapt coordinates to new top left and scale
113 var coordinates = document.coordinates(image)
114
115 var nit_name = image.name
116 var last_char = nit_name.chars.last
117 if last_char.to_s.is_numeric then
118 # Array of images
119 # TODO support more than 10 images in an array
120
121 nit_name = nit_name.substring(0, nit_name.length-1)
122 if not arrays_of_images.has(nit_name) then
123 # Create class attribute to store Array
124 arrays_of_images.add(nit_name)
125 attributes.add "\tvar {nit_name} = new Array[Image]\n"
126 end
127 load_exprs.add "\t\t{nit_name}.add(main_image.subimage({coordinates}))\n"
128 else
129 # Single image
130 attributes.add "\tvar {nit_name}: Image is noinit\n"
131 load_exprs.add "\t\t{nit_name} = main_image.subimage({coordinates})\n"
132 end
133 end
134
135 add """
136 # File generated by svg_to_png_and_nit, do not modify, redef instead
137
138 import mnit::image_set
139
140 class {{{document.nit_class_name}}}
141 super ImageSet
142
143 private var main_image: Image is noinit
144 """
145 add_all attributes
146 add """
147
148 redef fun load_all(app: App)
149 do
150 main_image = app.load_image(\"images/{{{document.drawing_name}}}.png\")
151 """
152 add_all load_exprs
153 add """
154 end
155 end
156 """
157 end
158 end
159
160 redef class Int
161 # Magic adaption of this coordinates to the given `margin` and `scale`
162 fun adapt(margin: Int, scale: Float): Int
163 do
164 var corrected = self-margin
165 return (corrected.to_f*scale).to_i
166 end
167
168 # The first power of to equal or greater than `self`
169 fun next_pow2: Int
170 do
171 var p = 2
172 while p < self do p = p*2
173 return p
174 end
175 end
176
177 var opt_out_src = new OptionString("Path to output source file (folder or file)", "--src", "-s")
178 var opt_assets = new OptionString("Path to assert dir where to put PNG files", "--assets", "-a")
179 var opt_scale = new OptionFloat("Apply scaling to exported images (default at 1.0 of 90dpi)", 1.0, "--scale", "-x")
180 var opt_help = new OptionBool("Print this help message", "--help", "-h")
181
182 var opt_context = new OptionContext
183 opt_context.add_option(opt_out_src, opt_assets, opt_scale, opt_help)
184
185 opt_context.parse(args)
186 var rest = opt_context.rest
187 var errors = opt_context.errors
188 if rest.is_empty and not opt_help.value then errors.add "You must specify at least one source drawing file"
189 if not errors.is_empty or opt_help.value then
190 print errors.join("\n")
191 print "Usage: svg_to_png_and_nit [Options] drawing.svg [Other files]"
192 print "Options:"
193 opt_context.usage
194 exit 1
195 end
196
197 if not "inkscape".program_is_in_path then
198 print "This tool needs the external program `inkscape`, make sure it is installed and in your PATH."
199 exit 1
200 end
201
202 var drawings = rest
203 for drawing in drawings do
204 if not drawing.file_exists then
205 stderr.write "Source drawing file '{drawing}' does not exist."
206 exit 1
207 end
208 end
209
210 var assets_path = opt_assets.value
211 if assets_path == null then assets_path = "assets"
212 if not assets_path.file_exists then
213 stderr.write "Assets dir '{assets_path}' does not exist (use --assets)\n"
214 exit 1
215 end
216
217 var src_path = opt_out_src.value
218 if src_path == null then src_path = "src"
219 if not src_path.file_exists and src_path.file_extension != "nit" then
220 stderr.write "Source dir '{src_path}' does not exist (use --src)\n"
221 exit 1
222 end
223
224 var scale = opt_scale.value
225
226 for drawing in drawings do
227 var drawing_name = drawing.basename(".svg")
228
229 # Get the page dimensions
230 # Inkscape doesn't give us this information
231 var page_width = -1
232 var page_height = -1
233 var svg_file = new FileReader.open(drawing)
234 while not svg_file.eof do
235 var line = svg_file.read_line
236
237 if page_width == -1 and line.search("width") != null then
238 var words = line.split("=")
239 var n = words[1]
240 n = n.substring(1, n.length-2) # remove ""
241 page_width = n.to_f.ceil.to_i
242 else if page_height == -1 and line.search("height") != null then
243 var words = line.split("=")
244 var n = words[1]
245 n = n.substring(1, n.length-2) # remove ""
246 page_height = n.to_f.ceil.to_i
247 end
248 end
249 svg_file.close
250
251 assert page_width != -1
252 assert page_height != -1
253
254 # Query Inkscape
255 var prog = "inkscape"
256 var proc = new ProcessReader.from_a(prog, ["--without-gui", "--query-all", drawing])
257
258 var min_x = 1000000
259 var min_y = 1000000
260 var max_x = -1
261 var max_y = -1
262 var images = new Array[Image]
263
264 # Gather all images beginning with 0
265 # also get the bounding box of all images
266 while not proc.eof do
267 var line = proc.read_line
268 var words = line.split(",")
269
270 if words.length == 5 then
271 var id = words[0]
272
273 var x = words[1].to_f.floor.to_i
274 var y = words[2].to_f.floor.to_i
275 var w = words[3].to_f.ceil.to_i+1
276 var h = words[4].to_f.ceil.to_i+1
277
278 if id.has_prefix("0") then
279 var nit_name = id.substring_from(1)
280 nit_name = nit_name.replace('-', "_")
281
282 var image = new Image(nit_name, x, y, w, h)
283 min_x = min_x.min(x)
284 min_y = min_y.min(y)
285 max_x = max_x.max(image.right)
286 max_y = max_y.max(image.bottom)
287
288 images.add image
289 end
290 end
291 end
292 proc.close
293
294
295 # Sort images by name, it prevents Array errors and looks better
296 alpha_comparator.sort(images)
297
298 var document = new Document(drawing_name, scale, min_x, max_x, min_y, max_y)
299
300 # Nit class
301 var nit_src = new MnitImageSetSrc(document, images)
302
303 if not src_path.file_extension == "nit" then
304 src_path = src_path/drawing_name+".nit"
305 end
306
307 # Output source file
308 var src_file = new FileWriter.open(src_path)
309 nit_src.write_to(src_file)
310 src_file.close
311
312 # Find closest power of 2
313 var dx = max_x - min_x
314 max_x = dx.next_pow2 + min_x
315
316 var dy = max_y - min_y
317 max_y = dy.next_pow2 + min_y
318
319 # Inkscape's --export-area inverts the Y axis. It uses the lower left corner of
320 # the drawing area where as queries return coordinates from the top left.
321 var y0 = page_height - max_y
322 var y1 = page_height - min_y
323
324 # Output png file to assets
325 var png_path = "{assets_path}/images/{drawing_name}.png"
326 var proc2 = new Process.from_a(prog, [drawing, "--without-gui",
327 "--export-dpi={(90.0*scale).to_i}",
328 "--export-png={png_path}",
329 "--export-area={min_x}:{y0}:{max_x}:{y1}",
330 "--export-background=#000000", "--export-background-opacity=0.0"])
331 proc2.wait
332 end