fix crash on videos on first run
[booh] / lib / booh / booh-lib.rb
1 #                         *  BOOH  *
2 #
3 # A.k.a `Best web-album Of the world, Or your money back, Humerus'.
4 #
5 # The acronyn sucks, however this is a tribute to Dragon Ball by
6 # Akira Toriyama, where the last enemy beaten by heroes of Dragon
7 # Ball is named "Boo". But there was already a free software project
8 # called Boo, so this one will be it "Booh". Or whatever.
9 #
10 #
11 # Copyright (c) 2004 Guillaume Cottenceau <gc3 at bluewin.ch>
12 #
13 # This software may be freely redistributed under the terms of the GNU
14 # public license version 2.
15 #
16 # You should have received a copy of the GNU General Public License
17 # along with this program; if not, write to the Free Software
18 # Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19
20 require 'iconv'
21 require 'timeout'
22
23 require 'rexml/document'
24
25 require 'gettext'
26 include GetText
27 bindtextdomain("booh")
28
29 require 'booh/config.rb'
30 require 'booh/version.rb'
31 begin
32     require 'gtk2'
33 rescue LoadError
34     $no_gtk2 = true
35 end
36 begin
37     require 'booh/libadds'
38 rescue LoadError
39     $no_libadds = true
40 end
41
42 module Booh
43     $verbose_level = 2
44     $CURRENT_CHARSET = `locale charmap`.chomp
45     $convert = 'convert -interlace line +profile "*"'
46     $convert_enhance = '-contrast -enhance -normalize'
47
48     def utf8(string)
49         return Iconv::iconv("UTF-8", $CURRENT_CHARSET, string).to_s
50     end
51
52     def utf8cut(string, maxlen)
53         begin
54             return Iconv::iconv("UTF-8", $CURRENT_CHARSET, string[0..maxlen-1]).to_s
55         rescue Iconv::InvalidCharacter
56             return utf8cut(string, maxlen-1)
57         end
58     end
59
60     def sizename(key)
61         #- fake for gettext to find these; if themes need more sizes, english name for them should be added here
62         sizenames = { 'small' => utf8(_("small")), 'medium' => utf8(_("medium")), 'large' => utf8(_("large")),
63                       'x-large' => utf8(_("x-large")), 'xx-large' => utf8(_("xx-large")),
64                       'original' => utf8(_("original")) }
65         return sizenames[key] || key
66     end
67     
68     def from_utf8(string)
69         return Iconv::iconv($CURRENT_CHARSET, "UTF-8", string).to_s
70     end
71
72     def from_utf8_safe(string)
73         begin
74             return Iconv::iconv($CURRENT_CHARSET, "UTF-8", string).to_s
75         rescue Iconv::IllegalSequence
76             return ''
77         end
78     end
79
80     def make_dest_filename_old(orig_filename)
81         #- we remove non alphanumeric characters but need to do that
82         #- cleverly to not end up with two similar dest filenames. we won't
83         #- urlencode because urldecode might happen in the browser.
84         return orig_filename.unpack("C*").collect { |v| v.chr =~ /[a-zA-Z\-_0-9\.\/]/ ? v.chr : sprintf("%2X", v) }.to_s
85     end
86
87     def make_dest_filename(orig_filename)
88         #- we remove non alphanumeric characters but need to do that
89         #- cleverly to not end up with two similar dest filenames. we won't
90         #- urlencode because urldecode might happen in the browser.
91         return orig_filename.unpack("C*").collect { |v| v.chr =~ /[a-zA-Z\-_0-9\.\/]/ ? v.chr : sprintf("~%02X", v) }.to_s
92     end
93
94     def msg(verbose_level, msg)
95         if verbose_level <= $verbose_level
96             if verbose_level == 0
97                 warn _("\t***ERROR***: %s\n") % msg
98             elsif verbose_level == 1
99                 warn _("\tWarning: %s\n") % msg
100             else
101                 puts msg
102             end
103         end
104     end
105
106     def msg_(verbose_level, msg)
107         if verbose_level <= $verbose_level
108             if verbose_level == 0
109                 warn _("\t***ERROR***: %s") % msg
110             elsif verbose_level == 1
111                 warn _("\tWarning: %s") % msg
112             else
113                 print msg
114             end
115         end
116     end
117
118     def die_(msg)
119         puts msg
120         exit 1
121     end
122
123     def select_theme(name, limit_sizes, optimizefor32, nperrow)
124         $theme = name
125         msg 3, _("Selecting theme '%s'") % $theme
126         themedir = "#{$FPATH}/themes/#{$theme}"
127         if !File.directory?(themedir)
128             die _("Theme was not found (tried %s directory).") % themedir
129         end
130         eval File.open("#{themedir}/metadata/parameters.rb").readlines.join
131
132         if limit_sizes
133             if limit_sizes != 'all'
134                 sizes = limit_sizes.split(/,/)
135                 $images_size = $images_size.find_all { |e| sizes.include?(e['name']) }
136                 if $images_size.length == 0
137                     die _("Can't carry on, no valid size selected.")
138                 end
139             end
140         else
141             $images_size = $images_size.find_all { |e| !e['optional'] }
142         end
143
144         if optimizefor32
145             $images_size.each { |e|
146                 e['fullscreen'].gsub!(/(\d+x)(\d+)/) { $1 + ($2.to_f*8/9).to_i.to_s }
147                 e['thumbnails'].gsub!(/(\d+x)(\d+)/) { $1 + ($2.to_f*8/9).to_i.to_s }
148             }
149             $albums_thumbnail_size.gsub!(/(\d+x)(\d+)/) { $1 + ($2.to_f*8/9).to_i.to_s }
150         end
151
152         if nperrow && nperrow != $default_N
153             ratio = nperrow.to_f / $default_N.to_f
154             $images_size.each { |e|
155                 e['thumbnails'].gsub!(/(\d+)x(\d+)/) { ($1.to_f/ratio).to_i.to_s + 'x' + ($2.to_f/ratio).to_i.to_s }
156             }
157         end
158
159         $default_size = $images_size.detect { |sizeobj| sizeobj['default'] }
160         if $default_size == nil
161             $default_size = $images_size[0]
162         end
163     end
164
165     def entry2type(entry)
166         if entry =~ /\.(jpg|jpeg|jpe|gif|bmp|png)$/i && entry !~ /['"\[\]]/
167             return 'image'
168         elsif !$ignore_videos && entry =~ /\.(mov|avi|mpg|mpeg|mpe|wmv|asx|3gp|mp4)$/i && entry !~ /['"\[\]]/
169             #- might consider using file magic later..
170             return 'video'
171         else
172             return nil
173         end
174     end
175
176     def sys(cmd)
177         msg 2, cmd
178         system(cmd)
179     end
180
181     def waitjob
182         finished = Process.wait2
183         $pids.delete(finished[0])
184         $pids = $pids.find_all { |pid| Process.waitpid(pid, Process::WNOHANG) == nil }
185     end
186
187     def waitjobs
188         while $pids && $pids.length > 0
189             waitjob
190         end
191     end
192
193     #- parallelizable sys
194     def psys(cmd)
195         if $mproc
196             if pid = fork
197                 $pids << pid
198             else
199                 msg 2, cmd + ' &'
200                 system(cmd)
201                 exit 0
202             end
203             if $pids.length == $mproc
204                 waitjob
205             end
206         else
207             sys(cmd)
208         end
209     end
210
211     #- grab the results of a command but don't sleep forever on a runaway process
212     def subproc_runaway_aware(command)
213         begin
214             timeout(10) {
215                 return `#{command}`
216             }
217         rescue Timeout::Error    
218             msg 1, _("forgetting runaway process (transcode sucks again?)")
219             #- todo should slay transcode but dunno how to do that
220             return nil
221         end
222     end
223
224     def get_image_size(fullpath)
225         if !$no_identify
226             if $sizes_cache.nil?
227                 $sizes_cache = {}
228             end
229             if $sizes_cache[fullpath].nil?
230                 #- identify is slow, try with gdk if available (negligible vs 35ms)
231                 if $no_gtk2
232                     if `identify '#{fullpath}'` =~ / JPEG (\d+)x(\d+) /
233                         $sizes_cache[fullpath] = { :x => $1.to_i, :y => $2.to_i }
234                     end
235                 else
236                     format, width, height = Gdk::Pixbuf.get_file_info(fullpath)
237                     if width
238                         $sizes_cache[fullpath] = { :x => width, :y => height }
239                     end
240                 end
241             end
242             return $sizes_cache[fullpath]
243         else
244             return nil
245         end
246     end
247
248     #- commify from http://pleac.sourceforge.net/ (pleac rulz)
249     def commify(n)
250         n.to_s =~ /([^\.]*)(\..*)?/
251         int, dec = $1.reverse, $2 ? $2 : ""
252         while int.gsub!(/(,|\.|^)(\d{3})(\d)/, '\1\2' + _(",") + '\3')
253         end
254         int.reverse + dec
255     end
256
257     def guess_rotate(filename)
258         #- identify is slow, try with libexif if available (4ms vs 35ms)
259         if $no_libadds
260             if $no_identify
261                 return 0
262             end
263             orientation = `identify -format "%[EXIF:orientation]" '#{filename}'`.chomp.to_i
264         else
265             orientation = Exif.orientation(filename)
266         end
267
268         if orientation == 6
269             angle = 90
270         elsif orientation == 8
271             angle = -90
272         else
273             return 0
274         end
275
276         #- remove rotate if image is obviously already in portrait (situation can come from gthumb)
277         size = get_image_size(filename)
278         if size && size[:x] < size[:y]
279             return 0
280         else
281             return angle
282         end
283     end
284
285     def rotate_pixbuf(pixbuf, angle)
286         return pixbuf.rotate(angle ==  90 ? Gdk::Pixbuf::ROTATE_CLOCKWISE :
287                              angle == 180 ? Gdk::Pixbuf::ROTATE_UPSIDEDOWN :
288                              (angle == 270 || angle == -90) ? Gdk::Pixbuf::ROTATE_COUNTERCLOCKWISE :
289                                             Gdk::Pixbuf::ROTATE_NONE)
290     end
291
292     def gen_thumbnails_element(orig, xmldirorelem, allow_background, dests)
293         if xmldirorelem.name == 'dir'
294             xmldirorelem = xmldirorelem.elements["*[@filename='#{utf8(File.basename(orig))}']"]
295         end
296         gen_thumbnails(orig, allow_background, dests, xmldirorelem, '')
297     end
298
299     def gen_thumbnails_subdir(orig, xmldirorelem, allow_background, dests, type)
300         #- type can be `subdirs' or `thumbnails' 
301         gen_thumbnails(orig, allow_background, dests, xmldirorelem, type + '-')
302     end
303
304     def gen_video_thumbnail(orig, dest_dir, orig_base, colorswap, frame_offset)
305         transcode_options = ''
306         if colorswap
307             transcode_options += '-k '
308         end
309         cmd = "mencoder '#{orig}' -nosound -ovc lavc -lavcopts vcodec=mjpeg -o '#{dest_dir}/#{orig_base}.avi' -frames #{frame_offset+26} -fps 25 >/dev/null 2>/dev/null"
310         msg 2, cmd
311         system cmd
312         if File.exists?("#{dest_dir}/#{orig_base}.avi")
313             cmd = "transcode -a 0 -c #{frame_offset}-#{frame_offset+1} -i '#{dest_dir}/#{orig_base}.avi' -y jpg -o '#{dest_dir}/#{orig_base}.jpg' #{transcode_options} 2>&1"
314             msg 2, cmd
315             results = subproc_runaway_aware(cmd)
316             File.delete(File.join(dest_dir, "#{orig_base}.avi"))
317             if results =~ /skipping frames/ && results =~ /encoded 0 frames/
318                 msg 0, _("specified frame-offset too large? max frame was: %s. that may also be another probleme. try another value.") % results.scan(/skipping frames \[000000-(\d+)\]/)[-1]
319                 return false
320             elsif results =~ /V: import format.*unknown/ || !File.exists?("#{dest_dir}/#{orig_base}.jpg000000.jpg")
321                 msg 0, _("could not extract first image of video %s encoded by mencoder") % "#{dest_dir}/#{orig_base}.avi"
322                 return false
323             end
324         else
325             msg 0, _("could not make mencoder to encode %s to mjpeg") % orig
326             return false
327         end
328         return true
329     end
330
331     def gen_thumbnails(orig, allow_background, dests, felem, attributes_prefix)
332         if !dests.detect { |dest| !File.exists?(dest['filename']) } 
333             return true
334         end
335
336         convert_options = ''
337         dest_dir = make_dest_filename(File.dirname(dests[0]['filename']))
338
339         if entry2type(orig) == 'image'
340             if felem
341                 if whitebalance = felem.attributes["#{attributes_prefix}white-balance"]
342                     neworig = "#{dest_dir}/#{File.basename(orig)}-whitebalance#{whitebalance}.jpg"
343                     cmd = "booh-fix-whitebalance '#{orig}' '#{neworig}' #{whitebalance}"
344                     sys(cmd)
345                     if File.exists?(neworig)
346                         orig = neworig
347                     end
348                 end
349                 if gammacorrect = felem.attributes["#{attributes_prefix}gamma-correction"]
350                     neworig = "#{dest_dir}/#{File.basename(orig)}-gammacorrect#{gammacorrect}.jpg"
351                     cmd = "booh-gamma-correction '#{orig}' '#{neworig}' #{gammacorrect}"
352                     sys(cmd)
353                     if File.exists?(neworig)
354                         orig = neworig
355                     end
356                 end
357                 rotate = felem.attributes["#{attributes_prefix}rotate"]
358                 if !rotate
359                     felem.add_attribute("#{attributes_prefix}rotate", rotate = guess_rotate(orig).to_s)
360                 end
361                 convert_options += "-rotate #{rotate} "
362                 if felem.attributes["#{attributes_prefix}enhance"]
363                     convert_options += ($config['convert-enhance'] || $convert_enhance) + " "
364                 end
365             end
366             for dest in dests
367                 if !File.exists?(dest['filename'])
368                     cmd = nil
369                     cmd ||= "#{$convert} #{convert_options}-size #{dest['size']} -resize '#{dest['size']}>' '#{orig}' '#{dest['filename']}'"
370                     if allow_background
371                         psys(cmd)
372                     else
373                         sys(cmd)
374                     end
375                 end
376             end
377             if neworig
378                 if allow_background
379                     waitjobs
380                 end
381                 File.delete(neworig)
382             end
383             return true
384
385         elsif entry2type(orig) == 'video'
386             if felem
387                 #- frame-offset is an attribute that allows to specify which frame to use for the thumbnail
388                 frame_offset = felem.attributes["#{attributes_prefix}frame-offset"]
389                 if !frame_offset
390                     felem.add_attribute("#{attributes_prefix}frame-offset", frame_offset = "0")
391                 end
392                 frame_offset = frame_offset.to_i
393                 if rotate = felem.attributes["#{attributes_prefix}rotate"]
394                     convert_options += "-rotate #{rotate} "
395                 end
396                 if felem.attributes["#{attributes_prefix}enhance"]
397                     convert_options += ($config['convert-enhance'] || $convert_enhance) + " "
398                 end
399             end
400             orig_base = File.basename(dests[0]['filename']) + File.basename(orig)
401             orig_image = "#{dest_dir}/#{orig_base}.jpg000000.jpg"
402             if File.exists?(orig_image)
403                 File.delete(orig_image)
404             end
405             for dest in dests
406                 if ! File.exists?(orig_image)
407                     if ! gen_video_thumbnail(orig, dest_dir, orig_base, felem && felem.attributes["#{attributes_prefix}color-swap"], frame_offset)
408                         return false
409                     end
410                     if felem && whitebalance = felem.attributes["#{attributes_prefix}white-balance"]
411                         if whitebalance.to_f != 0
412                             neworig = "#{dest_dir}/#{orig_base}-whitebalance#{whitebalance}.jpg"
413                             cmd = "booh-fix-whitebalance '#{orig_image}' '#{neworig}' #{whitebalance}"
414                             sys(cmd)
415                             if File.exists?(neworig)
416                                 orig_image = neworig
417                             end
418                         end
419                     end
420                     if felem && gammacorrect = felem.attributes["#{attributes_prefix}gamma-correction"]
421                         if gammacorrect.to_f != 0
422                             neworig = "#{dest_dir}/#{orig_base}-gammacorrect#{gammacorrect}.jpg"
423                             cmd = "booh-gamma-correction '#{orig_image}' '#{neworig}' #{gammacorrect}"
424                             sys(cmd)
425                             if File.exists?(neworig)
426                                 orig_image = neworig
427                             end
428                         end
429                     end
430                 end
431                 if !File.exists?(dest['filename'])
432                     sys("#{$convert} #{convert_options}-size #{dest['size']} -resize #{dest['size']} '#{orig_image}' '#{dest['filename']}'")
433                 end
434             end
435             if neworig
436                 File.delete(neworig)
437             end
438             return true
439         end
440     end
441
442     def invornil(obj, methodname)
443         if obj == nil
444             return nil
445         else
446             return obj.method(methodname).call
447         end
448     end
449
450     def find_subalbum_info_type(xmldir)
451         #- first look for subdirs info; if not, means there is no subdir
452         if xmldir.attributes['subdirs-caption']
453             return 'subdirs'
454         else
455             return 'thumbnails'
456         end
457     end
458
459     def find_subalbum_caption_info(xmldir)
460         type = find_subalbum_info_type(xmldir)
461         return [ from_utf8(xmldir.attributes["#{type}-captionfile"]), xmldir.attributes["#{type}-caption"] ]
462     end
463
464     def file_size(path)
465         begin
466             return File.size(path)
467         rescue
468             return -1
469         end
470     end
471
472     def max(a, b)
473         a > b ? a : b
474     end
475
476     def clamp(n, a, b)
477         n < a ? a : n > b ? b : n
478     end
479
480     def pano_amount(elem)
481         if pano_amount = elem.attributes['pano-amount']
482             if $N_per_row
483                 return clamp(pano_amount.to_f, 1, $N_per_row.to_i)
484             else
485                 return clamp(pano_amount.to_f, 1, $default_N.to_i)
486             end
487         else
488             return nil
489         end
490     end
491
492     def substInFile(name)
493         newcontent = IO.readlines(name).collect { |l| yield l }
494         ios = File.open(name, "w")
495         ios.write(newcontent)
496         ios.close
497     end
498
499     def open_url(url)
500         cmd = $config['browser'].gsub('%f', "'#{url}'") + ' &'
501         msg 2, cmd
502         system(cmd)
503     end
504
505     def get_license
506         return <<"EOF"
507                     GNU GENERAL PUBLIC LICENSE
508                        Version 2, June 1991
509
510  Copyright (C) 1989, 1991 Free Software Foundation, Inc.
511                           675 Mass Ave, Cambridge, MA 02139, USA
512  Everyone is permitted to copy and distribute verbatim copies
513  of this license document, but changing it is not allowed.
514
515                             Preamble
516
517   The licenses for most software are designed to take away your
518 freedom to share and change it.  By contrast, the GNU General Public
519 License is intended to guarantee your freedom to share and change free
520 software--to make sure the software is free for all its users.  This
521 General Public License applies to most of the Free Software
522 Foundation's software and to any other program whose authors commit to
523 using it.  (Some other Free Software Foundation software is covered by
524 the GNU Library General Public License instead.)  You can apply it to
525 your programs, too.
526
527   When we speak of free software, we are referring to freedom, not
528 price.  Our General Public Licenses are designed to make sure that you
529 have the freedom to distribute copies of free software (and charge for
530 this service if you wish), that you receive source code or can get it
531 if you want it, that you can change the software or use pieces of it
532 in new free programs; and that you know you can do these things.
533
534   To protect your rights, we need to make restrictions that forbid
535 anyone to deny you these rights or to ask you to surrender the rights.
536 These restrictions translate to certain responsibilities for you if you
537 distribute copies of the software, or if you modify it.
538
539   For example, if you distribute copies of such a program, whether
540 gratis or for a fee, you must give the recipients all the rights that
541 you have.  You must make sure that they, too, receive or can get the
542 source code.  And you must show them these terms so they know their
543 rights.
544
545   We protect your rights with two steps: (1) copyright the software, and
546 (2) offer you this license which gives you legal permission to copy,
547 distribute and/or modify the software.
548
549   Also, for each author's protection and ours, we want to make certain
550 that everyone understands that there is no warranty for this free
551 software.  If the software is modified by someone else and passed on, we
552 want its recipients to know that what they have is not the original, so
553 that any problems introduced by others will not reflect on the original
554 authors' reputations.
555
556   Finally, any free program is threatened constantly by software
557 patents.  We wish to avoid the danger that redistributors of a free
558 program will individually obtain patent licenses, in effect making the
559 program proprietary.  To prevent this, we have made it clear that any
560 patent must be licensed for everyone's free use or not licensed at all.
561
562   The precise terms and conditions for copying, distribution and
563 modification follow.
564
565
566                     GNU GENERAL PUBLIC LICENSE
567    TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
568
569   0. This License applies to any program or other work which contains
570 a notice placed by the copyright holder saying it may be distributed
571 under the terms of this General Public License.  The "Program", below,
572 refers to any such program or work, and a "work based on the Program"
573 means either the Program or any derivative work under copyright law:
574 that is to say, a work containing the Program or a portion of it,
575 either verbatim or with modifications and/or translated into another
576 language.  (Hereinafter, translation is included without limitation in
577 the term "modification".)  Each licensee is addressed as "you".
578
579 Activities other than copying, distribution and modification are not
580 covered by this License; they are outside its scope.  The act of
581 running the Program is not restricted, and the output from the Program
582 is covered only if its contents constitute a work based on the
583 Program (independent of having been made by running the Program).
584 Whether that is true depends on what the Program does.
585
586   1. You may copy and distribute verbatim copies of the Program's
587 source code as you receive it, in any medium, provided that you
588 conspicuously and appropriately publish on each copy an appropriate
589 copyright notice and disclaimer of warranty; keep intact all the
590 notices that refer to this License and to the absence of any warranty;
591 and give any other recipients of the Program a copy of this License
592 along with the Program.
593
594 You may charge a fee for the physical act of transferring a copy, and
595 you may at your option offer warranty protection in exchange for a fee.
596
597   2. You may modify your copy or copies of the Program or any portion
598 of it, thus forming a work based on the Program, and copy and
599 distribute such modifications or work under the terms of Section 1
600 above, provided that you also meet all of these conditions:
601
602     a) You must cause the modified files to carry prominent notices
603     stating that you changed the files and the date of any change.
604
605     b) You must cause any work that you distribute or publish, that in
606     whole or in part contains or is derived from the Program or any
607     part thereof, to be licensed as a whole at no charge to all third
608     parties under the terms of this License.
609
610     c) If the modified program normally reads commands interactively
611     when run, you must cause it, when started running for such
612     interactive use in the most ordinary way, to print or display an
613     announcement including an appropriate copyright notice and a
614     notice that there is no warranty (or else, saying that you provide
615     a warranty) and that users may redistribute the program under
616     these conditions, and telling the user how to view a copy of this
617     License.  (Exception: if the Program itself is interactive but
618     does not normally print such an announcement, your work based on
619     the Program is not required to print an announcement.)
620
621
622 These requirements apply to the modified work as a whole.  If
623 identifiable sections of that work are not derived from the Program,
624 and can be reasonably considered independent and separate works in
625 themselves, then this License, and its terms, do not apply to those
626 sections when you distribute them as separate works.  But when you
627 distribute the same sections as part of a whole which is a work based
628 on the Program, the distribution of the whole must be on the terms of
629 this License, whose permissions for other licensees extend to the
630 entire whole, and thus to each and every part regardless of who wrote it.
631
632 Thus, it is not the intent of this section to claim rights or contest
633 your rights to work written entirely by you; rather, the intent is to
634 exercise the right to control the distribution of derivative or
635 collective works based on the Program.
636
637 In addition, mere aggregation of another work not based on the Program
638 with the Program (or with a work based on the Program) on a volume of
639 a storage or distribution medium does not bring the other work under
640 the scope of this License.
641
642   3. You may copy and distribute the Program (or a work based on it,
643 under Section 2) in object code or executable form under the terms of
644 Sections 1 and 2 above provided that you also do one of the following:
645
646     a) Accompany it with the complete corresponding machine-readable
647     source code, which must be distributed under the terms of Sections
648     1 and 2 above on a medium customarily used for software interchange; or,
649
650     b) Accompany it with a written offer, valid for at least three
651     years, to give any third party, for a charge no more than your
652     cost of physically performing source distribution, a complete
653     machine-readable copy of the corresponding source code, to be
654     distributed under the terms of Sections 1 and 2 above on a medium
655     customarily used for software interchange; or,
656
657     c) Accompany it with the information you received as to the offer
658     to distribute corresponding source code.  (This alternative is
659     allowed only for noncommercial distribution and only if you
660     received the program in object code or executable form with such
661     an offer, in accord with Subsection b above.)
662
663 The source code for a work means the preferred form of the work for
664 making modifications to it.  For an executable work, complete source
665 code means all the source code for all modules it contains, plus any
666 associated interface definition files, plus the scripts used to
667 control compilation and installation of the executable.  However, as a
668 special exception, the source code distributed need not include
669 anything that is normally distributed (in either source or binary
670 form) with the major components (compiler, kernel, and so on) of the
671 operating system on which the executable runs, unless that component
672 itself accompanies the executable.
673
674 If distribution of executable or object code is made by offering
675 access to copy from a designated place, then offering equivalent
676 access to copy the source code from the same place counts as
677 distribution of the source code, even though third parties are not
678 compelled to copy the source along with the object code.
679
680
681   4. You may not copy, modify, sublicense, or distribute the Program
682 except as expressly provided under this License.  Any attempt
683 otherwise to copy, modify, sublicense or distribute the Program is
684 void, and will automatically terminate your rights under this License.
685 However, parties who have received copies, or rights, from you under
686 this License will not have their licenses terminated so long as such
687 parties remain in full compliance.
688
689   5. You are not required to accept this License, since you have not
690 signed it.  However, nothing else grants you permission to modify or
691 distribute the Program or its derivative works.  These actions are
692 prohibited by law if you do not accept this License.  Therefore, by
693 modifying or distributing the Program (or any work based on the
694 Program), you indicate your acceptance of this License to do so, and
695 all its terms and conditions for copying, distributing or modifying
696 the Program or works based on it.
697
698   6. Each time you redistribute the Program (or any work based on the
699 Program), the recipient automatically receives a license from the
700 original licensor to copy, distribute or modify the Program subject to
701 these terms and conditions.  You may not impose any further
702 restrictions on the recipients' exercise of the rights granted herein.
703 You are not responsible for enforcing compliance by third parties to
704 this License.
705
706   7. If, as a consequence of a court judgment or allegation of patent
707 infringement or for any other reason (not limited to patent issues),
708 conditions are imposed on you (whether by court order, agreement or
709 otherwise) that contradict the conditions of this License, they do not
710 excuse you from the conditions of this License.  If you cannot
711 distribute so as to satisfy simultaneously your obligations under this
712 License and any other pertinent obligations, then as a consequence you
713 may not distribute the Program at all.  For example, if a patent
714 license would not permit royalty-free redistribution of the Program by
715 all those who receive copies directly or indirectly through you, then
716 the only way you could satisfy both it and this License would be to
717 refrain entirely from distribution of the Program.
718
719 If any portion of this section is held invalid or unenforceable under
720 any particular circumstance, the balance of the section is intended to
721 apply and the section as a whole is intended to apply in other
722 circumstances.
723
724 It is not the purpose of this section to induce you to infringe any
725 patents or other property right claims or to contest validity of any
726 such claims; this section has the sole purpose of protecting the
727 integrity of the free software distribution system, which is
728 implemented by public license practices.  Many people have made
729 generous contributions to the wide range of software distributed
730 through that system in reliance on consistent application of that
731 system; it is up to the author/donor to decide if he or she is willing
732 to distribute software through any other system and a licensee cannot
733 impose that choice.
734
735 This section is intended to make thoroughly clear what is believed to
736 be a consequence of the rest of this License.
737
738
739   8. If the distribution and/or use of the Program is restricted in
740 certain countries either by patents or by copyrighted interfaces, the
741 original copyright holder who places the Program under this License
742 may add an explicit geographical distribution limitation excluding
743 those countries, so that distribution is permitted only in or among
744 countries not thus excluded.  In such case, this License incorporates
745 the limitation as if written in the body of this License.
746
747   9. The Free Software Foundation may publish revised and/or new versions
748 of the General Public License from time to time.  Such new versions will
749 be similar in spirit to the present version, but may differ in detail to
750 address new problems or concerns.
751
752 Each version is given a distinguishing version number.  If the Program
753 specifies a version number of this License which applies to it and "any
754 later version", you have the option of following the terms and conditions
755 either of that version or of any later version published by the Free
756 Software Foundation.  If the Program does not specify a version number of
757 this License, you may choose any version ever published by the Free Software
758 Foundation.
759
760   10. If you wish to incorporate parts of the Program into other free
761 programs whose distribution conditions are different, write to the author
762 to ask for permission.  For software which is copyrighted by the Free
763 Software Foundation, write to the Free Software Foundation; we sometimes
764 make exceptions for this.  Our decision will be guided by the two goals
765 of preserving the free status of all derivatives of our free software and
766 of promoting the sharing and reuse of software generally.
767
768                             NO WARRANTY
769
770   11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
771 FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
772 OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
773 PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
774 OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
775 MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE ENTIRE RISK AS
776 TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU.  SHOULD THE
777 PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
778 REPAIR OR CORRECTION.
779
780   12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
781 WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
782 REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
783 INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
784 OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
785 TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
786 YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
787 PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
788 POSSIBILITY OF SUCH DAMAGES.
789 EOF
790     end
791
792     def call_about
793         Gtk::AboutDialog.set_url_hook { |dialog, url| open_url(url) }
794         Gtk::AboutDialog.show($main_window, { :name => 'booh',
795                                               :version => $VERSION,
796                                               :copyright => 'Copyright (c) 2005-2007 Guillaume Cottenceau',
797                                               :license => get_license,
798                                               :website => 'http://booh.org/',
799                                               :authors => [ 'Guillaume Cottenceau' ],
800                                               :artists => [ 'Ayo73' ],
801                                               :comments => utf8(_("''The Web-Album of choice for discriminating Linux users''")),
802                                               :translator_credits => utf8(_('Esperanto: Stephane Fillod
803 Japanese: Masao Mutoh
804 German: Roland Eckert
805 French: Guillaume Cottenceau')),
806                                               :logo => Gdk::Pixbuf.new("#{$FPATH}/images/logo.png") })
807     end
808
809 end
810
811 class File
812     def File.reduce_path(path)
813         return path.gsub(/\w+\/\.\.\//, '')
814     end
815 end
816
817 module Enumerable
818     def collect_with_index
819         out = []
820         each_with_index { |e,i|
821             out << yield(e,i)
822         }
823         return out
824     end
825 end
826
827 class Array
828     def sum
829         retval = 0
830         each { |v| retval += v.to_i }
831         return retval
832     end
833 end
834
835 class REXML::Element
836     def previous_element_byname(name)
837         n = self
838         while n = n.previous_element
839             if n.name == name
840                 return n
841             end
842         end
843         return nil
844     end
845
846     def previous_element_byname_notattr(name, attr)
847         n = self
848         while n = n.previous_element
849             if n.name == name && !n.attributes[attr]
850                 return n
851             end
852         end
853         return nil
854     end
855
856     def next_element_byname(name)
857         n = self
858         while n = n.next_element
859             if n.name == name
860                 return n
861             end
862         end
863         return nil
864     end
865
866     def next_element_byname_notattr(name, attr)
867         n = self
868         while n = n.next_element
869             if n.name == name && !n.attributes[attr]
870                 return n
871             end
872         end
873         return nil
874     end
875
876     def child_byname_notattr(name, attr)
877         elements.each(name) { |element|
878             if !element.attributes[attr]
879                 return element
880             end
881         }
882         return nil
883     end
884 end
885
886