GenerateDocumentsService.php 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. <?php
  2. namespace App\Services;
  3. use App\Helpers\DateHelper;
  4. use App\Models\Contract;
  5. use App\Models\Order;
  6. use Exception;
  7. use Illuminate\Support\Facades\Storage;
  8. use Illuminate\Support\Str;
  9. use PhpOffice\PhpSpreadsheet\IOFactory;
  10. use PhpOffice\PhpSpreadsheet\Writer\Xlsx;
  11. class GenerateDocumentsService
  12. {
  13. const INSTALL_FILENAME = 'Монтаж ';
  14. const HANDOVER_FILENAME = 'Сдача ';
  15. /**
  16. * @param Order $order
  17. * @param int $userId
  18. * @return string
  19. * @throws Exception
  20. */
  21. public function generateInstallationPack(Order $order, int $userId): string
  22. {
  23. $techDocsPath = base_path('/tech-docs/');
  24. $products_sku = $order->products_sku;
  25. $articles = [];
  26. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/Схемы сборки/');
  27. foreach ($products_sku as $sku) {
  28. if(!in_array($sku->product->article, $articles)) {
  29. $articles[] = $sku->product->article;
  30. // find and copy scheme files to installation directory
  31. if(file_exists($techDocsPath . $sku->product->article . '/')) {
  32. foreach(Storage::disk('base')->allFiles('tech-docs/' . $sku->product->article) as $p) {
  33. $content = Storage::disk('base')->get($p);
  34. Storage::disk('public')->put('orders/' . $order->id . '/tmp/Схемы сборки/' . basename($p), $content);
  35. }
  36. }
  37. }
  38. }
  39. // generate xlsx order file
  40. $this->generateOrderForMount($order);
  41. // create zip archive
  42. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::INSTALL_FILENAME . $order->common_name . '.zip', $userId);
  43. // remove temp files
  44. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  45. $order->documents()->syncWithoutDetaching($fileModel);
  46. // return link
  47. return $fileModel?->link ?? '';
  48. }
  49. private function generateOrderForMount(Order $order): void
  50. {
  51. $inputFileType = 'Xlsx'; // Xlsx - Xml - Ods - Slk - Gnumeric - Csv
  52. $inputFileName = './templates/OrderForMount.xlsx';
  53. $reader = IOFactory::createReader($inputFileType);
  54. $spreadsheet = $reader->load($inputFileName);
  55. $sheet = $spreadsheet->getActiveSheet();
  56. // менеджер
  57. $sheet->setCellValue('F8', $order->user->name);
  58. $sheet->setCellValue('X8', $order->user->phone);
  59. // округ и район
  60. $sheet->setCellValue('L10', $order->district->shortname);
  61. $sheet->setCellValue('W10', $order->area->name);
  62. if($order->area->responsible) {
  63. // ответственный
  64. $sheet->setCellValue('C12', $order->area->responsible?->name
  65. . ', ' . $order->area->responsible?->phone . ', ' . $order->area->responsible?->post);
  66. } else {
  67. $sheet->setCellValue('C12', '');
  68. }
  69. // адрес
  70. $sheet->setCellValue('L14', $order->object_address);
  71. $str = Str::replace('<div>', '', $order->products_with_count);
  72. $str = Str::replace('</div>', "\n", $str);
  73. // мафы
  74. $sheet->setCellValue('G33', Str::trim($str));
  75. //
  76. $fileName = 'Заявка на монтаж - ' . $order->object_address . '.xlsx';
  77. $writer = new Xlsx($spreadsheet);
  78. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  79. $writer->save(storage_path('app/public/orders/') . $order->id .'/tmp/' . $fileName);
  80. }
  81. /**
  82. * @throws Exception
  83. */
  84. public function generateHandoverPack(Order $order, int $userId): string
  85. {
  86. $articles = [];
  87. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ПАСПОРТ/');
  88. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/СЕРТИФИКАТ/');
  89. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ПСТ/photos');
  90. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ТН/');
  91. // copy app photos
  92. foreach ($order->photos as $photo) {
  93. $from = $photo->path;
  94. $to = 'orders/' . $order->id . '/tmp/ФОТО ПСТ/photos/' . $photo->original_name;
  95. if(!Storage::disk('public')->exists($to)) {
  96. Storage::disk('public')->copy($from, $to);
  97. }
  98. }
  99. foreach ($order->products_sku as $sku) {
  100. // copy certificates
  101. if($sku->product->certificate_id) {
  102. $from = $sku->product->certificate->path;
  103. $to = 'orders/' . $order->id . '/tmp/СЕРТИФИКАТ/' . $sku->product->certificate->original_name;
  104. if(!Storage::disk('public')->exists($to)) {
  105. Storage::disk('public')->copy($from, $to);
  106. }
  107. }
  108. // copy passport
  109. if($sku->passport_id) {
  110. $from = $sku->passport->path;
  111. $to = 'orders/' . $order->id . '/tmp/ПАСПОРТ/' . $sku->passport->original_name;
  112. if(!Storage::disk('public')->exists($to)) {
  113. Storage::disk('public')->copy($from, $to);
  114. }
  115. }
  116. }
  117. // generate xlsx order files
  118. $this->generateStatement($order);
  119. // create zip archive
  120. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::HANDOVER_FILENAME . $order->common_name . '.zip', $userId);
  121. // remove temp files
  122. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  123. $order->documents()->syncWithoutDetaching($fileModel);
  124. // return link
  125. return $fileModel?->link ?? '';
  126. }
  127. private function generateStatement(Order $order): void
  128. {
  129. $inputFileType = 'Xlsx';
  130. $inputFileName = './templates/Statement.xlsx';
  131. $reader = IOFactory::createReader($inputFileType);
  132. $spreadsheet = $reader->load($inputFileName);
  133. $sheet = $spreadsheet->getActiveSheet();
  134. $contract = Contract::query()->where('contracts.year', $order->year)->first();
  135. $contract_number = $contract->contract_number ?? 'заполнить договор за ' . $order->year . ' год!';
  136. $contract_date = DateHelper::getHumanDate($contract->contract_date ?? '1970-01-01', true);
  137. $s = 'по Договору №' . $contract_number . ' от ' . $contract_date . ' г. Между ГБУ "Мосремонт" и ООО "НАШ ДВОР-СТ"';
  138. $sheet->setCellValue('B5', $s);
  139. // менеджер
  140. $sheet->setCellValue('G21', $order->user->name);
  141. // округ и район
  142. $sheet->setCellValue('C6', $order->district->shortname);
  143. $sheet->setCellValue('F6', $order->area->name);
  144. $i = 9; // start of table
  145. $nn = 1; // string number
  146. foreach ($order->products_sku as $sku) {
  147. if($nn > 1) { // inset row
  148. $sheet->insertNewRowBefore($i);
  149. }
  150. $sheet->setCellValue('A' . $i, $nn++);
  151. $sheet->setCellValue('B' . $i, $sku->product->statement_name);
  152. $sheet->setCellValue('C' . $i, $sku->product->passport_name);
  153. $sheet->setCellValue('D' . $i, 'шт');
  154. $sheet->setCellValue('E' . $i, '1');
  155. $sheet->setCellValue('F' . $i, $order->name);
  156. $sheet->setCellValue('G' . $i, $sku->factory_number);
  157. $sheet->setCellValue('H' . $i++, $sku->rfid);
  158. }
  159. // save file
  160. $fileName = '1.Ведомость.xlsx';
  161. $writer = new Xlsx($spreadsheet);
  162. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  163. $writer->save(storage_path('app/public/orders/') . $order->id .'/tmp/' . $fileName);
  164. }
  165. }