5 # A.k.a `Best web-album Of the world, Or your money back, Humerus'.
7 # The acronyn sucks, however this is a tribute to Dragon Ball by
8 # Akira Toriyama, where the last enemy beaten by heroes of Dragon
9 # Ball is named "Boo". But there was already a free software project
10 # called Boo, so this one will be it "Booh". Or whatever.
13 # Copyright (c) 2004 Guillaume Cottenceau <gc3 at bluewin.ch>
15 # This software may be freely redistributed under the terms of the GNU
16 # public license version 2.
18 # You should have received a copy of the GNU General Public License
19 # along with this program; if not, write to the Free Software
20 # Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
25 require 'rexml/document'
29 bindtextdomain("booh")
31 require 'booh/config.rb'
32 require 'booh/version.rb'
36 $CURRENT_CHARSET = `locale charmap`.chomp
37 $convert = 'convert -interlace line +profile "*"'
38 $convert_enhance = '-contrast -enhance -normalize'
41 return Iconv::iconv("UTF-8", $CURRENT_CHARSET, string).to_s
45 #- fake for gettext to find these; if themes need more sizes, english name for them should be added here
46 sizenames = { 'small' => utf8(_("small")), 'medium' => utf8(_("medium")), 'large' => utf8(_("large")),
47 'x-large' => utf8(_("x-large")), 'xx-large' => utf8(_("xx-large")),
48 'original' => utf8(_("original")) }
49 return sizenames[key] || key
53 return Iconv::iconv($CURRENT_CHARSET, "UTF-8", string).to_s
56 def make_dest_filename(orig_filename)
57 #- we remove non alphanumeric characters but need to do that
58 #- cleverly to not end up with two similar dest filenames. we won't
59 #- urlencode because urldecode might happen in the browser.
60 return orig_filename.unpack("C*").collect { |v| v.chr =~ /[a-zA-Z\-_0-9\.\/]/ ? v.chr : sprintf("%2X", v) }.to_s
63 def msg(verbose_level, msg)
64 if verbose_level <= $verbose_level
66 warn _("\t***ERROR***: %s\n") % msg
67 elsif verbose_level == 1
68 warn _("\tWarning: %s\n") % msg
75 def msg_(verbose_level, msg)
76 if verbose_level <= $verbose_level
78 warn _("\t***ERROR***: %s") % msg
79 elsif verbose_level == 1
80 warn _("\tWarning: %s") % msg
92 def select_theme(name, limit_sizes)
94 msg 3, _("Selecting theme `%s'") % $theme
95 themedir = "#{$FPATH}/themes/#{$theme}"
96 if !File.directory?(themedir)
97 die _("Theme was not found (tried %s directory).") % themedir
99 eval File.open("#{themedir}/metadata/parameters.rb").readlines.join
102 if limit_sizes != 'all'
103 sizes = limit_sizes.split(/,/)
104 $images_size = $images_size.find_all { |e| sizes.include?(e['name']) }
105 if $images_size.length == 0
106 die _("Can't carry on, no valid size selected.")
110 $images_size = $images_size.find_all { |e| !e['optional'] }
113 $default_size = $images_size.detect { |sizeobj| sizeobj['default'] }
114 if $default_size == nil
115 $default_size = $images_size[0]
119 def entry2type(entry)
120 if entry =~ /\.(jpg|jpeg|jpe|gif|bmp|png)$/i && entry !~ /['"\[\]]/
122 elsif entry =~ /\.(mov|avi|mpg|mpeg|mpe|wmv|asx)$/i && entry !~ /['"\[\]]/
123 #- might consider using file magic later..
135 #- parallelizable sys
145 if $pids.length == $mproc
146 finished = Process.wait2
147 $pids.delete(finished[0])
148 $pids = $pids.find_all { |pid| Process.waitpid(pid, Process::WNOHANG) == nil }
155 #- grab the results of a command but don't sleep forever on a runaway process
156 def subproc_runaway_aware(command)
161 rescue Timeout::Error
162 msg 1, _("forgetting runaway process (transcode sucks again?)")
163 #- todo should slay transcode but dunno how to do that
168 def get_image_size(fullpath)
169 if `identify '#{fullpath}'` =~ / JPEG (\d+)x(\d+) /
170 return { :x => $1.to_i, :y => $2.to_i }
176 #- commify from http://pleac.sourceforge.net/ (pleac rulz)
178 n.to_s =~ /([^\.]*)(\..*)?/
179 int, dec = $1.reverse, $2 ? $2 : ""
180 while int.gsub!(/(,|\.|^)(\d{3})(\d)/, '\1\2' + _(",") + '\3')
185 def guess_rotate(filename)
186 orientation = `exif '#{filename}'`.detect { |line| line =~ /^Orientation/ }
187 if orientation =~ /right - top/
189 elsif orientation =~ /left - bottom/
194 size = get_image_size(filename)
195 #- remove rotate if image is obviously already in portrait (situation can come from gthumb)
196 if size && size[:x] < size[:y]
203 def rotate_pixbuf(pixbuf, angle)
204 return pixbuf.rotate(angle == 90 ? Gdk::Pixbuf::ROTATE_CLOCKWISE :
205 angle == 180 ? Gdk::Pixbuf::ROTATE_UPSIDEDOWN :
206 angle == 270 ? Gdk::Pixbuf::ROTATE_COUNTERCLOCKWISE :
207 Gdk::Pixbuf::ROTATE_NONE)
210 def gen_thumbnails_element(orig, xmldirorelem, allow_background, dests)
211 if xmldirorelem.name == 'dir'
212 xmldirorelem = xmldirorelem.elements["[@filename='#{utf8(File.basename(orig))}']"]
214 gen_thumbnails(orig, allow_background, dests, xmldirorelem, '')
217 def gen_thumbnails_subdir(orig, xmldirorelem, allow_background, dests, type)
218 #- type can be `subdirs' or `thumbnails'
219 gen_thumbnails(orig, allow_background, dests, xmldirorelem, type + '-')
222 def gen_thumbnails(orig, allow_background, dests, felem, attributes_prefix)
223 if !dests.detect { |dest| !File.exists?(dest['filename']) }
228 dest_dir = make_dest_filename(File.dirname(dests[0]['filename']))
230 if entry2type(orig) == 'image'
232 if whitebalance = felem.attributes["#{attributes_prefix}white-balance"]
233 neworig = "#{dest_dir}/#{File.basename(orig)}-whitebalance#{whitebalance}.jpg"
234 cmd = "booh-fix-whitebalance '#{orig}' '#{neworig}' #{whitebalance}"
236 if File.exists?(neworig)
238 allow_background = false
241 rotate = felem.attributes["#{attributes_prefix}rotate"]
243 felem.add_attribute("#{attributes_prefix}rotate", rotate = guess_rotate(orig).to_i)
245 convert_options += "-rotate #{rotate} "
246 if felem.attributes["#{attributes_prefix}enhance"]
247 convert_options += ($config['convert-enhance'] || $convert_enhance) + " "
251 if !File.exists?(dest['filename'])
253 #- don't resize if image is already smaller than destination size
254 if size = get_image_size(orig)
255 dest['size'] =~ /(\d+)x(\d+)/
256 if (rotate == "90" || rotate == "270" || size[:x] < size[:y]) ? size[:y] < $1.to_i : size[:x] < $1.to_i
257 cmd = "#{$convert} #{convert_options} '#{orig}' '#{dest['filename']}'"
260 cmd ||= "#{$convert} #{convert_options}-size #{dest['size']} -resize #{dest['size']} '#{orig}' '#{dest['filename']}'"
269 system("rm -f '#{neworig}'")
273 elsif entry2type(orig) == 'video'
275 #- frame-offset is an attribute that allows to specify which frame to use for the thumbnail
276 frame_offset = felem.attributes["#{attributes_prefix}frame-offset"]
278 felem.add_attribute("#{attributes_prefix}frame-offset", frame_offset = "0")
280 frame_offset = frame_offset.to_i
281 if rotate = felem.attributes["#{attributes_prefix}rotate"]
282 convert_options += "-rotate #{rotate} "
284 if felem.attributes["#{attributes_prefix}enhance"]
285 convert_options += ($config['convert-enhance'] || $convert_enhance) + " "
288 orig_image = "#{dest_dir}/screenshot.jpg000000.jpg"
290 if !File.exists?(orig_image)
291 transcode_options = ''
293 if felem.attributes["#{attributes_prefix}color-swap"]
294 transcode_options += '-k '
297 cmd = "transcode -a 0 -c #{frame_offset}-#{frame_offset+1} -i '#{orig}' -y jpg -o '#{dest_dir}/screenshot.jpg' #{transcode_options} 2>&1"
299 results = subproc_runaway_aware(cmd)
300 if results =~ /skipping frames/ && results =~ /encoded 0 frames/
301 msg 0, _("specified frame-offset too large? max frame was: %s. that may also be another problem. try another value.") %
302 results.scan(/skipping frames \[000000-(\d+)\]/)[-1]
304 elsif results =~ /V: import format.*unknown/ || !File.exists?("#{dest_dir}/screenshot.jpg000000.jpg")
305 msg 2, _("* could not extract first image of video %s with transcode, will try first converting with mencoder") % orig
306 cmd = "mencoder '#{orig}' -nosound -ovc lavc -lavcopts vcodec=mjpeg -o '#{dest_dir}/foo.avi' -frames #{frame_offset+1} -fps 25 >/dev/null 2>/dev/null"
309 if File.exists?("#{dest_dir}/foo.avi")
310 cmd = "transcode -a 0 -c #{frame_offset}-#{frame_offset+1} -i '#{dest_dir}/foo.avi' -y jpg -o '#{dest_dir}/screenshot.jpg' #{transcode_options} 2>&1"
312 results = subproc_runaway_aware(cmd)
313 system("rm -f '#{dest_dir}/foo.avi'")
314 if results =~ /skipping frames/ && results =~ /encoded 0 frames/
315 msg 0, _("specified frame-offset too large? max frame was: %s. that may also be another probleme. try another value.") %
316 results.scan(/skipping frames \[000000-(\d+)\]/)[-1]
318 elsif results =~ /V: import format.*unknown/ || !File.exists?("#{dest_dir}/screenshot.jpg000000.jpg")
319 msg 0, _("could not extract first image of video %s encoded by mencoder") % "#{dest_dir}/foo.avi"
323 msg 0, _("could not make mencoder to encode %s to mpeg4") % orig
327 if felem && whitebalance = felem.attributes["#{attributes_prefix}white-balance"]
328 if whitebalance.to_f != 0
329 neworig = "#{dest_dir}/#{File.basename(orig)}-whitebalance#{whitebalance}.jpg"
330 cmd = "booh-fix-whitebalance '#{orig_image}' '#{neworig}' #{whitebalance}"
332 if File.exists?(neworig)
338 if !File.exists?(dest['filename'])
339 sys("#{$convert} #{convert_options}-size #{dest['size']} -resize #{dest['size']} '#{orig_image}' '#{dest['filename']}'")
343 system("rm -f '#{neworig}'")
349 def invornil(obj, methodname)
353 return obj.method(methodname).call
357 def find_subalbum_info_type(xmldir)
358 #- first look for subdirs info; if not, means there is no subdir
359 if xmldir.attributes['subdirs-caption']
366 def find_subalbum_caption_info(xmldir)
367 type = find_subalbum_info_type(xmldir)
368 return [ from_utf8(xmldir.attributes["#{type}-captionfile"]), xmldir.attributes["#{type}-caption"] ]
373 return File.size(path)
383 def substInFile(name)
384 newcontent = IO.readlines(name).collect { |l| yield l }
385 ios = File.open(name, "w")
386 ios.write(newcontent)
392 def File.reduce_path(path)
393 return path.gsub(/\w+\/\.\.\//, '')
398 def previous_element_byname(name)
400 while n = n.previous_element
408 def next_element_byname(name)
410 while n = n.next_element