ReclamationController.php 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Http\Requests\CreateReclamationRequest;
  4. use App\Http\Requests\StoreReclamationDetailsRequest;
  5. use App\Http\Requests\StoreReclamationRequest;
  6. use App\Http\Requests\StoreReclamationSparePartsRequest;
  7. use App\Jobs\GenerateFilesPack;
  8. use App\Jobs\GenerateReclamationPack;
  9. use App\Models\File;
  10. use App\Models\Order;
  11. use App\Models\Reclamation;
  12. use App\Models\ReclamationDetail;
  13. use App\Models\ReclamationStatus;
  14. use App\Models\ReclamationView;
  15. use App\Models\Role;
  16. use App\Models\User;
  17. use App\Services\FileService;
  18. use App\Services\SparePartReservationService;
  19. use Illuminate\Http\Request;
  20. use Illuminate\Support\Carbon;
  21. use Illuminate\Support\Facades\Storage;
  22. class ReclamationController extends Controller
  23. {
  24. protected array $data = [
  25. 'active' => 'reclamations',
  26. 'title' => 'Рекламации',
  27. 'id' => 'reclamations',
  28. 'header' => [
  29. 'id' => 'ID',
  30. 'user_name' => 'Менеджер',
  31. 'status_name' => 'Статус',
  32. 'district_name' => 'Округ',
  33. 'area_name' => 'Район',
  34. 'object_address' => 'Адрес объекта',
  35. 'maf_installation_year' => 'Год установки МАФ',
  36. 'create_date' => 'Дата создания',
  37. 'finish_date' => 'Дата завершения',
  38. 'start_work_date' => 'Дата начала работ',
  39. 'work_days' => 'Срок работ, дней',
  40. 'brigadier_name' => 'Бригадир',
  41. 'reason' => 'Причина',
  42. 'guarantee' => 'Гарантии',
  43. 'whats_done' => 'Что сделано',
  44. 'comment' => 'Комментарий',
  45. ],
  46. 'searchFields' => [
  47. 'reason',
  48. 'guarantee',
  49. 'whats_done',
  50. 'comment',
  51. ],
  52. 'ranges' => [],
  53. ];
  54. public function __construct()
  55. {
  56. $this->data['users'] = User::query()->whereIn('role', [Role::MANAGER, Role::ADMIN])->get()->pluck('name', 'id');
  57. $this->data['statuses'] = ReclamationStatus::query()->get()->pluck('name', 'id');
  58. }
  59. public function index(Request $request)
  60. {
  61. session(['gp_reclamations' => $request->all()]);
  62. $model = new ReclamationView();
  63. // fill filters
  64. $this->createFilters($model, 'user_name', 'status_name');
  65. $this->createDateFilters($model, 'create_date', 'finish_date');
  66. $q = $model::query();
  67. $this->acceptFilters($q, $request);
  68. $this->acceptSearch($q, $request);
  69. $this->setSortAndOrderBy($model, $request);
  70. $q->orderBy($this->data['sortBy'], $this->data['orderBy']);
  71. $this->data['reclamations'] = $q->paginate(session('per_page', config('pagination.per_page')))->withQueryString();
  72. return view('reclamations.index', $this->data);
  73. }
  74. public function create(CreateReclamationRequest $request, Order $order)
  75. {
  76. $reclamation = Reclamation::query()->create([
  77. 'order_id' => $order->id,
  78. 'user_id' => $request->user()->id,
  79. 'status_id' => Reclamation::STATUS_NEW,
  80. 'create_date' => Carbon::now(),
  81. 'finish_date' => Carbon::now()->addDays(30),
  82. ]);
  83. $skus = $request->validated('skus');
  84. $reclamation->skus()->attach($skus);
  85. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => url()->previous()]);
  86. }
  87. public function show(Request $request, Reclamation $reclamation)
  88. {
  89. $this->data['brigadiers'] = User::query()->where('role', Role::BRIGADIER)->get()->pluck('name', 'id');
  90. $this->data['reclamation'] = $reclamation;
  91. $this->data['previous_url'] = $request->get('previous_url');
  92. return view('reclamations.edit', $this->data);
  93. }
  94. public function update(StoreReclamationRequest $request, Reclamation $reclamation)
  95. {
  96. $data = $request->validated();
  97. $reclamation->update($data);
  98. return redirect()->route('reclamations.show', $reclamation->id);
  99. }
  100. public function delete(Reclamation $reclamation)
  101. {
  102. $reclamation->delete();
  103. return redirect()->route('reclamations.index');
  104. }
  105. public function uploadPhotoBefore(Request $request, Reclamation $reclamation, FileService $fileService)
  106. {
  107. $data = $request->validate([
  108. 'photo.*' => 'mimes:jpeg,jpg,png|max:8192',
  109. ]);
  110. $f = [];
  111. foreach ($data['photo'] as $photo) {
  112. $f[] = $fileService->saveUploadedFile('reclamations/' . $reclamation->id . '/photo_before', $photo);
  113. }
  114. $reclamation->photos_before()->syncWithoutDetaching($f);
  115. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  116. }
  117. public function uploadPhotoAfter(Request $request, Reclamation $reclamation, FileService $fileService)
  118. {
  119. $data = $request->validate([
  120. 'photo.*' => 'mimes:jpeg,jpg,png|max:8192',
  121. ]);
  122. $f = [];
  123. foreach ($data['photo'] as $photo) {
  124. $f[] = $fileService->saveUploadedFile('reclamations/' . $reclamation->id . '/photo_after', $photo);
  125. }
  126. $reclamation->photos_after()->syncWithoutDetaching($f);
  127. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  128. }
  129. public function deletePhotoBefore(Request $request, Reclamation $reclamation, File $file, FileService $fileService)
  130. {
  131. $reclamation->photos_before()->detach($file);
  132. Storage::disk('public')->delete($file->path);
  133. $file->delete();
  134. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  135. }
  136. public function deletePhotoAfter(Request $request, Reclamation $reclamation, File $file, FileService $fileService)
  137. {
  138. $reclamation->photos_after()->detach($file);
  139. Storage::disk('public')->delete($file->path);
  140. $file->delete();
  141. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  142. }
  143. public function uploadDocument(Request $request, Reclamation $reclamation, FileService $fileService)
  144. {
  145. $data = $request->validate([
  146. 'document.*' => 'file',
  147. ]);
  148. $f = [];
  149. $i = 0;
  150. foreach ($data['document'] as $document) {
  151. if($i++ >= 5) break;
  152. $f[] = $fileService->saveUploadedFile('reclamations/' . $reclamation->id . '/document', $document);
  153. }
  154. $reclamation->documents()->syncWithoutDetaching($f);
  155. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  156. }
  157. public function deleteDocument(Request $request, Reclamation $reclamation, File $file)
  158. {
  159. $reclamation->documents()->detach($file);
  160. Storage::disk('public')->delete($file->path);
  161. $file->delete();
  162. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  163. }
  164. public function uploadAct(Request $request, Reclamation $reclamation, FileService $fileService)
  165. {
  166. $data = $request->validate([
  167. 'acts.*' => 'file',
  168. ]);
  169. $f = [];
  170. $i = 0;
  171. foreach ($data['acts'] as $document) {
  172. if($i++ >= 5) break;
  173. $f[] = $fileService->saveUploadedFile('reclamations/' . $reclamation->id . '/act', $document);
  174. }
  175. $reclamation->acts()->syncWithoutDetaching($f);
  176. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  177. }
  178. public function deleteAct(Request $request, Reclamation $reclamation, File $file)
  179. {
  180. $reclamation->acts()->detach($file);
  181. Storage::disk('public')->delete($file->path);
  182. $file->delete();
  183. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  184. }
  185. public function updateDetails(StoreReclamationDetailsRequest $request, Reclamation $reclamation)
  186. {
  187. $names = $request->validated('name');
  188. $quantity = $request->validated('quantity');
  189. $withDocuments = $request->validated('with_documents');
  190. $reservationService = app(SparePartReservationService::class);
  191. foreach ($names as $key => $name) {
  192. if (!$name) continue;
  193. if ((int)$quantity[$key] >= 1) {
  194. // Проверяем, является ли это запчастью
  195. $sparePart = \App\Models\SparePart::where('article', $name)->first();
  196. if ($sparePart) {
  197. // Резервирование вместо прямого списания
  198. $withDocs = isset($withDocuments[$key]) && $withDocuments[$key];
  199. $qty = (int)$quantity[$key];
  200. // Получаем текущее количество в pivot
  201. $currentPivot = $reclamation->spareParts()->find($sparePart->id);
  202. $currentQty = $currentPivot?->pivot->quantity ?? 0;
  203. $diff = $qty - $currentQty;
  204. if ($diff > 0) {
  205. // Нужно зарезервировать дополнительное количество
  206. $result = $reservationService->reserve(
  207. $sparePart->id,
  208. $diff,
  209. $withDocs,
  210. $reclamation->id
  211. );
  212. // Обновляем pivot с учётом результата
  213. $reclamation->spareParts()->syncWithoutDetaching([
  214. $sparePart->id => [
  215. 'quantity' => $qty,
  216. 'with_documents' => $withDocs,
  217. 'status' => $result->isFullyReserved() ? 'reserved' : 'pending',
  218. 'reserved_qty' => $currentQty + $result->reserved,
  219. ]
  220. ]);
  221. } elseif ($diff < 0) {
  222. // Уменьшение — отменяем часть резерва
  223. $reservationService->adjustReservation(
  224. $reclamation->id,
  225. $sparePart->id,
  226. $withDocs,
  227. $qty
  228. );
  229. $reclamation->spareParts()->syncWithoutDetaching([
  230. $sparePart->id => [
  231. 'quantity' => $qty,
  232. 'with_documents' => $withDocs,
  233. 'reserved_qty' => $qty,
  234. ]
  235. ]);
  236. } else {
  237. // Количество не изменилось, возможно изменился with_documents
  238. $reclamation->spareParts()->syncWithoutDetaching([
  239. $sparePart->id => [
  240. 'quantity' => $qty,
  241. 'with_documents' => $withDocs,
  242. ]
  243. ]);
  244. }
  245. } else {
  246. // Обычная деталь
  247. ReclamationDetail::query()->updateOrCreate(
  248. ['reclamation_id' => $reclamation->id, 'name' => $name],
  249. ['quantity' => $quantity[$key]]
  250. );
  251. }
  252. } else {
  253. // Удаление
  254. // Проверяем, является ли это запчастью — отменяем резервы
  255. $sparePartToRemove = \App\Models\SparePart::where('article', $name)->first();
  256. if ($sparePartToRemove) {
  257. // Отменяем все резервы для этой запчасти в рекламации
  258. $reservationService->cancelForReclamation(
  259. $reclamation->id,
  260. $sparePartToRemove->id
  261. );
  262. // Удаляем связь
  263. $reclamation->spareParts()->detach($sparePartToRemove->id);
  264. } else {
  265. // Обычная деталь
  266. ReclamationDetail::query()
  267. ->where('reclamation_id', $reclamation->id)
  268. ->where('name', $name)
  269. ->delete();
  270. }
  271. }
  272. }
  273. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  274. }
  275. public function updateSpareParts(StoreReclamationSparePartsRequest $request, Reclamation $reclamation)
  276. {
  277. $rows = $request->validated('rows') ?? [];
  278. $reservationService = app(SparePartReservationService::class);
  279. // Получаем текущие привязки для сравнения
  280. $currentSpareParts = $reclamation->spareParts->keyBy('id');
  281. // Определяем какие запчасти были удалены
  282. $newSparePartIds = collect($rows)->pluck('spare_part_id')->filter()->toArray();
  283. $removedIds = $currentSpareParts->keys()->diff($newSparePartIds);
  284. // Отменяем резервы для удалённых запчастей
  285. foreach ($removedIds as $removedId) {
  286. $current = $currentSpareParts->get($removedId);
  287. if ($current) {
  288. $reservationService->cancelForReclamation(
  289. $reclamation->id,
  290. $removedId,
  291. $current->pivot->with_documents
  292. );
  293. }
  294. }
  295. // Собираем новые привязки
  296. $newSpareParts = [];
  297. foreach ($rows as $row) {
  298. $sparePartId = $row['spare_part_id'] ?? null;
  299. if (empty($sparePartId)) continue;
  300. $quantity = (int)($row['quantity'] ?? 0);
  301. if ($quantity < 1) continue;
  302. $withDocs = !empty($row['with_documents']) && $row['with_documents'] != '0';
  303. // Проверяем, изменилось ли количество
  304. $currentQty = $currentSpareParts->get($sparePartId)?->pivot->quantity ?? 0;
  305. $currentReserved = $currentSpareParts->get($sparePartId)?->pivot->reserved_qty ?? 0;
  306. $diff = $quantity - $currentQty;
  307. $status = 'pending';
  308. $reservedQty = $currentReserved;
  309. if ($diff > 0) {
  310. // Нужно зарезервировать дополнительное количество
  311. $result = $reservationService->reserve(
  312. $sparePartId,
  313. $diff,
  314. $withDocs,
  315. $reclamation->id
  316. );
  317. $reservedQty = $currentReserved + $result->reserved;
  318. $status = $reservedQty >= $quantity ? 'reserved' : 'pending';
  319. } elseif ($diff < 0) {
  320. // Уменьшение — отменяем часть резерва
  321. $reservationService->adjustReservation(
  322. $reclamation->id,
  323. $sparePartId,
  324. $withDocs,
  325. $quantity
  326. );
  327. $reservedQty = $quantity;
  328. $status = 'reserved';
  329. } else {
  330. // Количество не изменилось
  331. $status = $currentReserved >= $quantity ? 'reserved' : 'pending';
  332. }
  333. $newSpareParts[$sparePartId] = [
  334. 'quantity' => $quantity,
  335. 'with_documents' => $withDocs,
  336. 'status' => $status,
  337. 'reserved_qty' => $reservedQty,
  338. ];
  339. }
  340. // Синхронизируем (заменяем все старые привязки новыми)
  341. $reclamation->spareParts()->sync($newSpareParts);
  342. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')]);
  343. }
  344. public function generateReclamationPack(Request $request, Reclamation $reclamation)
  345. {
  346. GenerateReclamationPack::dispatch($reclamation, auth()->user()->id);
  347. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')])
  348. ->with(['success' => 'Задача генерации документов создана!']);
  349. }
  350. public function generatePhotosBeforePack(Request $request, Reclamation $reclamation)
  351. {
  352. GenerateFilesPack::dispatch($reclamation, $reclamation->photos_before, auth()->user()->id, 'Фото проблемы');
  353. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')])
  354. ->with(['success' => 'Задача архивации создана!']);
  355. }
  356. public function generatePhotosAfterPack(Request $request, Reclamation $reclamation)
  357. {
  358. GenerateFilesPack::dispatch($reclamation, $reclamation->photos_after, auth()->user()->id, 'Фото после');
  359. return redirect()->route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => $request->get('previous_url')])
  360. ->with(['success' => 'Задача архивации создана!']); }
  361. }