GenerateDocumentsService.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  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 Exception;
  8. use Illuminate\Support\Facades\Storage;
  9. use Illuminate\Support\Str;
  10. use PhpOffice\PhpSpreadsheet\Cell\Cell;
  11. use PhpOffice\PhpSpreadsheet\IOFactory;
  12. use PhpOffice\PhpSpreadsheet\Worksheet\Drawing;
  13. use PhpOffice\PhpSpreadsheet\Writer\Xlsx;
  14. class GenerateDocumentsService
  15. {
  16. const INSTALL_FILENAME = 'Монтаж ';
  17. const HANDOVER_FILENAME = 'Сдача ';
  18. /**
  19. * @param Order $order
  20. * @param int $userId
  21. * @return string
  22. * @throws Exception
  23. */
  24. public function generateInstallationPack(Order $order, int $userId): string
  25. {
  26. $techDocsPath = base_path('/tech-docs/');
  27. $products_sku = $order->products_sku;
  28. $articles = [];
  29. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/Схемы сборки/');
  30. foreach ($products_sku as $sku) {
  31. if (!in_array($sku->product->article, $articles)) {
  32. $articles[] = $sku->product->article;
  33. // find and copy scheme files to installation directory
  34. if (file_exists($techDocsPath . $sku->product->article . '/')) {
  35. foreach (Storage::disk('base')->allFiles('tech-docs/' . $sku->product->article) as $p) {
  36. $content = Storage::disk('base')->get($p);
  37. Storage::disk('public')->put('orders/' . $order->id . '/tmp/Схемы сборки/' . basename($p), $content);
  38. }
  39. }
  40. }
  41. }
  42. // generate xlsx order file
  43. $this->generateOrderForMount($order);
  44. // create zip archive
  45. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::INSTALL_FILENAME . $order->common_name . '.zip', $userId);
  46. // remove temp files
  47. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  48. $order->documents()->syncWithoutDetaching($fileModel);
  49. // return link
  50. return $fileModel?->link ?? '';
  51. }
  52. private function generateOrderForMount(Order $order): void
  53. {
  54. $inputFileType = 'Xlsx'; // Xlsx - Xml - Ods - Slk - Gnumeric - Csv
  55. $inputFileName = './templates/OrderForMount.xlsx';
  56. $reader = IOFactory::createReader($inputFileType);
  57. $spreadsheet = $reader->load($inputFileName);
  58. $sheet = $spreadsheet->getActiveSheet();
  59. // менеджер
  60. $sheet->setCellValue('F8', $order->user->name);
  61. $sheet->setCellValue('X8', $order->user->phone);
  62. // округ и район
  63. $sheet->setCellValue('L10', $order->district->shortname);
  64. $sheet->setCellValue('W10', $order->area->name);
  65. if ($order->area->responsible) {
  66. // ответственный
  67. $sheet->setCellValue('C12', $order->area->responsible?->name
  68. . ', ' . $order->area->responsible?->phone . ', ' . $order->area->responsible?->post);
  69. } else {
  70. $sheet->setCellValue('C12', '');
  71. }
  72. // адрес
  73. $sheet->setCellValue('L14', $order->object_address);
  74. $str = Str::replace('<div>', '', $order->products_with_count);
  75. $str = Str::replace('</div>', "\n", $str);
  76. // мафы
  77. $sheet->setCellValue('G33', Str::trim($str));
  78. //
  79. $fileName = 'Заявка на монтаж - ' . $order->object_address . '.xlsx';
  80. $writer = new Xlsx($spreadsheet);
  81. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  82. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  83. }
  84. /**
  85. * @throws Exception
  86. */
  87. public function generateHandoverPack(Order $order, int $userId): string
  88. {
  89. $articles = [];
  90. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ПАСПОРТ/');
  91. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/СЕРТИФИКАТ/');
  92. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ПСТ/photos');
  93. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp/ФОТО ТН/');
  94. // copy app photos
  95. foreach ($order->photos as $photo) {
  96. $from = $photo->path;
  97. $to = 'orders/' . $order->id . '/tmp/ФОТО ПСТ/photos/' . $photo->original_name;
  98. if (!Storage::disk('public')->exists($to)) {
  99. Storage::disk('public')->copy($from, $to);
  100. }
  101. }
  102. foreach ($order->products_sku as $sku) {
  103. // copy certificates
  104. if ($sku->product->certificate_id) {
  105. $from = $sku->product->certificate->path;
  106. $to = 'orders/' . $order->id . '/tmp/СЕРТИФИКАТ/' . $sku->product->certificate->original_name;
  107. if (!Storage::disk('public')->exists($to)) {
  108. Storage::disk('public')->copy($from, $to);
  109. }
  110. }
  111. // copy passport
  112. if ($sku->passport_id) {
  113. $from = $sku->passport->path;
  114. $to = 'orders/' . $order->id . '/tmp/ПАСПОРТ/';
  115. if (!Storage::disk('public')->exists($to . $sku->passport->original_name)) {
  116. $f = Storage::disk('public')->get($from);
  117. $ext = \File::extension($sku->passport->original_name);
  118. $targetName = $to . 'Паспорт ' . $sku->factory_number . ' арт. ' . $sku->product->article . '.' . $ext;
  119. Storage::disk('public')->put($targetName, $f);
  120. }
  121. }
  122. }
  123. // generate xlsx order files
  124. $this->generateStatement($order);
  125. $this->generateQualityDeclaration($order);
  126. $this->generateInventory($order);
  127. $this->generatePassport($order);
  128. // create zip archive
  129. $fileModel = (new FileService())->createZipArchive('orders/' . $order->id . '/tmp', self::HANDOVER_FILENAME . $order->common_name . '.zip', $userId);
  130. // remove temp files
  131. Storage::disk('public')->deleteDirectory('orders/' . $order->id . '/tmp');
  132. $order->documents()->syncWithoutDetaching($fileModel);
  133. // return link
  134. return $fileModel?->link ?? '';
  135. }
  136. private function generateStatement(Order $order): void
  137. {
  138. $inputFileType = 'Xlsx';
  139. $inputFileName = './templates/Statement.xlsx';
  140. $reader = IOFactory::createReader($inputFileType);
  141. $spreadsheet = $reader->load($inputFileName);
  142. $sheet = $spreadsheet->getActiveSheet();
  143. $contract = Contract::query()->where('contracts.year', $order->year)->first();
  144. $contract_number = $contract->contract_number ?? 'заполнить договор за ' . $order->year . ' год!';
  145. $contract_date = DateHelper::getHumanDate($contract->contract_date ?? '1970-01-01', true);
  146. $s = 'по Договору №' . $contract_number . ' от ' . $contract_date . ' г. Между ГБУ "Мосремонт" и ООО "НАШ ДВОР-СТ"';
  147. $sheet->setCellValue('B5', $s);
  148. // менеджер
  149. $sheet->setCellValue('G21', $order->user->name);
  150. // округ и район
  151. $sheet->setCellValue('C6', $order->district->shortname);
  152. $sheet->setCellValue('F6', $order->area->name);
  153. $i = 9; // start of table
  154. $nn = 1; // string number
  155. foreach ($order->products_sku as $sku) {
  156. if ($nn > 1) { // inset row
  157. $sheet->insertNewRowBefore($i);
  158. }
  159. $sheet->setCellValue('A' . $i, $nn++);
  160. $sheet->setCellValue('B' . $i, $sku->product->statement_name);
  161. $sheet->setCellValue('C' . $i, $sku->product->passport_name);
  162. $sheet->setCellValue('D' . $i, 'шт');
  163. $sheet->setCellValue('E' . $i, '1');
  164. $sheet->setCellValue('F' . $i, $order->name);
  165. $sheet->setCellValue('G' . $i, $sku->factory_number);
  166. $sheet->setCellValue('H' . $i++, $sku->rfid);
  167. }
  168. // save file
  169. $fileName = '1.Ведомость.xlsx';
  170. $writer = new Xlsx($spreadsheet);
  171. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  172. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  173. }
  174. private function generateQualityDeclaration(Order $order): void
  175. {
  176. $inputFileType = 'Xlsx';
  177. $inputFileName = './templates/QualityDeclaration.xlsx';
  178. $reader = IOFactory::createReader($inputFileType);
  179. $spreadsheet = $reader->load($inputFileName);
  180. $sheet = $spreadsheet->getActiveSheet();
  181. $address = 'г. Москва, ' . $order->district->shortname . ', район ' . $order->area->name . ', по адресу: ' . $order->object_address;
  182. $i = 1; // start of table
  183. $n = 1;
  184. foreach ($order->products_sku as $sku) {
  185. if ($n++ > 1) {
  186. $range = 'A' . $i . ':I' . $i + 56;
  187. $i = $i + 57;
  188. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  189. // add header img
  190. $drawing = new Drawing();
  191. $drawing->setName('Header');
  192. $drawing->setDescription('Header');
  193. $drawing->setPath('templates/header.png');
  194. $drawing->setCoordinates('A' . $i);
  195. $drawing->setOffsetX(16);
  196. $drawing->setOffsetY(8);
  197. $drawing->setResizeProportional(true);
  198. $drawing->setWidth(680);
  199. $drawing->setWorksheet($sheet);
  200. }
  201. // document date?
  202. $sheet->setCellValue('A' . $i + 7, DateHelper::getHumanDate(date('Y-m-d'), true));
  203. $sheet->setCellValue('B' . $i + 16, $sku->product->passport_name);
  204. $sheet->setCellValue('C' . $i + 18, $sku->rfid);
  205. $sheet->setCellValue('C' . $i + 20, $address);
  206. // add page break and copy prev page
  207. }
  208. // save file
  209. $fileName = '2.Декларация качества.xlsx';
  210. $writer = new Xlsx($spreadsheet);
  211. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  212. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  213. }
  214. private function generateInventory(Order $order): void
  215. {
  216. $inputFileType = 'Xlsx';
  217. $inputFileName = './templates/Inventory.xlsx';
  218. $reader = IOFactory::createReader($inputFileType);
  219. $spreadsheet = $reader->load($inputFileName);
  220. $sheet = $spreadsheet->getActiveSheet();
  221. $address = 'Округ: ' . $order->district->shortname . ' Район ' . $order->area->name;
  222. $sheet->setCellValue('C4', $order->name);
  223. $sheet->setCellValue('C5', $address);
  224. $sheet->setCellValue('C13', $order->user->name);
  225. $i = 8; // start of table
  226. $n = 1;
  227. foreach ($order->products_sku as $sku) {
  228. if ($n++ > 1) {
  229. $sheet->insertNewRowBefore($i + 3, 3);
  230. $range = 'A' . $i . ':E' . $i + 2;
  231. $i = $i + 3;
  232. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  233. }
  234. $sheet->setCellValue('A' . $i, $n - 1);
  235. $sheet->setCellValue('B' . $i, $sku->product->passport_name);
  236. $sheet->setCellValue('B' . $i + 2, $sku->rfid);
  237. }
  238. // save file
  239. $fileName = '3.Опись.xlsx';
  240. $writer = new Xlsx($spreadsheet);
  241. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  242. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  243. }
  244. private function generatePassport(Order $order): void
  245. {
  246. $inputFileType = 'Xlsx';
  247. $inputFileName = './templates/Passport.xlsx';
  248. $reader = IOFactory::createReader($inputFileType);
  249. $spreadsheet = $reader->load($inputFileName);
  250. $sheet = $spreadsheet->getActiveSheet();
  251. $i = 3; // start of table
  252. $n = 1;
  253. foreach ($order->products_sku as $sku) {
  254. if ($n++ > 1) {
  255. $sheet->insertNewRowBefore($i + 1, 1);
  256. $range = 'A' . $i . ':J' . $i;
  257. $i++;
  258. ExcelHelper::copyRows($sheet, $range, 'A' . $i);
  259. }
  260. $sheet->setCellValue('A' . $i, $n - 1);
  261. $sheet->setCellValue('B' . $i, $sku->product->passport_name);
  262. $sheet->setCellValue('C' . $i, $sku->product->type_tz);
  263. $sheet->setCellValue('D' . $i, $sku->rfid);
  264. $sheet->setCellValue('E' . $i, $sku->product->certificate_number);
  265. $sheet->setCellValue('G' . $i, $sku->factory_number);
  266. $sheet->setCellValue('H' . $i, DateHelper::getHumanDate($sku->manufacture_date, true));
  267. }
  268. // save file
  269. $fileName = '4.Паспорт объекта - ' . $order->object_address . '.xlsx';
  270. $writer = new Xlsx($spreadsheet);
  271. Storage::disk('public')->makeDirectory('orders/' . $order->id . '/tmp');
  272. $writer->save(storage_path('app/public/orders/') . $order->id . '/tmp/' . $fileName);
  273. }
  274. }