Photo.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701
  1. <?php
  2. ###
  3. # @name Photo Module
  4. # @author Tobias Reich
  5. # @copyright 2014 by Tobias Reich
  6. ###
  7. if (!defined('LYCHEE')) exit('Error: Direct access is not allowed!');
  8. class Photo extends Module {
  9. private $database = null;
  10. private $settings = null;
  11. private $photoIDs = null;
  12. private $uploadsBig = null;
  13. private $uploadsThumb = null;
  14. public function __construct($database, $plugins, $settings, $photoIDs) {
  15. # Init vars
  16. $this->database = $database;
  17. $this->plugins = $plugins;
  18. $this->settings = $settings;
  19. $this->photoIDs = $photoIDs;
  20. # Set upload dirs
  21. $this->uploadsBig = __DIR__ . '/../../uploads/big/';
  22. $this->uploadsThumb = __DIR__ . '/../../uploads/thumb/';
  23. return true;
  24. }
  25. public function add($files, $albumID, $description = '', $tags = '') {
  26. if (!isset($this->database)) return false;
  27. # Call plugins
  28. $this->plugins(__METHOD__, 0, func_get_args());
  29. switch($albumID) {
  30. case 's':
  31. # s for public (share)
  32. $public = 1;
  33. $star = 0;
  34. $albumID = 0;
  35. break;
  36. case 'f':
  37. # f for starred (fav)
  38. $star = 1;
  39. $public = 0;
  40. $albumID = 0;
  41. break;
  42. default:
  43. $star = 0;
  44. $public = 0;
  45. break;
  46. }
  47. foreach ($files as $file) {
  48. if ($file['type']!=='image/jpeg'&&
  49. $file['type']!=='image/png'&&
  50. $file['type']!=='image/gif')
  51. continue;
  52. $id = str_replace('.', '', microtime(true));
  53. while(strlen($id)<14) $id .= 0;
  54. $tmp_name = $file['tmp_name'];
  55. $extension = array_reverse(explode('.', $file['name']));
  56. $extension = $extension[0];
  57. $photo_name = md5($id) . ".$extension";
  58. $path = $this->uploadsBig . $photo_name;
  59. # Import if not uploaded via web
  60. if (!is_uploaded_file($tmp_name)) {
  61. if (!@copy($tmp_name, $path)) exit('Error: Could not copy photo to uploads!');
  62. else @unlink($tmp_name);
  63. } else {
  64. if (!@move_uploaded_file($tmp_name, $path)) exit('Error: Could not move photo to uploads!');
  65. }
  66. # Read infos
  67. $info = $this->getInfo($path);
  68. # Set original date
  69. if ($info['takestamp']!=='') @touch($path, $info['takestamp']);
  70. # Use title of file if IPTC title missing
  71. if ($info['title']==='') $info['title'] = mysqli_real_escape_string($this->database, substr(basename($file['name'], ".$extension"), 0, 30));
  72. # Use description parameter if set
  73. if ($description==='') $description = $info['description'];
  74. # Set orientation based on EXIF data
  75. if ($file['type']==='image/jpeg'&&isset($info['orientation'])&&$info['orientation']!==''&&isset($info['width'])&&isset($info['height'])) {
  76. if (!$this->adjustFile($path, $info)) exit('Error: Could not adjust photo!');
  77. }
  78. # Create Thumb
  79. if (!$this->createThumb($path, $photo_name)) exit('Error: Could not create thumbnail for photo!');
  80. # Save to DB
  81. $query = "INSERT INTO lychee_photos (id, title, url, description, tags, type, width, height, size, iso, aperture, make, model, shutter, focal, takestamp, thumbUrl, album, public, star)
  82. VALUES (
  83. '" . $id . "',
  84. '" . $info['title'] . "',
  85. '" . $photo_name . "',
  86. '" . $description . "',
  87. '" . $tags . "',
  88. '" . $info['type'] . "',
  89. '" . $info['width'] . "',
  90. '" . $info['height'] . "',
  91. '" . $info['size'] . "',
  92. '" . $info['iso'] . "',
  93. '" . $info['aperture'] . "',
  94. '" . $info['make'] . "',
  95. '" . $info['model'] . "',
  96. '" . $info['shutter'] . "',
  97. '" . $info['focal'] . "',
  98. '" . $info['takestamp'] . "',
  99. '" . md5($id) . ".jpeg',
  100. '" . $albumID . "',
  101. '" . $public . "',
  102. '" . $star . "');";
  103. $result = $this->database->query($query);
  104. if (!$result) exit('Error: Could not save photo in database!');
  105. }
  106. # Call plugins
  107. $this->plugins(__METHOD__, 1, func_get_args());
  108. return true;
  109. }
  110. private function createThumb($url, $filename, $width = 200, $height = 200) {
  111. if (!isset($this->settings, $url, $filename)) return false;
  112. # Call plugins
  113. $this->plugins(__METHOD__, 0, func_get_args());
  114. $info = getimagesize($url);
  115. $photoName = explode(".", $filename);
  116. $newUrl = $this->uploadsThumb . $photoName[0] . '.jpeg';
  117. $newUrl2x = $this->uploadsThumb . $photoName[0] . '@2x.jpeg';
  118. # create thumbnails with Imagick
  119. if(extension_loaded('imagick')) {
  120. # Read image
  121. $thumb = new Imagick();
  122. $thumb->readImage($url);
  123. $thumb->setImageCompressionQuality($this->settings['thumbQuality']);
  124. $thumb->setImageFormat('jpeg');
  125. # Copy image for 2nd thumb version
  126. $thumb2x = clone $thumb;
  127. # Create 1st version
  128. $thumb->cropThumbnailImage($width, $height);
  129. $thumb->writeImage($newUrl);
  130. # Create 2nd version
  131. $thumb2x->cropThumbnailImage($width*2, $height*2);
  132. $thumb2x->writeImage($newUrl2x);
  133. # Close thumb
  134. $thumb->clear();
  135. $thumb->destroy();
  136. # Close thumb2
  137. $thumb2x->clear();
  138. $thumb2x->destroy();
  139. } else {
  140. # Set position and size
  141. $thumb = imagecreatetruecolor($width, $height);
  142. $thumb2x = imagecreatetruecolor($width*2, $height*2);
  143. if ($info[0]<$info[1]) {
  144. $newSize = $info[0];
  145. $startWidth = 0;
  146. $startHeight = $info[1]/2 - $info[0]/2;
  147. } else {
  148. $newSize = $info[1];
  149. $startWidth = $info[0]/2 - $info[1]/2;
  150. $startHeight = 0;
  151. }
  152. # Fallback for older version
  153. if ($info['mime']==='image/webp'&&floatval(phpversion())<5.5) return false;
  154. # Create new image
  155. switch($info['mime']) {
  156. case 'image/jpeg': $sourceImg = imagecreatefromjpeg($url); break;
  157. case 'image/png': $sourceImg = imagecreatefrompng($url); break;
  158. case 'image/gif': $sourceImg = imagecreatefromgif($url); break;
  159. case 'image/webp': $sourceImg = imagecreatefromwebp($url); break;
  160. default: return false;
  161. }
  162. imagecopyresampled($thumb, $sourceImg, 0, 0, $startWidth, $startHeight, $width, $height, $newSize, $newSize);
  163. imagecopyresampled($thumb2x, $sourceImg, 0, 0, $startWidth, $startHeight, $width*2, $height*2, $newSize, $newSize);
  164. imagejpeg($thumb, $newUrl, $this->settings['thumbQuality']);
  165. imagejpeg($thumb2x, $newUrl2x, $this->settings['thumbQuality']);
  166. }
  167. # Call plugins
  168. $this->plugins(__METHOD__, 1, func_get_args());
  169. return true;
  170. }
  171. private function adjustFile($path, $info) {
  172. if (!isset($path, $info)) return false;
  173. # Call plugins
  174. $this->plugins(__METHOD__, 0, func_get_args());
  175. if (extension_loaded('imagick')) {
  176. $rotateImage = 0;
  177. switch ($info['orientation']) {
  178. case 3:
  179. $rotateImage = 180;
  180. $imageOrientation = 1;
  181. break;
  182. case 6:
  183. $rotateImage = 90;
  184. $imageOrientation = 1;
  185. break;
  186. case 8:
  187. $rotateImage = 270;
  188. $imageOrientation = 1;
  189. break;
  190. }
  191. if ($rotateImage) {
  192. $image = new Imagick();
  193. $image->readImage($path);
  194. $image->rotateImage(new ImagickPixel(), $rotateImage);
  195. $image->setImageOrientation($imageOrientation);
  196. $image->writeImage($path);
  197. $image->clear();
  198. $image->destroy();
  199. }
  200. } else {
  201. $newWidth = $info['width'];
  202. $newHeight = $info['height'];
  203. $sourceImg = imagecreatefromjpeg($path);
  204. switch ($info['orientation']) {
  205. case 2:
  206. # mirror
  207. # not yet implemented
  208. break;
  209. case 3:
  210. $sourceImg = imagerotate($sourceImg, -180, 0);
  211. break;
  212. case 4:
  213. # rotate 180 and mirror
  214. # not yet implemented
  215. break;
  216. case 5:
  217. # rotate 90 and mirror
  218. # not yet implemented
  219. break;
  220. case 6:
  221. $sourceImg = imagerotate($sourceImg, -90, 0);
  222. $newWidth = $info['height'];
  223. $newHeight = $info['width'];
  224. break;
  225. case 7:
  226. # rotate -90 and mirror
  227. # not yet implemented
  228. break;
  229. case 8:
  230. $sourceImg = imagerotate($sourceImg, 90, 0);
  231. $newWidth = $info['height'];
  232. $newHeight = $info['width'];
  233. break;
  234. }
  235. $newSourceImg = imagecreatetruecolor($newWidth, $newHeight);
  236. imagecopyresampled($newSourceImg, $sourceImg, 0, 0, 0, 0, $newWidth, $newHeight, $newWidth, $newHeight);
  237. imagejpeg($newSourceImg, $path, 100);
  238. }
  239. # Call plugins
  240. $this->plugins(__METHOD__, 1, func_get_args());
  241. return true;
  242. }
  243. public function get($albumID) {
  244. if (!isset($this->database, $this->photoIDs)) return false;
  245. # Call plugins
  246. $this->plugins(__METHOD__, 0, func_get_args());
  247. # Get photo
  248. $photos = $this->database->query("SELECT * FROM lychee_photos WHERE id = '$this->photoIDs' LIMIT 1;");
  249. $photo = $photos->fetch_assoc();
  250. # Parse photo
  251. $photo['sysdate'] = date('d M. Y', substr($photo['id'], 0, -4));
  252. if (strlen($photo['takestamp'])>0) $photo['takedate'] = date('d M. Y', $photo['takestamp']);
  253. if ($albumID!='false') {
  254. if ($photo['album']!=0) {
  255. # Get album
  256. $albums = $this->database->query("SELECT public FROM lychee_albums WHERE id = '" . $photo['album'] . " LIMIT 1';");
  257. $album = $albums->fetch_assoc();
  258. # Parse album
  259. $photo['public'] = ($album['public']=='1' ? '2' : $photo['public']);
  260. }
  261. $photo['original_album'] = $photo['album'];
  262. $photo['album'] = $albumID;
  263. }
  264. # Call plugins
  265. $this->plugins(__METHOD__, 1, func_get_args());
  266. return $photo;
  267. }
  268. private function getInfo($url) {
  269. if (!isset($this->database, $url)) return false;
  270. # Call plugins
  271. $this->plugins(__METHOD__, 0, func_get_args());
  272. $iptcArray = array();
  273. $info = getimagesize($url, $iptcArray);
  274. # General information
  275. $return['type'] = $info['mime'];
  276. $return['width'] = $info[0];
  277. $return['height'] = $info[1];
  278. # Size
  279. $size = filesize($url)/1024;
  280. if ($size>=1024) $return['size'] = round($size/1024, 1) . ' MB';
  281. else $return['size'] = round($size, 1) . ' KB';
  282. # IPTC Metadata Fallback
  283. $return['title'] = '';
  284. $return['description'] = '';
  285. # IPTC Metadata
  286. if(isset($iptcArray['APP13'])) {
  287. $iptcInfo = iptcparse($iptcArray['APP13']);
  288. if (is_array($iptcInfo)) {
  289. $temp = @$iptcInfo['2#105'][0];
  290. if (isset($temp)&&strlen($temp)>0) $return['title'] = $temp;
  291. $temp = @$iptcInfo['2#120'][0];
  292. if (isset($temp)&&strlen($temp)>0) $return['description'] = $temp;
  293. }
  294. }
  295. # EXIF Metadata Fallback
  296. $return['orientation'] = '';
  297. $return['iso'] = '';
  298. $return['aperture'] = '';
  299. $return['make'] = '';
  300. $return['model'] = '';
  301. $return['shutter'] = '';
  302. $return['focal'] = '';
  303. $return['takestamp'] = '';
  304. # Read EXIF
  305. if ($info['mime']=='image/jpeg') $exif = @exif_read_data($url, 'EXIF', 0);
  306. else $exif = false;
  307. # EXIF Metadata
  308. if ($exif!==false) {
  309. if (isset($exif['Orientation'])) $return['orientation'] = $exif['Orientation'];
  310. else if (isset($exif['IFD0']['Orientation'])) $return['orientation'] = $exif['IFD0']['Orientation'];
  311. $temp = @$exif['ISOSpeedRatings'];
  312. if (isset($temp)) $return['iso'] = $temp;
  313. $temp = @$exif['COMPUTED']['ApertureFNumber'];
  314. if (isset($temp)) $return['aperture'] = $temp;
  315. $temp = @$exif['Make'];
  316. if (isset($temp)) $return['make'] = $exif['Make'];
  317. $temp = @$exif['Model'];
  318. if (isset($temp)) $return['model'] = $temp;
  319. $temp = @$exif['ExposureTime'];
  320. if (isset($temp)) $return['shutter'] = $exif['ExposureTime'] . ' Sec.';
  321. $temp = @$exif['FocalLength'];
  322. if (isset($temp)) $return['focal'] = ($temp/1) . ' mm';
  323. $temp = @$exif['DateTimeOriginal'];
  324. if (isset($temp)) $return['takestamp'] = strtotime($temp);
  325. }
  326. # Security
  327. foreach(array_keys($return) as $key) $return[$key] = mysqli_real_escape_string($this->database, $return[$key]);
  328. # Call plugins
  329. $this->plugins(__METHOD__, 1, func_get_args());
  330. return $return;
  331. }
  332. public function getArchive() {
  333. if (!isset($this->database, $this->photoIDs)) return false;
  334. # Call plugins
  335. $this->plugins(__METHOD__, 0, func_get_args());
  336. # Get photo
  337. $photos = $this->database->query("SELECT title, url FROM lychee_photos WHERE id = '$this->photoIDs' LIMIT 1;");
  338. $photo = $photos->fetch_object();
  339. # Get extension
  340. $extension = array_reverse(explode('.', $photo->url));
  341. # Parse title
  342. if ($photo->title=='') $photo->title = 'Untitled';
  343. # Set headers
  344. header("Content-Type: application/octet-stream");
  345. header("Content-Disposition: attachment; filename=\"$photo->title.$extension[0]\"");
  346. header("Content-Length: " . filesize($this->uploadsBig . $photo->url));
  347. # Send file
  348. readfile($this->uploadsBig . $photo->url);
  349. # Call plugins
  350. $this->plugins(__METHOD__, 1, func_get_args());
  351. return true;
  352. }
  353. function setTitle($title) {
  354. if (!isset($this->database, $this->photoIDs)) return false;
  355. # Call plugins
  356. $this->plugins(__METHOD__, 0, func_get_args());
  357. # Parse
  358. if (strlen($title)>50) $title = substr($title, 0, 50);
  359. # Set title
  360. $result = $this->database->query("UPDATE lychee_photos SET title = '$title' WHERE id IN ($this->photoIDs);");
  361. # Call plugins
  362. $this->plugins(__METHOD__, 1, func_get_args());
  363. if (!$result) return false;
  364. return true;
  365. }
  366. function setDescription($description) {
  367. if (!isset($this->database, $this->photoIDs)) return false;
  368. # Call plugins
  369. $this->plugins(__METHOD__, 0, func_get_args());
  370. # Parse
  371. $description = htmlentities($description);
  372. if (strlen($description)>1000) $description = substr($description, 0, 1000);
  373. # Set description
  374. $result = $this->database->query("UPDATE lychee_photos SET description = '$description' WHERE id IN ('$this->photoIDs');");
  375. # Call plugins
  376. $this->plugins(__METHOD__, 1, func_get_args());
  377. if (!$result) return false;
  378. return true;
  379. }
  380. public function setStar() {
  381. if (!isset($this->database, $this->photoIDs)) return false;
  382. # Call plugins
  383. $this->plugins(__METHOD__, 0, func_get_args());
  384. # Init vars
  385. $error = false;
  386. # Get photos
  387. $photos = $this->database->query("SELECT id, star FROM lychee_photos WHERE id IN ($this->photoIDs);");
  388. # For each photo
  389. while ($photo = $photos->fetch_object()) {
  390. # Invert star
  391. $star = ($photo->star==0 ? 1 : 0);
  392. # Set star
  393. $star = $this->database->query("UPDATE lychee_photos SET star = '$star' WHERE id = '$photo->id';");
  394. if (!$star) $error = true;
  395. }
  396. # Call plugins
  397. $this->plugins(__METHOD__, 1, func_get_args());
  398. if ($error) return false;
  399. return true;
  400. }
  401. function getPublic($password) {
  402. if (!isset($this->database, $this->photoIDs)) return false;
  403. # Call plugins
  404. $this->plugins(__METHOD__, 0, func_get_args());
  405. # Get photo
  406. $photos = $this->database->query("SELECT public, album FROM lychee_photos WHERE id = '$this->photoIDs' LIMIT 1;");
  407. $photo = $photos->fetch_object();
  408. # Check if public
  409. if ($photo->public==1) return true;
  410. else {
  411. $album = new Album($this->database, null, null, $photo->album);
  412. $acP = $album->checkPassword($password);
  413. $agP = $album->getPublic();
  414. if ($acP===true&&$agP===true) return true;
  415. }
  416. # Call plugins
  417. $this->plugins(__METHOD__, 1, func_get_args());
  418. return false;
  419. }
  420. public function setPublic() {
  421. if (!isset($this->database, $this->photoIDs)) return false;
  422. # Call plugins
  423. $this->plugins(__METHOD__, 0, func_get_args());
  424. # Get public
  425. $photos = $this->database->query("SELECT public FROM lychee_photos WHERE id = '$this->photoIDs' LIMIT 1;");
  426. $photo = $photos->fetch_object();
  427. # Invert public
  428. $public = ($photo->public==0 ? 1 : 0);
  429. # Set public
  430. $result = $this->database->query("UPDATE lychee_photos SET public = '$public' WHERE id = '$this->photoIDs';");
  431. # Call plugins
  432. $this->plugins(__METHOD__, 1, func_get_args());
  433. if (!$result) return false;
  434. return true;
  435. }
  436. function setAlbum($albumID) {
  437. if (!isset($this->database, $this->photoIDs)) return false;
  438. # Call plugins
  439. $this->plugins(__METHOD__, 0, func_get_args());
  440. # Set album
  441. $result = $this->database->query("UPDATE lychee_photos SET album = '$albumID' WHERE id IN ($this->photoIDs);");
  442. # Call plugins
  443. $this->plugins(__METHOD__, 1, func_get_args());
  444. if (!$result) return false;
  445. return true;
  446. }
  447. public function setTags($tags) {
  448. if (!isset($this->database, $this->photoIDs)) return false;
  449. # Call plugins
  450. $this->plugins(__METHOD__, 0, func_get_args());
  451. # Parse tags
  452. $tags = preg_replace('/(\ ,\ )|(\ ,)|(,\ )|(,{1,}\ {0,})|(,$|^,)/', ',', $tags);
  453. $tags = preg_replace('/,$|^,/', ',', $tags);
  454. if (strlen($tags)>1000) return false;
  455. # Set tags
  456. $result = $this->database->query("UPDATE lychee_photos SET tags = '$tags' WHERE id IN ($this->photoIDs);");
  457. # Call plugins
  458. $this->plugins(__METHOD__, 1, func_get_args());
  459. if (!$result) return false;
  460. return true;
  461. }
  462. public function delete() {
  463. if (!isset($this->database, $this->photoIDs)) return false;
  464. # Call plugins
  465. $this->plugins(__METHOD__, 0, func_get_args());
  466. # Get photos
  467. $photos = $this->database->query("SELECT id, url, thumbUrl FROM lychee_photos WHERE id IN ($this->photoIDs);");
  468. # For each photo
  469. while ($photo = $photos->fetch_object()) {
  470. # Get retina thumb url
  471. $thumbUrl2x = explode(".", $photo->thumbUrl);
  472. $thumbUrl2x = $thumbUrl2x[0] . '@2x.' . $thumbUrl2x[1];
  473. # Delete files
  474. if (file_exists($this->uploadsBig . $photo->url)&&!unlink($this->uploadsBig . $photo->url)) return false;
  475. if (file_exists($this->uploadsThumb . $photo->thumbUrl)&&!unlink($this->uploadsThumb . $photo->thumbUrl)) return false;
  476. if (file_exists($this->uploadsThumb . $thumbUrl2x)&&!unlink($this->uploadsThumb . $thumbUrl2x)) return false;
  477. # Delete db entry
  478. $delete = $this->database->query("DELETE FROM lychee_photos WHERE id = '$photo->id';");
  479. if (!$delete) return false;
  480. }
  481. # Call plugins
  482. $this->plugins(__METHOD__, 1, func_get_args());
  483. if (!$photos) return false;
  484. return true;
  485. }
  486. }
  487. ?>