Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.

207 wiersze
6.3KB

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Controllers;
  4. use App\Models\SwimLane;
  5. use App\Repositories\CardRepository;
  6. use App\Repositories\SwimLaneRepository;
  7. use App\Services\AuthService;
  8. use Core\Controller;
  9. use Core\Request;
  10. class SwimLanesController extends Controller
  11. {
  12. private function lanes(): SwimLaneRepository
  13. {
  14. return new SwimLaneRepository(database());
  15. }
  16. private function cards(): CardRepository
  17. {
  18. return new CardRepository(database());
  19. }
  20. public function store(Request $request): mixed
  21. {
  22. if (!AuthService::isLoggedIn()) {
  23. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  24. }
  25. $boardId = (int) $request->input('board_id', 0);
  26. $name = trim((string) $request->input('name', ''));
  27. if ($boardId === 0 || $name === '') {
  28. return $this->json(['ok' => false, 'error' => 'board_id and name are required']);
  29. }
  30. $now = date('Y-m-d H:i:s');
  31. $username = AuthService::getCurrentUsername();
  32. $lane = new SwimLane();
  33. $lane->boardId = $boardId;
  34. $lane->name = $name;
  35. $lane->position = $this->lanes()->maxPosition($boardId) + 1;
  36. $lane->createdAt = $now;
  37. $lane->createdBy = $username;
  38. $lane->updatedAt = $now;
  39. $lane->updatedBy = $username;
  40. $this->lanes()->insert($lane);
  41. return $this->json(['ok' => true, 'id' => $lane->id, 'name' => $lane->name, 'position' => $lane->position]);
  42. }
  43. public function update(Request $request, int $id): mixed
  44. {
  45. if (!AuthService::isLoggedIn()) {
  46. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  47. }
  48. $name = trim((string) $request->input('name', ''));
  49. if ($name === '') {
  50. return $this->json(['ok' => false, 'error' => 'name is required']);
  51. }
  52. $row = $this->lanes()->find($id);
  53. if ($row === null) {
  54. return $this->json(['ok' => false, 'error' => 'Not found'], 404);
  55. }
  56. $lane = SwimLane::fromRow($row);
  57. $lane->name = $name;
  58. $lane->updatedAt = date('Y-m-d H:i:s');
  59. $lane->updatedBy = AuthService::getCurrentUsername();
  60. $this->lanes()->update($lane);
  61. return $this->json(['ok' => true]);
  62. }
  63. public function toggleExport(Request $request, int $id): mixed
  64. {
  65. if (!AuthService::isLoggedIn()) {
  66. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  67. }
  68. $row = $this->lanes()->find($id);
  69. if ($row === null) {
  70. return $this->json(['ok' => false, 'error' => 'Not found'], 404);
  71. }
  72. $show = (string) $request->input('show_export_button', '0') === '1';
  73. $this->lanes()->updateShowExportButton($id, $show, date('Y-m-d H:i:s'), AuthService::getCurrentUsername());
  74. return $this->json(['ok' => true, 'show_export_button' => $show]);
  75. }
  76. public function updateCardAgeSettings(Request $request, int $id): mixed
  77. {
  78. if (!AuthService::isLoggedIn()) {
  79. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  80. }
  81. $row = $this->lanes()->find($id);
  82. if ($row === null) {
  83. return $this->json(['ok' => false, 'error' => 'Not found'], 404);
  84. }
  85. $showCardAge = (string) $request->input('show_card_age', '0') === '1';
  86. $cardAgeWarningDays = max(0, (int) $request->input('card_age_warning_days', 0));
  87. $this->lanes()->updateCardAgeSettings(
  88. $id,
  89. $showCardAge,
  90. $cardAgeWarningDays,
  91. date('Y-m-d H:i:s'),
  92. AuthService::getCurrentUsername()
  93. );
  94. return $this->json([
  95. 'ok' => true,
  96. 'show_card_age' => $showCardAge,
  97. 'card_age_warning_days' => $cardAgeWarningDays,
  98. ]);
  99. }
  100. public function export(int $id): mixed
  101. {
  102. if (!AuthService::isLoggedIn()) {
  103. return $this->redirect('/auth/login');
  104. }
  105. $row = $this->lanes()->find($id);
  106. if ($row === null) {
  107. return new \Core\Response('Not found', 404);
  108. }
  109. $lane = \App\Models\SwimLane::fromRow($row);
  110. $cards = $this->cards()->findBySwimLaneId($id);
  111. $out = fopen('php://memory', 'wb');
  112. fputcsv($out, ['Job #', 'Job Name', 'Customer', 'Delivery Date', 'Quantity', 'Notes'], ',', '"', '\\');
  113. foreach ($cards as $card) {
  114. fputcsv($out, [
  115. $card->jobNumber,
  116. $card->jobName,
  117. $card->customerName,
  118. $card->deliveryDate ?? '',
  119. $card->quantity ?? '',
  120. $card->notes,
  121. ], ',', '"', '\\');
  122. }
  123. rewind($out);
  124. $csv = (string) stream_get_contents($out);
  125. fclose($out);
  126. $slug = trim((string) preg_replace('/[^a-z0-9]+/i', '-', $lane->name), '-');
  127. $filename = ($slug ?: 'lane') . '-' . date('Y-m-d') . '.csv';
  128. return new \Core\Response($csv, 200, [
  129. 'Content-Type' => 'text/csv; charset=utf-8',
  130. 'Content-Disposition' => 'attachment; filename="' . $filename . '"',
  131. ]);
  132. }
  133. public function destroy(int $id): mixed
  134. {
  135. if (!AuthService::isLoggedIn()) {
  136. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  137. }
  138. $this->cards()->deleteBySwimLaneId($id);
  139. $this->lanes()->delete($id);
  140. return $this->json(['ok' => true]);
  141. }
  142. public function reorder(): mixed
  143. {
  144. if (!AuthService::isLoggedIn()) {
  145. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  146. }
  147. $raw = file_get_contents('php://input');
  148. $items = json_decode((string) $raw, true);
  149. if (!is_array($items)) {
  150. return $this->json(['ok' => false, 'error' => 'Invalid JSON payload']);
  151. }
  152. $now = date('Y-m-d H:i:s');
  153. $username = AuthService::getCurrentUsername();
  154. foreach ($items as $item) {
  155. $laneId = (int) ($item['id'] ?? 0);
  156. $position = (int) ($item['position'] ?? 0);
  157. if ($laneId > 0) {
  158. $this->lanes()->updatePosition($laneId, $position, $now, $username);
  159. }
  160. }
  161. return $this->json(['ok' => true]);
  162. }
  163. }

Powered by TurnKey Linux.