edit.blade.php 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599
  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. @endif
  16. @if(hasRole('admin'))
  17. <a href="#" onclick="if(confirm('Удалить рекламацию?')) $('form#destroy').submit();"
  18. class="btn btn-sm mb-1 btn-danger">Удалить</a>
  19. <form action="{{ route('reclamations.delete', $reclamation) }}" method="post" class="d-none" id="destroy">
  20. @csrf
  21. @method('DELETE')
  22. </form>
  23. @endif
  24. </div>
  25. </div>
  26. <div class="row">
  27. <div class="col-xl-5 border-end mb-3 mb-xl-0">
  28. <form action="{{ route('reclamations.update', $reclamation) }}" method="post">
  29. @csrf
  30. <input type="hidden" id="order_id" name="order_id" value="{{ $reclamation->order_id}}">
  31. <input type="hidden" name="previous_url" value="{{ $previous_url ?? '' }}">
  32. @include('partials.input', ['name' => 'order_name', 'title' => 'Площадка', 'disabled' => true, 'value' => $reclamation->order->common_name ?? ''])
  33. @include('partials.select', ['name' => 'status_id', 'title' => 'Статус', 'options' => $statuses, 'value' => $reclamation->status_id ?? old('status_id'), 'disabled' => !hasRole('admin,manager')])
  34. @include('partials.select', ['name' => 'user_id', 'title' => 'Менеджер', 'options' => $users, 'value' => $reclamation->user_id ?? old('user_id') ?? auth()->user()->id, 'disabled' => !hasRole('admin,manager')])
  35. @include('partials.input', ['name' => 'create_date', 'title' => 'Дата создания', 'type' => 'date', 'required' => true, 'value' => $reclamation->create_date ?? date('Y-m-d'), 'disabled' => !hasRole('admin,manager')])
  36. @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')])
  37. @include('partials.select', ['name' => 'brigadier_id', 'title' => 'Бригадир', 'options' => $brigadiers, 'value' => $reclamation->brigadier_id ?? old('brigadier_id'), 'disabled' => !hasRole('admin,manager'), 'first_empty' => true])
  38. @include('partials.input', ['name' => 'start_work_date', 'title' => 'Дата начала работ', 'type' => 'date', 'value' => $reclamation->start_work_date, 'disabled' => !hasRole('admin,manager')])
  39. @include('partials.input', ['name' => 'work_days', 'title' => 'Срок работ, дней', 'type' => 'number', 'min' => 1, 'value' => $reclamation->work_days, 'disabled' => !hasRole('admin,manager')])
  40. @include('partials.select', ['name' => 'reason', 'title' => 'Причина', 'size' => 6, 'value' => $reclamation->reason ?? '', 'options' => ['Вандализм', 'Гарантия', 'Сервисное обслуживание'], 'key_as_val' => true, 'disabled' => !hasRole('admin,manager')])
  41. @include('partials.textarea', ['name' => 'guarantee', 'title' => 'Гарантии', 'size' => 6, 'value' => $reclamation->guarantee ?? '', 'disabled' => !hasRole('admin,manager')])
  42. @include('partials.textarea', ['name' => 'whats_done', 'title' => 'Что сделано', 'size' => 6, 'value' => $reclamation->whats_done ?? '', 'disabled' => !hasRole('admin,manager')])
  43. @include('partials.textarea', ['name' => 'comment', 'title' => 'Комментарий', 'size' => 6, 'value' => $reclamation->comment ?? '', 'disabled' => !hasRole('admin,manager')])
  44. @include('partials.submit', ['name' => 'Сохранить', 'offset' => 5, 'disabled' => !hasRole('admin,manager'), 'backurl' => route('reclamations.index', session('gp_reclamations'))])
  45. </form>
  46. </div>
  47. <div class="col-xl-7 ">
  48. <div class="col-12 overflow-x-scroll">
  49. <table class="table">
  50. <thead>
  51. <tr>
  52. <th>Картинка</th>
  53. <th>МАФ</th>
  54. <th>Тип</th>
  55. <th>Номер заказа МАФ</th>
  56. <th>RFID</th>
  57. <th>Заводской номер</th>
  58. <th>Дата производства</th>
  59. </tr>
  60. </thead>
  61. <tbody>
  62. @foreach($reclamation->skus as $p)
  63. <tr>
  64. <td>
  65. @if($p->product->image)
  66. <a href="{{ $p->product->image }}" data-toggle="lightbox" data-gallery="maf"
  67. data-size="fullscreen">
  68. <img src="{{ $p->product->image }}" alt="" class="img-thumbnail maf-img">
  69. </a>
  70. @endif
  71. </td>
  72. <td>
  73. @if(hasRole('admin,manager'))
  74. <a href="{{ route('product_sku.show', $p) }}">
  75. {!! $p->product->article !!}
  76. </a>
  77. <br>
  78. <a class="small" href="{{ route('catalog.show', $p->product) }}">каталог</a>
  79. @endif
  80. </td>
  81. <td>{!! $p->product->nomenclature_number !!}</td>
  82. <td>
  83. @if($p->maf_order_id && hasRole('admin,manager'))
  84. <a href="{{ route('maf_order.show', $p->maf_order) }}">{{ $p->maf_order->order_number }}</a>
  85. @endif
  86. </td>
  87. <td>{{ $p->rfid }}</td>
  88. <td>{{ $p->factory_number }}</td>
  89. <td>{{ $p->manufacture_date }}</td>
  90. </tr>
  91. @endforeach
  92. </tbody>
  93. </table>
  94. </div>
  95. <div class="spare-parts">
  96. <a href="#spare_parts" data-bs-toggle="collapse">Запчасти ({{ $reclamation->spareParts->count() }})</a>
  97. <form method="post" action="{{ route('reclamations.update-spare-parts', $reclamation) }}"
  98. class="my-2 collapse" id="spare_parts">
  99. @csrf
  100. <div class="spare-parts-rows">
  101. @forelse($reclamation->spareParts as $idx => $sp)
  102. <div class="row mb-1 spare-part-row" data-index="{{ $idx }}">
  103. <div class="col-6 position-relative">
  104. <input type="hidden" name="rows[{{ $idx }}][spare_part_id]" value="{{ $sp->id }}" class="spare-part-id">
  105. <input type="text" class="form-control form-control-sm spare-part-search"
  106. value="{{ $sp->article }}@if($sp->used_in_maf) ({{ $sp->used_in_maf }})@endif"
  107. placeholder="Введите артикул или название"
  108. autocomplete="off"
  109. @disabled(!hasRole('admin,manager'))>
  110. <div class="spare-part-dropdown"></div>
  111. </div>
  112. <div class="col-2">
  113. <input type="number" name="rows[{{ $idx }}][quantity]" value="{{ $sp->pivot->quantity }}"
  114. min="0" class="form-control form-control-sm text-end"
  115. @disabled(!hasRole('admin,manager'))
  116. placeholder="кол-во">
  117. </div>
  118. <div class="col-3">
  119. <div class="form-check form-check-inline mt-1">
  120. <input type="hidden" name="rows[{{ $idx }}][with_documents]" value="0">
  121. <input type="checkbox" name="rows[{{ $idx }}][with_documents]" value="1"
  122. class="form-check-input" id="with_docs_{{ $idx }}"
  123. @checked($sp->pivot->with_documents)
  124. @disabled(!hasRole('admin,manager'))>
  125. <label class="form-check-label small" for="with_docs_{{ $idx }}">с док.</label>
  126. </div>
  127. </div>
  128. <div class="col-1">
  129. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  130. <i class="bi bi-x-circle"></i>
  131. </span>
  132. </div>
  133. </div>
  134. @empty
  135. <div class="row mb-1 spare-part-row" data-index="0">
  136. <div class="col-6 position-relative">
  137. <input type="hidden" name="rows[0][spare_part_id]" value="" class="spare-part-id">
  138. <input type="text" class="form-control form-control-sm spare-part-search"
  139. value=""
  140. placeholder="Введите артикул или название"
  141. autocomplete="off"
  142. @disabled(!hasRole('admin,manager'))>
  143. <div class="spare-part-dropdown"></div>
  144. </div>
  145. <div class="col-2">
  146. <input type="number" name="rows[0][quantity]" value=""
  147. min="0" class="form-control form-control-sm text-end"
  148. @disabled(!hasRole('admin,manager'))
  149. placeholder="кол-во">
  150. </div>
  151. <div class="col-3">
  152. <div class="form-check form-check-inline mt-1">
  153. <input type="hidden" name="rows[0][with_documents]" value="0">
  154. <input type="checkbox" name="rows[0][with_documents]" value="1"
  155. class="form-check-input" id="with_docs_0"
  156. @disabled(!hasRole('admin,manager'))>
  157. <label class="form-check-label small" for="with_docs_0">с док.</label>
  158. </div>
  159. </div>
  160. <div class="col-1">
  161. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  162. <i class="bi bi-x-circle"></i>
  163. </span>
  164. </div>
  165. </div>
  166. @endforelse
  167. </div>
  168. <div class="row mb-1 spare-part-row spare-part-template d-none" data-index="__INDEX__">
  169. <div class="col-6 position-relative">
  170. <input type="hidden" name="rows[__INDEX__][spare_part_id]" value="" class="spare-part-id">
  171. <input type="text" class="form-control form-control-sm spare-part-search"
  172. value=""
  173. placeholder="Введите артикул или название"
  174. autocomplete="off"
  175. disabled>
  176. <div class="spare-part-dropdown"></div>
  177. </div>
  178. <div class="col-2">
  179. <input type="number" name="rows[__INDEX__][quantity]" value=""
  180. min="0" class="form-control form-control-sm text-end" disabled
  181. placeholder="кол-во">
  182. </div>
  183. <div class="col-3">
  184. <div class="form-check form-check-inline mt-1">
  185. <input type="hidden" name="rows[__INDEX__][with_documents]" value="0">
  186. <input type="checkbox" name="rows[__INDEX__][with_documents]" value="1"
  187. class="form-check-input" disabled>
  188. <label class="form-check-label small">с док.</label>
  189. </div>
  190. </div>
  191. <div class="col-1">
  192. <span class="btn btn-sm text-danger remove-spare-part-row" title="Удалить строку">
  193. <i class="bi bi-x-circle"></i>
  194. </span>
  195. </div>
  196. </div>
  197. <div class="row">
  198. <div class="col-12 text-end">
  199. <span class="btn btn-light btn-sm cursor-pointer" id="add-spare-part-row"
  200. @disabled(!hasRole('admin,manager'))>Добавить строку</span>
  201. <button class="btn btn-primary btn-sm" type="submit" @disabled(!hasRole('admin,manager'))>Сохранить запчасти</button>
  202. </div>
  203. </div>
  204. @if($errors->any())
  205. @dump($errors->all())
  206. @endif
  207. </form>
  208. </div>
  209. <hr>
  210. <div class="documents">
  211. Документы
  212. @if(hasRole('admin,manager'))
  213. <button class="btn btn-sm text-success" onclick="$('#upl-documents').trigger('click');"><i
  214. class="bi bi-plus-circle-fill"></i> Загрузить
  215. </button>
  216. <form action="{{ route('reclamations.upload-document', $reclamation) }}"
  217. enctype="multipart/form-data" method="post" class="visually-hidden">
  218. @csrf
  219. <input required type="file" id="upl-documents" onchange="$(this).parent().submit()" multiple
  220. name="document[]" class="form-control form-control-sm">
  221. </form>
  222. @endif
  223. <div class="row my-2 g-1">
  224. @foreach($reclamation->documents as $document)
  225. <div class="col-12">
  226. <a href="{{ $document->link }}" target="_blank">
  227. {{ $document->original_name }}
  228. </a>
  229. @if(hasRole('admin'))
  230. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  231. onclick="if(confirm('Удалить?')) $('#document-{{ $document->id }}').submit()"
  232. title="Удалить"></i>
  233. @endif
  234. <form action="{{ route('reclamations.delete-document', [$reclamation, $document]) }}"
  235. method="POST" id="document-{{ $document->id }}" class="visually-hidden">
  236. @csrf
  237. @method('DELETE')
  238. </form>
  239. </div>
  240. @endforeach
  241. </div>
  242. </div>
  243. <hr>
  244. <div class="acts">
  245. Акты
  246. @if(hasRole('admin,manager'))
  247. <button class="btn btn-sm text-success" onclick="$('#upl-acts').trigger('click');"><i
  248. class="bi bi-plus-circle-fill"></i> Загрузить
  249. </button>
  250. <form action="{{ route('reclamations.upload-act', $reclamation) }}" enctype="multipart/form-data"
  251. method="post" class="visually-hidden">
  252. @csrf
  253. <input required type="file" id="upl-acts" onchange="$(this).parent().submit()" multiple
  254. name="acts[]" class="form-control form-control-sm">
  255. </form>
  256. @endif
  257. <div class="row my-2 g-1">
  258. @foreach($reclamation->acts as $act)
  259. <div class="col-12">
  260. <a href="{{ $act->link }}" target="_blank">
  261. {{ $act->original_name }}
  262. </a>
  263. @if(hasRole('admin'))
  264. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  265. onclick="if(confirm('Удалить?')) $('#act-{{ $act->id }}').submit()"
  266. title="Удалить"></i>
  267. <form action="{{ route('reclamations.delete-act', [$reclamation, $act]) }}"
  268. method="POST"
  269. id="act-{{ $act->id }}" class="visually-hidden">
  270. @csrf
  271. @method('DELETE')
  272. </form>
  273. @endif
  274. </div>
  275. @endforeach
  276. </div>
  277. </div>
  278. <hr>
  279. <div class="photo_before">
  280. <a href="#photos_before" data-bs-toggle="collapse">Фотографии проблемы
  281. ({{ $reclamation->photos_before->count() }})</a>
  282. <button class="btn btn-sm text-success" onclick="$('#upl-photo-before').trigger('click');"><i
  283. class="bi bi-plus-circle-fill"></i> Загрузить
  284. </button>
  285. @if($reclamation->photos_before->count())
  286. <a href="{{ route('reclamation.generate-photos-before-pack', $reclamation) }}"
  287. class="btn btn-sm text-primary"><i
  288. class="bi bi-download"></i> Скачать все
  289. </a>
  290. @endif
  291. <form action="{{ route('reclamations.upload-photo-before', $reclamation) }}"
  292. enctype="multipart/form-data" method="post" class="visually-hidden">
  293. @csrf
  294. <input required type="file" id="upl-photo-before" onchange="$(this).parent().submit()" multiple
  295. name="photo[]" class="form-control form-control-sm" accept=".jpg,.jpeg,.png">
  296. </form>
  297. <div class="row my-2 g-1 collapse" id="photos_before">
  298. @foreach($reclamation->photos_before as $photo)
  299. <div class="col-4">
  300. <a href="{{ $photo->link }}"
  301. data-toggle="lightbox" data-gallery="photos-before" data-size="fullscreen">
  302. <img class="img-thumbnail" src="{{ $photo->link }}" alt="">
  303. </a>
  304. @if(hasRole('admin'))
  305. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer rm-but"
  306. onclick="if(confirm('Удалить фото?')) $('#photo-{{ $photo->id }}').submit()"
  307. title="Удалить"></i>
  308. @endif
  309. <form action="{{ route('reclamations.delete-photo-before', [$reclamation, $photo]) }}"
  310. method="POST" id="photo-{{ $photo->id }}" class="visually-hidden">
  311. @csrf
  312. @method('DELETE')
  313. </form>
  314. </div>
  315. @endforeach
  316. </div>
  317. </div>
  318. <hr>
  319. <div class="photo_after">
  320. <a href="#photos_after" data-bs-toggle="collapse">Фотографии после устранения
  321. ({{ $reclamation->photos_after->count() }})</a>
  322. <button class="btn btn-sm text-success" onclick="$('#upl-photo-after').trigger('click');"><i
  323. class="bi bi-plus-circle-fill"></i> Загрузить
  324. </button>
  325. @if($reclamation->photos_after->count())
  326. <a href="{{ route('reclamation.generate-photos-after-pack', $reclamation) }}"
  327. class="btn btn-sm text-primary"><i
  328. class="bi bi-download"></i> Скачать все
  329. </a>
  330. @endif
  331. <form action="{{ route('reclamations.upload-photo-after', $reclamation) }}"
  332. enctype="multipart/form-data" method="post" class="visually-hidden">
  333. @csrf
  334. <input required type="file" id="upl-photo-after" onchange="$(this).parent().submit()" multiple
  335. name="photo[]" class="form-control form-control-sm" accept=".jpg,.jpeg,.png">
  336. </form>
  337. <div class="row my-2 g-1 collapse" id="photos_after">
  338. @foreach($reclamation->photos_after as $photo)
  339. <div class="col-4">
  340. <a href="{{ $photo->link }}"
  341. data-toggle="lightbox" data-gallery="photos-after" data-size="fullscreen">
  342. <img class="img-thumbnail" src="{{ $photo->link }}" alt="">
  343. </a>
  344. @if(hasRole('admin'))
  345. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer rm-but"
  346. onclick="if(confirm('Удалить фото?')) $('#photo-{{ $photo->id }}').submit()"
  347. title="Удалить"></i>
  348. @endif
  349. <form action="{{ route('reclamations.delete-photo-after', [$reclamation, $photo]) }}"
  350. method="POST" id="photo-{{ $photo->id }}" class="visually-hidden">
  351. @csrf
  352. @method('DELETE')
  353. </form>
  354. </div>
  355. @endforeach
  356. </div>
  357. </div>
  358. </div>
  359. </div>
  360. </div>
  361. <!-- Модальное окно графика -->
  362. <div class="modal fade" id="copySchedule" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
  363. <div class="modal-dialog modal-fullscreen-sm-down modal-lg">
  364. <div class="modal-content">
  365. <div class="modal-header">
  366. <h1 class="modal-title fs-5" id="addModalLabel">Перенести в график монтажей</h1>
  367. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  368. </div>
  369. <div class="modal-body">
  370. <form action="{{ route('schedule.create-from-reclamation') }}" method="post" id="scheduleCreateForm">
  371. @csrf
  372. <div>
  373. <input type="hidden" name="reclamation_id" value="{{ $reclamation->id }}">
  374. {{-- <textarea name="comment" placeholder="Комментарий для графика" class="form-control mb-3"></textarea>--}}
  375. <input type="checkbox" checked="checked" id="sendNotifications" name="send_notifications" class="form-check-inline">
  376. <label for="sendNotifications" class="form-check-label mb-2">Уведомить менеджера и бригадира</label><br>
  377. <input type="checkbox" id="deleteOldRecords" name="delete_old_records" class="form-check-inline">
  378. <label for="deleteOldRecords" class="form-check-label mb-2">Удалить старые записи в графике для этой рекламации?</label><br>
  379. <button type="submit" class="btn btn-primary btn-sm">Обновить график</button>
  380. </div>
  381. </form>
  382. </div>
  383. </div>
  384. </div>
  385. </div>
  386. @endsection
  387. @push('styles')
  388. <style>
  389. .spare-part-dropdown {
  390. position: absolute;
  391. top: 100%;
  392. left: 0;
  393. right: 0;
  394. z-index: 1050;
  395. background: white;
  396. border: 1px solid #dee2e6;
  397. border-top: none;
  398. border-radius: 0 0 .375rem .375rem;
  399. max-height: 250px;
  400. overflow-y: auto;
  401. display: none;
  402. box-shadow: 0 4px 6px rgba(0,0,0,.1);
  403. }
  404. .spare-part-dropdown .sp-item {
  405. padding: 8px 12px;
  406. cursor: pointer;
  407. border-bottom: 1px solid #f0f0f0;
  408. }
  409. .spare-part-dropdown .sp-item:last-child {
  410. border-bottom: none;
  411. }
  412. .spare-part-dropdown .sp-item:hover,
  413. .spare-part-dropdown .sp-item.active {
  414. background-color: #e9ecef;
  415. }
  416. .spare-part-dropdown .sp-item .sp-article {
  417. font-weight: 600;
  418. color: #495057;
  419. }
  420. .spare-part-dropdown .sp-item .sp-used {
  421. font-size: 0.875em;
  422. color: #6c757d;
  423. }
  424. </style>
  425. @endpush
  426. @push('scripts')
  427. <script type="module">
  428. $('#createScheduleButton').on('click', function () {
  429. let myModalSchedule = new bootstrap.Modal(document.getElementById("copySchedule"), {});
  430. myModalSchedule.show();
  431. });
  432. let sparePartIndex = {{ max($reclamation->spareParts->count(), 1) }};
  433. const searchUrl = '{{ route('spare_parts.search') }}';
  434. // Инициализация автокомплита для поля
  435. function initSparePartAutocomplete($row) {
  436. const $input = $row.find('.spare-part-search');
  437. const $dropdown = $row.find('.spare-part-dropdown');
  438. const $hiddenId = $row.find('.spare-part-id');
  439. let currentFocus = -1;
  440. let searchTimeout;
  441. $input.on('input', function() {
  442. const query = $(this).val().trim();
  443. clearTimeout(searchTimeout);
  444. // Сбрасываем ID при изменении текста
  445. $hiddenId.val('');
  446. if (query.length >= 1) {
  447. searchTimeout = setTimeout(function() {
  448. $.get(searchUrl, { query: query }).done(function(data) {
  449. showDropdown(data, query);
  450. });
  451. }, 200);
  452. } else {
  453. $dropdown.hide();
  454. }
  455. });
  456. function showDropdown(items, query) {
  457. $dropdown.empty();
  458. currentFocus = -1;
  459. if (items.length === 0) {
  460. $dropdown.hide();
  461. return;
  462. }
  463. items.forEach(function(item) {
  464. const $item = $('<div class="sp-item"></div>');
  465. const highlightedArticle = highlightMatch(item.article, query);
  466. const highlightedUsed = item.used_in_maf ? highlightMatch(item.used_in_maf, query) : '';
  467. $item.html('<span class="sp-article">' + highlightedArticle + '</span>' +
  468. (item.used_in_maf ? ' <span class="sp-used">(' + highlightedUsed + ')</span>' : ''));
  469. $item.data('id', item.id);
  470. $item.data('article', item.article);
  471. $item.data('used', item.used_in_maf || '');
  472. $item.on('click', function() {
  473. selectItem($(this));
  474. });
  475. $dropdown.append($item);
  476. });
  477. $dropdown.show();
  478. }
  479. function highlightMatch(text, query) {
  480. if (!text || !query) return text || '';
  481. const regex = new RegExp('(' + escapeRegex(query) + ')', 'gi');
  482. return text.replace(regex, '<strong>$1</strong>');
  483. }
  484. function escapeRegex(str) {
  485. return str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
  486. }
  487. function selectItem($item) {
  488. const article = $item.data('article');
  489. const used = $item.data('used');
  490. const displayText = article + (used ? ' (' + used + ')' : '');
  491. $input.val(displayText);
  492. $hiddenId.val($item.data('id'));
  493. $dropdown.hide();
  494. }
  495. // Навигация клавиатурой
  496. $input.on('keydown', function(e) {
  497. const $items = $dropdown.find('.sp-item');
  498. if (e.keyCode === 40) { // Стрелка вниз
  499. e.preventDefault();
  500. currentFocus++;
  501. if (currentFocus >= $items.length) currentFocus = 0;
  502. setActive($items);
  503. } else if (e.keyCode === 38) { // Стрелка вверх
  504. e.preventDefault();
  505. currentFocus--;
  506. if (currentFocus < 0) currentFocus = $items.length - 1;
  507. setActive($items);
  508. } else if (e.keyCode === 13) { // Enter
  509. e.preventDefault();
  510. if (currentFocus > -1 && $items.length > 0) {
  511. selectItem($items.eq(currentFocus));
  512. }
  513. } else if (e.keyCode === 27) { // Escape
  514. $dropdown.hide();
  515. }
  516. });
  517. function setActive($items) {
  518. $items.removeClass('active');
  519. if (currentFocus >= 0 && currentFocus < $items.length) {
  520. $items.eq(currentFocus).addClass('active');
  521. }
  522. }
  523. // Закрытие при клике вне
  524. $(document).on('click', function(e) {
  525. if (!$(e.target).closest($row).length) {
  526. $dropdown.hide();
  527. }
  528. });
  529. }
  530. // Инициализация для существующих строк
  531. $('.spare-part-row').not('.spare-part-template').each(function() {
  532. initSparePartAutocomplete($(this));
  533. });
  534. // Добавление новой строки запчасти
  535. $('#add-spare-part-row').on('click', function() {
  536. let template = $('.spare-part-template').first();
  537. let newRow = template.clone();
  538. newRow.removeClass('spare-part-template d-none');
  539. newRow.attr('data-index', sparePartIndex);
  540. // Заменяем __INDEX__ на реальный индекс во всех name атрибутах
  541. newRow.find('[name]').each(function() {
  542. let name = $(this).attr('name');
  543. $(this).attr('name', name.replace('__INDEX__', sparePartIndex));
  544. $(this).prop('disabled', false);
  545. });
  546. // Сбрасываем значения
  547. newRow.find('.spare-part-search').val('');
  548. newRow.find('.spare-part-id').val('');
  549. newRow.find('input[type="number"]').val('');
  550. newRow.find('input[type="checkbox"]').prop('checked', false);
  551. $('.spare-parts-rows').append(newRow);
  552. // Инициализируем автокомплит для новой строки
  553. initSparePartAutocomplete(newRow);
  554. sparePartIndex++;
  555. });
  556. // Удаление строки
  557. $(document).on('click', '.remove-spare-part-row', function() {
  558. $(this).closest('.spare-part-row').remove();
  559. });
  560. </script>
  561. @endpush