GenerateDocumentsService.php 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611
  1. <?php
  2. namespace App\Services;
  3. use App\Helpers\CountHelper;
  4. use App\Helpers\DateHelper;
  5. use App\Helpers\ExcelHelper;
  6. use App\Models\Contract;
  7. use App\Models\File;
  8. use App\Models\Order;
  9. use App\Models\ProductSKU;
  10. use App\Models\Reclamation;
  11. use App\Models\Ttn;
  12. use Exception;
  13. use Faker\Guesser\Name;
  14. use Illuminate\Support\Collection;
  15. use Illuminate\Support\Facades\Storage;
  16. use Illuminate\Support\Str;
  17. use PhpOffice\PhpSpreadsheet\IOFactory;
  18. use PhpOffice\PhpSpreadsheet\Worksheet\Drawing;
  19. use PhpOffice\PhpSpreadsheet\Writer\Pdf\Dompdf;
  20. use PhpOffice\PhpSpreadsheet\Writer\Pdf\Mpdf;
  21. use PhpOffice\PhpSpreadsheet\Writer\Pdf\Tcpdf;
  22. use PhpOffice\PhpSpreadsheet\Writer\Xlsx;
  23. class GenerateDocumentsService
  24. {
  25. const INSTALL_FILENAME = 'Монтаж ';
  26. const HANDOVER_FILENAME = 'Сдача ';
  27. const RECLAMATION_FILENAME = 'Рекламация ';
  28. /**
  29. * @param Order $order
  30. * @param int $userId
  31. * @return string
  32. * @throws Exception
  33. */
  34. public function generateInstallationPack(Order $order, int $userId): string
  35. {
  36. $techDocsPath = base_path('/tech-docs/');
  37. $products_sku = $order->products_sku;
  38. $articles = [];
  39. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/Схемы сборки/');
  40. foreach ($products_sku as $sku) {
  41. if (!in_array($sku->product->article, $articles)) {
  42. $articles[] = $sku->product->article;
  43. // find and copy scheme files to installation directory
  44. if (file_exists($techDocsPath . $sku->product->article . '/')) {
  45. foreach (Storage::disk('base')->allFiles('tech-docs/' . $sku->product->article) as $p) {
  46. $content = Storage::disk('base')->get($p);
  47. Storage::disk('public')->put('orders/' . $order->id . '/tmp/Схемы сборки/' . basename($p), $content);
  48. }
  49. }
  50. }
  51. }
  52. // generate xlsx order file
  53. $this->generateOrderForMount($order);
  54. // create zip archive
  55. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::INSTALL_FILENAME . $order->common_name . '.zip', $userId);
  56. // remove temp files
  57. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  58. $order->documents()->syncWithoutDetaching($fileModel);
  59. // return link
  60. return $fileModel?->link ?? '';
  61. }
  62. private function generateOrderForMount(Order $order): void
  63. {
  64. $inputFileType = 'Xlsx'; // Xlsx - Xml - Ods - Slk - Gnumeric - Csv
  65. $inputFileName = './templates/OrderForMount.xlsx';
  66. $reader = IOFactory::createReader($inputFileType);
  67. $spreadsheet = $reader->load($inputFileName);
  68. $sheet = $spreadsheet->getActiveSheet();
  69. // менеджер
  70. $sheet->setCellValue('F8', $order->user->name);
  71. $sheet->setCellValue('X8', $order->user->phone);
  72. // округ и район
  73. $sheet->setCellValue('L10', $order->district->shortname);
  74. $sheet->setCellValue('W10', $order->area->name);
  75. if ($order->area->responsible) {
  76. // ответственный
  77. $sheet->setCellValue('C12', $order->area->responsible?->name
  78. . ', ' . $order->area->responsible?->phone . ', ' . $order->area->responsible?->post);
  79. } else {
  80. $sheet->setCellValue('C12', '');
  81. }
  82. // адрес
  83. $sheet->setCellValue('L14', $order->object_address);
  84. $str = Str::replace('<div>', '', $order->products_with_count);
  85. $str = Str::replace('</div>', "\n", $str);
  86. // мафы
  87. $sheet->setCellValue('G33', Str::trim($str));
  88. //
  89. $fileName = 'Заявка на монтаж - ' . $order->object_address . '.xlsx';
  90. $writer = new Xlsx($spreadsheet);
  91. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  92. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  93. }
  94. /**
  95. * @throws Exception
  96. */
  97. public function generateHandoverPack(Order $order, int $userId): string
  98. {
  99. $articles = [];
  100. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ПАСПОРТ/');
  101. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/СЕРТИФИКАТ/');
  102. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ПСТ/');
  103. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ТН/');
  104. // copy app photos
  105. foreach ($order->photos as $photo) {
  106. $from = $photo->path;
  107. $to = 'orders/' . $order->id . '/tmp/ФОТО ПСТ/' . $photo->original_name;
  108. if (!Storage::disk('public')->exists($to)) {
  109. Storage::disk('public')->copy($from, $to);
  110. }
  111. }
  112. foreach ($order->products_sku as $sku) {
  113. // copy certificates
  114. if ($sku->product->certificate_id) {
  115. $from = $sku->product->certificate->path;
  116. $to = 'orders/' . $order->id . '/tmp/СЕРТИФИКАТ/' . $sku->product->certificate->original_name;
  117. if (!Storage::disk('public')->exists($to)) {
  118. Storage::disk('public')->copy($from, $to);
  119. }
  120. }
  121. // copy passport
  122. if ($sku->passport_id) {
  123. $from = $sku->passport->path;
  124. $to = 'orders/' . $order->id . '/tmp/ПАСПОРТ/';
  125. if (!Storage::disk('public')->exists($to . $sku->passport->original_name)) {
  126. $f = Storage::disk('public')->get($from);
  127. $ext = \File::extension($sku->passport->original_name);
  128. $targetName = $to . 'Паспорт ' . $sku->factory_number . ' арт. ' . $sku->product->article . '.' . $ext;
  129. Storage::disk('public')->put($targetName, $f);
  130. }
  131. }
  132. }
  133. // generate xlsx order files
  134. $this->generateStatement($order);
  135. $this->generateQualityDeclaration($order);
  136. $this->generateInventory($order);
  137. $this->generatePassport($order);
  138. // create zip archive
  139. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::HANDOVER_FILENAME . $order->common_name . '.zip', $userId);
  140. // remove temp files
  141. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  142. $order->documents()->syncWithoutDetaching($fileModel);
  143. // return link
  144. return $fileModel?->link ?? '';
  145. }
  146. private function generateStatement(Order $order): void
  147. {
  148. $inputFileType = 'Xlsx';
  149. $inputFileName = './templates/Statement.xlsx';
  150. $reader = IOFactory::createReader($inputFileType);
  151. $spreadsheet = $reader->load($inputFileName);
  152. $sheet = $spreadsheet->getActiveSheet();
  153. $contract = Contract::query()->where('contracts.year', $order->year)->first();
  154. $contract_number = $contract->contract_number ?? 'заполнить договор за ' . $order->year . ' год!';
  155. $contract_date = DateHelper::getHumanDate($contract->contract_date ?? '1970-01-01', true);
  156. $s = 'по Договору №' . $contract_number . ' от ' . $contract_date . ' г. Между ГБУ "Мосремонт" и ООО "НАШ ДВОР-СТ"';
  157. $sheet->setCellValue('B5', $s);
  158. // менеджер
  159. $sheet->setCellValue('G21', $order->user->name);
  160. // округ и район
  161. $sheet->setCellValue('C6', $order->district->shortname);
  162. $sheet->setCellValue('F6', $order->area->name);
  163. $i = 9; // start of table
  164. $nn = 1; // string number
  165. foreach ($order->products_sku as $sku) {
  166. if ($nn > 1) { // inset row
  167. $sheet->insertNewRowBefore($i);
  168. }
  169. $sheet->setCellValue('A' . $i, $nn++);
  170. $sheet->setCellValue('B' . $i, $sku->product->statement_name);
  171. $sheet->setCellValue('C' . $i, $sku->product->passport_name);
  172. $sheet->setCellValue('D' . $i, 'шт');
  173. $sheet->setCellValue('E' . $i, '1');
  174. $sheet->setCellValue('F' . $i, $order->name);
  175. $sheet->setCellValue('G' . $i, $sku->factory_number);
  176. $sheet->setCellValue('H' . $i++, $sku->rfid);
  177. }
  178. // save file
  179. $fileName = '1.Ведомость.xlsx';
  180. $writer = new Xlsx($spreadsheet);
  181. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  182. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  183. }
  184. private function generateQualityDeclaration(Order $order): void
  185. {
  186. $inputFileType = 'Xlsx';
  187. $inputFileName = './templates/QualityDeclaration.xlsx';
  188. $reader = IOFactory::createReader($inputFileType);
  189. $spreadsheet = $reader->load($inputFileName);
  190. $sheet = $spreadsheet->getActiveSheet();
  191. $address = 'г. Москва, ' . $order->district->shortname . ', район ' . $order->area->name . ', по адресу: ' . $order->object_address;
  192. $i = 1; // start of table
  193. $n = 1;
  194. foreach ($order->products_sku as $sku) {
  195. if ($n++ > 1) {
  196. $range = 'A' . $i . ':I' . $i + 56;
  197. $i = $i + 57;
  198. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  199. // add header img
  200. $drawing = new Drawing();
  201. $drawing->setName('Header');
  202. $drawing->setDescription('Header');
  203. $drawing->setPath('templates/header.png');
  204. $drawing->setCoordinates('A' . $i);
  205. $drawing->setOffsetX(16);
  206. $drawing->setOffsetY(8);
  207. $drawing->setResizeProportional(true);
  208. $drawing->setWidth(680);
  209. $drawing->setWorksheet($sheet);
  210. }
  211. // document date?
  212. $sheet->setCellValue('A' . $i + 7, DateHelper::getHumanDate(date('Y-m-d'), true));
  213. $sheet->setCellValue('B' . $i + 16, $sku->product->passport_name);
  214. $sheet->setCellValue('C' . $i + 18, $sku->rfid);
  215. $sheet->setCellValue('C' . $i + 20, $address);
  216. // add page break and copy prev page
  217. }
  218. // save file
  219. $fileName = '2.Декларация качества.xlsx';
  220. $writer = new Xlsx($spreadsheet);
  221. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  222. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  223. }
  224. private function generateInventory(Order $order): void
  225. {
  226. $inputFileType = 'Xlsx';
  227. $inputFileName = './templates/Inventory.xlsx';
  228. $reader = IOFactory::createReader($inputFileType);
  229. $spreadsheet = $reader->load($inputFileName);
  230. $sheet = $spreadsheet->getActiveSheet();
  231. $address = 'Округ: ' . $order->district->shortname . ' Район ' . $order->area->name;
  232. $sheet->setCellValue('C4', $order->name);
  233. $sheet->setCellValue('C5', $address);
  234. $sheet->setCellValue('C13', $order->user->name);
  235. $i = 8; // start of table
  236. $n = 1;
  237. foreach ($order->products_sku as $sku) {
  238. if ($n++ > 1) {
  239. $sheet->insertNewRowBefore($i + 3, 3);
  240. $range = 'A' . $i . ':E' . $i + 2;
  241. $i = $i + 3;
  242. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  243. }
  244. $sheet->setCellValue('A' . $i, $n - 1);
  245. $sheet->setCellValue('B' . $i, $sku->product->passport_name);
  246. $sheet->setCellValue('B' . $i + 2, $sku->rfid);
  247. }
  248. // save file
  249. $fileName = '3.Опись.xlsx';
  250. $writer = new Xlsx($spreadsheet);
  251. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  252. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  253. }
  254. private function generatePassport(Order $order): void
  255. {
  256. $inputFileType = 'Xlsx';
  257. $inputFileName = './templates/Passport.xlsx';
  258. $reader = IOFactory::createReader($inputFileType);
  259. $spreadsheet = $reader->load($inputFileName);
  260. $sheet = $spreadsheet->getActiveSheet();
  261. $i = 3; // start of table
  262. $n = 1;
  263. foreach ($order->products_sku as $sku) {
  264. if ($n++ > 1) {
  265. $sheet->insertNewRowBefore($i + 1, 1);
  266. $range = 'A' . $i . ':J' . $i;
  267. $i++;
  268. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  269. }
  270. $sheet->setCellValue('A' . $i, $n - 1);
  271. $sheet->setCellValue('B' . $i, $sku->product->passport_name);
  272. $sheet->setCellValue('C' . $i, $sku->product->type_tz);
  273. $sheet->setCellValue('D' . $i, $sku->rfid);
  274. $sheet->setCellValue('E' . $i, $sku->product->certificate_number);
  275. $sheet->setCellValue('G' . $i, $sku->factory_number);
  276. $sheet->setCellValue('H' . $i, DateHelper::getHumanDate($sku->manufacture_date, true));
  277. }
  278. // save file
  279. $fileName = '4.Паспорт объекта - ' . $order->object_address . '.xlsx';
  280. $writer = new Xlsx($spreadsheet);
  281. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  282. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  283. }
  284. /**
  285. * @param Reclamation $reclamation
  286. * @param int $userId
  287. * @return string
  288. * @throws Exception
  289. */
  290. public function generateReclamationPack(Reclamation $reclamation, int $userId): string
  291. {
  292. Storage::disk('public')->makeDirectory('reclamations/' . $reclamation->id . '/tmp/' . $reclamation->order->object_address . '/ФОТО НАРУШЕНИЯ/');
  293. // copy photos
  294. foreach ($reclamation->photos_before as $photo) {
  295. $from = $photo->path;
  296. $to = 'reclamations/' . $reclamation->id . '/tmp/' . $reclamation->order->object_address . '/ФОТО НАРУШЕНИЯ/' . $photo->original_name;
  297. if (!Storage::disk('public')->exists($to)) {
  298. Storage::disk('public')->copy($from, $to);
  299. }
  300. }
  301. // create xls and pdf
  302. $this->generateReclamationOrder($reclamation);
  303. $this->generateReclamationAct($reclamation);
  304. $this->generateReclamationGuarantee($reclamation);
  305. // create zip archive
  306. $fileModel = (new FileService())->createZipArchive('reclamations/' . $reclamation->id . '/tmp', self::RECLAMATION_FILENAME . $reclamation->order->object_address . '.zip', $userId);
  307. // remove temp files
  308. Storage::disk('public')->deleteDirectory('reclamations/' . $reclamation->id . '/tmp');
  309. $reclamation->documents()->syncWithoutDetaching($fileModel);
  310. // return link
  311. return $fileModel?->link ?? '';
  312. }
  313. /**
  314. * @throws Exception
  315. */
  316. private function generateReclamationOrder(Reclamation $reclamation): void
  317. {
  318. $inputFileType = 'Xlsx';
  319. $inputFileName = './templates/ReclamationOrder.xlsx';
  320. $reader = IOFactory::createReader($inputFileType);
  321. $spreadsheet = $reader->load($inputFileName);
  322. $sheet = $spreadsheet->getActiveSheet();
  323. $articles = [];
  324. foreach ($reclamation->skus as $p) {
  325. $articles[] = $p->product->article;
  326. }
  327. $sheet->setCellValue('J4', DateHelper::getHumanDate($reclamation->create_date, true));
  328. $sheet->setCellValue('L10', $reclamation->order->common_name);
  329. $sheet->setCellValue('L11', $reclamation->order->area?->responsible?->name);
  330. $sheet->setCellValue('W11', $reclamation->order->area?->responsible?->phone);
  331. $sheet->setCellValue('L12', $reclamation->order->year);
  332. $sheet->setCellValue('G13', $reclamation->guarantee);
  333. $sheet->setCellValue('G14', implode(', ', $articles));
  334. $sheet->setCellValue('Y15', DateHelper::getHumanDate($reclamation->finish_date, true));
  335. $sheet->setCellValue('U20', DateHelper::getHumanDate($reclamation->create_date, true));
  336. // save file
  337. $fileName = 'Монтажная заявка - ' . $reclamation->order->object_address . '.xlsx';
  338. $writer = new Xlsx($spreadsheet);
  339. $fd = 'reclamations/' . $reclamation->id . '/tmp/' . $reclamation->order->object_address;
  340. Storage::disk('public')->makeDirectory($fd);
  341. $fp = storage_path('app/public/reclamations/') . $reclamation->id . '/tmp/' . $reclamation->order->object_address . '/' . $fileName;
  342. Storage::disk('public')->delete($fd . '/' . $fileName);
  343. $writer->save($fp);
  344. PdfConverterClient::convert($fp);
  345. }
  346. /**
  347. * @throws Exception
  348. */
  349. private function generateReclamationAct(Reclamation $reclamation): void
  350. {
  351. $inputFileType = 'Xlsx';
  352. $inputFileName = './templates/ReclamationAct.xlsx';
  353. $reader = IOFactory::createReader($inputFileType);
  354. $spreadsheet = $reader->load($inputFileName);
  355. $sheet = $spreadsheet->getActiveSheet();
  356. $mafs = [];
  357. foreach ($reclamation->skus as $p) {
  358. $mafs[] = $p->product->passport_name . ', тип ' . $p->product->nomenclature_number;
  359. }
  360. $sheet->setCellValue('A17', $reclamation->order->object_address);
  361. $sheet->setCellValue('A22', implode('; ', $mafs));
  362. $sheet->setCellValue('A27', $reclamation->whats_done);
  363. $i = 24;
  364. $n = 1;
  365. foreach ($reclamation->skus as $p) {
  366. if ($n++ > 1) {
  367. $i++;
  368. $sheet->insertNewRowBefore($i, 1);
  369. $range = 'D' . $i . ':I' . $i;
  370. $sheet->mergeCells($range);
  371. }
  372. $sheet->setCellValue('D' . $i, $p->rfid);
  373. }
  374. // save file
  375. $fileName = 'Акт - ' . $reclamation->order->object_address . '.xlsx';
  376. $writer = new Xlsx($spreadsheet);
  377. $fd = 'reclamations/' . $reclamation->id . '/tmp/' . $reclamation->order->object_address;
  378. Storage::disk('public')->makeDirectory($fd);
  379. $fp = storage_path('app/public/reclamations/') . $reclamation->id . '/tmp/' . $reclamation->order->object_address . '/' . $fileName;
  380. Storage::disk('public')->delete($fd . '/' . $fileName);
  381. $writer->save($fp);
  382. PdfConverterClient::convert($fp);
  383. }
  384. /**
  385. * @throws Exception
  386. */
  387. private function generateReclamationGuarantee(Reclamation $reclamation): void
  388. {
  389. $inputFileType = 'Xlsx';
  390. $inputFileName = './templates/ReclamationGuarantee.xlsx';
  391. $reader = IOFactory::createReader($inputFileType);
  392. $spreadsheet = $reader->load($inputFileName);
  393. $sheet = $spreadsheet->getActiveSheet();
  394. $mafs = [];
  395. foreach ($reclamation->skus as $p) {
  396. $mafs[] = 'Тип ' . $p->product->nomenclature_number . ' (' . $p->product->passport_name . ')' ;
  397. }
  398. $contract = Contract::query()->where('contracts.year', $reclamation->order->year)->first();
  399. $text = "ООО «НАШ ДВОР-СТ» в рамках обязательств по Договору №{$contract?->contract_number}" .
  400. " от " . DateHelper::getHumanDate($contract?->contract_date ?? '1970-01-01', true) .
  401. " г. на выполнение комплекса работ по поставке, монтажу устанавливаемых на городских территориях малых архитектурных форм гарантирует " .
  402. $reclamation->guarantee . " на оборудовании «" . implode('; ', $mafs) .
  403. "» установленному по адресу г. Москва, " . $reclamation->order->object_address . " в срок до " .
  404. DateHelper::getHumanDate($reclamation->finish_date, true). " г. в связи с отсутствием детали в наличии и ее производством.";
  405. $sheet->setCellValue('B9', $reclamation->id);
  406. $sheet->setCellValue('D9', DateHelper::getHumanDate($reclamation->create_date, true));
  407. $sheet->setCellValue('A19', $text);
  408. // save file
  409. $fileName = 'Гарантийное письмо - ' . $reclamation->order->object_address . '.xlsx';
  410. $writer = new Xlsx($spreadsheet);
  411. $fd = 'reclamations/' . $reclamation->id . '/tmp/' . $reclamation->order->object_address;
  412. Storage::disk('public')->makeDirectory($fd);
  413. $fp = storage_path('app/public/reclamations/') . $reclamation->id . '/tmp/' . $reclamation->order->object_address . '/' . $fileName;
  414. Storage::disk('public')->delete($fd . '/' . $fileName);
  415. $writer->save($fp);
  416. PdfConverterClient::convert($fp);
  417. }
  418. public function generateFilePack(Collection $files, int $userId, string $name = 'files'): \App\Models\File
  419. {
  420. $dir = Str::random(2);
  421. Storage::disk('public')->makeDirectory('files/' . $dir . '/tmp/');
  422. // copy files
  423. foreach ($files as $file) {
  424. $from = $file->path;
  425. $to = 'files/' . $dir . '/tmp/' . $file->original_name;
  426. if (!Storage::disk('public')->exists($to)) {
  427. Storage::disk('public')->copy($from, $to);
  428. }
  429. }
  430. // create zip archive
  431. $fileModel = (new FileService())->createZipArchive('files/' . $dir . '/tmp', $name .'_' . date('Y-m-d_h-i-s') . '.zip', $userId);
  432. // remove temp files
  433. Storage::disk('public')->deleteDirectory('files/' . $dir . '/tmp');
  434. // return link
  435. return $fileModel;
  436. }
  437. public function generateTtnPack(Ttn $ttn, int $userId): string
  438. {
  439. $skus = ProductSKU::query()->whereIn('id', json_decode($ttn->skus))->get();
  440. $volume = $weight = $places = 0;
  441. foreach ($skus as $sku) {
  442. if(!isset($order)) {
  443. $order = $sku->order;
  444. }
  445. $volume += $sku->product->volume;
  446. $weight += $sku->product->weight;
  447. $places += $sku->product->places;
  448. }
  449. $installationDate = ($order->installation_date) ? DateHelper::getHumanDate($order->installation_date, true) : '-';
  450. $ttnNumber = ($ttn->ttn_number_suffix) ? $ttn->ttn_number . '-' . $ttn->ttn_number_suffix : $ttn->ttn_number;
  451. $inputFileType = 'Xlsx';
  452. $inputFileName = './templates/Ttn.xlsx';
  453. $reader = IOFactory::createReader($inputFileType);
  454. $spreadsheet = $reader->load($inputFileName);
  455. $sheet = $spreadsheet->getActiveSheet();
  456. $sheet->setCellValue('D8', $installationDate);
  457. $sheet->setCellValue('R8', $ttnNumber);
  458. $sheet->setCellValue('AF8', DateHelper::getHumanDate($ttn->order_date, true));
  459. $sheet->setCellValue('AT8', $ttn->order_number);
  460. $sheet->setCellValue('B19', $order->object_address ?? '');
  461. $sheet->setCellValue('AD22', CountHelper::humanCount($places, 'место', 'места', 'мест') . ', способ упаковки: поддон, ящик, картон, полиэтилен');
  462. $sheet->setCellValue('B24', $weight . ' кг, ' . $volume . ' м.куб');
  463. $sheet->setCellValue('B36', $installationDate);
  464. $sheet->setCellValue('AB57', $installationDate . ' 8-00');
  465. $sheet->setCellValue('B59', $installationDate . ' 8-10');
  466. $sheet->setCellValue('AB59', $installationDate . ' 8-40');
  467. $sheet->setCellValue('B61', $weight . ' кг');
  468. $sheet->setCellValue('B63', CountHelper::humanCount($places, 'место', 'места', 'мест'));
  469. $sheet->setCellValue('B75', $order->object_address ?? '');
  470. $sheet->setCellValue('AB75', $installationDate);
  471. $sheet->setCellValue('B77', $installationDate);
  472. $sheet->setCellValue('AB77', $installationDate);
  473. $sheet->setCellValue('AB79', CountHelper::humanCount($places, 'место', 'места', 'мест'));
  474. $sheet->setCellValue('B81', $weight . ' кг');
  475. $sheet->setCellValue('B83', $order->brigadier?->name ?? '');
  476. $sheet->setCellValue('B89', $ttn->order_sum);
  477. $sheet->setCellValue('AD89', $ttn->order_sum);
  478. $sheet->setCellValue('AS89', $ttn->order_sum);
  479. // save file
  480. $fileName = 'ТТН №' . $ttn->ttn_number . ' от ' . DateHelper::getHumanDate($ttn->order_date) . '.xlsx';
  481. $writer = new Xlsx($spreadsheet);
  482. $fd = 'ttn/' . $ttn->year;
  483. Storage::disk('public')->makeDirectory($fd);
  484. $fp = storage_path('app/public/ttn/') . $ttn->year . '/' . $fileName;
  485. Storage::disk('public')->delete($fd . '/' . $fileName);
  486. $writer->save($fp);
  487. $fileModel = File::query()->updateOrCreate([
  488. 'link' => url('/storage/') . '/ttn/' . $ttn->year . '/' .$fileName,
  489. 'path' => $fp . '/' .$fileName,
  490. 'user_id' => $userId,
  491. 'original_name' => $fileName,
  492. 'mime_type' => 'application/xlsx',
  493. ]);
  494. $ttn->file_id = $fileModel->id;
  495. $ttn->save();
  496. $order->documents()->attach($fileModel->id);
  497. return $fileModel->link ?? '';
  498. }
  499. }