Total Complexity | 106 |
Total Lines | 695 |
Duplicated Lines | 0 % |
Changes | 0 |
Complex classes like Media often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use Media, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
8 | class Media extends Model |
||
9 | { |
||
10 | public static $useCache = true; |
||
11 | public static $singularNoun = 'media item'; |
||
12 | public static $pluralNoun = 'media items'; |
||
13 | |||
14 | // support subclassing |
||
15 | public static $rootClass = __CLASS__; |
||
16 | public static $defaultClass = __CLASS__; |
||
17 | public static $subClasses = [__CLASS__, Image::class, PDF::class, Video::class, Audio::class]; |
||
18 | public static $collectionRoute = '/media'; |
||
19 | |||
20 | // get rid of these?? |
||
21 | public static $Namespaces = []; |
||
22 | public static $Types = []; |
||
23 | |||
24 | |||
25 | public static $tableName = 'media'; |
||
26 | |||
27 | public static $fields = [ |
||
28 | 'ContextClass' => [ |
||
29 | 'type' => 'string' |
||
30 | ,'notnull' => false, |
||
31 | ] |
||
32 | ,'ContextID' => [ |
||
33 | 'type' => 'integer' |
||
34 | ,'notnull' => false, |
||
35 | ] |
||
36 | ,'MIMEType' => 'string' |
||
37 | ,'Width' => [ |
||
38 | 'type' => 'integer' |
||
39 | ,'unsigned' => true |
||
40 | ,'notnull' => false, |
||
41 | ] |
||
42 | ,'Height' => [ |
||
43 | 'type' => 'integer' |
||
44 | ,'unsigned' => true |
||
45 | ,'notnull' => false, |
||
46 | ] |
||
47 | ,'Duration' => [ |
||
48 | 'type' => 'float' |
||
49 | ,'unsigned' => true |
||
50 | ,'notnull' => false, |
||
51 | ] |
||
52 | ,'Caption' => [ |
||
53 | 'type' => 'string' |
||
54 | ,'notnull' => false, |
||
55 | ], |
||
56 | ]; |
||
57 | |||
58 | public static $relationships = [ |
||
59 | 'Creator' => [ |
||
60 | 'type' => 'one-one' |
||
61 | ,'class' => 'Person' |
||
62 | ,'local' => 'CreatorID', |
||
63 | ] |
||
64 | ,'Context' => [ |
||
65 | 'type' => 'context-parent', |
||
66 | ], |
||
67 | ]; |
||
68 | |||
69 | public static $searchConditions = [ |
||
70 | 'Caption' => [ |
||
71 | 'qualifiers' => ['any','caption'] |
||
72 | ,'points' => 2 |
||
73 | ,'sql' => 'Caption LIKE "%%%s%%"', |
||
74 | ] |
||
75 | ,'CaptionLike' => [ |
||
76 | 'qualifiers' => ['caption-like'] |
||
77 | ,'points' => 2 |
||
78 | ,'sql' => 'Caption LIKE "%s"', |
||
79 | ] |
||
80 | ,'CaptionNot' => [ |
||
81 | 'qualifiers' => ['caption-not'] |
||
82 | ,'points' => 2 |
||
83 | ,'sql' => 'Caption NOT LIKE "%%%s%%"', |
||
84 | ] |
||
85 | ,'CaptionNotLike' => [ |
||
86 | 'qualifiers' => ['caption-not-like'] |
||
87 | ,'points' => 2 |
||
88 | ,'sql' => 'Caption NOT LIKE "%s"', |
||
89 | ], |
||
90 | ]; |
||
91 | |||
92 | public static $webPathFormat = '/media/open/%u'; // 1=mediaID |
||
93 | public static $thumbnailRequestFormat = '/thumbnail/%1$u/%2$ux%3$u%4$s'; // 1=media_id 2=width 3=height 4=fill_color |
||
94 | public static $blankThumbnailRequestFormat = '/thumbnail/%1$s/%2$ux%3$u%4$s'; // 1=class 2=width 3=height 4=fill_color |
||
95 | public static $thumbnailJPEGCompression = 90; |
||
96 | public static $thumbnailPNGCompression = 9; |
||
97 | public static $defaultFilenameFormat = 'default.%s.jpg'; |
||
98 | public static $newDirectoryPermissions = 0775; |
||
99 | public static $newFilePermissions = 0664; |
||
100 | public static $magicPath = null;//'/usr/share/misc/magic.mgc'; |
||
101 | public static $useFaceDetection = true; |
||
102 | public static $faceDetectionTimeLimit = 10; |
||
103 | |||
104 | public static $mimeHandlers = []; |
||
105 | |||
106 | public static $mimeRewrites = [ |
||
107 | 'image/photoshop' => 'application/psd' |
||
108 | ,'image/x-photoshop' => 'application/psd' |
||
109 | ,'image/psd' => 'application/psd' |
||
110 | ,'application/photoshop' => 'application/psd' |
||
111 | ,'image/vnd.adobe.photoshop' => 'application/psd', |
||
112 | ]; |
||
113 | |||
114 | |||
115 | // privates |
||
116 | protected $_webPath; |
||
117 | protected $_filesystemPath; |
||
118 | protected $_mediaInfo; |
||
119 | |||
120 | |||
121 | // magic methods |
||
122 | public function getValue($name) |
||
123 | { |
||
124 | switch ($name) { |
||
125 | case 'Data': |
||
126 | case 'SummaryData': |
||
127 | case 'JsonTranslation': |
||
128 | return [ |
||
129 | 'ID' => $this->ID |
||
130 | ,'Class' => $this->Class |
||
131 | ,'ContextClass' => $this->ContextClass |
||
132 | ,'ContextID' => $this->ContextID |
||
|
|||
133 | ,'MIMEType' => $this->MIMEType |
||
134 | ,'Width' => $this->Width |
||
135 | ,'Height' => $this->Height |
||
136 | ,'Duration' => $this->Duration, |
||
137 | ]; |
||
138 | |||
139 | case 'Filename': |
||
140 | return $this->getFilename(); |
||
141 | |||
142 | case 'ThumbnailMIMEType': |
||
143 | return $this->MIMEType; |
||
144 | |||
145 | case 'Extension': |
||
146 | throw new Exception('Unable to find extension for mime-type: '.$this->MIMEType); |
||
147 | |||
148 | case 'WebPath': |
||
149 | |||
150 | if (!isset($this->_webPath)) { |
||
151 | $this->_webPath = sprintf( |
||
152 | static::$webPathFormat, |
||
153 | $this->ID |
||
154 | ); |
||
155 | } |
||
156 | |||
157 | return $this->_webPath; |
||
158 | |||
159 | |||
160 | case 'FilesystemPath': |
||
161 | return $this->getFilesystemPath(); |
||
162 | |||
163 | |||
164 | case 'BlankPath': |
||
165 | |||
166 | return static::getBlankPath($this->ContextClass); |
||
167 | |||
168 | |||
169 | default: |
||
170 | return parent::getValue($name); |
||
171 | } |
||
172 | } |
||
173 | |||
174 | |||
175 | // public methods |
||
176 | public static function getBlankThumbnailRequest($class, $width, $height, $fillColor = null) |
||
177 | { |
||
178 | return sprintf( |
||
179 | static::$blankThumbnailRequestFormat, |
||
180 | $class, |
||
181 | $width, |
||
182 | $height, |
||
183 | (isset($fillColor) ? 'x'.$fillColor : '') |
||
184 | ); |
||
185 | } |
||
186 | |||
187 | public function getThumbnailRequest($width, $height = null, $fillColor = null, $cropped = false) |
||
188 | { |
||
189 | return sprintf( |
||
190 | static::$thumbnailRequestFormat, |
||
191 | $this->ID, |
||
192 | $width, |
||
193 | $height ?: $width, |
||
194 | (is_string($fillColor) ? 'x'.$fillColor : '') |
||
195 | ).($cropped ? '/cropped' : ''); |
||
196 | } |
||
197 | |||
198 | public function getImage($sourceFile = null) |
||
199 | { |
||
200 | if (!isset($sourceFile)) { |
||
201 | $sourceFile = $this->FilesystemPath ? $this->FilesystemPath : $this->BlankPath; |
||
202 | } |
||
203 | |||
204 | switch ($this->MIMEType) { |
||
205 | case 'application/psd': |
||
206 | case 'image/tiff': |
||
207 | |||
208 | //Converts PSD to PNG temporarily on the real file system. |
||
209 | $tempFile = tempnam('/tmp', 'media_convert'); |
||
210 | exec("convert -density 100 ".$this->FilesystemPath."[0] -flatten $tempFile.png"); |
||
211 | |||
212 | return imagecreatefrompng("$tempFile.png"); |
||
213 | |||
214 | case 'application/pdf': |
||
215 | |||
216 | return PDF::getImage($sourceFile); |
||
217 | |||
218 | case 'application/postscript': |
||
219 | |||
220 | return imagecreatefromstring(shell_exec("gs -r150 -dEPSCrop -dNOPAUSE -dBATCH -sDEVICE=png48 -sOutputFile=- -q $this->FilesystemPath")); |
||
221 | |||
222 | default: |
||
223 | |||
224 | if (!$fileData = @file_get_contents($sourceFile)) { |
||
225 | throw new Exception('Could not load media source: '.$sourceFile); |
||
226 | } |
||
227 | |||
228 | $image = imagecreatefromstring($fileData); |
||
229 | |||
230 | if ($this->MIMEType == 'image/jpeg' && ($exifData = @exif_read_data($sourceFile)) && !empty($exifData['Orientation'])) { |
||
231 | switch ($exifData['Orientation']) { |
||
232 | case 1: // nothing |
||
233 | break; |
||
234 | case 2: // horizontal flip |
||
235 | imageflip($image, IMG_FLIP_HORIZONTAL); // TODO: need PHP 5.3 compat method |
||
236 | break; |
||
237 | case 3: // 180 rotate left |
||
238 | $image = imagerotate($image, 180, null); |
||
239 | break; |
||
240 | case 4: // vertical flip |
||
241 | imageflip($image, IMG_FLIP_VERTICAL); // TODO: need PHP 5.3 compat method |
||
242 | break; |
||
243 | case 5: // vertical flip + 90 rotate right |
||
244 | imageflip($image, IMG_FLIP_VERTICAL); // TODO: need PHP 5.3 compat method |
||
245 | $image = imagerotate($image, -90, null); |
||
246 | break; |
||
247 | case 6: // 90 rotate right |
||
248 | $image = imagerotate($image, -90, null); |
||
249 | break; |
||
250 | case 7: // horizontal flip + 90 rotate right |
||
251 | imageflip($image, IMG_FLIP_HORIZONTAL); // TODO: need PHP 5.3 compat method |
||
252 | $image = imagerotate($image, -90, null); |
||
253 | break; |
||
254 | case 8: // 90 rotate left |
||
255 | $image = imagerotate($image, 90, null); |
||
256 | break; |
||
257 | } |
||
258 | } |
||
259 | |||
260 | return $image; |
||
261 | } |
||
262 | } |
||
263 | |||
264 | public function getThumbnail($maxWidth, $maxHeight, $fillColor = false, $cropped = false) |
||
265 | { |
||
266 | // init thumbnail path |
||
267 | $thumbFormat = sprintf('%ux%u', $maxWidth, $maxHeight); |
||
268 | |||
269 | if ($fillColor) { |
||
270 | $thumbFormat .= 'x'.strtoupper($fillColor); |
||
271 | } |
||
272 | |||
273 | if ($cropped) { |
||
274 | $thumbFormat .= '.cropped'; |
||
275 | } |
||
276 | |||
277 | $thumbPath = App::$ApplicationPath.'/media/'.$thumbFormat.'/'.$this->Filename; |
||
278 | |||
279 | // look for cached thumbnail |
||
280 | if (!file_exists($thumbPath)) { |
||
281 | // ensure directory exists |
||
282 | $thumbDir = dirname($thumbPath); |
||
283 | if (!is_dir($thumbDir)) { |
||
284 | mkdir($thumbDir, static::$newDirectoryPermissions, true); |
||
285 | } |
||
286 | |||
287 | // create new thumbnail |
||
288 | $this->createThumbnailImage($thumbPath, $maxWidth, $maxHeight, $fillColor, $cropped); |
||
289 | } |
||
290 | |||
291 | |||
292 | // return path |
||
293 | return $thumbPath; |
||
294 | } |
||
295 | |||
296 | public function createThumbnailImage($thumbPath, $maxWidth, $maxHeight, $fillColor = false, $cropped = false) |
||
297 | { |
||
298 | $thumbWidth = $maxWidth; |
||
299 | $thumbHeight = $maxHeight; |
||
300 | |||
301 | if ($cropped && extension_loaded('imagick')) { |
||
302 | $originalTimeLimit = ini_get('max_execution_time'); |
||
303 | |||
304 | // check for existing facedetect job |
||
305 | $cacheKey = "facedetect:{$thumbPath}"; |
||
306 | $faceDetectTime = Cache::fetch($cacheKey); |
||
307 | |||
308 | // a parallel or dead worker is already working on this thumb |
||
309 | if ($faceDetectTime) { |
||
310 | // wait for existing job to finish or timeout |
||
311 | while (time() - $faceDetectTime < static::$faceDetectionTimeLimit) { |
||
312 | sleep(1); |
||
313 | } |
||
314 | |||
315 | // other worker succeeded, we're done |
||
316 | if (file_exists($thumbPath)) { |
||
317 | return true; |
||
318 | } |
||
319 | |||
320 | // disable face detection because it already failed for this thumb |
||
321 | static::$useFaceDetection = false; |
||
322 | } |
||
323 | |||
324 | if (static::$useFaceDetection && extension_loaded('facedetect')) { |
||
325 | Cache::store($cacheKey, time()); |
||
326 | set_time_limit(static::$faceDetectionTimeLimit); |
||
327 | |||
328 | $cropper = new CropFace($this->FilesystemPath); |
||
329 | } else { |
||
330 | $cropper = new stojg\crop\CropEntropy($this->FilesystemPath); |
||
331 | } |
||
332 | |||
333 | $croppedImage = $cropper->resizeAndCrop($thumbWidth, $thumbHeight); |
||
334 | |||
335 | $croppedImage->writeimage($thumbPath); |
||
336 | |||
337 | set_time_limit($originalTimeLimit); |
||
338 | Cache::delete($cacheKey); |
||
339 | } else { |
||
340 | // load source image |
||
341 | $srcImage = $this->getImage(); |
||
342 | $srcWidth = imagesx($srcImage); |
||
343 | $srcHeight = imagesy($srcImage); |
||
344 | |||
345 | // calculate |
||
346 | if ($srcWidth && $srcHeight) { |
||
347 | $widthRatio = ($srcWidth > $maxWidth) ? ($maxWidth / $srcWidth) : 1; |
||
348 | $heightRatio = ($srcHeight > $maxHeight) ? ($maxHeight / $srcHeight) : 1; |
||
349 | |||
350 | // crop width/height to scale size if fill disabled |
||
351 | if ($cropped) { |
||
352 | $ratio = max($widthRatio, $heightRatio); |
||
353 | } else { |
||
354 | $ratio = min($widthRatio, $heightRatio); |
||
355 | } |
||
356 | |||
357 | $scaledWidth = round($srcWidth * $ratio); |
||
358 | $scaledHeight = round($srcHeight * $ratio); |
||
359 | } else { |
||
360 | $scaledWidth = $maxWidth; |
||
361 | $scaledHeight = $maxHeight; |
||
362 | } |
||
363 | |||
364 | if (!$fillColor && !$cropped) { |
||
365 | $thumbWidth = $scaledWidth; |
||
366 | $thumbHeight = $scaledHeight; |
||
367 | } |
||
368 | |||
369 | // create thumbnail images |
||
370 | $image = imagecreatetruecolor($thumbWidth, $thumbHeight); |
||
371 | |||
372 | // paint fill color |
||
373 | if ($fillColor) { |
||
374 | // extract decimal values from hex triplet |
||
375 | $fillColor = sscanf($fillColor, '%2x%2x%2x'); |
||
376 | |||
377 | // convert to color index |
||
378 | $fillColor = imagecolorallocate($image, $fillColor[0], $fillColor[1], $fillColor[2]); |
||
379 | |||
380 | // fill background |
||
381 | imagefill($image, 0, 0, $fillColor); |
||
382 | } elseif (($this->MIMEType == 'image/gif') || ($this->MIMEType == 'image/png')) { |
||
383 | $trans_index = imagecolortransparent($srcImage); |
||
384 | |||
385 | // check if there is a specific transparent color |
||
386 | if ($trans_index >= 0 && $trans_index < imagecolorstotal($srcImage)) { |
||
387 | $trans_color = imagecolorsforindex($srcImage, $trans_index); |
||
388 | |||
389 | // allocate in thumbnail |
||
390 | $trans_index = imagecolorallocate($image, $trans_color['red'], $trans_color['green'], $trans_color['blue']); |
||
391 | |||
392 | // fill background |
||
393 | imagefill($image, 0, 0, $trans_index); |
||
394 | imagecolortransparent($image, $trans_index); |
||
395 | } elseif ($this->MIMEType == 'image/png') { |
||
396 | imagealphablending($image, false); |
||
397 | $trans_color = imagecolorallocatealpha($image, 0, 0, 0, 127); |
||
398 | imagefill($image, 0, 0, $trans_color); |
||
399 | imagesavealpha($image, true); |
||
400 | } |
||
401 | |||
402 | /* |
||
403 | $trans_index = imagecolorallocate($image, 218, 0, 245); |
||
404 | ImageColorTransparent($image, $background); // make the new temp image all transparent |
||
405 | imagealphablending($image, false); // turn off the alpha blending to keep the alpha channel |
||
406 | */ |
||
407 | } |
||
408 | |||
409 | // resize photo to thumbnail |
||
410 | if ($cropped) { |
||
411 | imagecopyresampled( |
||
412 | $image, |
||
413 | $srcImage, |
||
414 | ($thumbWidth - $scaledWidth) / 2, |
||
415 | ($thumbHeight - $scaledHeight) / 2, |
||
416 | 0, |
||
417 | 0, |
||
418 | $scaledWidth, |
||
419 | $scaledHeight, |
||
420 | $srcWidth, |
||
421 | $srcHeight |
||
422 | ); |
||
423 | } else { |
||
424 | imagecopyresampled( |
||
425 | $image, |
||
426 | $srcImage, |
||
427 | round(($thumbWidth - $scaledWidth) / 2), |
||
428 | round(($thumbHeight - $scaledHeight) / 2), |
||
429 | 0, |
||
430 | 0, |
||
431 | $scaledWidth, |
||
432 | $scaledHeight, |
||
433 | $srcWidth, |
||
434 | $srcHeight |
||
435 | ); |
||
436 | } |
||
437 | |||
438 | // save thumbnail to disk |
||
439 | switch ($this->ThumbnailMIMEType) { |
||
440 | case 'image/gif': |
||
441 | imagegif($image, $thumbPath); |
||
442 | break; |
||
443 | |||
444 | case 'image/jpeg': |
||
445 | imagejpeg($image, $thumbPath, static::$thumbnailJPEGCompression); |
||
446 | break; |
||
447 | |||
448 | case 'image/png': |
||
449 | imagepng($image, $thumbPath, static::$thumbnailPNGCompression); |
||
450 | break; |
||
451 | |||
452 | default: |
||
453 | throw new Exception('Unhandled thumbnail format'); |
||
454 | } |
||
455 | } |
||
456 | |||
457 | chmod($thumbPath, static::$newFilePermissions); |
||
458 | return true; |
||
459 | } |
||
460 | |||
461 | /* |
||
462 | public function delete() |
||
463 | { |
||
464 | // remove file |
||
465 | @unlink($this->FilesystemPath); |
||
466 | |||
467 | // delete record |
||
468 | return $this->deleteRecord(); |
||
469 | } |
||
470 | */ |
||
471 | |||
472 | |||
473 | // static methods |
||
474 | public static function createFromUpload($uploadedFile, $fieldValues = []) |
||
475 | { |
||
476 | // handle recieving a field array from $_FILES |
||
477 | if (is_array($uploadedFile)) { |
||
478 | if (isset($uploadedFile['error']) && $uploadedFile['error'] != ERR_UPLOAD_OK) { |
||
479 | return null; |
||
480 | } |
||
481 | |||
482 | if (!empty($uploadedFile['name']) && empty($fieldValues['Caption'])) { |
||
483 | $fieldValues['Caption'] = preg_replace('/\.[^.]+$/', '', $uploadedFile['name']); |
||
484 | } |
||
485 | |||
486 | $uploadedFile = $uploadedFile['tmp_name']; |
||
487 | } |
||
488 | |||
489 | // sanity check |
||
490 | if (!is_uploaded_file($uploadedFile)) { |
||
491 | throw new Exception('Supplied file is not a valid upload'); |
||
492 | } |
||
493 | |||
494 | return static::createFromFile($uploadedFile, $fieldValues); |
||
495 | } |
||
496 | |||
497 | public static function createFromFile($file, $fieldValues = []) |
||
498 | { |
||
499 | try { |
||
500 | // handle url input |
||
501 | if (filter_var($file, FILTER_VALIDATE_URL)) { |
||
502 | $tempName = tempnam('/tmp', 'remote_media'); |
||
503 | copy($file, $tempName); |
||
504 | $file = $tempName; |
||
505 | } |
||
506 | |||
507 | // analyze file |
||
508 | $mediaInfo = static::analyzeFile($file); |
||
509 | |||
510 | // create media object |
||
511 | $Media = $mediaInfo['className']::create($fieldValues); |
||
512 | |||
513 | // init media |
||
514 | $Media->initializeFromAnalysis($mediaInfo); |
||
515 | |||
516 | // save media |
||
517 | $Media->save(); |
||
518 | |||
519 | // write file |
||
520 | $Media->writeFile($file); |
||
521 | |||
522 | return $Media; |
||
523 | } catch (Exception $e) { |
||
524 | \Emergence\Logger::general_warning('Caught exception while processing media upload, aborting upload and returning null', [ |
||
525 | 'exceptionClass' => get_class($e) |
||
526 | ,'exceptionMessage' => $e->getMessage() |
||
527 | ,'exceptionCode' => $e->getCode() |
||
528 | ,'recordData' => $Media ? $Media->getData() : null |
||
529 | ,'mediaInfo' => $mediaInfo, |
||
530 | ]); |
||
531 | // fall through to cleanup below |
||
532 | } |
||
533 | |||
534 | // remove photo record |
||
535 | if ($Media) { |
||
536 | $Media->destroy(); |
||
537 | } |
||
538 | |||
539 | return null; |
||
540 | } |
||
541 | |||
542 | public function initializeFromAnalysis($mediaInfo) |
||
543 | { |
||
544 | $this->MIMEType = $mediaInfo['mimeType']; |
||
545 | $this->Width = $mediaInfo['width']; |
||
546 | $this->Height = $mediaInfo['height']; |
||
547 | $this->Duration = $mediaInfo['duration']; |
||
548 | } |
||
549 | |||
550 | |||
551 | public static function analyzeFile($filename) |
||
552 | { |
||
553 | // DO NOT CALL FROM decendent's override, parent calls child |
||
554 | |||
555 | // check file |
||
556 | if (!is_readable($filename)) { |
||
557 | throw new Exception('Unable to read media file for analysis: "'.$filename.'"'); |
||
558 | } |
||
559 | |||
560 | // get mime type |
||
561 | $finfo = finfo_open(FILEINFO_MIME_TYPE, static::$magicPath); |
||
562 | |||
563 | if (!$finfo || !($mimeType = finfo_file($finfo, $filename))) { |
||
564 | throw new Exception('Unable to load media file info'); |
||
565 | } |
||
566 | |||
567 | finfo_close($finfo); |
||
568 | |||
569 | // dig deeper if only generic mimetype returned |
||
570 | if ($mimeType == 'application/octet-stream') { |
||
571 | $finfo = finfo_open(FILEINFO_NONE, static::$magicPath); |
||
572 | |||
573 | if (!$finfo || !($fileInfo = finfo_file($finfo, $filename))) { |
||
574 | throw new Exception('Unable to load media file info'); |
||
575 | } |
||
576 | |||
577 | finfo_close($finfo); |
||
578 | |||
579 | // detect EPS |
||
580 | if (preg_match('/^DOS EPS/i', $fileInfo)) { |
||
581 | $mimeType = 'application/postscript'; |
||
582 | } |
||
583 | } elseif (array_key_exists($mimeType, static::$mimeRewrites)) { |
||
584 | $mimeType = static::$mimeRewrites[$mimeType]; |
||
585 | } |
||
586 | |||
587 | // condense |
||
588 | |||
589 | |||
590 | // compile mime data |
||
591 | $mediaInfo = [ |
||
592 | 'mimeType' => $mimeType, |
||
593 | ]; |
||
594 | |||
595 | // determine handler |
||
596 | $staticClass = get_called_class(); |
||
597 | |||
598 | if (!isset(static::$mimeHandlers[$mediaInfo['mimeType']]) || $staticClass != 'Media') { |
||
599 | // MICS::dump(static::$mimeHandlers, 'MIME Handlers'); |
||
600 | // throw new Exception('No class registered for mime type "' . $mediaInfo['mimeType'] . '"'); |
||
601 | |||
602 | $mediaInfo['className'] = $staticClass; |
||
603 | } else { |
||
604 | $mediaInfo['className'] = static::$mimeHandlers[$mediaInfo['mimeType']]; |
||
605 | |||
606 | // call registered type's analyzer |
||
607 | $mediaInfo = call_user_func([$mediaInfo['className'], 'analyzeFile'], $filename, $mediaInfo); |
||
608 | } |
||
609 | |||
610 | return $mediaInfo; |
||
611 | } |
||
612 | |||
613 | public static function getBlankPath($contextClass) |
||
614 | { |
||
615 | $path = ['site-root','img',sprintf(static::$defaultFilenameFormat, $contextClass)]; |
||
616 | |||
617 | if ($node = Site::resolvePath($path)) { |
||
618 | return $node->RealPath; |
||
619 | } else { |
||
620 | throw new Exception('Could not load '.implode('/', $path)); |
||
621 | } |
||
622 | } |
||
623 | |||
624 | public static function getBlank($contextClass) |
||
652 | } |
||
653 | |||
654 | public static function getSupportedTypes() |
||
655 | { |
||
656 | return array_unique(array_merge(array_keys(static::$mimeHandlers), array_keys(static::$mimeRewrites))); |
||
657 | } |
||
658 | |||
659 | public function getFilesystemPath($variant = 'original', $filename = null) |
||
660 | { |
||
661 | if ($this->isPhantom) { |
||
662 | return null; |
||
663 | } |
||
664 | |||
665 | return App::$ApplicationPath.'/media/'.$variant.'/'.($filename ?: $this->getFilename($variant)); |
||
666 | } |
||
667 | |||
668 | public function getFilename($variant = 'original') |
||
669 | { |
||
670 | if ($this->isPhantom) { |
||
671 | return 'default.'.$this->Extension; |
||
672 | } |
||
673 | |||
674 | return $this->ID.'.'.$this->Extension; |
||
675 | } |
||
676 | |||
677 | public function getMIMEType($variant = 'original') |
||
680 | } |
||
681 | |||
682 | public function writeFile($sourceFile) |
||
683 | { |
||
684 | $targetDirectory = dirname($this->FilesystemPath); |
||
685 | |||
686 | // create target directory if needed |
||
687 | if (!is_dir($targetDirectory)) { |
||
688 | mkdir($targetDirectory, static::$newDirectoryPermissions, true); |
||
689 | } |
||
690 | |||
691 | // move source file to target path |
||
692 | if (!rename($sourceFile, $this->FilesystemPath)) { |
||
693 | throw new \Exception('Failed to move source file to destination'); |
||
694 | } |
||
695 | |||
696 | // set file permissions |
||
697 | chmod($this->FilesystemPath, static::$newFilePermissions); |
||
698 | } |
||
699 | |||
700 | public function isVariantAvailable($variant) |
||
703 | } |
||
704 | } |
||
705 |