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.

189 wiersze
5.7KB

  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 updateCardAgeSettings(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. $showCardAge = (string) $request->input('show_card_age', '0') === '1';
  73. $cardAgeWarningDays = max(0, (int) $request->input('card_age_warning_days', 0));
  74. $this->lanes()->updateCardAgeSettings(
  75. $id,
  76. $showCardAge,
  77. $cardAgeWarningDays,
  78. date('Y-m-d H:i:s'),
  79. AuthService::getCurrentUsername()
  80. );
  81. return $this->json([
  82. 'ok' => true,
  83. 'show_card_age' => $showCardAge,
  84. 'card_age_warning_days' => $cardAgeWarningDays,
  85. ]);
  86. }
  87. public function export(int $id): mixed
  88. {
  89. if (!AuthService::isLoggedIn()) {
  90. return $this->redirect('/auth/login');
  91. }
  92. $row = $this->lanes()->find($id);
  93. if ($row === null) {
  94. return new \Core\Response('Not found', 404);
  95. }
  96. $lane = \App\Models\SwimLane::fromRow($row);
  97. $cards = $this->cards()->findBySwimLaneId($id);
  98. $out = fopen('php://memory', 'wb');
  99. fputcsv($out, ['Job #', 'Job Name', 'Customer', 'Delivery Date', 'Quantity', 'Notes']);
  100. foreach ($cards as $card) {
  101. fputcsv($out, [
  102. $card->jobNumber,
  103. $card->jobName,
  104. $card->customerName,
  105. $card->deliveryDate ?? '',
  106. $card->quantity ?? '',
  107. $card->notes,
  108. ]);
  109. }
  110. rewind($out);
  111. $csv = (string) stream_get_contents($out);
  112. fclose($out);
  113. $slug = trim((string) preg_replace('/[^a-z0-9]+/i', '-', $lane->name), '-');
  114. $filename = ($slug ?: 'lane') . '-' . date('Y-m-d') . '.csv';
  115. return new \Core\Response($csv, 200, [
  116. 'Content-Type' => 'text/csv; charset=utf-8',
  117. 'Content-Disposition' => 'attachment; filename="' . $filename . '"',
  118. ]);
  119. }
  120. public function destroy(int $id): mixed
  121. {
  122. if (!AuthService::isLoggedIn()) {
  123. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  124. }
  125. $this->cards()->deleteBySwimLaneId($id);
  126. $this->lanes()->delete($id);
  127. return $this->json(['ok' => true]);
  128. }
  129. public function reorder(): mixed
  130. {
  131. if (!AuthService::isLoggedIn()) {
  132. return $this->json(['ok' => false, 'error' => 'Unauthorized'], 401);
  133. }
  134. $raw = file_get_contents('php://input');
  135. $items = json_decode((string) $raw, true);
  136. if (!is_array($items)) {
  137. return $this->json(['ok' => false, 'error' => 'Invalid JSON payload']);
  138. }
  139. $now = date('Y-m-d H:i:s');
  140. $username = AuthService::getCurrentUsername();
  141. foreach ($items as $item) {
  142. $laneId = (int) ($item['id'] ?? 0);
  143. $position = (int) ($item['position'] ?? 0);
  144. if ($laneId > 0) {
  145. $this->lanes()->updatePosition($laneId, $position, $now, $username);
  146. }
  147. }
  148. return $this->json(['ok' => true]);
  149. }
  150. }

Powered by TurnKey Linux.