edit.blade.php 51 KB

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