edit.blade.php 65 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966
  1. @extends('layouts.app')
  2. @section('content')
  3. <div class="px-3">
  4. <div class="row">
  5. <div class="col-xl-6">
  6. <h4 class="d-flex align-items-center gap-2">
  7. Рекламация
  8. <span class="badge text-bg-{{ \App\Models\ReclamationStatus::STATUS_COLOR[$reclamation->status_id] ?? 'secondary' }}">
  9. {{ $reclamation->status?->name ?? (\App\Models\Reclamation::STATUS_NAMES[$reclamation->status_id] ?? '-') }}
  10. </span>
  11. </h4>
  12. </div>
  13. <div class="col-xl-6 action-toolbar mb-2">
  14. <a href="{{ $back_url ?? route('reclamations.index', session('gp_reclamations')) }}"
  15. class="btn btn-sm btn-outline-secondary">Назад</a>
  16. @if(hasRole('admin') && !is_null($reclamation->brigadier_id) && !is_null($reclamation->start_work_date))
  17. <button class="btn btn-sm btn-primary" id="createScheduleButton">Перенести в график</button>
  18. @endif
  19. @if(hasRole('admin,manager'))
  20. <a href="{{ route('order.generate-reclamation-pack', ['reclamation' => $reclamation, 'nav' => $nav ?? null]) }}"
  21. class="btn btn-primary btn-sm">Пакет документов рекламации</a>
  22. <a href="{{ route('reclamation.generate-reclamation-payment-pack', ['reclamation' => $reclamation, 'nav' => $nav ?? null]) }}"
  23. class="btn btn-primary btn-sm">Пакет документов на оплату</a>
  24. @endif
  25. @if(hasRole('admin'))
  26. <a href="#" onclick="customConfirm('Удалить рекламацию?', function () { $('form#destroy').submit(); }, 'Подтверждение удаления'); return false;"
  27. class="btn btn-sm btn-danger">Удалить</a>
  28. <form action="{{ route('reclamations.delete', $reclamation) }}" method="post" class="d-none" id="destroy">
  29. @csrf
  30. @method('DELETE')
  31. </form>
  32. @endif
  33. </div>
  34. </div>
  35. <div class="row">
  36. <div class="col-xl-5 border-end mb-3 mb-xl-0">
  37. <form action="{{ route('reclamations.update', $reclamation) }}" method="post">
  38. @csrf
  39. <input type="hidden" id="order_id" name="order_id" value="{{ $reclamation->order_id}}">
  40. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  41. @include('partials.link', ['title' => 'Площадка', 'href' => route('order.show', ['order' => $reclamation->order_id, 'sync_year' => 1, 'nav' => $nav ?? null]), 'text' => $reclamation->order->common_name ?? ''])
  42. @include('partials.select', ['name' => 'status_id', 'title' => 'Статус', 'options' => $statuses, 'value' => $reclamation->status_id ?? old('status_id'), 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  43. @include('partials.select', ['name' => 'user_id', 'title' => 'Менеджер', 'options' => $users, 'value' => $reclamation->user_id ?? old('user_id') ?? auth()->user()->id, 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  44. @include('partials.input', ['name' => 'maf_installation_year', 'title' => 'Год установки МАФ', 'type' => 'text', 'value' => $reclamation->order->year, 'disabled' => true])
  45. @include('partials.input', ['name' => 'create_date', 'title' => 'Дата создания', 'type' => 'date', 'required' => true, 'value' => $reclamation->create_date ?? date('Y-m-d'), 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  46. @include('partials.input', ['name' => 'finish_date', 'title' => 'Дата завершения', 'type' => 'date', 'required' => true, 'value' => $reclamation->finish_date ?? date('Y-m-d', strtotime('+30 days')), 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  47. @include('partials.select', ['name' => 'brigadier_id', 'title' => 'Бригадир', 'options' => $brigadiers, 'value' => $reclamation->brigadier_id ?? old('brigadier_id'), 'disabled' => !hasRole('admin,manager'), 'first_empty' => true, 'classes' => ['update-once']])
  48. @include('partials.input', ['name' => 'start_work_date', 'title' => 'Дата начала работ', 'type' => 'date', 'value' => $reclamation->start_work_date, 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  49. @include('partials.input', ['name' => 'work_days', 'title' => 'Срок работ, дней', 'type' => 'number', 'min' => 1, 'value' => $reclamation->work_days, 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  50. @include('partials.select', ['name' => 'reason', 'title' => 'Причина', 'size' => 6, 'value' => $reclamation->reason ?? '', 'options' => ['Вандализм', 'Гарантия', 'Сервисное обслуживание'], 'key_as_val' => true, 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  51. @include('partials.input', ['name' => 'factory_reclamation_number', 'title' => '№ рекламации на фабрике', 'type' => 'text', 'value' => $reclamation->factory_reclamation_number ?? '', 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  52. @include('partials.textarea', ['name' => 'guarantee', 'title' => 'Гарантии', 'size' => 6, 'value' => $reclamation->guarantee ?? '', 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  53. @include('partials.textarea', ['name' => 'whats_done', 'title' => 'Что сделано', 'size' => 6, 'value' => $reclamation->whats_done ?? '', 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  54. @include('partials.textarea', ['name' => 'comment', 'title' => 'Комментарий', 'size' => 6, 'value' => $reclamation->comment ?? '', 'disabled' => !hasRole('admin,manager'), 'classes' => ['update-once']])
  55. @include('partials.submit', ['name' => 'Сохранить', 'offset' => 5, 'disabled' => !hasRole('admin,manager'), 'back_url' => $back_url ?? route('reclamations.index', session('gp_reclamations'))])
  56. </form>
  57. </div>
  58. <div class="col-xl-7 ">
  59. <div class="col-12 overflow-x-scroll">
  60. <table class="table">
  61. <thead>
  62. <tr>
  63. <th>Картинка</th>
  64. <th>МАФ</th>
  65. <th>Тип</th>
  66. <th>Номер заказа МАФ</th>
  67. <th>RFID</th>
  68. <th>Заводской номер</th>
  69. <th>Дата производства</th>
  70. </tr>
  71. </thead>
  72. <tbody>
  73. @foreach($reclamation->skus as $p)
  74. <tr>
  75. <td>
  76. @if($p->product->image)
  77. <a href="{{ $p->product->image }}" data-toggle="lightbox" data-gallery="maf"
  78. data-size="fullscreen">
  79. <img src="{{ $p->product->image }}" alt="" class="img-thumbnail maf-img">
  80. </a>
  81. @endif
  82. </td>
  83. <td>
  84. @if(hasRole('admin,manager'))
  85. <a href="{{ route('product_sku.show', ['product_sku' => $p, 'nav' => $nav ?? null]) }}">
  86. {{ $p->product->article }}
  87. </a>
  88. <br>
  89. <a class="small" href="{{ route('catalog.show', ['product' => $p->product, 'nav' => $nav ?? null]) }}">каталог</a>
  90. @else
  91. {{ $p->product->article }}
  92. @endif
  93. </td>
  94. <td>{!! $p->product->nomenclature_number !!}</td>
  95. <td>
  96. @if($p->maf_order_id && hasRole('admin'))
  97. <a href="{{ route('maf_order.show', $p->maf_order) }}">{{ $p->maf_order->order_number }}</a>
  98. @else
  99. {{ $p->maf_order?->order_number }}
  100. @endif
  101. </td>
  102. <td>{{ $p->rfid }}</td>
  103. <td>{{ $p->factory_number }}</td>
  104. <td>{{ $p->manufacture_date }}</td>
  105. </tr>
  106. @endforeach
  107. </tbody>
  108. </table>
  109. </div>
  110. <div class="spare-parts">
  111. @php
  112. $activeReservationsCount = $reclamation->activeReservations()->count();
  113. $openShortagesCount = $reclamation->openShortages()->count();
  114. @endphp
  115. <a href="#spare_parts" data-bs-toggle="collapse">
  116. Запчасти ({{ $reclamation->spareParts->count() }})
  117. @if($activeReservationsCount > 0)
  118. <span class="badge bg-warning" title="Активных резервов">{{ $activeReservationsCount }} резерв</span>
  119. @endif
  120. @if($openShortagesCount > 0)
  121. <span class="badge bg-danger" title="Открытых дефицитов">{{ $openShortagesCount }} дефицит</span>
  122. @endif
  123. </a>
  124. <form method="post" action="{{ route('reclamations.update-spare-parts', $reclamation) }}"
  125. class="my-2 collapse" id="spare_parts">
  126. @csrf
  127. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  128. <div class="spare-parts-rows">
  129. @forelse($reclamation->spareParts as $idx => $sp)
  130. <div class="row mb-1 spare-part-row g-1" data-index="{{ $idx }}">
  131. <div class="col-12 col-md-6 position-relative">
  132. <input type="hidden" name="rows[{{ $idx }}][spare_part_id]" value="{{ $sp->id }}" class="spare-part-id">
  133. <input type="text" class="form-control form-control-sm spare-part-search"
  134. value="{{ $sp->article }}@if($sp->note) ({{ $sp->note }})@endif"
  135. placeholder="Введите артикул или название"
  136. autocomplete="off"
  137. @disabled(!hasRole('admin,manager'))>
  138. <div class="spare-part-dropdown"></div>
  139. </div>
  140. <div class="col-4 col-md-2">
  141. <input type="number" name="rows[{{ $idx }}][quantity]" value="{{ $sp->pivot->quantity }}"
  142. min="0" class="form-control form-control-sm text-end"
  143. @disabled(!hasRole('admin,manager'))
  144. placeholder="кол-во">
  145. </div>
  146. <div class="col-5 col-md-3">
  147. <div class="form-check form-check-inline mt-1">
  148. <input type="hidden" name="rows[{{ $idx }}][with_documents]" value="0">
  149. <input type="checkbox" name="rows[{{ $idx }}][with_documents]" value="1"
  150. class="form-check-input" id="with_docs_{{ $idx }}"
  151. @checked($sp->pivot->with_documents)
  152. @disabled(!hasRole('admin,manager'))>
  153. <label class="form-check-label small" for="with_docs_{{ $idx }}">с док.</label>
  154. </div>
  155. </div>
  156. <div class="col-3 col-md-1">
  157. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  158. <i class="bi bi-x-circle"></i>
  159. </span>
  160. </div>
  161. </div>
  162. @empty
  163. <div class="row mb-1 spare-part-row g-1" data-index="0">
  164. <div class="col-12 col-md-6 position-relative">
  165. <input type="hidden" name="rows[0][spare_part_id]" value="" class="spare-part-id">
  166. <input type="text" class="form-control form-control-sm spare-part-search"
  167. value=""
  168. placeholder="Введите артикул или название"
  169. autocomplete="off"
  170. @disabled(!hasRole('admin,manager'))>
  171. <div class="spare-part-dropdown"></div>
  172. </div>
  173. <div class="col-4 col-md-2">
  174. <input type="number" name="rows[0][quantity]" value=""
  175. min="0" class="form-control form-control-sm text-end"
  176. @disabled(!hasRole('admin,manager'))
  177. placeholder="кол-во">
  178. </div>
  179. <div class="col-5 col-md-3">
  180. <div class="form-check form-check-inline mt-1">
  181. <input type="hidden" name="rows[0][with_documents]" value="0">
  182. <input type="checkbox" name="rows[0][with_documents]" value="1"
  183. class="form-check-input" id="with_docs_0"
  184. @disabled(!hasRole('admin,manager'))>
  185. <label class="form-check-label small" for="with_docs_0">с док.</label>
  186. </div>
  187. </div>
  188. <div class="col-3 col-md-1">
  189. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  190. <i class="bi bi-x-circle"></i>
  191. </span>
  192. </div>
  193. </div>
  194. @endforelse
  195. </div>
  196. <div class="row mb-1 spare-part-row spare-part-template g-1 d-none" data-index="__INDEX__">
  197. <div class="col-12 col-md-6 position-relative">
  198. <input type="hidden" name="rows[__INDEX__][spare_part_id]" value="" class="spare-part-id">
  199. <input type="text" class="form-control form-control-sm spare-part-search"
  200. value=""
  201. placeholder="Введите артикул или название"
  202. autocomplete="off"
  203. disabled>
  204. <div class="spare-part-dropdown"></div>
  205. </div>
  206. <div class="col-4 col-md-2">
  207. <input type="number" name="rows[__INDEX__][quantity]" value=""
  208. min="0" class="form-control form-control-sm text-end" disabled
  209. placeholder="кол-во">
  210. </div>
  211. <div class="col-5 col-md-3">
  212. <div class="form-check form-check-inline mt-1">
  213. <input type="hidden" name="rows[__INDEX__][with_documents]" value="0">
  214. <input type="checkbox" name="rows[__INDEX__][with_documents]" value="1"
  215. class="form-check-input" disabled>
  216. <label class="form-check-label small">с док.</label>
  217. </div>
  218. </div>
  219. <div class="col-3 col-md-1">
  220. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  221. <i class="bi bi-x-circle"></i>
  222. </span>
  223. </div>
  224. </div>
  225. <div class="row">
  226. <div class="col-12 text-end">
  227. <span class="btn btn-light btn-sm cursor-pointer" id="add-spare-part-row"
  228. @disabled(!hasRole('admin,manager'))>Добавить строку</span>
  229. <button class="btn btn-primary btn-sm" type="submit" @disabled(!hasRole('admin,manager'))>Сохранить запчасти</button>
  230. </div>
  231. </div>
  232. </form>
  233. {{-- Резервы и дефициты --}}
  234. @php
  235. $reservations = $reclamation->sparePartReservations()
  236. ->with('sparePart', 'sparePartOrder')
  237. ->orderByDesc('created_at')
  238. ->get();
  239. $shortages = $reclamation->sparePartShortages()
  240. ->with('sparePart')
  241. ->orderByDesc('created_at')
  242. ->get();
  243. @endphp
  244. @if($reservations->count() > 0 || $shortages->count() > 0)
  245. <div class="mt-3 p-2 border rounded bg-light">
  246. {{-- Активные резервы --}}
  247. @if($reservations->where('status', 'active')->count() > 0)
  248. <div class="mb-3">
  249. <strong class="text-warning"><i class="bi bi-bookmark-fill"></i> Активные резервы</strong>
  250. <table class="table table-sm table-striped mt-1 mb-0">
  251. <thead>
  252. <tr>
  253. <th>Запчасть</th>
  254. <th class="text-center">Кол-во</th>
  255. <th class="text-center">С док.</th>
  256. <th>Номер заказа</th>
  257. @if(hasRole('admin,manager'))
  258. <th></th>
  259. @endif
  260. </tr>
  261. </thead>
  262. <tbody>
  263. @foreach($reservations->where('status', 'active') as $reservation)
  264. @php
  265. $issueCandidateOrders = \App\Models\SparePartOrder::query()
  266. ->where('spare_part_id', $reservation->spare_part_id)
  267. ->where('with_documents', $reservation->with_documents)
  268. ->where(function ($query) use ($reservation) {
  269. $query->where(function ($inner) {
  270. $inner->where('status', \App\Models\SparePartOrder::STATUS_IN_STOCK)
  271. ->where('available_qty', '>', 0);
  272. })->orWhere('id', $reservation->spare_part_order_id);
  273. })
  274. ->orderBy('created_at')
  275. ->get();
  276. @endphp
  277. <tr>
  278. <td>
  279. @if($reservation->sparePart)
  280. @if(hasRole('admin,manager'))
  281. <a href="{{ route('spare_parts.show', $reservation->sparePart->id) }}">
  282. {{ $reservation->sparePart->article }}
  283. </a>
  284. @else
  285. {{ $reservation->sparePart->article }}
  286. @endif
  287. @else
  288. -
  289. @endif
  290. </td>
  291. <td class="text-center">{{ $reservation->reserved_qty }}</td>
  292. <td class="text-center">
  293. @if($reservation->with_documents)
  294. <i class="bi bi-check-circle text-success"></i>
  295. @else
  296. <i class="bi bi-x-circle text-muted"></i>
  297. @endif
  298. </td>
  299. <td>
  300. @if($reservation->sparePartOrder)
  301. @if(hasRole('admin,manager'))
  302. <a href="{{ route('spare_part_orders.show', $reservation->sparePartOrder->id) }}">
  303. {{ $reservation->sparePartOrder->display_order_number }}
  304. </a>
  305. @else
  306. {{ $reservation->sparePartOrder->display_order_number }}
  307. @endif
  308. @else
  309. -
  310. @endif
  311. </td>
  312. @if(hasRole('admin,manager'))
  313. <td class="text-end">
  314. @if($issueCandidateOrders->count() > 1)
  315. <button type="button"
  316. class="btn btn-sm btn-outline-primary"
  317. title="Изменить заказ резерва"
  318. data-bs-toggle="modal"
  319. data-bs-target="#reassignReservationModal-{{ $reservation->id }}">
  320. <i class="bi bi-pencil"></i>
  321. </button>
  322. <form action="{{ route('spare_part_reservations.issue', $reservation) }}" method="POST" class="d-inline">
  323. @csrf
  324. <button type="submit" class="btn btn-sm btn-success" title="Списать">
  325. <i class="bi bi-check-lg"></i>
  326. </button>
  327. </form>
  328. <div class="modal fade" id="reassignReservationModal-{{ $reservation->id }}" tabindex="-1" aria-labelledby="reassignReservationModalLabel-{{ $reservation->id }}" aria-hidden="true">
  329. <div class="modal-dialog modal-dialog-scrollable">
  330. <div class="modal-content">
  331. <div class="modal-header">
  332. <h1 class="modal-title fs-5" id="reassignReservationModalLabel-{{ $reservation->id }}">Выбор заказа для резерва</h1>
  333. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  334. </div>
  335. <form action="{{ route('spare_part_reservations.reassign', $reservation) }}" method="POST">
  336. @csrf
  337. <div class="modal-body">
  338. <div class="mb-2 small text-muted">
  339. {{ $reservation->sparePart?->article ?? 'Запчасть' }}, {{ $reservation->reserved_qty }} шт.
  340. </div>
  341. @foreach($issueCandidateOrders as $candidateOrder)
  342. @php
  343. $activeReservedForCandidate = \App\Models\Reservation::query()
  344. ->where('spare_part_order_id', $candidateOrder->id)
  345. ->where('status', \App\Models\Reservation::STATUS_ACTIVE)
  346. ->sum('reserved_qty');
  347. $freeCandidateQty = max(0, (int) $candidateOrder->available_qty - (int) $activeReservedForCandidate);
  348. $isCurrentCandidate = (int) $candidateOrder->id === (int) $reservation->spare_part_order_id;
  349. @endphp
  350. <label class="form-check mb-2">
  351. <input class="form-check-input"
  352. type="radio"
  353. name="selected_order_id"
  354. value="{{ $candidateOrder->id }}"
  355. @checked($isCurrentCandidate)>
  356. <span class="form-check-label">
  357. {{ $candidateOrder->display_order_number }}
  358. @if($isCurrentCandidate)
  359. <span class="text-muted">, текущий резерв</span>
  360. @endif
  361. <span class="text-muted">, доступно {{ $freeCandidateQty }}</span>
  362. </span>
  363. </label>
  364. @endforeach
  365. </div>
  366. <div class="modal-footer">
  367. <button type="button" class="btn btn-outline-secondary btn-sm" data-bs-dismiss="modal">Закрыть</button>
  368. <button type="submit" class="btn btn-primary btn-sm">Сохранить резерв</button>
  369. </div>
  370. </form>
  371. </div>
  372. </div>
  373. </div>
  374. @else
  375. <form action="{{ route('spare_part_reservations.issue', $reservation) }}" method="POST" class="d-inline">
  376. @csrf
  377. <button type="submit" class="btn btn-sm btn-success" title="Списать">
  378. <i class="bi bi-check-lg"></i>
  379. </button>
  380. </form>
  381. @endif
  382. <form action="{{ route('spare_part_reservations.cancel', $reservation) }}" method="POST" class="d-inline">
  383. @csrf
  384. <button type="submit" class="btn btn-sm btn-outline-warning" title="Отменить резерв">
  385. <i class="bi bi-x-lg"></i>
  386. </button>
  387. </form>
  388. </td>
  389. @endif
  390. </tr>
  391. @endforeach
  392. </tbody>
  393. </table>
  394. @if(hasRole('admin,manager') && $reservations->where('status', 'active')->count() > 1)
  395. <div class="text-end mt-1">
  396. <form action="{{ route('spare_part_reservations.issue_all', $reclamation->id) }}" method="POST" class="d-inline">
  397. @csrf
  398. <button type="submit" class="btn btn-sm btn-success">Списать все</button>
  399. </form>
  400. <form action="{{ route('spare_part_reservations.cancel_all', $reclamation->id) }}" method="POST" class="d-inline">
  401. @csrf
  402. <button type="submit" class="btn btn-sm btn-outline-warning">Отменить все резервы</button>
  403. </form>
  404. </div>
  405. @endif
  406. </div>
  407. @endif
  408. {{-- Списанные --}}
  409. @if($reservations->where('status', 'issued')->count() > 0)
  410. <div class="mb-3">
  411. <strong class="text-success"><i class="bi bi-check-circle-fill"></i> Списано</strong>
  412. <table class="table table-sm table-success mt-1 mb-0">
  413. <thead>
  414. <tr>
  415. <th>Запчасть</th>
  416. <th class="text-center">Кол-во</th>
  417. <th class="text-center">С док.</th>
  418. <th>Номер заказа</th>
  419. <th>Дата</th>
  420. </tr>
  421. </thead>
  422. <tbody>
  423. @foreach($reservations->where('status', 'issued') as $reservation)
  424. <tr>
  425. <td>
  426. @if($reservation->sparePart)
  427. @if(hasRole('admin,manager'))
  428. <a href="{{ route('spare_parts.show', $reservation->sparePart->id) }}">
  429. {{ $reservation->sparePart->article }}
  430. </a>
  431. @else
  432. {{ $reservation->sparePart->article }}
  433. @endif
  434. @else
  435. -
  436. @endif
  437. </td>
  438. <td class="text-center">{{ $reservation->reserved_qty }}</td>
  439. <td class="text-center">
  440. @if($reservation->with_documents)
  441. <i class="bi bi-check-circle text-success"></i>
  442. @else
  443. <i class="bi bi-x-circle text-muted"></i>
  444. @endif
  445. </td>
  446. <td>
  447. @if($reservation->sparePartOrder)
  448. @if(hasRole('admin,manager'))
  449. <a href="{{ route('spare_part_orders.show', $reservation->sparePartOrder->id) }}">
  450. {{ $reservation->sparePartOrder->display_order_number }}
  451. </a>
  452. @else
  453. {{ $reservation->sparePartOrder->display_order_number }}
  454. @endif
  455. @else
  456. -
  457. @endif
  458. </td>
  459. <td>{{ $reservation->updated_at->format('d.m.Y H:i') }}</td>
  460. </tr>
  461. @endforeach
  462. </tbody>
  463. </table>
  464. </div>
  465. @endif
  466. {{-- Открытые дефициты --}}
  467. @if($shortages->where('status', 'open')->count() > 0)
  468. <div class="mb-3">
  469. <strong class="text-danger"><i class="bi bi-exclamation-triangle-fill"></i> Дефициты (нехватка)</strong>
  470. <table class="table table-sm table-danger mt-1 mb-0">
  471. <thead>
  472. <tr>
  473. <th>Запчасть</th>
  474. <th class="text-center">Требуется</th>
  475. <th class="text-center">Зарезервировано</th>
  476. <th class="text-center">Не хватает</th>
  477. <th class="text-center">С док.</th>
  478. </tr>
  479. </thead>
  480. <tbody>
  481. @foreach($shortages->where('status', 'open') as $shortage)
  482. <tr>
  483. <td>
  484. @if($shortage->sparePart)
  485. @if(hasRole('admin,manager'))
  486. <a href="{{ route('spare_parts.show', $shortage->sparePart->id) }}">
  487. {{ $shortage->sparePart->article }}
  488. </a>
  489. @else
  490. {{ $shortage->sparePart->article }}
  491. @endif
  492. @else
  493. -
  494. @endif
  495. </td>
  496. <td class="text-center">{{ $shortage->required_qty }}</td>
  497. <td class="text-center">{{ $shortage->reserved_qty }}</td>
  498. <td class="text-center fw-bold">{{ $shortage->missing_qty }}</td>
  499. <td class="text-center">
  500. @if($shortage->with_documents)
  501. <i class="bi bi-check-circle text-success"></i>
  502. @else
  503. <i class="bi bi-x-circle text-muted"></i>
  504. @endif
  505. </td>
  506. </tr>
  507. @endforeach
  508. </tbody>
  509. </table>
  510. </div>
  511. @endif
  512. </div>
  513. @endif
  514. </div>
  515. <hr>
  516. <div class="documents">
  517. Документы
  518. @if(hasRole('admin,manager'))
  519. <button class="btn btn-sm text-success" onclick="$('#upl-documents').trigger('click');"><i
  520. class="bi bi-plus-circle-fill"></i> Загрузить
  521. </button>
  522. <form action="{{ route('reclamations.upload-document', $reclamation) }}"
  523. enctype="multipart/form-data" method="post" class="visually-hidden">
  524. @csrf
  525. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  526. <input required type="file" id="upl-documents" onchange="$(this).parent().submit()" multiple
  527. name="document[]" class="form-control form-control-sm">
  528. </form>
  529. @endif
  530. <div class="row my-2 g-1">
  531. @foreach($reclamation->documents as $document)
  532. <div class="col-12">
  533. <a href="{{ $document->link }}" target="_blank">
  534. {{ $document->original_name }}
  535. </a>
  536. @if(hasRole('admin,manager'))
  537. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  538. onclick="customConfirm('Удалить?', function () { $('#document-{{ $document->id }}').submit(); }, 'Подтверждение удаления')"
  539. title="Удалить"></i>
  540. @endif
  541. <form action="{{ route('reclamations.delete-document', [$reclamation, $document]) }}"
  542. method="POST" id="document-{{ $document->id }}" class="visually-hidden">
  543. @csrf
  544. @method('DELETE')
  545. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  546. </form>
  547. </div>
  548. @endforeach
  549. </div>
  550. </div>
  551. <hr>
  552. <div class="acts">
  553. Акты
  554. @if(hasRole('admin,manager,brigadier,warehouse_head'))
  555. <button class="btn btn-sm text-success" onclick="$('#upl-acts').trigger('click');"><i
  556. class="bi bi-plus-circle-fill"></i> Загрузить
  557. </button>
  558. <form action="{{ route('reclamations.upload-act', $reclamation) }}" enctype="multipart/form-data"
  559. method="post" class="visually-hidden">
  560. @csrf
  561. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  562. <input required type="file" id="upl-acts" onchange="$(this).parent().submit()" multiple
  563. name="acts[]" class="form-control form-control-sm">
  564. </form>
  565. @endif
  566. <div class="row my-2 g-1">
  567. @foreach($reclamation->acts as $act)
  568. <div class="col-12">
  569. <a href="{{ $act->link }}" target="_blank">
  570. {{ $act->original_name }}
  571. </a>
  572. @if(hasRole('admin,manager'))
  573. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  574. onclick="customConfirm('Удалить?', function () { $('#act-{{ $act->id }}').submit(); }, 'Подтверждение удаления')"
  575. title="Удалить"></i>
  576. @endif
  577. <form action="{{ route('reclamations.delete-act', [$reclamation, $act]) }}"
  578. method="POST"
  579. id="act-{{ $act->id }}" class="visually-hidden">
  580. @csrf
  581. @method('DELETE')
  582. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  583. </form>
  584. </div>
  585. @endforeach
  586. </div>
  587. </div>
  588. <hr>
  589. <div class="photo_before">
  590. <a href="#photos_before" data-bs-toggle="collapse">Фотографии проблемы
  591. ({{ $reclamation->photos_before->count() }})</a>
  592. @if(hasRole('admin,manager'))
  593. <button class="btn btn-sm text-success" onclick="$('#upl-photo-before').trigger('click');"><i
  594. class="bi bi-plus-circle-fill"></i> Загрузить
  595. </button>
  596. @endif
  597. @if($reclamation->photos_before->count())
  598. <a href="{{ route('reclamation.generate-photos-before-pack', ['reclamation' => $reclamation, 'nav' => $nav ?? null]) }}"
  599. class="btn btn-sm text-primary"><i
  600. class="bi bi-download"></i> Скачать все
  601. </a>
  602. @endif
  603. @if(hasRole('admin,manager'))
  604. <form action="{{ route('reclamations.upload-photo-before', $reclamation) }}"
  605. enctype="multipart/form-data" method="post" class="visually-hidden">
  606. @csrf
  607. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  608. <input required type="file" id="upl-photo-before" onchange="$(this).parent().submit()" multiple
  609. name="photo[]" class="form-control form-control-sm" accept=".jpg,.jpeg,.png,.webp">
  610. </form>
  611. @endif
  612. <div class="row my-2 g-1 collapse" id="photos_before">
  613. @foreach($reclamation->photos_before as $photo)
  614. <div class="col-4">
  615. <a href="{{ $photo->link }}"
  616. data-toggle="lightbox" data-gallery="photos-before" data-size="fullscreen">
  617. <img class="img-thumbnail" src="{{ $photo->link }}" alt="">
  618. </a>
  619. @if(hasRole('admin,manager'))
  620. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer rm-but"
  621. onclick="customConfirm('Удалить фото?', function () { $('#photo-{{ $photo->id }}').submit(); }, 'Подтверждение удаления')"
  622. title="Удалить"></i>
  623. @endif
  624. <form action="{{ route('reclamations.delete-photo-before', [$reclamation, $photo]) }}"
  625. method="POST" id="photo-{{ $photo->id }}" class="visually-hidden">
  626. @csrf
  627. @method('DELETE')
  628. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  629. </form>
  630. </div>
  631. @endforeach
  632. </div>
  633. </div>
  634. <hr>
  635. <div class="photo_after">
  636. <a href="#photos_after" data-bs-toggle="collapse">Фотографии после устранения
  637. ({{ $reclamation->photos_after->count() }})</a>
  638. @if(hasRole('admin,manager,brigadier,warehouse_head'))
  639. <button class="btn btn-sm text-success" onclick="$('#upl-photo-after').trigger('click');"><i
  640. class="bi bi-plus-circle-fill"></i> Загрузить
  641. </button>
  642. @endif
  643. @if($reclamation->photos_after->count())
  644. <a href="{{ route('reclamation.generate-photos-after-pack', ['reclamation' => $reclamation, 'nav' => $nav ?? null]) }}"
  645. class="btn btn-sm text-primary"><i
  646. class="bi bi-download"></i> Скачать все
  647. </a>
  648. @endif
  649. @if(hasRole('admin,manager,brigadier,warehouse_head'))
  650. <form action="{{ route('reclamations.upload-photo-after', $reclamation) }}"
  651. enctype="multipart/form-data" method="post" class="visually-hidden">
  652. @csrf
  653. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  654. <input required type="file" id="upl-photo-after" onchange="$(this).parent().submit()" multiple
  655. name="photo[]" class="form-control form-control-sm" accept=".jpg,.jpeg,.png,.webp">
  656. </form>
  657. @endif
  658. <div class="row my-2 g-1 collapse" id="photos_after">
  659. @foreach($reclamation->photos_after as $photo)
  660. <div class="col-4">
  661. <a href="{{ $photo->link }}"
  662. data-toggle="lightbox" data-gallery="photos-after" data-size="fullscreen">
  663. <img class="img-thumbnail" src="{{ $photo->link }}" alt="">
  664. </a>
  665. @if(hasRole('admin,manager'))
  666. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer rm-but"
  667. onclick="customConfirm('Удалить фото?', function () { $('#photo-{{ $photo->id }}').submit(); }, 'Подтверждение удаления')"
  668. title="Удалить"></i>
  669. @endif
  670. <form action="{{ route('reclamations.delete-photo-after', [$reclamation, $photo]) }}"
  671. method="POST" id="photo-{{ $photo->id }}" class="visually-hidden">
  672. @csrf
  673. @method('DELETE')
  674. <input type="hidden" name="nav" value="{{ $nav ?? '' }}">
  675. </form>
  676. </div>
  677. @endforeach
  678. </div>
  679. </div>
  680. @include('partials.chat', [
  681. 'title' => 'Чат рекламации',
  682. 'messages' => $reclamation->chatMessages,
  683. 'users' => $chatUsers,
  684. 'responsibleUserIds' => $chatResponsibleUserIds,
  685. 'managerUserId' => $chatManagerUserId ?? null,
  686. 'brigadierUserId' => $chatBrigadierUserId ?? null,
  687. 'action' => route('reclamations.chat-messages.store', $reclamation),
  688. 'contextKey' => 'reclamation-' . $reclamation->id,
  689. 'submitLabel' => 'Отправить в чат',
  690. ])
  691. </div>
  692. </div>
  693. </div>
  694. @if(hasRole('admin'))
  695. <!-- Модальное окно графика -->
  696. <div class="modal fade" id="copySchedule" tabindex="-1" aria-labelledby="copyScheduleLabel" aria-hidden="true">
  697. <div class="modal-dialog modal-fullscreen-sm-down modal-lg">
  698. <div class="modal-content">
  699. <div class="modal-header">
  700. <h1 class="modal-title fs-5" id="copyScheduleLabel">Перенести в график монтажей</h1>
  701. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  702. </div>
  703. <div class="modal-body">
  704. <form action="{{ route('schedule.create-from-reclamation') }}" method="post" id="scheduleCreateForm">
  705. @csrf
  706. <div>
  707. <input type="hidden" name="reclamation_id" value="{{ $reclamation->id }}">
  708. {{-- <textarea name="comment" placeholder="Комментарий для графика" class="form-control mb-3"></textarea>--}}
  709. <input type="checkbox" checked="checked" id="sendNotifications" name="send_notifications" value="1" class="form-check-inline">
  710. <label for="sendNotifications" class="form-check-label mb-2">Уведомить менеджера и бригадира</label><br>
  711. <input type="checkbox" id="deleteOldRecords" name="delete_old_records" class="form-check-inline">
  712. <label for="deleteOldRecords" class="form-check-label mb-2">Удалить старые записи в графике для этой рекламации?</label><br>
  713. <button type="submit" class="btn btn-primary btn-sm">Обновить график</button>
  714. </div>
  715. </form>
  716. </div>
  717. </div>
  718. </div>
  719. </div>
  720. @endif
  721. @endsection
  722. @push('scripts')
  723. <script type="module">
  724. @if(hasRole('admin'))
  725. $('#createScheduleButton').on('click', function () {
  726. let myModalSchedule = new bootstrap.Modal(document.getElementById("copySchedule"), {});
  727. myModalSchedule.show();
  728. });
  729. @endif
  730. let sparePartIndex = {{ max($reclamation->spareParts->count(), 1) }};
  731. const searchUrl = '{{ route('spare_parts.search') }}';
  732. // Инициализация автокомплита для поля
  733. function initSparePartAutocomplete($row) {
  734. const $input = $row.find('.spare-part-search');
  735. const $dropdown = $row.find('.spare-part-dropdown');
  736. const $hiddenId = $row.find('.spare-part-id');
  737. let currentFocus = -1;
  738. let searchTimeout;
  739. $input.on('input', function() {
  740. const query = $(this).val().trim();
  741. clearTimeout(searchTimeout);
  742. // Сбрасываем ID при изменении текста
  743. $hiddenId.val('');
  744. if (query.length >= 1) {
  745. searchTimeout = setTimeout(function() {
  746. $.get(searchUrl, { query: query }).done(function(data) {
  747. showDropdown(data, query);
  748. });
  749. }, 200);
  750. } else {
  751. $dropdown.hide();
  752. }
  753. });
  754. function showDropdown(items, query) {
  755. $dropdown.empty();
  756. currentFocus = -1;
  757. if (items.length === 0) {
  758. $dropdown.hide();
  759. return;
  760. }
  761. items.forEach(function(item) {
  762. const $item = $('<div class="sp-item"></div>');
  763. const highlightedArticle = highlightMatch(item.article, query);
  764. const highlightedNote = item.note ? highlightMatch(item.note, query) : '';
  765. $item.html('<span class="sp-article">' + highlightedArticle + '</span>' +
  766. (item.note ? ' <span class="sp-used">(' + highlightedNote + ')</span>' : ''));
  767. $item.data('id', item.id);
  768. $item.data('article', item.article);
  769. $item.data('note', item.note || '');
  770. $item.on('click', function() {
  771. selectItem($(this));
  772. });
  773. $dropdown.append($item);
  774. });
  775. $dropdown.show();
  776. }
  777. function highlightMatch(text, query) {
  778. if (!text || !query) return text || '';
  779. const regex = new RegExp('(' + escapeRegex(query) + ')', 'gi');
  780. return text.replace(regex, '<strong>$1</strong>');
  781. }
  782. function escapeRegex(str) {
  783. return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
  784. }
  785. function selectItem($item) {
  786. const article = $item.data('article');
  787. const note = $item.data('note');
  788. const displayText = article + (note ? ' (' + note + ')' : '');
  789. $input.val(displayText);
  790. $hiddenId.val($item.data('id'));
  791. $dropdown.hide();
  792. }
  793. // Навигация клавиатурой
  794. $input.on('keydown', function(e) {
  795. const $items = $dropdown.find('.sp-item');
  796. if (e.keyCode === 40) { // Стрелка вниз
  797. e.preventDefault();
  798. currentFocus++;
  799. if (currentFocus >= $items.length) currentFocus = 0;
  800. setActive($items);
  801. } else if (e.keyCode === 38) { // Стрелка вверх
  802. e.preventDefault();
  803. currentFocus--;
  804. if (currentFocus < 0) currentFocus = $items.length - 1;
  805. setActive($items);
  806. } else if (e.keyCode === 13) { // Enter
  807. e.preventDefault();
  808. if (currentFocus > -1 && $items.length > 0) {
  809. selectItem($items.eq(currentFocus));
  810. }
  811. } else if (e.keyCode === 27) { // Escape
  812. $dropdown.hide();
  813. }
  814. });
  815. function setActive($items) {
  816. $items.removeClass('active');
  817. if (currentFocus >= 0 && currentFocus < $items.length) {
  818. $items.eq(currentFocus).addClass('active');
  819. }
  820. }
  821. // Закрытие при клике вне
  822. $(document).on('click', function(e) {
  823. if (!$(e.target).closest($row).length) {
  824. $dropdown.hide();
  825. }
  826. });
  827. }
  828. $('.chat-notification-type').on('change', function () {
  829. const target = $(this).data('chat-target');
  830. $(target).toggleClass('d-none', $(this).val() !== 'user');
  831. }).trigger('change');
  832. $('.update-once').on('focus', function () {
  833. $(this).data('previous-value', $(this).val());
  834. });
  835. $('.update-once').on('change', function () {
  836. let $field = $(this);
  837. let value = $field.val();
  838. let fieldName = $field.attr('name');
  839. let previousValue = $field.data('previous-value');
  840. $.post(
  841. '{{ route('reclamations.update', $reclamation) }}',
  842. {
  843. '_token': '{{ csrf_token() }}',
  844. order_id: '{{ $reclamation->order_id }}',
  845. nav: '{{ $nav ?? '' }}',
  846. user_id: $('#user_id').val(),
  847. status_id: $('#status_id').val(),
  848. create_date: $('#create_date').val(),
  849. finish_date: $('#finish_date').val(),
  850. brigadier_id: $('#brigadier_id').val(),
  851. start_work_date: $('#start_work_date').val(),
  852. work_days: $('#work_days').val(),
  853. reason: $('#reason').val(),
  854. factory_reclamation_number: $('#factory_reclamation_number').val(),
  855. guarantee: $('#guarantee').val(),
  856. whats_done: $('#whats_done').val(),
  857. comment: $('#comment').val(),
  858. [fieldName]: value
  859. },
  860. function () {
  861. $field.data('previous-value', value);
  862. $('.alerts').append(
  863. '<div class="main-alert alert alert-success" role="alert">Рекламация обновлена!</div>'
  864. );
  865. setTimeout(function () {
  866. $('.main-alert').fadeTo(2000, 500).slideUp(500, function () {
  867. $('.main-alert').slideUp(500);
  868. });
  869. }, 3000);
  870. }
  871. ).fail(function (xhr) {
  872. if (previousValue !== undefined) {
  873. $field.val(previousValue);
  874. }
  875. let errorText = xhr.responseJSON?.message || 'Не удалось обновить рекламацию!';
  876. $('.alerts').append(
  877. '<div class="main-alert alert alert-danger" role="alert">' + errorText + '</div>'
  878. );
  879. setTimeout(function () {
  880. $('.main-alert').fadeTo(2000, 500).slideUp(500, function () {
  881. $('.main-alert').slideUp(500);
  882. });
  883. }, 3000);
  884. });
  885. });
  886. // Инициализация для существующих строк
  887. $('.spare-part-row').not('.spare-part-template').each(function() {
  888. initSparePartAutocomplete($(this));
  889. });
  890. // Добавление новой строки запчасти
  891. $('#add-spare-part-row').on('click', function() {
  892. let template = $('.spare-part-template').first();
  893. let newRow = template.clone();
  894. newRow.removeClass('spare-part-template d-none');
  895. newRow.attr('data-index', sparePartIndex);
  896. // Заменяем __INDEX__ на реальный индекс во всех name атрибутах
  897. newRow.find('[name]').each(function() {
  898. let name = $(this).attr('name');
  899. $(this).attr('name', name.replace('__INDEX__', sparePartIndex));
  900. $(this).prop('disabled', false);
  901. });
  902. // Сбрасываем значения и разблокируем поле поиска
  903. newRow.find('.spare-part-search').val('').prop('disabled', false);
  904. newRow.find('.spare-part-id').val('');
  905. newRow.find('input[type="number"]').val('');
  906. newRow.find('input[type="checkbox"]').prop('checked', false);
  907. $('.spare-parts-rows').append(newRow);
  908. // Инициализируем автокомплит для новой строки
  909. initSparePartAutocomplete(newRow);
  910. sparePartIndex++;
  911. });
  912. // Удаление строки
  913. $(document).on('click', '.remove-spare-part-row', function() {
  914. $(this).closest('.spare-part-row').remove();
  915. });
  916. </script>
  917. @endpush