issue34, fixed wickeld tag column when too many tags are selected
[phpfspot.git] / phpfspot.class.php
1 <?php
2
3 require_once "phpfspot_cfg.php";
4 require_once "phpfspot_db.php";
5 require_once "phpfspot_tmpl.php";
6
7 class PHPFSPOT {
8
9    var $cfg;
10    var $db;
11    var $cfg_db;
12    var $tmpl;
13    var $tags;
14    var $avail_tags;
15    var $current_tags;
16
17    public function __construct()
18    {
19       /* Check necessary requirements */
20       if(!$this->checkRequirements()) {
21          exit(1);
22       }
23
24       $this->cfg = new PHPFSPOT_CFG;
25
26       $this->db  = new PHPFSPOT_DB(&$this, $this->cfg->fspot_db);
27       $this->cfg_db = new PHPFSPOT_DB(&$this, $this->cfg->phpfspot_db);
28       $this->check_config_table();
29
30       $this->tmpl = new PHPFSPOT_TMPL($this);
31
32       $this->get_tags();
33
34       session_start();
35
36       if(!isset($_SESSION['tag_condition']))
37          $_SESSION['tag_condition'] = 'or';
38
39       if(!isset($_SESSION['searchfor']))
40          $_SESSION['searchfor'] = '';
41
42       // if begin_with is still set but rows_per_page is now 0, unset it
43       if(isset($_SESSION['begin_with']) && $this->cfg->rows_per_page == 0)
44          unset($_SESSION['begin_with']);
45
46    } // __construct()
47
48    public function __destruct()
49    {
50
51    } // __destruct()
52
53    public function show()
54    {
55       $this->tmpl->assign('searchfor', $_SESSION['searchfor']);
56       $this->tmpl->assign('page_title', $this->cfg->page_title);
57       $this->tmpl->assign('current_condition', $_SESSION['tag_condition']);
58       $this->tmpl->assign('from_date', $this->get_calendar());
59       $this->tmpl->assign('to_date', $this->get_calendar());
60
61       switch($_GET['mode']) {
62          case 'showpi':
63             if(isset($_GET['tags'])) {
64                $_SESSION['selected_tags'] = split(',', $_GET['tags']);
65             }
66             break;
67       }
68
69       $this->tmpl->assign('content_page', 'welcome.tpl');
70       $this->tmpl->show("index.tpl");
71
72
73    } // show()
74
75    private function get_tags()
76    {
77    
78       $this->avail_tags = Array();
79       $count = 0;
80    
81       $result = $this->db->db_query("
82          SELECT id,name
83          FROM tags
84          ORDER BY sort_priority ASC
85       ");
86       
87       while($row = $this->db->db_fetch_object($result)) {
88
89          $tag_id = $row['id'];
90          $tag_name = $row['name'];
91
92          /* check if config requests to ignore this tag */
93          if(in_array($row['name'], $this->cfg->hide_tags))
94             continue;
95
96          $this->tags[$tag_id] = $tag_name; 
97          $this->avail_tags[$count] = $tag_id;
98
99          $count++;
100
101       }
102
103    } // get_tags()
104
105    public function get_photo_details($idx)
106    {
107       $result = $this->db->db_query("
108          SELECT *
109          FROM photos
110          WHERE id='". $idx ."'
111       ");
112       
113       return $this->db->db_fetch_object($result);
114
115    } // get_photo_details
116
117    public function getPhotoName($idx)
118    {
119       if($details = $this->get_photo_details($idx)) {
120
121          $name = $details['name'];
122
123          if(strlen($name) > 15) {
124             $name = substr($name, 0, 10) ."...". substr($name, -10);
125          }
126
127          return $name;
128    
129       }
130
131    } // getPhotoName()
132
133    public function translate_path($path, $width = 0)
134    {  
135       return str_replace($this->cfg->path_replace_from, $this->cfg->path_replace_to, $path);
136
137    } // translate_path
138
139    public function showPhoto($photo)
140    {
141       $all_photos = $this->getPhotoSelection();
142       $count = count($all_photos);
143
144       for($i = 0; $i < $count; $i++) {
145          
146          if($get_next) {
147             $next_img = $all_photos[$i];
148             break;
149          }
150
151          if($all_photos[$i] == $photo) {
152             $get_next = 1;
153          }
154          else {
155             $previous_img = $all_photos[$i];
156          }
157
158          if($photo == $all_photos[$i]) {
159                $current = $i;
160          }
161       }
162
163       $details = $this->get_photo_details($photo);
164       $orig_path = $this->translate_path($details['directory_path']) ."/". $details['name'];
165       $thumb_path = $this->cfg->base_path ."/thumbs/". $this->cfg->photo_width ."_". $this->getMD5($photo);
166
167       if(!file_exists($orig_path)) {
168          $this->_warning("Photo ". $orig_path ." does not exist!<br />\n");
169       }
170
171       if(!is_readable($orig_path)) {
172          $this->_warning("Photo ". $orig_path ." is not readable for user ". get_current_user() ."<br />\n");
173       }
174
175       /* If the thumbnail doesn't exist yet, try to create it */
176       if(!file_exists($thumb_path)) {
177          $this->gen_thumb($photo, true);
178       }
179
180       $meta = $this->get_meta_informations($orig_path);
181
182       /* If EXIF data are available, use them */
183       if(isset($meta['ExifImageWidth'])) {
184          $meta_res = $meta['ExifImageWidth'] ."x". $meta['ExifImageLength'];
185       } else {
186          $info = getimagesize($orig_path);
187          $meta_res = $info[0] ."x". $info[1]; 
188       }
189
190       $meta_date = isset($meta['FileDateTime']) ? strftime("%a %x %X", $meta['FileDateTime']) : "n/a";
191       $meta_make = isset($meta['Make']) ? $meta['Make'] ." ". $meta['Model'] : "n/a";
192       $meta_size = isset($meta['FileSize']) ? round($meta['FileSize']/1024, 1) ."kbyte" : "n/a";
193
194       if(file_exists($thumb_path)) {
195
196          $info = getimagesize($thumb_path);
197
198          $this->tmpl->assign('description', $details['description']);
199          $this->tmpl->assign('image_name', $details['name']);
200
201          $this->tmpl->assign('width', $info[0]);
202          $this->tmpl->assign('height', $info[1]);
203          $this->tmpl->assign('ExifMadeOn', $meta_date);
204          $this->tmpl->assign('ExifMadeWith', $meta_make);
205          $this->tmpl->assign('ExifOrigResolution', $meta_res);
206          $this->tmpl->assign('ExifFileSize', $meta_size);
207     
208          $this->tmpl->assign('image_url', 'phpfspot_img.php?idx='. $photo ."&amp;width=". $this->cfg->photo_width);
209          $this->tmpl->assign('image_url_full', 'phpfspot_img.php?idx='. $photo);
210
211          $this->tmpl->assign('tags', $this->get_photo_tags($photo));
212          $this->tmpl->assign('current', $current);
213       }
214       else {
215          $this->_warning("Can't open file ". $thumb_path ."\n");
216       }
217
218       if($previous_img) {
219          $this->tmpl->assign('previous_url', "javascript:showImage(". $previous_img .");");
220          $this->tmpl->assign('prev_img', $previous_img);
221       }
222
223       if($next_img) {
224          $this->tmpl->assign('next_url', "javascript:showImage(". $next_img .");");
225          $this->tmpl->assign('next_img', $next_img);
226       }
227       $this->tmpl->assign('mini_width', $this->cfg->mini_width);
228
229       $this->tmpl->show("single_photo.tpl");
230
231    } // showPhoto()
232
233    public function getAvailableTags()
234    {
235       $result = $this->db->db_query("
236          SELECT tag_id as id, count(tag_id) as quantity
237          FROM photo_tags
238          INNER JOIN tags t
239             ON t.id = tag_id
240          GROUP BY tag_id
241          ORDER BY t.name ASC
242       ");
243
244       $tags = Array();
245
246       while($row = $this->db->db_fetch_object($result)) {
247          $tags[$row['id']] = $row['quantity'];
248       }
249
250       // change these font sizes if you will
251       $max_size = 125; // max font size in %
252       $min_size = 75; // min font size in %
253
254       // get the largest and smallest array values
255       $max_qty = max(array_values($tags));
256       $min_qty = min(array_values($tags));
257
258       // find the range of values
259       $spread = $max_qty - $min_qty;
260       if (0 == $spread) { // we don't want to divide by zero
261          $spread = 1;
262       }
263
264       // determine the font-size increment
265       // this is the increase per tag quantity (times used)
266       $step = ($max_size - $min_size)/($spread);
267
268       // loop through our tag array
269       foreach ($tags as $key => $value) {
270
271          if(isset($_SESSION['selected_tags']) && in_array($key, $_SESSION['selected_tags']))
272             continue;
273
274           // calculate CSS font-size
275           // find the $value in excess of $min_qty
276           // multiply by the font-size increment ($size)
277           // and add the $min_size set above
278          $size = $min_size + (($value - $min_qty) * $step);
279           // uncomment if you want sizes in whole %:
280          $size = ceil($size);
281
282          print "<a href=\"javascript:Tags('add', ". $key .");\" class=\"tag\" style=\"font-size: ". $size ."%;\">". $this->tags[$key] ."</a>, ";
283
284       }
285
286    } // getAvailableTags()
287
288    public function getSelectedTags()
289    {
290       foreach($this->avail_tags as $tag)
291       {
292          // return all selected tags
293          if(isset($_SESSION['selected_tags']) && in_array($tag, $_SESSION['selected_tags'])) {
294             print "<a href=\"javascript:Tags('del', ". $tag .");\" class=\"tag\">". $this->tags[$tag] ."</a>, ";
295          }
296
297       }
298
299    } // getSelectedTags()
300
301    public function addTag($tag)
302    {
303       if(!isset($_SESSION['selected_tags']))
304          $_SESSION['selected_tags'] = Array();
305
306       if(!in_array($tag, $_SESSION['selected_tags']))
307          array_push($_SESSION['selected_tags'], $tag);
308    
309    } // addTag()
310
311    public function delTag($tag)
312    {
313       if(isset($_SESSION['selected_tags'])) {
314          $key = array_search($tag, $_SESSION['selected_tags']);
315          unset($_SESSION['selected_tags'][$key]);
316          sort($_SESSION['selected_tags']);
317       }
318
319    } // delTag()
320
321    public function resetTags()
322    {
323       unset($_SESSION['selected_tags']);
324
325    } // resetTags()
326
327    public function resetTagSearch()
328    {
329       unset($_SESSION['searchfor']);
330
331    } // resetTagSearch()
332
333    public function getPhotoSelection()
334    {  
335       $tagged_photos = Array();
336
337       /* return a search result */
338       if(isset($_SESSION['searchfor']) && $_SESSION['searchfor'] != '') {
339          $result = $this->db->db_query("
340             SELECT DISTINCT photo_id
341                FROM photo_tags pt
342             INNER JOIN photos p
343                ON p.id=pt.photo_id
344             INNER JOIN tags t
345                ON pt.tag_id=t.id
346             WHERE t.name LIKE '%". $_SESSION['searchfor'] ."%'
347                ORDER BY p.time ASC
348          ");
349          while($row = $this->db->db_fetch_object($result)) {
350             array_push($tagged_photos, $row['photo_id']);
351          }
352          return $tagged_photos;
353       }
354
355       /* return according the selected tags */
356       if(isset($_SESSION['selected_tags']) && !empty($_SESSION['selected_tags'])) {
357          $selected = "";
358          foreach($_SESSION['selected_tags'] as $tag)
359             $selected.= $tag .",";
360          $selected = substr($selected, 0, strlen($selected)-1);
361
362          if($_SESSION['tag_condition'] == 'or') {
363             $result = $this->db->db_query("
364                SELECT DISTINCT photo_id
365                   FROM photo_tags pt
366                INNER JOIN photos p
367                   ON p.id=pt.photo_id
368                WHERE pt.tag_id IN (". $selected .")
369                ORDER BY p.time ASC
370             ");
371          }
372          elseif($_SESSION['tag_condition'] == 'and') {
373
374             /* Join together a table looking like
375
376                pt1.photo_id pt1.tag_id pt2.photo_id pt2.tag_id ...
377
378                so the query can quickly return all images matching the
379                selected tags in an AND condition
380
381             */
382
383             $query_str = "
384                SELECT DISTINCT pt1.photo_id
385                   FROM photo_tags pt1
386             ";
387
388             for($i = 0; $i < count($_SESSION['selected_tags']); $i++) {
389                $query_str.= "
390                   INNER JOIN photo_tags pt". ($i+2) ."
391                      ON pt1.photo_id=pt". ($i+2) .".photo_id
392                ";
393             }
394             $query_str.= "WHERE pt1.tag_id=". $_SESSION['selected_tags'][0];
395             for($i = 1; $i < count($_SESSION['selected_tags']); $i++) {
396                $query_str.= "
397                   AND pt". ($i+1) .".tag_id=". $_SESSION['selected_tags'][$i] ."
398                "; 
399             }
400             $result = $this->db->db_query($query_str);
401          }
402
403          while($row = $this->db->db_fetch_object($result)) {
404             array_push($tagged_photos, $row['photo_id']);
405          }
406          return $tagged_photos;
407       }
408
409       /* return all available photos */
410       $result = $this->db->db_query("
411          SELECT DISTINCT photo_id
412             FROM photo_tags pt
413          INNER JOIN photos p
414             ON p.id=pt.photo_id
415          ORDER BY p.time ASC
416       ");
417       while($row = $this->db->db_fetch_object($result)) {
418          array_push($tagged_photos, $row['photo_id']);
419       }
420       return $tagged_photos;
421
422    } // getPhotoSelection()
423
424    public function showPhotoIndex()
425    {
426       $photos = $this->getPhotoSelection();
427
428       $count = count($photos);
429
430       if(isset($_SESSION['begin_with']) && $_SESSION['begin_with'] != "")
431          $anchor = $_SESSION['begin_with'];
432
433       if(!isset($this->cfg->rows_per_page) || $this->cfg->rows_per_page == 0) {
434
435          $begin_with = 0;
436          $end_with = $count;
437
438       }
439       elseif($this->cfg->rows_per_page > 0) {
440
441          if(!$_SESSION['begin_with'] || $_SESSION['begin_with'] == 0)
442             $begin_with = 0;
443          else {
444
445             $begin_with = $_SESSION['begin_with'];
446
447             // verify $begin_with - perhaps the thumbs-per-rows or
448             // rows-per-page variables have changed or the jump back
449             // from a photo wasn't exact - so calculate the real new
450             // starting point
451             $multiplicator = $this->cfg->rows_per_page * $this->cfg->thumbs_per_row;
452             for($i = 0; $i <= $count; $i+=$multiplicator) {
453                if($begin_with >= $i && $begin_with < $i+$multiplicator) {
454                   $begin_with = $i;
455                   break;
456                }
457             }
458          }
459
460          $end_with = $begin_with + ($this->cfg->rows_per_page * $this->cfg->thumbs_per_row);
461       }
462
463    
464       $rows = 0;
465       $cols = 0;
466       $images[$rows] = Array();
467       $img_height[$rows] = Array();
468       $img_width[$rows] = Array();
469       $img_id[$rows] = Array();
470       $img_name[$rows] = Array();
471
472       for($i = $begin_with; $i < $end_with; $i++) {
473
474          $images[$rows][$cols] = $photos[$i];
475          $img_id[$rows][$cols] = $i;
476          $img_name[$rows][$cols] = $this->getPhotoName($photos[$i]);
477
478          $thumb_path = $this->cfg->base_path ."/thumbs/". $this->cfg->thumb_width ."_". $this->getMD5($photos[$i]);
479
480          if(file_exists($thumb_path)) {
481             $info = getimagesize($thumb_path); 
482             $img_width[$rows][$cols] = $info[0];
483             $img_height[$rows][$cols] = $info[1];
484          }
485
486          if($cols == $this->cfg->thumbs_per_row-1) {
487             $cols = 0;
488             $rows++;
489             $images[$rows] = Array();
490             $img_width[$rows] = Array();
491             $img_height[$rows] = Array();
492          }
493          else {
494             $cols++;
495          }
496       } 
497
498       // +1 for for smarty's selection iteration
499       $rows++;
500
501       if(isset($_SESSION['searchfor']) && $_SESSION['searchfor'] != '')
502          $this->tmpl->assign('searchfor', $_SESSION['searchfor']);
503
504       /* do we have to display the page selector ? */
505       if($this->cfg->rows_per_page != 0) {
506       
507          /* calculate the page switchers */
508          $previous_start = $begin_with - ($this->cfg->rows_per_page * $this->cfg->thumbs_per_row);
509          $next_start = $begin_with + ($this->cfg->rows_per_page * $this->cfg->thumbs_per_row);
510
511          if($begin_with != 0) 
512             $this->tmpl->assign("previous_url", "javascript:showPhotoIndex(". $previous_start .");"); 
513          if($end_with < $count)
514             $this->tmpl->assign("next_url", "javascript:showPhotoIndex(". $next_start .");"); 
515
516          $photo_per_page  = $this->cfg->rows_per_page * $this->cfg->thumbs_per_row;
517          $last_page = ceil($count / $photo_per_page);
518
519          /* get the current selected page */
520          if($begin_with == 0) {
521             $current_page = 1;
522          } else {
523             $current_page = 0;
524             for($i = $begin_with; $i >= 0; $i-=$photo_per_page) {
525                $current_page++;
526             }
527          } 
528
529          for($i = 1; $i <= $last_page; $i++) {
530
531             if($current_page == $i)
532                $style = "style=\"font-size: 125%;\"";
533             elseif($current_page-1 == $i || $current_page+1 == $i)
534                $style = "style=\"font-size: 105%;\"";
535             elseif(($current_page-5 >= $i) && ($i != 1) ||
536                ($current_page+5 <= $i) && ($i != $last_page))
537                $style = "style=\"font-size: 75%;\"";
538             else
539                $style = "";
540
541             $select = "<a href=\"javascript:showPhotoIndex(". (($i*$photo_per_page)-$photo_per_page) .");\"";
542                if($style != "")
543                   $select.= $style;
544             $select.= ">". $i ."</a>&nbsp;";
545
546             // until 9 pages we show the selector from 1-9
547             if($last_page <= 9) {
548                $page_select.= $select;
549                continue;
550             } else {
551                if($i == 1 /* first page */ || 
552                   $i == $last_page /* last page */ ||
553                   $i == $current_page /* current page */ ||
554                   $i == ceil($last_page * 0.25) /* first quater */ ||
555                   $i == ceil($last_page * 0.5) /* half */ ||
556                   $i == ceil($last_page * 0.75) /* third quater */ ||
557                   (in_array($i, array(1,2,3,4,5,6)) && $current_page <= 4) /* the first 6 */ ||
558                   (in_array($i, array($last_page, $last_page-1, $last_page-2, $last_page-3, $last_page-4, $last_page-5)) && $current_page >= $last_page-4) /* the last 6 */ ||
559                   $i == $current_page-3 || $i == $current_page-2 || $i == $current_page-1 /* three before */ ||
560                   $i == $current_page+3 || $i == $current_page+2 || $i == $current_page+1 /* three after */) {
561
562                   $page_select.= $select;
563                   continue;
564
565                }
566             }
567
568             $page_select.= ".";
569          }
570
571          /* only show the page selector if we have more then one page */
572          if($last_page > 1)
573             $this->tmpl->assign('page_selector', $page_select);
574       }
575
576       $current_tags = "";
577       if($_SESSION['selected_tags'] != "") {
578          foreach($_SESSION['selected_tags'] as $tag)
579             $current_tags.= $tag .",";
580          $current_tags = substr($current_tags, 0, strlen($current_tags)-1);
581       }
582
583       $extern_link = "http://". $_SERVER['SERVER_NAME'] ."/index.php?mode=showpi";
584       if($current_tags != "") {
585          $extern_link.= "&tags=". $current_tags;
586       }
587
588       $this->tmpl->assign('extern_link', $extern_link);
589       $this->tmpl->assign('count', $count);
590       $this->tmpl->assign('width', $this->cfg->thumb_width);
591       $this->tmpl->assign('images', $images);
592       $this->tmpl->assign('img_width', $img_width);
593       $this->tmpl->assign('img_height', $img_height);
594       $this->tmpl->assign('img_id', $img_id);
595       $this->tmpl->assign('img_name', $img_name);
596       $this->tmpl->assign('rows', $rows);
597       $this->tmpl->assign('columns', $this->cfg->thumbs_per_row);
598
599       $this->tmpl->show("photo_index.tpl");
600
601       if(isset($anchor))
602          print "<script language=\"JavaScript\">self.location.hash = '#image". $anchor ."';</script>\n";
603
604    } // showPhotoIndex()
605
606    public function showBubbleDetails($photo, $direction)
607    {
608       if($direction == "up")
609          $direction = "bubbleimg_up";
610       else
611          $direction = "bubbleimg_down";
612
613       $details = $this->get_photo_details($photo);
614       $orig_path = $this->translate_path($details['directory_path'])  ."/". $details['name'];
615
616       $image_url = "phpfspot_img.php?idx=". $photo ."&amp;width=". $this->cfg->bubble_width;
617
618       $filesize = filesize($orig_path);
619       $filesize = rand($filesize/1024, 2);
620
621       if(!file_exists($orig_path)) {
622          $this->_warning("Photo ". $orig_path ." does not exist!<br />\n");
623          return;
624       }
625       
626       if(!is_readable($orig_path)) {
627          $this->_warning("Photo ". $orig_path ." is not readable for user ". get_current_user() ."<br />\n");
628          return;
629       }
630
631       $img = getimagesize($orig_path);
632
633       $this->tmpl->assign('file_size', $filesize);
634       $this->tmpl->assign('width', $img[0]);
635       $this->tmpl->assign('height', $img[1]);
636       $this->tmpl->assign('file_name', $details['name']);
637       $this->tmpl->assign('image_id', $direction);
638       $this->tmpl->assign('image_url', $image_url);
639       $this->tmpl->show("bubble_details.tpl");
640
641    } // showBubbleDetails()
642
643    public function showCredits()
644    {
645       $this->tmpl->assign('version', $this->cfg->version);
646       $this->tmpl->assign('product', $this->cfg->product);
647       $this->tmpl->show("credits.tpl");
648
649    } // showCredits()
650
651    public function create_thumbnail($orig_image, $thumb_image, $width)
652    {  
653       if(!file_exists($orig_image))
654          return false;
655
656       $details = getimagesize($orig_image);
657       
658       /* check if original photo is a support image type */
659       if(!$this->checkifImageSupported($details['mime']))
660          return false;
661
662       $meta = $this->get_meta_informations($orig_image);
663
664       $rotate = 0;
665       $flip = false;
666
667       switch($meta['Orientation']) {
668
669          case 1: /* top, left */
670             $rotate = 0; $flip = false; break;
671          case 2: /* top, right */
672             $rotate = 0; $flip = true; break;
673          case 3: /* bottom, left */
674             $rotate = 180; $flip = false; break;
675          case 4: /* bottom, right */
676             $rotate = 180; $flip = true; break;
677          case 5: /* left side, top */
678             $rotate = 90; $flip = true; break;
679          case 6: /* right side, top */
680             $rotate = 90; $flip = false; break;
681          case 7: /* left side, bottom */
682             $rotate = 270; $flip = true; break;
683          case 8: /* right side, bottom */
684             $rotate = 270; $flip = false; break;
685       }
686
687       $src_img = @imagecreatefromjpeg($orig_image);
688
689       if(!$src_img) {
690          print "Can't load image from ". $orig_image ."\n";
691          return false;
692       }
693
694       /* grabs the height and width */
695       $cur_width = imagesx($src_img);
696       $cur_height = imagesy($src_img);
697
698       // If requested width is more then the actual image width,
699       // do not generate a thumbnail
700
701       if($width >= $cur_width) {
702          imagedestroy($src_img);
703          return true;
704       }
705
706       // If the image will be rotate because EXIF orientation said so
707       // 'virtually rotate' the image for further calculations
708       if($rotate == 90 || $rotate == 270) {
709          $tmp = $cur_width;
710          $cur_width = $cur_height;
711          $cur_height = $tmp;
712       }
713
714       /* calculates aspect ratio */
715       $aspect_ratio = $cur_height / $cur_width;
716
717       /* sets new size */
718       if($aspect_ratio < 1) {
719          $new_w = $width;
720          $new_h = abs($new_w * $aspect_ratio);
721       } else {
722          /* 'virtually' rotate the image and calculate it's ratio */
723          $tmp_w = $cur_height;
724          $tmp_h = $cur_width;
725          /* now get the ratio from the 'rotated' image */
726          $tmp_ratio = $tmp_h/$tmp_w;
727          /* now calculate the new dimensions */
728          $tmp_w = $width;
729          $tmp_h = abs($tmp_w * $tmp_ratio);
730
731          // now that we know, how high they photo should be, if it
732          // gets rotated, use this high to scale the image
733          $new_h = $tmp_h;
734          $new_w = abs($new_h / $aspect_ratio);
735
736          // If the image will be rotate because EXIF orientation said so
737          // now 'virtually rotate' back the image for the image manipulation
738          if($rotate == 90 || $rotate == 270) {
739             $tmp = $new_w;
740             $new_w = $new_h;
741             $new_h = $tmp;
742          }
743       }
744
745       /* creates new image of that size */
746       $dst_img = imagecreatetruecolor($new_w, $new_h);
747
748       imagefill($dst_img, 0, 0, ImageColorAllocate($dst_img, 255, 255, 255));
749
750       /* copies resized portion of original image into new image */
751       imagecopyresampled($dst_img, $src_img, 0, 0, 0, 0, $new_w, $new_h, imagesx($src_img), imagesy($src_img));
752
753       /* needs the image to be flipped horizontal? */
754       if($flip) {
755          print "(FLIP)";
756          $image = $dst_img;
757          for($x = 0; $x < $new_w; $x++) {
758             imagecopy($dst_img, $image, $x, 0, $w - $x - 1, 0, 1, $h);
759          }
760       }
761
762       if($rotate) {
763          $this->_debug("(ROTATE)");
764          $dst_img = $this->rotateImage($dst_img, $rotate);
765       }
766
767       /* write down new generated file */
768       $result = imagejpeg($dst_img, $thumb_image, 75);
769
770       /* free your mind */
771       imagedestroy($dst_img);
772       imagedestroy($src_img);
773
774       if($result === false) {
775          print "Can't write thumbnail ". $thumb_image ."\n";
776          return false;
777       }
778
779       return true;
780
781    } // create_thumbnail()
782
783    public function get_meta_informations($file)
784    {
785       return exif_read_data($file);
786
787    } // get_meta_informations()
788
789    public function check_config_table()
790    {
791       // if the config table doesn't exist yet, create it
792       if(!$this->cfg_db->db_check_table_exists("images")) {
793          $this->cfg_db->db_exec("
794             CREATE TABLE images (
795                img_idx int primary key,
796                img_md5 varchar(32)
797             )
798             ");
799       }
800
801    } // check_config_table
802
803    /**
804     * Generates a thumbnail from photo idx
805     *
806     * This function will generate JPEG thumbnails from provided F-Spot photo
807     * indizes.
808     *
809     * 1. Check if all thumbnail generations (width) are already in place and
810     *    readable
811     * 2. Check if the md5sum of the original file has changed
812     * 3. Generate the thumbnails if needed
813     */
814    public function gen_thumb($idx = 0, $force = 0)
815    {
816       $error = 0;
817
818       $resolutions = Array(
819          $this->cfg->thumb_width,
820          $this->cfg->bubble_width,
821          $this->cfg->photo_width,
822          $this->cfg->mini_width,
823       );
824
825       /* get details from F-Spot's database */
826       $details = $this->get_photo_details($idx);
827
828       /* calculate file MD5 sum */
829       $full_path = $this->translate_path($details['directory_path'])  ."/". $details['name'];
830
831       if(!file_exists($full_path)) {
832          $this->_warning("File ". $full_path ." does not exist\n");
833          return;
834       }
835
836       if(!is_readable($full_path)) {
837          $this->_warning("File ". $full_path ." is not readable for ". get_current_user() ."\n");
838          return;
839       }
840
841       $file_md5 = md5_file($full_path);
842
843       $this->_debug("Image [". $idx ."] ". $details['name'] ." Thumbnails:");
844
845       foreach($resolutions as $resolution) {
846
847          $thumb_path = $this->cfg->base_path ."/thumbs/". $resolution ."_". $file_md5;
848
849          /* if the thumbnail file doesn't exist, create it */
850          if(!file_exists($thumb_path)) {
851
852             $this->_debug(" ". $resolution ."px");
853             if(!$this->create_thumbnail($full_path, $thumb_path, $resolution))
854                $error = 1;
855          }
856
857          /* if the file hasn't changed there is no need to regen the thumb */
858          elseif($file_md5 != $this->getMD5($idx) || $force) {
859
860             $this->_debug(" ". $resolution ."px");
861             if(!$this->create_thumbnail($full_path, $thumb_path, $resolution))
862                $error = 1;
863
864          }
865       }
866
867       /* set the new/changed MD5 sum for the current photo */
868       if(!$error) {
869          $this->setMD5($idx, $file_md5);
870       }
871
872       $this->_debug("\n");
873
874    } // gen_thumb()
875
876    public function getMD5($idx)
877    {
878       $result = $this->cfg_db->db_query("
879          SELECT img_md5 
880          FROM images
881          WHERE img_idx='". $idx ."'
882       ");
883
884       if(!$result)
885          return 0;
886
887       $img = $this->cfg_db->db_fetch_object($result);
888       return $img['img_md5'];
889       
890    } // getMD5()
891
892    private function setMD5($idx, $md5)
893    {
894       $result = $this->cfg_db->db_exec("
895          REPLACE INTO images (img_idx, img_md5)
896          VALUES ('". $idx ."', '". $md5 ."')
897       ");
898
899    } // setMD5()
900
901    public function setTagCondition($mode)
902    {
903       $_SESSION['tag_condition'] = $mode;
904
905    } // setTagCondition()
906
907    public function startTagSearch($searchfor)
908    {
909       $_SESSION['searchfor'] = $searchfor;
910       $_SESSION['selected_tags'] = Array();
911
912       foreach($this->avail_tags as $tag) {
913          if(preg_match('/'. $searchfor .'/i', $this->tags[$tag]))
914             array_push($_SESSION['selected_tags'], $tag);
915       }
916
917    } // startTagSearch()
918
919    private function rotateImage($img, $degrees)
920    {
921       if(function_exists("imagerotate"))
922          $img = imagerotate($img, $degrees, 0);
923       else
924       {
925          function imagerotate($src_img, $angle)
926          {
927             $src_x = imagesx($src_img);
928             $src_y = imagesy($src_img);
929             if ($angle == 180) {
930                $dest_x = $src_x;
931                $dest_y = $src_y;
932             }
933             elseif ($src_x <= $src_y) {
934                $dest_x = $src_y;
935                $dest_y = $src_x;
936             }
937             elseif ($src_x >= $src_y) {
938                $dest_x = $src_y;
939                $dest_y = $src_x;
940             }
941                
942             $rotate=imagecreatetruecolor($dest_x,$dest_y);
943             imagealphablending($rotate, false);
944                
945             switch ($angle) {
946             
947                case 90:
948                   for ($y = 0; $y < ($src_y); $y++) {
949                      for ($x = 0; $x < ($src_x); $x++) {
950                         $color = imagecolorat($src_img, $x, $y);
951                         imagesetpixel($rotate, $dest_x - $y - 1, $x, $color);
952                      }
953                   }
954                   break;
955
956                case 270:
957                   for ($y = 0; $y < ($src_y); $y++) {
958                      for ($x = 0; $x < ($src_x); $x++) {
959                         $color = imagecolorat($src_img, $x, $y);
960                         imagesetpixel($rotate, $y, $dest_y - $x - 1, $color);
961                      }
962                   }
963                   break;
964
965                case 180:
966                   for ($y = 0; $y < ($src_y); $y++) {
967                      for ($x = 0; $x < ($src_x); $x++) {
968                         $color = imagecolorat($src_img, $x, $y);
969                         imagesetpixel($rotate, $dest_x - $x - 1, $dest_y - $y - 1, $color);
970                      }
971                   }
972                   break;
973
974                default: $rotate = $src_img;
975             };
976
977             return $rotate;
978
979          }
980
981          $img = imagerotate($img, $degrees);
982
983       }
984
985       return $img;
986
987    } // rotateImage()
988
989    private function get_photo_tags($idx)
990    {
991       $result = $this->db->db_query("
992          SELECT t.id, t.name
993          FROM tags t
994          INNER JOIN photo_tags pt
995             ON t.id=pt.tag_id
996          WHERE pt.photo_id='". $idx ."'
997       ");
998
999       $tags = Array();
1000
1001       while($row = $this->db->db_fetch_object($result))
1002          $tags[$row['id']] = $row['name'];
1003
1004       return $tags;
1005
1006    } // get_photo_tags()
1007
1008    public function showTextImage($txt, $color=000000, $space=4, $font=4, $w=300)
1009    {
1010       if (strlen($color) != 6) 
1011          $color = 000000;
1012
1013       $int = hexdec($color);
1014       $h = imagefontheight($font);
1015       $fw = imagefontwidth($font);
1016       $txt = explode("\n", wordwrap($txt, ($w / $fw), "\n"));
1017       $lines = count($txt);
1018       $im = imagecreate($w, (($h * $lines) + ($lines * $space)));
1019       $bg = imagecolorallocate($im, 255, 255, 255);
1020       $color = imagecolorallocate($im, 0xFF & ($int >> 0x10), 0xFF & ($int >> 0x8), 0xFF & $int);
1021       $y = 0;
1022
1023       foreach ($txt as $text) {
1024          $x = (($w - ($fw * strlen($text))) / 2);
1025          imagestring($im, $font, $x, $y, $text, $color);
1026          $y += ($h + $space);
1027       }
1028
1029       Header("Content-type: image/png");
1030       ImagePng($im);
1031
1032    } // showTextImage()
1033
1034    private function checkRequirements()
1035    {
1036       if(!function_exists("imagecreatefromjpeg")) {
1037          print "PHP GD library extension is missing<br />\n";
1038          $missing = true;
1039       }
1040
1041       if(!function_exists("sqlite3_open")) {
1042          print "PHP SQLite3 library extension is missing<br />\n";
1043          $missing = true;
1044       }
1045
1046       /* Check for HTML_AJAX PEAR package, lent from Horde project */
1047       ini_set('track_errors', 1);
1048       @include_once 'HTML/AJAX/Server.php';
1049       if(isset($php_errormsg) && preg_match('/Failed opening.*for inclusion/i', $php_errormsg)) {
1050          print "PEAR HTML_AJAX package is missing<br />\n";
1051          $missing = true;
1052       }
1053       @include_once 'Calendar/Calendar.php';
1054       if(isset($php_errormsg) && preg_match('/Failed opening.*for inclusion/i', $php_errormsg)) {
1055          print "PEAR Calendar package is missing<br />\n";
1056          $missing = true;
1057       }
1058       ini_restore('track_errors');
1059
1060       if(isset($missing))
1061          return false;
1062
1063       return true;
1064
1065    } // checkRequirements()
1066
1067    private function _debug($text)
1068    {
1069       if($this->fromcmd) {
1070          print $text;
1071       }
1072
1073    } // _debug()
1074
1075    public function checkifImageSupported($mime)
1076    {
1077       if(in_array($mime, Array("image/jpeg")))
1078          return true;
1079
1080       return false;
1081
1082    } // checkifImageSupported()
1083
1084    public function _warning($text)
1085    {
1086       print "<img src=\"resources/green_info.png\" alt=\"warning\" />\n";
1087       print $text;
1088
1089    } // _warning()
1090
1091    private function get_calendar()
1092    {
1093 //      require_once CALENDAR_ROOT.'Year.php';
1094 //      require_once CALENDAR_ROOT.'Month.php';
1095 //      require_once CALENDAR_ROOT.'Day.php';
1096
1097 //      $Year = new Calendar_Year();
1098 //      $Month = new Calendar_Month();
1099 //      $Day = new Calendar_Day();
1100
1101       $output = "<select name=\"year\">\n";
1102       for($year = 1990; $year <= date("Y"); $year++) {
1103          $output.= "<option value=\"". $year ."\">". $year ."</option>\n";
1104       }
1105       $output.= "</select>\n";
1106
1107       $output.= "<select name=\"month\">\n";
1108       for($month = 1; $month <= 12; $month++) {
1109          $output.= "<option value=\"". $month ."\">". $month ."</option>\n";
1110       }
1111       $output.= "</select>\n";
1112
1113       $output.= "<select name=\"day\">\n";
1114       for($day = 1; $day <= 31; $day++) {
1115          $output.= "<option value=\"". $day ."\">". $day ."</option>\n";
1116       }
1117       $output.= "</select>\n";
1118
1119       return $output;
1120
1121    } // get_calendar()
1122
1123 }
1124
1125 ?>