upload_image.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. <?php
  2. /**
  3. * Image Upload Handler
  4. * Handles image uploads for the WYSIWYG editor
  5. */
  6. // Start session
  7. if (session_status() === PHP_SESSION_NONE) {
  8. session_start();
  9. }
  10. require_once '../includes/config.php';
  11. require_once '../includes/database.php';
  12. require_once '../includes/auth.php';
  13. // Require authentication
  14. $auth = new Auth();
  15. $auth->requireAuth();
  16. // Set headers for JSON response
  17. header('Content-Type: application/json');
  18. // Create uploads directory if it doesn't exist
  19. $uploadsDir = '../uploads/images';
  20. if (!file_exists($uploadsDir)) {
  21. mkdir($uploadsDir, 0755, true);
  22. }
  23. // Handle file upload
  24. if ($_SERVER['REQUEST_METHOD'] === 'POST' && isset($_FILES['images'])) {
  25. $uploadedFiles = [];
  26. $errors = [];
  27. // Handle multiple files
  28. $files = $_FILES['images'];
  29. $fileCount = count($files['name']);
  30. for ($i = 0; $i < $fileCount; $i++) {
  31. if ($files['error'][$i] === UPLOAD_ERR_OK) {
  32. $file = [
  33. 'name' => $files['name'][$i],
  34. 'type' => $files['type'][$i],
  35. 'tmp_name' => $files['tmp_name'][$i],
  36. 'error' => $files['error'][$i],
  37. 'size' => $files['size'][$i]
  38. ];
  39. // Validate file
  40. $validation = validateImageFile($file);
  41. if ($validation['valid']) {
  42. // Generate unique filename
  43. $extension = pathinfo($file['name'], PATHINFO_EXTENSION);
  44. $filename = uniqid() . '.' . $extension;
  45. $filepath = $uploadsDir . '/' . $filename;
  46. // Move file to uploads directory
  47. if (move_uploaded_file($file['tmp_name'], $filepath)) {
  48. // Save to database
  49. $imageId = saveImageToDatabase($filename, $file['name'], $file['size']);
  50. if ($imageId) {
  51. $uploadedFiles[] = [
  52. 'id' => $imageId,
  53. 'filename' => $filename,
  54. 'original_name' => $file['name'],
  55. 'size' => $file['size'],
  56. 'url' => '/uploads/images/' . $filename,
  57. 'thumbnail_url' => '/uploads/images/' . $filename
  58. ];
  59. } else {
  60. $errors[] = 'Failed to save ' . $file['name'] . ' to database';
  61. unlink($filepath); // Remove uploaded file
  62. }
  63. } else {
  64. $errors[] = 'Failed to upload ' . $file['name'];
  65. }
  66. } else {
  67. $errors[] = $validation['error'];
  68. }
  69. } else {
  70. $errors[] = 'Error uploading file: ' . getUploadErrorMessage($files['error'][$i]);
  71. }
  72. }
  73. echo json_encode([
  74. 'success' => empty($errors),
  75. 'files' => $uploadedFiles,
  76. 'errors' => $errors
  77. ]);
  78. exit;
  79. }
  80. // Handle GET request to fetch existing images
  81. if ($_SERVER['REQUEST_METHOD'] === 'GET') {
  82. $images = getExistingImages();
  83. echo json_encode([
  84. 'success' => true,
  85. 'images' => $images
  86. ]);
  87. exit;
  88. }
  89. // Handle DELETE request to remove image
  90. if ($_SERVER['REQUEST_METHOD'] === 'DELETE' && isset($_GET['id'])) {
  91. $imageId = (int)$_GET['id'];
  92. $success = deleteImage($imageId);
  93. echo json_encode([
  94. 'success' => $success
  95. ]);
  96. exit;
  97. }
  98. echo json_encode(['success' => false, 'error' => 'Invalid request']);
  99. /**
  100. * Validate uploaded image file
  101. */
  102. function validateImageFile($file) {
  103. // Check file size (5MB max)
  104. $maxSize = 5 * 1024 * 1024; // 5MB
  105. if ($file['size'] > $maxSize) {
  106. return ['valid' => false, 'error' => 'File too large. Maximum size is 5MB'];
  107. }
  108. // Check file type
  109. $allowedTypes = ['image/jpeg', 'image/png', 'image/gif', 'image/webp'];
  110. if (!in_array($file['type'], $allowedTypes)) {
  111. return ['valid' => false, 'error' => 'Invalid file type. Only JPG, PNG, GIF, and WebP are allowed'];
  112. }
  113. // Check file extension
  114. $allowedExtensions = ['jpg', 'jpeg', 'png', 'gif', 'webp'];
  115. $extension = strtolower(pathinfo($file['name'], PATHINFO_EXTENSION));
  116. if (!in_array($extension, $allowedExtensions)) {
  117. return ['valid' => false, 'error' => 'Invalid file extension'];
  118. }
  119. return ['valid' => true];
  120. }
  121. /**
  122. * Save image information to database
  123. */
  124. function saveImageToDatabase($filename, $originalName, $size) {
  125. $db = Database::getInstance();
  126. try {
  127. $db->insert('images', [
  128. 'filename' => $filename,
  129. 'original_name' => $originalName,
  130. 'file_size' => $size,
  131. 'uploaded_at' => date('Y-m-d H:i:s'),
  132. 'uploaded_by' => $_SESSION['user_id'] ?? 1
  133. ]);
  134. return $db->lastInsertId();
  135. } catch (Exception $e) {
  136. error_log('Failed to save image to database: ' . $e->getMessage());
  137. return false;
  138. }
  139. }
  140. /**
  141. * Get existing images from database
  142. */
  143. function getExistingImages() {
  144. $db = Database::getInstance();
  145. try {
  146. $images = $db->fetchAll("SELECT * FROM images ORDER BY uploaded_at DESC");
  147. $result = [];
  148. foreach ($images as $image) {
  149. $result[] = [
  150. 'id' => $image['id'],
  151. 'filename' => $image['filename'],
  152. 'original_name' => $image['original_name'],
  153. 'size' => $image['file_size'],
  154. 'url' => '/uploads/images/' . $image['filename'],
  155. 'thumbnail_url' => '/uploads/images/' . $image['filename'],
  156. 'uploaded_at' => $image['uploaded_at']
  157. ];
  158. }
  159. return $result;
  160. } catch (Exception $e) {
  161. error_log('Failed to fetch images: ' . $e->getMessage());
  162. return [];
  163. }
  164. }
  165. /**
  166. * Delete image from database and filesystem
  167. */
  168. function deleteImage($imageId) {
  169. $db = Database::getInstance();
  170. try {
  171. // Get image info
  172. $image = $db->fetch("SELECT * FROM images WHERE id = ?", [$imageId]);
  173. if (!$image) {
  174. return false;
  175. }
  176. // Delete from database
  177. $db->delete('images', 'id = ?', [$imageId]);
  178. // Delete file from filesystem
  179. $filepath = '../uploads/images/' . $image['filename'];
  180. if (file_exists($filepath)) {
  181. unlink($filepath);
  182. }
  183. return true;
  184. } catch (Exception $e) {
  185. error_log('Failed to delete image: ' . $e->getMessage());
  186. return false;
  187. }
  188. }
  189. /**
  190. * Get upload error message
  191. */
  192. function getUploadErrorMessage($errorCode) {
  193. switch ($errorCode) {
  194. case UPLOAD_ERR_INI_SIZE:
  195. return 'The uploaded file exceeds the upload_max_filesize directive in php.ini';
  196. case UPLOAD_ERR_FORM_SIZE:
  197. return 'The uploaded file exceeds the MAX_FILE_SIZE directive that was specified in the HTML form';
  198. case UPLOAD_ERR_PARTIAL:
  199. return 'The uploaded file was only partially uploaded';
  200. case UPLOAD_ERR_NO_FILE:
  201. return 'No file was uploaded';
  202. case UPLOAD_ERR_NO_TMP_DIR:
  203. return 'Missing a temporary folder';
  204. case UPLOAD_ERR_CANT_WRITE:
  205. return 'Failed to write file to disk';
  206. case UPLOAD_ERR_EXTENSION:
  207. return 'A PHP extension stopped the file upload';
  208. default:
  209. return 'Unknown upload error';
  210. }
  211. }
  212. ?>