ReclamationController.php 22 KB

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