| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Uploads; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Illuminate\Http\UploadedFile; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FaviconHandler | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |     protected string $path; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     public function __construct( | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |         protected ImageResizer $imageResizer, | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     ) { | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |         $this->path = public_path('favicon.ico'); | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Save the given UploadedFile instance as the application favicon. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function saveForUploadedImage(UploadedFile $file): void | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |         if (!is_writeable($this->path)) { | 
					
						
							| 
									
										
										
										
											2023-02-09 21:24:43 +08:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |         $imageData = file_get_contents($file->getRealPath()); | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |         $pngData = $this->imageResizer->resizeImageData($imageData, 32, 32, false, 'png'); | 
					
						
							|  |  |  |         $icoData = $this->pngToIco($pngData, 32, 32); | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |         file_put_contents($this->path, $icoData); | 
					
						
							| 
									
										
										
										
											2023-02-09 21:24:43 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Restore the original favicon image. | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |      * Returned boolean indicates if the copy occurred. | 
					
						
							| 
									
										
										
										
											2023-02-09 21:24:43 +08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |     public function restoreOriginal(): bool | 
					
						
							| 
									
										
										
										
											2023-02-09 21:24:43 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |         $permissionItem = file_exists($this->path) ? $this->path : dirname($this->path); | 
					
						
							|  |  |  |         if (!is_writeable($permissionItem)) { | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2023-02-09 21:24:43 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |         return copy($this->getOriginalPath(), $this->path); | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Restore the original favicon image if no favicon image is already in use. | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |      * Returns a boolean to indicate if the file exists. | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |     public function restoreOriginalIfNotExists(): bool | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |         if (file_exists($this->path)) { | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $this->restoreOriginal(); | 
					
						
							| 
									
										
										
										
											2023-02-10 04:57:35 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the path to the favicon file. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getPath(): string | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->path; | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 05:16:27 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the path of the original favicon copy. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getOriginalPath(): string | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return public_path('icon.ico'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2023-02-10 01:47:33 +08:00
										 |  |  |      * Convert PNG image data to ICO file format. | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |      * Built following the file format info from Wikipedia: | 
					
						
							|  |  |  |      * https://en.wikipedia.org/wiki/ICO_(file_format) | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |     protected function pngToIco(string $pngData, int $width, int $height): string | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |         // ICO header
 | 
					
						
							|  |  |  |         $header = pack('v', 0x00); // Reserved. Must always be 0
 | 
					
						
							|  |  |  |         $header .= pack('v', 0x01); // Specifies ico image
 | 
					
						
							|  |  |  |         $header .= pack('v', 0x01); // Specifies number of images
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // ICO Image Directory
 | 
					
						
							|  |  |  |         $entry = hex2bin(dechex($width)); // Image width
 | 
					
						
							|  |  |  |         $entry .= hex2bin(dechex($height)); // Image height
 | 
					
						
							|  |  |  |         $entry .= "\0"; // Color palette, typically 0
 | 
					
						
							|  |  |  |         $entry .= "\0"; // Reserved
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Color planes, Appears to remain 1 for bmp image data
 | 
					
						
							|  |  |  |         $entry .= pack('v', 0x01); | 
					
						
							|  |  |  |         // Bits per pixel, can range from 1 to 32. From testing conversion
 | 
					
						
							| 
									
										
										
										
											2023-02-09 23:14:41 +08:00
										 |  |  |         // via intervention from png typically provides this as 24.
 | 
					
						
							| 
									
										
										
										
											2023-02-10 01:47:33 +08:00
										 |  |  |         $entry .= pack('v', 0x00); | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |         // Size of the image data in bytes
 | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |         $entry .= pack('V', strlen($pngData)); | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |         // Offset of the bmp data from file start
 | 
					
						
							|  |  |  |         $entry .= pack('V', strlen($header) + strlen($entry) + 4); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Join & return the combined parts of the ICO image data
 | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |         return $header . $entry . $pngData; | 
					
						
							| 
									
										
										
										
											2023-02-09 07:06:42 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | } |