2016-06-22 15:49:20 +10:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Model;
|
|
|
|
|
|
|
|
use DB;
|
|
|
|
use Illuminate\Database\Eloquent\Model;
|
|
|
|
|
|
|
|
class Photo extends Model
|
|
|
|
{
|
|
|
|
protected $table = 'photo';
|
|
|
|
|
|
|
|
// Imagick Object
|
|
|
|
private $_io;
|
|
|
|
|
|
|
|
// How should the image be rotated, based on the value of orientation
|
|
|
|
private $_rotate = [
|
|
|
|
3=>180,
|
|
|
|
6=>90,
|
|
|
|
8=>-90,
|
|
|
|
];
|
|
|
|
|
2016-07-01 14:37:55 +10:00
|
|
|
public function People()
|
|
|
|
{
|
|
|
|
return $this->belongsToMany('App\Model\Person');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function Tags()
|
|
|
|
{
|
|
|
|
return $this->belongsToMany('App\Model\Tag');
|
|
|
|
}
|
|
|
|
|
2016-06-29 14:04:02 +10:00
|
|
|
/**
|
|
|
|
* Photo's NOT pending removal.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Database\Eloquent\Builder
|
|
|
|
*/
|
|
|
|
public function scopeNotRemove($query)
|
|
|
|
{
|
2016-06-30 09:32:57 +10:00
|
|
|
return $query->where(function($query)
|
|
|
|
{
|
2016-06-29 14:04:02 +10:00
|
|
|
$query->where('remove','!=',TRUE)
|
|
|
|
->orWhere('remove','=',NULL);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Photo's NOT duplicate.
|
|
|
|
*
|
|
|
|
* @return \Illuminate\Database\Eloquent\Builder
|
|
|
|
*/
|
|
|
|
public function scopeNotDuplicate($query)
|
|
|
|
{
|
2016-06-30 09:32:57 +10:00
|
|
|
return $query->where(function($query)
|
|
|
|
{
|
2016-06-29 14:04:02 +10:00
|
|
|
$query->where('duplicate','!=',TRUE)
|
|
|
|
->orWhere('duplicate','=',NULL);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
public function date_taken()
|
|
|
|
{
|
2016-06-22 16:51:31 +10:00
|
|
|
return $this->date_taken ? (date('Y-m-d H:i:s',$this->date_taken).($this->subsectime ? '.'.$this->subsectime : '')) : 'UNKNOWN';
|
|
|
|
}
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
/**
|
|
|
|
* Return the date of the file
|
|
|
|
*/
|
|
|
|
public function file_date($type,$format=FALSE)
|
|
|
|
{
|
2016-06-30 16:01:12 +10:00
|
|
|
if (! is_readable($this->file_path()))
|
|
|
|
return NULL;
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
switch ($type)
|
|
|
|
{
|
|
|
|
case 'a': $t = fileatime($this->file_path());
|
|
|
|
break;
|
|
|
|
case 'c': $t = filectime($this->file_path());
|
|
|
|
break;
|
|
|
|
case 'm': $t = filemtime($this->file_path());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $format ? date('d-m-Y H:i:s',$t) : $t;
|
|
|
|
}
|
|
|
|
|
2016-06-22 15:49:20 +10:00
|
|
|
/**
|
|
|
|
* Determine the new name for the image
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function file_path($short=FALSE,$new=FALSE)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
$file = $this->filename;
|
|
|
|
|
|
|
|
if ($new)
|
|
|
|
$file = sprintf('%s.%s',((is_null($this->date_taken) OR ! $this->date_taken)
|
|
|
|
? sprintf('UNKNOWN/%07s',$this->file_path_id())
|
|
|
|
: sprintf('%s_%03s',date('Y/m/d-His',$this->date_taken),$this->subsectime).($this->subsectime ? '' : sprintf('-%05s',$this->id))),$this->type());
|
|
|
|
|
|
|
|
return (($short OR preg_match('/^\//',$file)) ? '' : config('photo.dir').DIRECTORY_SEPARATOR).$file;
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Calculate a file path ID based on the id of the file
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function file_path_id($sep=3,$depth=9)
|
|
|
|
{
|
2016-06-22 16:51:31 +10:00
|
|
|
return trim(chunk_split(sprintf("%0{$depth}s",$this->id),$sep,'/'),'/');
|
|
|
|
}
|
|
|
|
|
2016-06-30 16:01:12 +10:00
|
|
|
/**
|
|
|
|
* Return the photo size
|
|
|
|
*/
|
|
|
|
public function file_size()
|
|
|
|
{
|
|
|
|
return (! is_readable($this->file_path())) ? NULL : filesize($this->file_path());
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Display the GPS coordinates
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function gps()
|
|
|
|
{
|
2016-06-22 16:51:31 +10:00
|
|
|
return ($this->gps_lat AND $this->gps_lon) ? sprintf('%s/%s',$this->gps_lat,$this->gps_lon) : 'UNKNOWN';
|
|
|
|
}
|
|
|
|
|
2016-06-29 20:49:02 +10:00
|
|
|
/**
|
|
|
|
* Return the image, rotated, minus exif data
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function image()
|
|
|
|
{
|
2016-06-29 20:49:02 +10:00
|
|
|
$imo = $this->io();
|
|
|
|
|
|
|
|
if (array_key_exists('exif',$imo->getImageProfiles()))
|
|
|
|
$imo->removeImageProfile('exif');
|
|
|
|
|
|
|
|
$this->rotate($imo);
|
|
|
|
|
|
|
|
return $imo->getImageBlob();
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Return an Imagick object or attribute
|
|
|
|
*
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
protected function io($attr=NULL)
|
|
|
|
{
|
2016-06-22 16:51:31 +10:00
|
|
|
if (is_null($this->_io))
|
|
|
|
$this->_io = new \Imagick($this->file_path());
|
|
|
|
|
|
|
|
return is_null($attr) ? $this->_io : $this->_io->getImageProperty($attr);
|
|
|
|
}
|
|
|
|
|
2016-07-01 12:19:17 +10:00
|
|
|
public function isParentWritable($dir)
|
|
|
|
{
|
|
|
|
if (file_exists($dir) AND is_writable($dir) AND is_dir($dir))
|
|
|
|
return TRUE;
|
|
|
|
elseif ($dir == dirname($dir) OR file_exists($dir))
|
|
|
|
return FALSE;
|
|
|
|
else
|
|
|
|
return ($this->isParentWritable(dirname($dir)));
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Calculate the GPS coordinates
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public static function latlon(array $coordinate,$hemisphere)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
if (! $coordinate OR ! $hemisphere)
|
|
|
|
return NULL;
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
for ($i=0; $i<3; $i++)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
$part = explode('/', $coordinate[$i]);
|
|
|
|
|
|
|
|
if (count($part) == 1)
|
|
|
|
$coordinate[$i] = $part[0];
|
|
|
|
|
|
|
|
elseif (count($part) == 2)
|
|
|
|
$coordinate[$i] = floatval($part[0])/floatval($part[1]);
|
|
|
|
|
|
|
|
else
|
|
|
|
$coordinate[$i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
list($degrees, $minutes, $seconds) = $coordinate;
|
|
|
|
|
|
|
|
$sign = ($hemisphere == 'W' || $hemisphere == 'S') ? -1 : 1;
|
|
|
|
|
|
|
|
return round($sign*($degrees+$minutes/60+$seconds/3600),$degrees > 100 ? 3 : 4);
|
|
|
|
}
|
|
|
|
|
2016-06-29 14:04:02 +10:00
|
|
|
/**
|
|
|
|
* Determine if a file is moveable
|
|
|
|
*
|
|
|
|
* useID boolean Determine if the path is based on the the ID or date
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function moveable()
|
|
|
|
{
|
2016-06-29 14:04:02 +10:00
|
|
|
// If the source and target are the same, we dont need to move it
|
|
|
|
if ($this->file_path() == $this->file_path(FALSE,TRUE))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
// If there is already a file in the target.
|
|
|
|
// @todo If the target file is to be deleted, we could move this file
|
|
|
|
if (file_exists($this->file_path(FALSE,TRUE)))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
// Test if the source is readable
|
|
|
|
if (! is_readable($this->file_path()))
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
// Test if the dir is writable (so we can remove the file)
|
2016-07-01 12:19:17 +10:00
|
|
|
if (! $this->isParentWritable(dirname($this->file_path())))
|
2016-06-29 14:04:02 +10:00
|
|
|
return 3;
|
|
|
|
|
|
|
|
// Test if the target dir is writable
|
|
|
|
// @todo The target dir may not exist yet, so we should check that a parent exists and is writable.
|
2016-07-01 12:19:17 +10:00
|
|
|
if (! $this->isParentWritable($this->file_path(FALSE,TRUE)))
|
2016-06-29 14:04:02 +10:00
|
|
|
return 4;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2016-06-29 20:49:02 +10:00
|
|
|
/**
|
|
|
|
* Get the id of the previous photo
|
|
|
|
*/
|
|
|
|
public function next()
|
|
|
|
{
|
|
|
|
$po = DB::table('photo');
|
|
|
|
$po->where('id','>',$this->id);
|
|
|
|
$po->orderby('id','ASC');
|
|
|
|
return $po->first();
|
|
|
|
}
|
|
|
|
|
2016-06-30 16:01:12 +10:00
|
|
|
/**
|
|
|
|
* Display the orientation of a photo
|
|
|
|
*/
|
|
|
|
public function orientation() {
|
|
|
|
switch ($this->orientation) {
|
|
|
|
case 1: return 'None!';
|
|
|
|
case 3: return 'Upside Down';
|
|
|
|
case 6: return 'Rotate Right';
|
|
|
|
case 8: return 'Rotate Left';
|
|
|
|
default:
|
|
|
|
return 'unknown?';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-22 15:49:20 +10:00
|
|
|
/**
|
|
|
|
* Rotate the image
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
private function rotate(\Imagick $imo)
|
|
|
|
{
|
|
|
|
if (array_key_exists($this->orientation,$this->_rotate))
|
|
|
|
$imo->rotateImage(new \ImagickPixel('none'),$this->_rotate[$this->orientation]);
|
|
|
|
|
|
|
|
return $imo->getImageBlob();
|
|
|
|
}
|
|
|
|
|
2016-06-29 20:49:02 +10:00
|
|
|
public static function path($path)
|
|
|
|
{
|
2016-06-29 14:04:02 +10:00
|
|
|
return preg_replace('/^\//','',str_replace(config('photo.dir'),'',$path));
|
|
|
|
}
|
|
|
|
|
2016-06-29 20:49:02 +10:00
|
|
|
/**
|
|
|
|
* Get the id of the previous photo
|
|
|
|
*/
|
|
|
|
public function previous()
|
|
|
|
{
|
|
|
|
$po = DB::table('photo');
|
|
|
|
$po->where('id','<',$this->id);
|
|
|
|
$po->orderby('id','DEC');
|
|
|
|
return $po->first();
|
|
|
|
}
|
|
|
|
|
2016-06-30 16:01:12 +10:00
|
|
|
public function property($property)
|
|
|
|
{
|
|
|
|
if (! $this->io())
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
switch ($property)
|
|
|
|
{
|
|
|
|
case 'height': return $this->_io->getImageHeight(); break;
|
|
|
|
case 'orientation': return $this->_io->getImageOrientation(); break;
|
|
|
|
case 'signature': return $this->_io->getImageSignature(); break;
|
|
|
|
case 'width': return $this->_io->getImageWidth(); break;
|
|
|
|
default:
|
|
|
|
return $this->_io->getImageProperty($property);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function properties()
|
2016-06-29 20:49:02 +10:00
|
|
|
{
|
2016-06-30 16:01:12 +10:00
|
|
|
return $this->io() ? $this->_io->getImageProperties() : [];
|
2016-06-29 20:49:02 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Display the photo signature
|
|
|
|
*/
|
|
|
|
public function signature($short=FALSE)
|
|
|
|
{
|
2016-06-30 09:32:57 +10:00
|
|
|
return $short ? static::signaturetrim($this->signature) : $this->signature;
|
2016-06-29 20:49:02 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
public static function signaturetrim($signature,$chars=6)
|
|
|
|
{
|
|
|
|
return sprintf('%s...%s',substr($signature,0,$chars),substr($signature,-1*$chars));
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Determine if the image should be moved
|
|
|
|
*/
|
|
|
|
public function shouldMove()
|
|
|
|
{
|
|
|
|
return ($this->filename != $this->file_path(TRUE,TRUE));
|
|
|
|
}
|
|
|
|
|
2016-06-22 15:49:20 +10:00
|
|
|
/**
|
|
|
|
* Return the image's thumbnail
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public function thumbnail($rotate=TRUE)
|
|
|
|
{
|
|
|
|
if (! $this->thumbnail)
|
2016-06-29 20:49:02 +10:00
|
|
|
{
|
2016-06-30 16:01:12 +10:00
|
|
|
return $this->io()->thumbnailimage(200,200,true,true) ? $this->_io->getImageBlob() : NULL;
|
2016-06-29 20:49:02 +10:00
|
|
|
}
|
2016-06-22 15:49:20 +10:00
|
|
|
|
|
|
|
if (! $rotate OR ! array_key_exists($this->orientation,$this->_rotate) OR ! extension_loaded('imagick'))
|
|
|
|
return $this->thumbnail;
|
|
|
|
|
|
|
|
$imo = new \Imagick();
|
|
|
|
$imo->readImageBlob($this->thumbnail);
|
|
|
|
|
|
|
|
return $this->rotate($imo);
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:51:31 +10:00
|
|
|
/**
|
|
|
|
* Return the extension of the image
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function type($mime=FALSE)
|
|
|
|
{
|
2016-06-22 16:51:31 +10:00
|
|
|
return strtolower($mime ? File::mime_by_ext(pathinfo($this->filename,PATHINFO_EXTENSION)) : pathinfo($this->filename,PATHINFO_EXTENSION));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find duplicate images based on some attributes of the current image
|
|
|
|
*/
|
2016-06-30 09:32:57 +10:00
|
|
|
public function list_duplicate($includeme=FALSE)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
$po = DB::table('photo');
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
if ($this->id AND ! $includeme)
|
2016-06-22 15:49:20 +10:00
|
|
|
$po->where('id','!=',$this->id);
|
|
|
|
|
|
|
|
// Ignore photo's pending removal.
|
2016-06-30 16:01:12 +10:00
|
|
|
if (! $includeme)
|
|
|
|
$po->where(function($query)
|
|
|
|
{
|
|
|
|
$query->where('remove','!=',TRUE)
|
|
|
|
->orWhere('remove','=',NULL);
|
|
|
|
});
|
2016-06-22 15:49:20 +10:00
|
|
|
|
|
|
|
// Where the signature is the same
|
2016-06-30 09:32:57 +10:00
|
|
|
$po->where(function($query)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
$query->where('signature','=',$this->signature);
|
|
|
|
|
|
|
|
// Or they have the same time taken with the same camera
|
2016-06-30 09:32:57 +10:00
|
|
|
if ($this->date_taken AND ($this->model OR $this->make))
|
|
|
|
{
|
|
|
|
$query->orWhere(function($query)
|
|
|
|
{
|
2016-06-22 15:49:20 +10:00
|
|
|
$query->where('date_taken','=',$this->date_taken ? $this->date_taken : NULL);
|
|
|
|
$query->where('subsectime','=',$this->subsectime ? $this->subsectime : NULL);
|
|
|
|
|
|
|
|
if (! is_null($this->model))
|
|
|
|
$query->where('model','=',$this->model);
|
|
|
|
|
|
|
|
if (! is_null($this->make))
|
|
|
|
$query->where('make','=',$this->make);
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-06-30 09:32:57 +10:00
|
|
|
return $po->pluck('id');
|
2016-06-22 15:49:20 +10:00
|
|
|
}
|
|
|
|
}
|