GenerateDocumentsService.php 20 KB

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