ReclamationController.php 20 KB

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