edit.blade.php 49 KB

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