show.blade.php 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  1. @php
  2. use App\Models\Order;
  3. @endphp
  4. @extends('layouts.app')
  5. @section('content')
  6. <div class="px-3">
  7. <div class="row mb-2">
  8. <div class="col-md-6">
  9. <h3>
  10. Площадка {{ $order->object_address }}
  11. <div class="badge text-bg-{{ Order::STATUS_COLOR[$order->order_status_id] }}">{{ $order->orderStatus->name }}</div>
  12. </h3>
  13. </div>
  14. <div class="col-md-6 text-end">
  15. <a href="{{ route('order.edit', ['order' => $order, 'previous_url' => $previous_url]) }}"
  16. class="btn btn-sm mb-1 btn-primary">Редактировать</a>
  17. @if(hasRole('admin') && ($order->order_status_id == Order::STATUS_NEW))
  18. <a href="#" onclick="if(confirm('Удалить площадку?')) $('form#destroy').submit();"
  19. class="btn btn-sm mb-1 btn-danger">Удалить</a>
  20. <form action="{{ route('order.destroy', $order) }}" method="post" class="d-none" id="destroy">
  21. @csrf
  22. @method('DELETE')
  23. </form>
  24. @endif
  25. @if(in_array($order->order_status_id, [Order::STATUS_READY_TO_MOUNT, Order::STATUS_IN_MOUNT]) && $order->isAllMafConnected())
  26. <a href="{{ route('order.generate-installation-pack', $order) }}"
  27. class="btn btn-sm mb-1 btn-primary">Документы для монтажа</a>
  28. <button @disabled(is_null($order->brigadier_id)) class="btn btn-primary btn-sm mb-1"
  29. id="createScheduleButton">Перенести в график
  30. </button>
  31. @endif
  32. @if($order->canCreateHandover())
  33. <a href="{{ route('order.generate-handover-pack', $order) }}" class="btn btn-sm mb-1 btn-primary">Документы
  34. для сдачи</a>
  35. @endif
  36. <a href="{{ $previous_url ?? route('order.index', session('gp_orders')) }}"
  37. class="btn btn-sm mb-1 btn-outline-secondary">Назад</a>
  38. </div>
  39. </div>
  40. <div class="row">
  41. <div class="col-xl-3 border-end">
  42. <h4>Общая информация об объекте</h4>
  43. <div>Название: {{ $order->name }}</div>
  44. <div>ID площадки: {{ $order->id }}</div>
  45. <div>
  46. Адрес: {{ $order->common_name }}
  47. </div>
  48. <div>Тип объекта: {{ $order->objectType->name }}</div>
  49. <div class="text-{{ Order::STATUS_COLOR[$order->order_status_id] }}">Статус
  50. объекта: {{ $order->orderStatus->name }}</div>
  51. <div>Комментарий: {{ $order->comment }}</div>
  52. <div class="row">
  53. <div class="col-md-8">Дата выхода на монтаж:</div>
  54. <div class="col-md-4">
  55. <input type="date" name="installation_date" id="installation_date" value="{{ $order->installation_date }}" class="form-control form-control-sm">
  56. </div>
  57. </div>
  58. <div class="row">
  59. <div class="col-md-8">Дней на монтаж:</div>
  60. <div class="col-md-4">
  61. <input type="number" min="1" name="install_days" id="install_days" value="{{ $order->install_days }}" class="form-control form-control-sm">
  62. </div>
  63. </div>
  64. <div>Дата готовности площадки: {{ $order->ready_date }}</div>
  65. <div>Бригадир: {{ $order?->brigadier?->name }}, тел. {{ $order?->brigadier?->phone }}</div>
  66. <div>Менеджер: {{ $order->user->name }}, тел. {{ $order->user->phone }}</div>
  67. <div>Название группы в ТГ: {{ $order->tg_group_name }}</div>
  68. <div>Ссылка на группу в ТГ: {{ $order->tg_group_link }}</div>
  69. <hr>
  70. <div class="reclamations">
  71. Рекламации
  72. @foreach($order->reclamations as $reclamation)
  73. <div>
  74. <a href="{{ route('reclamations.show', ['reclamation' => $reclamation, 'previous_url' => url()->current()]) }}">
  75. Рекламация № {{ $reclamation->id }} от {{ $reclamation->create_date }}
  76. </a>
  77. </div>
  78. @endforeach
  79. </div>
  80. <hr>
  81. <div class="documents">
  82. Документы
  83. <button class="btn btn-sm text-success" onclick="$('#upl-documents').trigger('click');"><i
  84. class="bi bi-plus-circle-fill"></i> Загрузить
  85. </button>
  86. <form action="{{ route('order.upload-document', $order) }}" enctype="multipart/form-data"
  87. method="post" class="visually-hidden">
  88. @csrf
  89. <input required type="file" id="upl-documents" onchange="$(this).parent().submit()" multiple
  90. name="document[]" class="form-control form-control-sm">
  91. </form>
  92. <div class="row my-2 g-1">
  93. @foreach($order->documents as $document)
  94. <div class="col-12">
  95. <a href="{{ $document->link }}" target="_blank">
  96. {{ $document->original_name }}
  97. </a>
  98. @if(hasRole('admin'))
  99. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  100. onclick="if(confirm('Удалить?')) $('#document-{{ $document->id }}').submit()"
  101. title="Удалить"></i>
  102. @endif
  103. <form action="{{ route('order.delete-document', [$order, $document]) }}" method="POST"
  104. id="document-{{ $document->id }}" class="visually-hidden">
  105. @csrf
  106. @method('DELETE')
  107. </form>
  108. </div>
  109. @endforeach
  110. </div>
  111. </div>
  112. <hr>
  113. <div class="statements">
  114. Ведомости
  115. <button class="btn btn-sm text-success" onclick="$('#upl-statements').trigger('click');"><i
  116. class="bi bi-plus-circle-fill"></i> Загрузить
  117. </button>
  118. <form action="{{ route('order.upload-statement', $order) }}" enctype="multipart/form-data"
  119. method="post" class="visually-hidden">
  120. @csrf
  121. <input required type="file" id="upl-statements" onchange="$(this).parent().submit()" multiple
  122. name="statement[]" class="form-control form-control-sm">
  123. </form>
  124. <div class="row my-2 g-1">
  125. @foreach($order->statements as $statement)
  126. <div class="col-12">
  127. <a href="{{ $statement->link }}" target="_blank">
  128. {{ $statement->original_name }}
  129. </a>
  130. @if(hasRole('admin'))
  131. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer ms-2"
  132. onclick="if(confirm('Удалить?')) $('#statement-{{ $statement->id }}').submit()"
  133. title="Удалить"></i>
  134. @endif
  135. <form action="{{ route('order.delete-statement', [$order, $statement]) }}" method="POST"
  136. id="statement-{{ $statement->id }}" class="visually-hidden">
  137. @csrf
  138. @method('DELETE')
  139. </form>
  140. </div>
  141. @endforeach
  142. </div>
  143. </div>
  144. <hr>
  145. <div class="photo">
  146. <a href="#photos" data-bs-toggle="collapse">Фотографии ({{ $order->photos->count() }})</a>
  147. <button class="btn btn-sm text-success" onclick="$('#upl-photo').trigger('click');"><i
  148. class="bi bi-plus-circle-fill"></i> Загрузить
  149. </button>
  150. @if($order->photos->count())
  151. <a href="{{ route('order.generate-photos-pack', $order) }}" class="btn btn-sm text-primary"><i
  152. class="bi bi-download"></i> Скачать все
  153. </a>
  154. @endif
  155. <form action="{{ route('order.upload-photo', $order) }}" enctype="multipart/form-data" method="post"
  156. class="visually-hidden">
  157. @csrf
  158. <input required type="file" id="upl-photo" onchange="$(this).parent().submit()" multiple
  159. name="photo[]" class="form-control form-control-sm" accept=".jpg,.jpeg,.png">
  160. </form>
  161. <div class="row my-2 g-1 collapse" id="photos">
  162. @foreach($order->photos as $photo)
  163. <div class="col-4">
  164. <a href="{{ $photo->link }}"
  165. data-toggle="lightbox" data-gallery="photos" data-size="fullscreen">
  166. <img class="img-thumbnail" src="{{ $photo->link }}" alt="">
  167. </a>
  168. @if(hasRole('admin'))
  169. <i class="bi bi-x-circle-fill fs-6 text-danger cursor-pointer rm-but"
  170. onclick="if(confirm('Удалить фото?')) $('#photo-{{ $photo->id }}').submit()"
  171. title="Удалить"></i>
  172. @endif
  173. <form action="{{ route('order.delete-photo', [$order, $photo]) }}" method="POST"
  174. id="photo-{{ $photo->id }}" class="visually-hidden">
  175. @csrf
  176. @method('DELETE')
  177. </form>
  178. </div>
  179. @endforeach
  180. </div>
  181. </div>
  182. </div>
  183. <div class="col-xl-9">
  184. <h4>МАФы заказа</h4>
  185. <div id="selected_maf">
  186. @if(isset($order) && $order->products_sku)
  187. <div class="col-12 overflow-x-scroll mb-3">
  188. <table class="table">
  189. <thead>
  190. <tr>
  191. <th><input type="checkbox" class="form-check" id="check-all-maf"></th>
  192. <th>Картинка</th>
  193. <th>МАФ</th>
  194. <th>Тип</th>
  195. <th>Статус</th>
  196. <th>Номер заказа МАФ</th>
  197. <th>RFID</th>
  198. <th>Заводской номер</th>
  199. <th>Дата производства</th>
  200. <th>Склад</th>
  201. <th>Паспорт</th>
  202. </tr>
  203. </thead>
  204. <tbody>
  205. @php
  206. $needs = $order->getNeeds();
  207. @endphp
  208. @foreach($order->products_sku as $p)
  209. <tr>
  210. <td>
  211. <input type="checkbox" class="form-check check-maf"
  212. data-maf-id="{{ $p->id }}">
  213. </td>
  214. <td>
  215. @if($p->product->image)
  216. <a href="{{ $p->product->image }}" data-toggle="lightbox"
  217. data-gallery="maf" data-size="fullscreen">
  218. <img src="{{ $p->product->image }}" alt=""
  219. class="img-thumbnail maf-img">
  220. </a>
  221. @endif
  222. </td>
  223. <td>
  224. <a href="{{ route('product_sku.show', ['product_sku' =>$p, 'previous_url' => url()->current()]) }}">
  225. {!! $p->product->article !!}
  226. </a>
  227. <br>
  228. <a class="small"
  229. href="{{ route('catalog.show', ['product' => $p->product, 'previous_url' => request()->fullUrl()]) }}">каталог</a>
  230. </td>
  231. <td>{!! $p->product->nomenclature_number !!}</td>
  232. <td>{{ $p->status }}</td>
  233. <td>
  234. @if($p->maf_order_id)
  235. <a href="{{ route('maf_order.show', $p->maf_order) }}">{{ $p->maf_order->order_number }}</a>
  236. @endif
  237. </td>
  238. <td>{{ $p->rfid }}</td>
  239. <td>{{ $p->factory_number }}</td>
  240. <td>{{ $p->manufacture_date }}</td>
  241. <td class="text-center">
  242. @if($p->maf_order?->order_number)
  243. <i class="bi bi-check-all text-success fw-bold"></i>
  244. @else
  245. @if($needs[$p->product_id]['sku']-- > 0)
  246. <i class="bi bi-check text-success fw-bold"></i>
  247. @else
  248. <i class="bi bi-x text-danger fw-bold"></i>
  249. @endif
  250. @endif
  251. </td>
  252. <td class="text-center">
  253. @if($p->passport)
  254. <i class="bi bi-check text-success fw-bold"></i>
  255. @else
  256. <i class="bi bi-x text-danger fw-bold"></i>
  257. @endif
  258. </td>
  259. </tr>
  260. @endforeach
  261. </tbody>
  262. </table>
  263. </div>
  264. <div>
  265. <a href="{{ route('order.get-maf', $order) }}"
  266. class="btn btn-primary btn-sm mb-1 @disabled($order->ready_to_mount == 'Нет' )">Привязать
  267. все МАФы</a>
  268. <br class="d-md-none">
  269. <a href="{{ route('order.revert-maf', $order) }}" class="btn btn-primary btn-sm mb-1">Отвязать
  270. все МАФы</a>
  271. <br class="d-md-none">
  272. <button class="btn btn-primary btn-sm mb-1" data-bs-toggle="modal"
  273. data-bs-target="#moveModal">Перенести МАФы
  274. </button>
  275. <br class="d-md-none">
  276. <button class="btn btn-sm mb-1 btn-warning" id="create-reclamation-button">Создать
  277. рекламацию
  278. </button>
  279. <form action="{{ route('reclamations.create', $order) }}" method="post"
  280. class="visually-hidden" id="create-reclamation-form">
  281. @csrf
  282. </form>
  283. <a href="#" class="btn btn-primary btn-sm mb-1" id="ttnBtn">ТН</a>
  284. </div>
  285. @endif
  286. </div>
  287. </div>
  288. </div>
  289. </div>
  290. <!-- Модальное окно графика -->
  291. <div class="modal fade" id="copySchedule" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
  292. <div class="modal-dialog modal-fullscreen-sm-down modal-lg">
  293. <div class="modal-content">
  294. <div class="modal-header">
  295. <h1 class="modal-title fs-5" id="addModalLabel">Перенести в график монтажей</h1>
  296. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  297. </div>
  298. <div class="modal-body">
  299. <form action="{{ route('schedule.create-from-order') }}" method="post" id="scheduleCreateForm">
  300. @csrf
  301. <div>
  302. <input type="hidden" name="order_id" value="{{ $order->id }}">
  303. <textarea name="comment" placeholder="Комментарий для графика" class="form-control mb-3"></textarea>
  304. <input type="checkbox" id="deleteOldRecords" name="delete_old_records" class="form-check-inline">
  305. <label for="deleteOldRecords" class="form-check-label mb-2">Удалить старые записи в графике для этой площадки?</label><br>
  306. <button type="submit" class="btn btn-primary btn-sm">Обновить график</button>
  307. </div>
  308. </form>
  309. </div>
  310. </div>
  311. </div>
  312. </div>
  313. <!-- Модальное окно ТН -->
  314. <div class="modal fade" id="createTtnModal" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
  315. <div class="modal-dialog modal-fullscreen-sm-down modal-lg">
  316. <div class="modal-content">
  317. <div class="modal-header">
  318. <h1 class="modal-title fs-5" id="addModalLabel">Введите данные для ТН</h1>
  319. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  320. </div>
  321. <div class="modal-body">
  322. <form action="{{ route('order.create-ttn') }}" method="post" id="ttnForm">
  323. @csrf
  324. <div>
  325. <input type="text" class="form-control mb-2" name="order_number" placeholder="Номер заказа">
  326. <input type="date" class="form-control mb-2" name="order_date" placeholder="Дата заказа" value="{{ date('Y-m-d') }}">
  327. <input type="number" class="form-control mb-2" name="order_sum" placeholder="Сумма заказа" value="0">
  328. <button href="#" class="btn btn-primary" id="createTtn">Создать ТН</button>
  329. </div>
  330. </form>
  331. </div>
  332. </div>
  333. </div>
  334. </div>
  335. <!-- Модальное окно переноса -->
  336. <div class="modal fade" id="moveModal" tabindex="-1" aria-labelledby="exampleModalLabel" aria-hidden="true">
  337. <div class="modal-dialog modal-fullscreen-sm-down modal-lg">
  338. <div class="modal-content">
  339. <div class="modal-header">
  340. <h1 class="modal-title fs-5" id="addModalLabel">Выбрать площадку, куда переносим</h1>
  341. <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Закрыть"></button>
  342. </div>
  343. <div class="modal-body">
  344. <form action="{{ route('order.move-maf') }}" method="post">
  345. @csrf
  346. <div id="select_maf_form">
  347. <input type="text" class="form-control mb-2" placeholder="Поиск площадки" id="search_order">
  348. <select id="select_order" name="new_order_id" class="form-select mb-3" size="20" multiple
  349. required></select>
  350. <a href="#" class="btn btn-primary" id="moveMaf">Перенести</a>
  351. </div>
  352. </form>
  353. </div>
  354. </div>
  355. </div>
  356. </div>
  357. @if($errors->any())
  358. @dump($errors)
  359. @endif
  360. @endsection
  361. @push('scripts')
  362. <script type="module">
  363. // select order
  364. $('#search_order').on('keyup', function () {
  365. // search products on backend
  366. $.get('{{ route('order.search') }}?s=' + $(this).val(),
  367. function (data) {
  368. $('#select_order').children().remove()
  369. $.each(data, function (id, name) {
  370. $('#select_order').append('<option value=\'' + id + '\'>' + name + '</option>');
  371. });
  372. }
  373. );
  374. }).trigger('keyup');
  375. $('#check-all-maf').on('change', function () {
  376. $('input:checkbox.check-maf').not(this).prop('checked', this.checked);
  377. });
  378. // move maf
  379. $('#moveMaf').on('click', function () {
  380. let ids = Array();
  381. $('.check-maf').each(function () {
  382. if ($(this).prop('checked')) {
  383. ids.push($(this).attr('data-maf-id'));
  384. }
  385. });
  386. $.post('{{ route('order.move-maf') }}',
  387. {
  388. '_token': $('meta[name=csrf-token]').attr('content'),
  389. ids: JSON.stringify(ids),
  390. 'new_order_id': $('#select_order').find(":selected").val()
  391. },
  392. function () {
  393. location.reload();
  394. }
  395. );
  396. });
  397. $('#create-reclamation-button').on('click', function () {
  398. let ids = Array();
  399. $('.check-maf').each(function () {
  400. if ($(this).prop('checked')) {
  401. ids.push($(this).attr('data-maf-id'));
  402. $('#create-reclamation-form').append('<input type="hidden" name="skus[]" value="' + $(this).attr('data-maf-id') + '">');
  403. }
  404. });
  405. if (ids.length) {
  406. $('#create-reclamation-form').submit();
  407. } else {
  408. alert('Нужно выбрать МАФ для рекламации!');
  409. }
  410. });
  411. $('#createScheduleButton').on('click', function () {
  412. let ids = Array();
  413. $('.check-maf').each(function () {
  414. if ($(this).prop('checked')) {
  415. ids.push($(this).attr('data-maf-id'));
  416. $('#scheduleCreateForm').append('<input type="hidden" name="skus[]" value="' + $(this).attr('data-maf-id') + '">');
  417. }
  418. });
  419. let myModalSchedule = new bootstrap.Modal(document.getElementById("copySchedule"), {});
  420. myModalSchedule.show();
  421. });
  422. $('#ttnBtn').on('click', function () {
  423. if ($('input.check-maf:checkbox:checked').length > 0) {
  424. let myModalTtn = new bootstrap.Modal(document.getElementById("createTtnModal"), {});
  425. myModalTtn.show();
  426. } else {
  427. alert('Нужно выбрать МАФ для ТН!');
  428. }
  429. });
  430. $('#createTtn').on('click', function () {
  431. let ids = Array();
  432. $('.check-maf').each(function () {
  433. if ($(this).prop('checked')) {
  434. ids.push($(this).attr('data-maf-id'));
  435. $('#ttnForm').append('<input type="hidden" name="skus[]" value="' + $(this).attr('data-maf-id') + '">');
  436. }
  437. });
  438. if (ids.length) {
  439. $('#ttnForm').submit();
  440. } else {
  441. alert('Нужно выбрать МАФ для ТН!');
  442. }
  443. });
  444. $('#installation_date').on('change', function () {
  445. let installationDate = $(this).val();
  446. $.post(
  447. '{{ route('order.update', $order->id) }}',
  448. {
  449. '_token': '{{ csrf_token() }}',
  450. id: '{{ $order->id }}',
  451. installation_date: installationDate
  452. },
  453. function () {
  454. $('.alerts').append(
  455. '<div class="main-alert alert alert-success" role="alert">Обновлена дата выхода на монтаж!</div>'
  456. );
  457. setTimeout(function () {
  458. $('.main-alert').fadeTo(2000, 500).slideUp(500, function () {
  459. $(".main-alert").slideUp(500);
  460. })
  461. }, 3000);
  462. }
  463. );
  464. });
  465. $('#install_days').on('change', function () {
  466. let installDays = $(this).val();
  467. $.post(
  468. '{{ route('order.update', $order->id) }}',
  469. {
  470. '_token' : '{{ csrf_token() }}',
  471. id: '{{ $order->id }}',
  472. install_days: installDays
  473. },
  474. function () {
  475. $('.alerts').append(
  476. '<div class="main-alert alert alert-success" role="alert">Обновлено количество дней на монтаж!</div>'
  477. );
  478. setTimeout(function () {
  479. $('.main-alert').fadeTo(2000, 500).slideUp(500, function () {
  480. $(".main-alert").slideUp(500);
  481. })
  482. }, 3000);
  483. }
  484. );
  485. });
  486. </script>
  487. @endpush