database = $database; $this->plugins = $plugins; $this->settings = $settings; $this->photoIDs = $photoIDs; return true; } public function add($files, $albumID, $description = '', $tags = '') { # Check dependencies self::dependencies(isset($this->database)); # Check permissions if (hasPermissions(LYCHEE_UPLOADS_BIG)===false||hasPermissions(LYCHEE_UPLOADS_THUMB)===false) { Log::error($this->database, __METHOD__, __LINE__, 'Wrong permissions in uploads/'); exit('Error: Wrong permissions in uploads-folder!'); } # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); switch($albumID) { case 's': # s for public (share) $public = 1; $star = 0; $albumID = 0; break; case 'f': # f for starred (fav) $star = 1; $public = 0; $albumID = 0; break; case 'r': # r for recent $public = 0; $star = 0; $albumID = 0; break; default: $star = 0; $public = 0; break; } foreach ($files as $file) { # Verify extension $extension = getExtension($file['name']); if (!in_array(strtolower($extension), $this->validExtensions, true)) continue; # Verify image $type = @exif_imagetype($file['tmp_name']); if (!in_array($type, $this->allowedTypes, true)) continue; # Generate id $id = str_replace('.', '', microtime(true)); while(strlen($id)<14) $id .= 0; # Set paths $tmp_name = $file['tmp_name']; $photo_name = md5($id) . $extension; $path = LYCHEE_UPLOADS_BIG . $photo_name; # Calculate checksum $checksum = sha1_file($tmp_name); if ($checksum===false) { Log::error($this->database, __METHOD__, __LINE__, 'Could not calculate checksum for photo'); exit('Error: Could not calculate checksum for photo!'); } # Check if image exists based on checksum if ($checksum===false) { $checksum = ''; $exists = false; } else { $exists = $this->exists($checksum); if ($exists!==false) { $photo_name = $exists['photo_name']; $path = $exists['path']; $path_thumb = $exists['path_thumb']; $exists = true; } } if ($exists===false) { # Import if not uploaded via web if (!is_uploaded_file($tmp_name)) { if (!@copy($tmp_name, $path)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not copy photo to uploads'); exit('Error: Could not copy photo to uploads!'); } else @unlink($tmp_name); } else { if (!@move_uploaded_file($tmp_name, $path)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not move photo to uploads'); exit('Error: Could not move photo to uploads!'); } } } # Read infos $info = $this->getInfo($path); # Use title of file if IPTC title missing if ($info['title']==='') $info['title'] = substr(basename($file['name'], $extension), 0, 30); # Use description parameter if set if ($description==='') $description = $info['description']; if ($exists===false) { # Set orientation based on EXIF data if ($file['type']==='image/jpeg'&&isset($info['orientation'], $info['width'], $info['height'])&&$info['orientation']!=='') { if (!$this->adjustFile($path, $info)) Log::notice($this->database, __METHOD__, __LINE__, 'Could not adjust photo (' . $info['title'] . ')'); } # Set original date if ($info['takestamp']!=='') @touch($path, $info['takestamp']); # Create Thumb if (!$this->createThumb($path, $photo_name)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not create thumbnail for photo'); exit('Error: Could not create thumbnail for photo!'); } # Set thumb url $path_thumb = md5($id) . '.jpeg'; } # Save to DB $values = [LYCHEE_TABLE_PHOTOS, $id, $info['title'], $photo_name, $description, $tags, $info['type'], $info['width'], $info['height'], $info['size'], $info['iso'], $info['aperture'], $info['make'], $info['model'], $info['shutter'], $info['focal'], $info['takestamp'], $path_thumb, $albumID, $public, $star, $checksum]; $query = Database::prepare($this->database, "INSERT INTO ? (id, title, url, description, tags, type, width, height, size, iso, aperture, make, model, shutter, focal, takestamp, thumbUrl, album, public, star, checksum) VALUES ('?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?', '?')", $values); $result = $this->database->query($query); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); exit('Error: Could not save photo in database!'); } } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return true; } private function exists($checksum, $photoID = null) { # Check dependencies self::dependencies(isset($this->database, $checksum)); # Exclude $photoID from select when $photoID is set if (isset($photoID)) $query = Database::prepare($this->database, "SELECT id, url, thumbUrl FROM ? WHERE checksum = '?' AND id <> '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $checksum, $photoID]); else $query = Database::prepare($this->database, "SELECT id, url, thumbUrl FROM ? WHERE checksum = '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $checksum]); $result = $this->database->query($query); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, 'Could not check for existing photos with the same checksum'); return false; } if ($result->num_rows===1) { $result = $result->fetch_object(); $return = array( 'photo_name' => $result->url, 'path' => LYCHEE_UPLOADS_BIG . $result->url, 'path_thumb' => $result->thumbUrl ); return $return; } return false; } private function createThumb($url, $filename, $width = 200, $height = 200) { # Check dependencies self::dependencies(isset($this->database, $this->settings, $url, $filename)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); $info = getimagesize($url); $photoName = explode(".", $filename); $newUrl = LYCHEE_UPLOADS_THUMB . $photoName[0] . '.jpeg'; $newUrl2x = LYCHEE_UPLOADS_THUMB . $photoName[0] . '@2x.jpeg'; # create thumbnails with Imagick if(extension_loaded('imagick')&&$this->settings['imagick']==='1') { # Read image $thumb = new Imagick(); $thumb->readImage($url); $thumb->setImageCompressionQuality($this->settings['thumbQuality']); $thumb->setImageFormat('jpeg'); # Copy image for 2nd thumb version $thumb2x = clone $thumb; # Create 1st version $thumb->cropThumbnailImage($width, $height); $thumb->writeImage($newUrl); $thumb->clear(); $thumb->destroy(); # Create 2nd version $thumb2x->cropThumbnailImage($width*2, $height*2); $thumb2x->writeImage($newUrl2x); $thumb2x->clear(); $thumb2x->destroy(); } else { # Set position and size $thumb = imagecreatetruecolor($width, $height); $thumb2x = imagecreatetruecolor($width*2, $height*2); if ($info[0]<$info[1]) { $newSize = $info[0]; $startWidth = 0; $startHeight = $info[1]/2 - $info[0]/2; } else { $newSize = $info[1]; $startWidth = $info[0]/2 - $info[1]/2; $startHeight = 0; } # Create new image switch($info['mime']) { case 'image/jpeg': $sourceImg = imagecreatefromjpeg($url); break; case 'image/png': $sourceImg = imagecreatefrompng($url); break; case 'image/gif': $sourceImg = imagecreatefromgif($url); break; default: Log::error($this->database, __METHOD__, __LINE__, 'Type of photo is not supported'); return false; break; } # Create thumb fastimagecopyresampled($thumb, $sourceImg, 0, 0, $startWidth, $startHeight, $width, $height, $newSize, $newSize); imagejpeg($thumb, $newUrl, $this->settings['thumbQuality']); imagedestroy($thumb); # Create retina thumb fastimagecopyresampled($thumb2x, $sourceImg, 0, 0, $startWidth, $startHeight, $width*2, $height*2, $newSize, $newSize); imagejpeg($thumb2x, $newUrl2x, $this->settings['thumbQuality']); imagedestroy($thumb2x); # Free memory imagedestroy($sourceImg); } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return true; } public function adjustFile($path, $info) { # Check dependencies self::dependencies(isset($path, $info)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); if (extension_loaded('imagick')&&$this->settings['imagick']==='1') { $rotateImage = 0; switch ($info['orientation']) { case 3: $rotateImage = 180; $imageOrientation = 1; break; case 6: $rotateImage = 90; $imageOrientation = 1; break; case 8: $rotateImage = 270; $imageOrientation = 1; break; } if ($rotateImage!==0) { $image = new Imagick(); $image->readImage($path); $image->rotateImage(new ImagickPixel(), $rotateImage); $image->setImageOrientation($imageOrientation); $image->writeImage($path); $image->clear(); $image->destroy(); } } else { $newWidth = $info['width']; $newHeight = $info['height']; $process = false; $sourceImg = imagecreatefromjpeg($path); switch ($info['orientation']) { case 2: # mirror # not yet implemented break; case 3: $process = true; $sourceImg = imagerotate($sourceImg, -180, 0); break; case 4: # rotate 180 and mirror # not yet implemented break; case 5: # rotate 90 and mirror # not yet implemented break; case 6: $process = true; $sourceImg = imagerotate($sourceImg, -90, 0); $newWidth = $info['height']; $newHeight = $info['width']; break; case 7: # rotate -90 and mirror # not yet implemented break; case 8: $process = true; $sourceImg = imagerotate($sourceImg, 90, 0); $newWidth = $info['height']; $newHeight = $info['width']; break; } # Need to adjust photo? if ($process===true) { # Recreate photo $newSourceImg = imagecreatetruecolor($newWidth, $newHeight); imagecopyresampled($newSourceImg, $sourceImg, 0, 0, 0, 0, $newWidth, $newHeight, $newWidth, $newHeight); imagejpeg($newSourceImg, $path, 100); # Free memory imagedestroy($sourceImg); imagedestroy($newSourceImg); } } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return true; } public function get($albumID) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get photo $query = Database::prepare($this->database, "SELECT * FROM ? WHERE id = '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); $photo = $photos->fetch_assoc(); # Parse photo $photo['sysdate'] = date('d M. Y', substr($photo['id'], 0, -4)); if (strlen($photo['takestamp'])>1) $photo['takedate'] = date('d M. Y', $photo['takestamp']); # Parse url $photo['url'] = LYCHEE_URL_UPLOADS_BIG . $photo['url']; $photo['thumbUrl'] = LYCHEE_URL_UPLOADS_THUMB . $photo['thumbUrl']; if ($albumID!='false') { # Show photo as public when parent album is public # Check if parent album is available and not photo not unsorted if ($photo['album']!=0) { # Get album $query = Database::prepare($this->database, "SELECT public FROM ? WHERE id = '?' LIMIT 1", [LYCHEE_TABLE_ALBUMS, $photo['album']]); $albums = $this->database->query($query); $album = $albums->fetch_assoc(); # Parse album $photo['public'] = ($album['public']=='1' ? '2' : $photo['public']); } $photo['original_album'] = $photo['album']; $photo['album'] = $albumID; } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return $photo; } public function getInfo($url) { # Check dependencies self::dependencies(isset($this->database, $url)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); $iptcArray = array(); $info = getimagesize($url, $iptcArray); # General information $return['type'] = $info['mime']; $return['width'] = $info[0]; $return['height'] = $info[1]; # Size $size = filesize($url)/1024; if ($size>=1024) $return['size'] = round($size/1024, 1) . ' MB'; else $return['size'] = round($size, 1) . ' KB'; # IPTC Metadata Fallback $return['title'] = ''; $return['description'] = ''; # IPTC Metadata if(isset($iptcArray['APP13'])) { $iptcInfo = iptcparse($iptcArray['APP13']); if (is_array($iptcInfo)) { $temp = @$iptcInfo['2#105'][0]; if (isset($temp)&&strlen($temp)>0) $return['title'] = $temp; $temp = @$iptcInfo['2#120'][0]; if (isset($temp)&&strlen($temp)>0) $return['description'] = $temp; $temp = @$iptcInfo['2#005'][0]; if (isset($temp)&&strlen($temp)>0&&$return['title']==='') $return['title'] = $temp; } } # EXIF Metadata Fallback $return['orientation'] = ''; $return['iso'] = ''; $return['aperture'] = ''; $return['make'] = ''; $return['model'] = ''; $return['shutter'] = ''; $return['focal'] = ''; $return['takestamp'] = 0; # Read EXIF if ($info['mime']=='image/jpeg') $exif = @exif_read_data($url, 'EXIF', 0); else $exif = false; # EXIF Metadata if ($exif!==false) { if (isset($exif['Orientation'])) $return['orientation'] = $exif['Orientation']; else if (isset($exif['IFD0']['Orientation'])) $return['orientation'] = $exif['IFD0']['Orientation']; $temp = @$exif['ISOSpeedRatings']; if (isset($temp)) $return['iso'] = $temp; $temp = @$exif['COMPUTED']['ApertureFNumber']; if (isset($temp)) $return['aperture'] = $temp; $temp = @$exif['Make']; if (isset($temp)) $return['make'] = trim($temp); $temp = @$exif['Model']; if (isset($temp)) $return['model'] = trim($temp); $temp = @$exif['ExposureTime']; if (isset($temp)) $return['shutter'] = $exif['ExposureTime'] . ' Sec.'; $temp = @$exif['FocalLength']; if (isset($temp)) $return['focal'] = ($temp/1) . ' mm'; $temp = @$exif['DateTimeOriginal']; if (isset($temp)) $return['takestamp'] = strtotime($temp); } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return $return; } public function getArchive() { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get photo $query = Database::prepare($this->database, "SELECT title, url FROM ? WHERE id = '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); $photo = $photos->fetch_object(); # Get extension $extension = getExtension($photo->url); if ($extension===false) { Log::error($this->database, __METHOD__, __LINE__, 'Invalid photo extension'); return false; } # Parse title if ($photo->title=='') $photo->title = 'Untitled'; # Set headers header("Content-Type: application/octet-stream"); header("Content-Disposition: attachment; filename=\"" . $photo->title . $extension . "\""); header("Content-Length: " . filesize(LYCHEE_UPLOADS_BIG . $photo->url)); # Send file readfile(LYCHEE_UPLOADS_BIG . $photo->url); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return true; } public function setTitle($title) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Parse if (strlen($title)>50) $title = substr($title, 0, 50); # Set title $query = Database::prepare($this->database, "UPDATE ? SET title = '?' WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $title, $this->photoIDs]); $result = $this->database->query($query); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } public function setDescription($description) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Parse $description = htmlentities($description, ENT_COMPAT | ENT_HTML401, 'UTF-8'); if (strlen($description)>1000) $description = substr($description, 0, 1000); # Set description $query = Database::prepare($this->database, "UPDATE ? SET description = '?' WHERE id IN ('?')", [LYCHEE_TABLE_PHOTOS, $description, $this->photoIDs]); $result = $this->database->query($query); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } public function setStar() { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Init vars $error = false; # Get photos $query = Database::prepare($this->database, "SELECT id, star FROM ? WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); # For each photo while ($photo = $photos->fetch_object()) { # Invert star $star = ($photo->star==0 ? 1 : 0); # Set star $query = Database::prepare($this->database, "UPDATE ? SET star = '?' WHERE id = '?'", [LYCHEE_TABLE_PHOTOS, $star, $photo->id]); $star = $this->database->query($query); if (!$star) $error = true; } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if ($error===true) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } public function getPublic($password) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get photo $query = Database::prepare($this->database, "SELECT public, album FROM ? WHERE id = '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); $photo = $photos->fetch_object(); # Check if public if ($photo->public==1) return true; else { $album = new Album($this->database, null, null, $photo->album); $acP = $album->checkPassword($password); $agP = $album->getPublic(); if ($acP===true&&$agP===true) return true; } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return false; } public function setPublic() { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get public $query = Database::prepare($this->database, "SELECT public FROM ? WHERE id = '?' LIMIT 1", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); $photo = $photos->fetch_object(); # Invert public $public = ($photo->public==0 ? 1 : 0); # Set public $query = Database::prepare($this->database, "UPDATE ? SET public = '?' WHERE id = '?'", [LYCHEE_TABLE_PHOTOS, $public, $this->photoIDs]); $result = $this->database->query($query); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } function setAlbum($albumID) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Set album $query = Database::prepare($this->database, "UPDATE ? SET album = '?' WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $albumID, $this->photoIDs]); $result = $this->database->query($query); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } public function setTags($tags) { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Parse tags $tags = preg_replace('/(\ ,\ )|(\ ,)|(,\ )|(,{1,}\ {0,})|(,$|^,)/', ',', $tags); $tags = preg_replace('/,$|^,|(\ ){0,}$/', '', $tags); if (strlen($tags)>1000) { Log::notice($this->database, __METHOD__, __LINE__, 'Length of tags higher than 1000'); return false; } # Set tags $query = Database::prepare($this->database, "UPDATE ? SET tags = '?' WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $tags, $this->photoIDs]); $result = $this->database->query($query); # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); if (!$result) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } return true; } public function duplicate() { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get photos $query = Database::prepare($this->database, "SELECT id, checksum FROM ? WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); if (!$photos) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } # For each photo while ($photo = $photos->fetch_object()) { # Generate id $id = str_replace('.', '', microtime(true)); while(strlen($id)<14) $id .= 0; # Duplicate entry $values = [LYCHEE_TABLE_PHOTOS, $id, LYCHEE_TABLE_PHOTOS, $photo->id]; $query = Database::prepare($this->database, "INSERT INTO ? (id, title, url, description, tags, type, width, height, size, iso, aperture, make, model, shutter, focal, takestamp, thumbUrl, album, public, star, checksum) SELECT '?' AS id, title, url, description, tags, type, width, height, size, iso, aperture, make, model, shutter, focal, takestamp, thumbUrl, album, public, star, checksum FROM ? WHERE id = '?'", $values); $duplicate = $this->database->query($query); if (!$duplicate) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } } return true; } public function delete() { # Check dependencies self::dependencies(isset($this->database, $this->photoIDs)); # Call plugins $this->plugins(__METHOD__, 0, func_get_args()); # Get photos $query = Database::prepare($this->database, "SELECT id, url, thumbUrl, checksum FROM ? WHERE id IN (?)", [LYCHEE_TABLE_PHOTOS, $this->photoIDs]); $photos = $this->database->query($query); if (!$photos) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } # For each photo while ($photo = $photos->fetch_object()) { # Check if other photos are referring to this images # If so, only delete the db entry if ($this->exists($photo->checksum, $photo->id)===false) { # Get retina thumb url $thumbUrl2x = explode(".", $photo->thumbUrl); $thumbUrl2x = $thumbUrl2x[0] . '@2x.' . $thumbUrl2x[1]; # Delete big if (file_exists(LYCHEE_UPLOADS_BIG . $photo->url)&&!unlink(LYCHEE_UPLOADS_BIG . $photo->url)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not delete photo in uploads/big/'); return false; } # Delete thumb if (file_exists(LYCHEE_UPLOADS_THUMB . $photo->thumbUrl)&&!unlink(LYCHEE_UPLOADS_THUMB . $photo->thumbUrl)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not delete photo in uploads/thumb/'); return false; } # Delete thumb@2x if (file_exists(LYCHEE_UPLOADS_THUMB . $thumbUrl2x)&&!unlink(LYCHEE_UPLOADS_THUMB . $thumbUrl2x)) { Log::error($this->database, __METHOD__, __LINE__, 'Could not delete high-res photo in uploads/thumb/'); return false; } } # Delete db entry $query = Database::prepare($this->database, "DELETE FROM ? WHERE id = '?'", [LYCHEE_TABLE_PHOTOS, $photo->id]); $delete = $this->database->query($query); if (!$delete) { Log::error($this->database, __METHOD__, __LINE__, $this->database->error); return false; } } # Call plugins $this->plugins(__METHOD__, 1, func_get_args()); return true; } } ?>