Proyectos de Subversion LeadersLinked - Backend

Rev

| Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
7218 eleazar 1
<?php
2
 
3
declare(strict_types=1);
4
 
5
namespace LeadersLinked\Controller;
6
 
7
use Laminas\Db\Adapter\AdapterInterface;
16768 efrain 8
 
7218 eleazar 9
use Laminas\Mvc\Controller\AbstractActionController;
10
use Laminas\Log\LoggerInterface;
11
use Laminas\View\Model\ViewModel;
12
use Laminas\View\Model\JsonModel;
13
use LeadersLinked\Mapper\QueryMapper;
14
use Laminas\Db\Sql\Select;
15
use LeadersLinked\Library\Functions;
16
use LeadersLinked\Mapper\OrganizationalClimateTestMapper;
17
use LeadersLinked\Mapper\OrganizationalClimateMapper;
18
use LeadersLinked\Mapper\OrganizationalClimateFormMapper;
19
use LeadersLinked\Form\OrganizationalClimateTestForm;
20
use LeadersLinked\Model\OrganizationalClimateTest;
21
use LeadersLinked\Mapper\UserMapper;
22
use LeadersLinked\Hydrator\ObjectPropertyHydrator;
7530 eleazar 23
use LeadersLinked\Library\SurveyReport;
7218 eleazar 24
use PhpOffice\PhpSpreadsheet\Spreadsheet;
25
use PhpOffice\PhpSpreadsheet\Writer\Xlsx;
26
use PhpOffice\PhpSpreadsheet\IOFactory;
27
 
28
class OrganizationalClimateReportController extends AbstractActionController {
29
 
30
    /**
31
     *
16769 efrain 32
     * @var \Laminas\Db\Adapter\AdapterInterface
7218 eleazar 33
     */
34
    private $adapter;
16768 efrain 35
 
7218 eleazar 36
    /**
37
     *
16769 efrain 38
     * @var \LeadersLinked\Cache\CacheInterface
7218 eleazar 39
     */
16769 efrain 40
    private $cache;
41
 
42
 
43
    /**
44
     *
45
     * @var \Laminas\Log\LoggerInterface
46
     */
7218 eleazar 47
    private $logger;
16768 efrain 48
 
7218 eleazar 49
    /**
50
     *
51
     * @var array
52
     */
53
    private $config;
16768 efrain 54
 
16769 efrain 55
 
7218 eleazar 56
    /**
57
     *
16769 efrain 58
     * @var \Laminas\Mvc\I18n\Translator
59
     */
60
    private $translator;
61
 
62
 
63
    /**
64
     *
65
     * @param \Laminas\Db\Adapter\AdapterInterface $adapter
66
     * @param \LeadersLinked\Cache\CacheInterface $cache
67
     * @param \Laminas\Log\LoggerInterface LoggerInterface $logger
7218 eleazar 68
     * @param array $config
16769 efrain 69
     * @param \Laminas\Mvc\I18n\Translator $translator
7218 eleazar 70
     */
16769 efrain 71
    public function __construct($adapter, $cache, $logger, $config, $translator)
16768 efrain 72
    {
16769 efrain 73
        $this->adapter      = $adapter;
74
        $this->cache        = $cache;
75
        $this->logger       = $logger;
76
        $this->config       = $config;
77
        $this->translator   = $translator;
7218 eleazar 78
    }
79
 
80
    public function indexAction() {
81
 
82
        $request = $this->getRequest();
83
        $currentUserPlugin = $this->plugin('currentUserPlugin');
84
        $currentCompany = $currentUserPlugin->getCompany();
85
        $currentUser = $currentUserPlugin->getUser();
86
 
87
        try{
88
        $request = $this->getRequest();
89
        if ($request->isGet()) {
90
 
91
            $headers = $request->getHeaders();
92
 
93
            $isJson = false;
94
            if ($headers->has('Accept')) {
95
                $accept = $headers->get('Accept');
96
 
97
                $prioritized = $accept->getPrioritized();
98
 
99
                foreach ($prioritized as $key => $value) {
100
                    $raw = trim($value->getRaw());
101
 
102
                    if (!$isJson) {
103
                        $isJson = strpos($raw, 'json');
104
                    }
105
                }
106
            }
107
 
108
            if ($isJson) {
109
 
110
                $search = $this->params()->fromQuery('search', []);
16766 efrain 111
                $search = empty($search['value']) ? '' :  Functions::sanitizeFilterString($search['value']);
7218 eleazar 112
 
15371 efrain 113
                $start = intval($this->params()->fromQuery('start', 0), 10);
7218 eleazar 114
                $records_x_page = intval($this->params()->fromQuery('length', 10), 10);
15371 efrain 115
                $page =  intval($start / $records_x_page);
116
                $page++;
117
 
7218 eleazar 118
                $order = $this->params()->fromQuery('order', []);
119
                $order_field = empty($order[0]['column']) ? 99 : intval($order[0]['column'], 10);
16766 efrain 120
                $order_direction = empty($order[0]['dir']) ? 'ASC' : Functions::sanitizeFilterString(filter_var($order[0]['dir']));
7218 eleazar 121
 
122
                $fields = ['name'];
123
                $order_field = isset($fields[$order_field]) ? $fields[$order_field] : 'name';
124
 
125
                if (!in_array($order_direction, ['ASC', 'DESC'])) {
126
                    $order_direction = 'ASC';
127
                }
128
 
129
                $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
130
                $paginator = $organizationalClimateMapper->fetchAllDataTableByCompanyId($currentCompany->id, $search, $page, $records_x_page, $order_field, $order_direction);
131
 
132
                $organizationalClimateFormMapper = OrganizationalClimateFormMapper::getInstance($this->adapter);
133
 
134
                $items = [];
135
                $records = $paginator->getCurrentItems();
136
 
137
                foreach ($records as $record) {
138
                    $organizationalClimateForm = $organizationalClimateFormMapper->fetchOne($record->form_id);
139
                    $params = [
140
 
141
                        'id' => $record->uuid,
142
                    ];
143
                    $item = [
144
                        'id' => $record->id,
145
                        'name' => $record->name,
146
                        'form' => $organizationalClimateForm->name,
147
                        'status' => $record->status,
148
                        'actions' => [
12997 efrain 149
                            'link_report_all' => $this->url()->fromRoute('organizational-climate/report/all', ['organizational_climate_id' => $record->uuid]),
150
                            'link_report_csv' => $this->url()->fromRoute('organizational-climate/report/csv', ['organizational_climate_id' => $record->uuid]),
151
                            'link_overview' => $this->url()->fromRoute('organizational-climate/report/overview', ['organizational_climate_id' => $record->uuid])
7218 eleazar 152
                        ]
153
                    ];
154
 
155
                    array_push($items, $item);
156
                }
157
 
158
                return new JsonModel([
159
                    'success' => true,
160
                    'data' => [
161
                        'items' => $items,
162
                        'total' => $paginator->getTotalItemCount(),
163
                    ]
164
                ]);
165
            } else {
166
                $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
167
                $survies = $organizationalClimateMapper->fetchAllByCompanyId($currentCompany->id);
168
 
169
                $form = new OrganizationalClimateTestForm($this->adapter, $currentCompany->id);
170
 
171
                $this->layout()->setTemplate('layout/layout-backend');
172
                $viewModel = new ViewModel();
173
                $viewModel->setTemplate('leaders-linked/organizational-climate-report/index.phtml');
174
                $viewModel->setVariables([
175
                    'form'      => $form,
176
                    'survies' => $survies
177
                ]);
178
                return $viewModel;
179
            }
180
        } else {
181
            return new JsonModel([
182
                'success' => false,
183
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
184
            ]);
185
 
186
        }
187
        } catch (\Throwable $e) {
188
            $e->getMessage();
189
            return new JsonModel([
190
                'success' => false,
191
                'data' => $e
192
            ]);
193
        }
194
 
195
    }
196
 
197
    public function overviewAction(){
198
        $request = $this->getRequest();
199
        $currentUserPlugin = $this->plugin('currentUserPlugin');
200
        $currentCompany = $currentUserPlugin->getCompany();
201
        $currentUser = $currentUserPlugin->getUser();
202
 
203
        $request = $this->getRequest();
12997 efrain 204
        $uuid = $this->params()->fromRoute('organizational_climate_id');
7218 eleazar 205
 
206
        if (!$uuid) {
207
            $data = [
208
                'success' => false,
209
                'data' => 'ERROR_INVALID_PARAMETER'
210
            ];
211
 
212
            return new JsonModel($data);
213
        }
214
 
215
        $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
216
        $organizationalClimate = $organizationalClimateMapper->fetchOneByUuid($uuid);
217
 
218
        $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
219
        $organizationalClimate = $organizationalClimateMapper->fetchOneByUuid($uuid);
220
 
221
        $organizationalClimateTestMapper = OrganizationalClimateTestMapper::getInstance($this->adapter);
7532 eleazar 222
        $organizationalClimateTests = $organizationalClimateTestMapper->fetchAllBySurveyId($organizationalClimate->id);
7218 eleazar 223
 
224
        $organizationalClimateFormMapper = OrganizationalClimateFormMapper::getInstance($this->adapter);
225
        $organizationalClimateForm = $organizationalClimateFormMapper->fetchOne($organizationalClimate->form_id);
226
 
227
        $sections = json_decode($organizationalClimateForm->content, true);
228
 
229
        $allTests = array_map(
230
            function($response) {
231
                return json_decode($response->content, true);
232
            },
233
            $organizationalClimateTests,
234
        );
235
 
236
        $averages = [];
237
 
238
        foreach($sections as $section) {
239
            foreach($section['questions'] as $question) {
240
                switch ($question['type']) {
241
                    case 'multiple':
242
                        $totals = [];
243
 
244
                        foreach($question['options'] as $option) {
245
                            $totals[$option['slug_option']] = 0;
246
                        }
247
 
248
                        foreach($question['options'] as $option) {
249
                            $totals[$option['slug_option']] = count(array_filter(
250
                                $allTests,
251
                                function ($test) use($option, $question) {
252
                                    return in_array($option['slug_option'], $test[$question['slug_question']]);
253
                                }
254
                            ));
255
                        }
256
 
257
                        $averages[$question['slug_question']] = $totals;
258
 
259
                        break;
260
 
261
                    case 'simple':
262
                        $totals = [];
263
 
264
                        foreach($question['options'] as $option) {
265
                            $totals[$option['slug_option']] = 0;
266
                        }
267
 
268
                        foreach ($allTests as $test) {
269
                            $totals[$test[$question['slug_question']]]++;
270
                        }
271
 
272
                        foreach($totals as $slug => $amount) {
273
                            $totals[$slug] = ($amount / count($allTests)) * 100;
274
                        }
275
 
276
                        $averages[$question['slug_question']] = $totals;
277
                    break;
278
 
279
                }
280
            }
281
        }
282
 
283
        $this->layout()->setTemplate('layout/layout-backend.phtml');
284
        $viewModel = new ViewModel();
285
        $viewModel->setTemplate('leaders-linked/organizational-climate-report/overview.phtml');
286
        $viewModel->setVariables([
287
            'sections' => $sections,
288
            'averages' => $averages,
289
        ]);
290
        return $viewModel ;
291
 
292
    }
293
 
294
    public function allAction() {
295
        $request = $this->getRequest();
296
        $currentUserPlugin = $this->plugin('currentUserPlugin');
297
        $currentCompany = $currentUserPlugin->getCompany();
298
        $currentUser = $currentUserPlugin->getUser();
299
 
300
        $request = $this->getRequest();
12997 efrain 301
        $uuid = $this->params()->fromRoute('organizational_climate_id');
7218 eleazar 302
 
303
 
304
 
305
        if (!$uuid) {
306
            $data = [
307
                'success' => false,
308
                'data' => 'ERROR_INVALID_PARAMETER'
309
            ];
310
 
311
            return new JsonModel($data);
312
        }
313
 
314
        $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
315
        $organizationalClimate = $organizationalClimateMapper->fetchOneByUuid($uuid);
316
 
317
        $organizationalClimateTestMapper = OrganizationalClimateTestMapper::getInstance($this->adapter);
7528 eleazar 318
        $organizationalClimateTests = $organizationalClimateTestMapper->fetchAllBySurveyId($organizationalClimate->id);
7218 eleazar 319
 
320
        if (!$organizationalClimateTests) {
321
            $data = [
322
                'success' => false,
323
                'data' => 'ERROR_RECORD_NOT_FOUND'
324
            ];
325
 
326
            return new JsonModel($data);
327
        }
328
 
329
 
330
        if ($request->isGet()) {
331
            $hydrator = new ObjectPropertyHydrator();
332
 
333
            //get form data
334
            $organizationalClimateFormMapper = OrganizationalClimateFormMapper::getInstance($this->adapter);
335
            $organizationalClimateForm = $organizationalClimateFormMapper->fetchOne($organizationalClimate->form_id);
336
 
337
            if ($organizationalClimateForm) {
338
 
339
                return $this->renderPDF($organizationalClimateForm, $organizationalClimateTests, $organizationalClimate);
340
            } else {
341
 
342
                $data = [
343
                    'success' => false,
344
                    'data' => 'ERROR_METHOD_NOT_ALLOWED'
345
                ];
346
 
347
                return new JsonModel($data);
348
            }
349
        } else {
350
            $data = [
351
                'success' => false,
352
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
353
            ];
354
 
355
            return new JsonModel($data);
356
        }
357
 
358
        return new JsonModel($data);
359
    }
360
 
361
 
362
    public function renderPDF($organizationalClimateForm, $organizationalClimateTests, $organizationalClimate) {
363
 
364
        $target_path = $this->config['leaderslinked.fullpath.company'] . DIRECTORY_SEPARATOR . $organizationalClimate->uuid;
365
 
366
 
367
        if(file_exists($target_path)) {
368
            Functions::deleteFiles($target_path);
369
        } else {
370
            @mkdir($target_path, 0755, true);
371
        }
372
 
373
        // Set Data
16768 efrain 374
        $headerFormName = Functions::utf8_decode($organizationalClimateForm->name);
375
        $headerSurveyName = Functions::utf8_decode('Informe de Encuesta: ' . trim($organizationalClimate->name) . ' al ' . date("m-d-Y H:i:s", strtotime($organizationalClimate->added_on)));
7218 eleazar 376
        $sections = json_decode($organizationalClimateForm->content, true);
377
 
378
        $allTests = array_map(
379
            function($response) {
380
                return json_decode($response->content, true);
381
            },
382
            $organizationalClimateTests,
383
        );
384
 
385
        $averages = [];
386
 
387
        foreach($sections as $section) {
388
            foreach($section['questions'] as $question) {
389
                switch ($question['type']) {
390
                    case 'multiple':
391
                        $totals = [];
392
 
393
                        foreach($question['options'] as $option) {
394
                            $totals[$option['slug_option']] = 0;
395
                        }
396
 
397
                        foreach($question['options'] as $option) {
398
                            $totals[$option['slug_option']] = count(array_filter(
399
                                $allTests,
400
                                function ($test) use($option, $question) {
401
                                    return in_array($option['slug_option'], $test[$question['slug_question']]);
402
                                }
403
                            ));
404
                        }
405
 
406
                        $averages[$question['slug_question']] = $totals;
407
 
408
                        break;
409
 
410
                    case 'simple':
411
                        $totals = [];
412
 
413
                        foreach($question['options'] as $option) {
414
                            $totals[$option['slug_option']] = 0;
415
                        }
416
 
417
                        foreach ($allTests as $test) {
418
                            $totals[$test[$question['slug_question']]]++;
419
                        }
420
 
421
                        foreach($totals as $slug => $amount) {
422
                            $totals[$slug] = ($amount / count($allTests)) * 100;
423
                        }
424
 
425
                        $averages[$question['slug_question']] = $totals;
426
                    break;
427
 
428
                }
429
            }
430
        }
431
 
432
        //Generate New PDF
7529 eleazar 433
        $pdf = new SurveyReport();
7218 eleazar 434
 
435
        $pdf->AliasNbPages();
436
        $pdf->AddPage();
437
 
438
        // Set header secundary
7529 eleazar 439
        $pdf->customHeader($headerFormName, $headerSurveyName);
7218 eleazar 440
 
441
        $countSection = 0;
442
 
443
        for ($i = 0; $i < count($sections); $i++) {
444
            if ($countSection > 1) {
445
                $countSection = 0;
446
                $pdf->AddPage();
447
                $pdf->customHeader($headerFormName, $headerUserName);
448
            }
449
            $section = $sections[$i];
450
 
451
            foreach ($section['questions'] as $question) {
452
 
453
                    switch ($question['type']) {
454
                        case 'simple':
455
                            $labels = [];
456
                            $values = [];
457
 
458
                            foreach($question['options'] as $option) {
459
                                $labels []= strip_tags($option['text']) . "\n(%.1f%%)";
460
 
461
                                $values []= $averages[$question['slug_question']][$option['slug_option']];
462
                            }
463
 
464
                            $filename = $target_path . DIRECTORY_SEPARATOR .  $question['slug_question'] . '.png';
465
                            $pdf->Cell(0,10,strip_tags(trim(str_replace(' ', ' ', html_entity_decode($question['text'])))),0,1);
466
                            $pdf->pieChart($labels, $values, '', $filename);
467
                            $pdf->SetFont('Arial', '', 12);
468
                            $pdf->Image($filename, 45, $pdf->getY(), 120);
469
                            $pdf->setY($pdf->getY() + 90);
470
 
471
 
472
                            break;
473
 
474
                        case 'multiple':
475
                            $labels = [];
476
                            $values = [];
477
 
478
                            foreach($question['options'] as $option) {
479
                                $labels []= strip_tags($option['text']);
480
 
481
                                $values []= $averages[$question['slug_question']][$option['slug_option']];
482
                            }
483
 
484
                            $filename = $target_path . DIRECTORY_SEPARATOR .  $question['slug_question'] . '.png';
485
                            $pdf->Cell(0,10,strip_tags($question['text']),0,1);
486
                            $pdf->barChart($labels, $values, '', $filename);
487
                            $pdf->SetFont('Arial', '', 12);
488
                            $pdf->Image($filename, 12, $pdf->getY());
489
                            $pdf->setY($pdf->getY() + (count($values) * 13) + 10);
490
 
491
                            break;
492
                    }
493
                $countSection++;
494
            }
495
 
496
 
497
        }
498
 
499
        return $pdf->Output();
500
    }
501
 
502
    public function csvAction() {
503
        $request = $this->getRequest();
504
        $currentUserPlugin = $this->plugin('currentUserPlugin');
505
        $currentCompany = $currentUserPlugin->getCompany();
506
        $currentUser = $currentUserPlugin->getUser();
507
 
508
        $request = $this->getRequest();
12997 efrain 509
        $uuid = $this->params()->fromRoute('organizational_climate_id');
7218 eleazar 510
 
511
 
512
 
513
        if (!$uuid) {
514
            $data = [
515
                'success' => false,
516
                'data' => 'ERROR_INVALID_PARAMETER'
517
            ];
518
 
519
            return new JsonModel($data);
520
        }
521
 
522
        $organizationalClimateMapper = OrganizationalClimateMapper::getInstance($this->adapter);
523
        $organizationalClimate = $organizationalClimateMapper->fetchOneByUuid($uuid);
524
 
525
        $organizationalClimateTestMapper = OrganizationalClimateTestMapper::getInstance($this->adapter);
7529 eleazar 526
        $organizationalClimateTests = $organizationalClimateTestMapper->fetchAllBySurveyId($organizationalClimate->id);
7218 eleazar 527
 
528
        if (!$organizationalClimateTests) {
529
            $data = [
530
                'success' => false,
531
                'data' => 'ERROR_RECORD_NOT_FOUND'
532
            ];
533
 
534
            return new JsonModel($data);
535
        }
536
 
537
 
538
        if ($request->isGet()) {
539
            $hydrator = new ObjectPropertyHydrator();
540
 
541
            //get form data
542
            $organizationalClimateFormMapper = OrganizationalClimateFormMapper::getInstance($this->adapter);
543
            $organizationalClimateForm = $organizationalClimateFormMapper->fetchOne($organizationalClimate->form_id);
544
 
545
            if ($organizationalClimateForm) {
546
 
547
                return $this->csvRender($organizationalClimateForm, $organizationalClimateTests, $organizationalClimate);
548
            } else {
549
 
550
                $data = [
551
                    'success' => false,
552
                    'data' => 'ERROR_METHOD_NOT_ALLOWED'
553
                ];
554
 
555
                return new JsonModel($data);
556
            }
557
        } else {
558
            $data = [
559
                'success' => false,
560
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
561
            ];
562
 
563
            return new JsonModel($data);
564
        }
565
 
566
        return new JsonModel($data);
567
    }
568
 
569
 
570
    public function csvRender($organizationalClimateForm, $organizationalClimateTests, $organizationalClimate)
571
    {
572
        $sections = json_decode($organizationalClimateForm->content, true);
573
 
574
        $spreadsheet = new Spreadsheet();
575
        $sheet = $spreadsheet->getActiveSheet();
576
 
577
        $allTests = array_map(
578
            function($response) {
579
                return json_decode($response->content, true);
580
            },
581
            $organizationalClimateTests,
582
        );
583
 
584
        $averages = [];
585
 
586
        foreach($sections as $section) {
587
            foreach($section['questions'] as $question) {
588
                switch ($question['type']) {
589
                    case 'multiple':
590
                        $totals = [];
591
 
592
                        foreach($question['options'] as $option) {
593
                            $totals[$option['slug_option']] = 0;
594
                        }
595
 
596
                        foreach($question['options'] as $option) {
597
                            $totals[$option['slug_option']] = count(array_filter(
598
                                $allTests,
599
                                function ($test) use($option, $question) {
600
                                    return in_array($option['slug_option'], $test[$question['slug_question']]);
601
                                }
602
                            ));
603
                        }
604
 
605
                        $averages[$question['slug_question']] = $totals;
606
 
607
                        break;
608
 
609
                    case 'simple':
610
                        $totals = [];
611
 
612
                        foreach($question['options'] as $option) {
613
                            $totals[$option['slug_option']] = 0;
614
                        }
615
 
616
                        foreach ($allTests as $test) {
617
                            $totals[$test[$question['slug_question']]]++;
618
                        }
619
 
620
                        foreach($totals as $slug => $amount) {
621
                            $totals[$slug] = ($amount / count($allTests)) * 100;
622
                        }
623
 
624
                        $averages[$question['slug_question']] = $totals;
625
                    break;
626
 
627
                }
628
            }
629
        }
630
 
631
        $row = 1;
632
 
633
        for ($i = 0; $i < count($sections); $i++) {
634
            $section = $sections[$i];
635
 
636
           for($j = 0; $j < count($section['questions']); $j++) {
637
                $question = $section['questions'][$j];
638
 
639
                if (!in_array($question['type'], ['simple', 'multiple'])){
640
                    continue;
641
                }
642
                $sheet->setCellValueByColumnAndRow(1, $row++, strip_tags($question['text']));
643
 
644
                switch ($question['type']) {
645
                    case 'simple':
646
                        for($k = 0; $k < count($question['options']); $k++) {
647
                            $option = $question['options'][$k];
648
                            $sheet->setCellValueByColumnAndRow(1, $row, strip_tags($option['text']));
649
                            $sheet->setCellValueByColumnAndRow(2, $row, round($averages[$question['slug_question']][$option['slug_option']], 2) . '%');
650
                            $row++;
651
                        }
652
 
653
                        break;
654
 
655
                    case 'multiple':
656
                        for($k = 0; $k < count($question['options']); $k++) {
657
                            $option = $question['options'][$k];
658
                            $sheet->setCellValueByColumnAndRow(1, $row, strip_tags($option['text']));
659
                            $sheet->setCellValueByColumnAndRow(2, $row,$averages[$question['slug_question']][$option['slug_option']]);
660
                            $row++;
661
                        }
662
 
663
                        break;
664
                }
665
            }
666
 
667
 
668
        }
669
 
670
        header('Content-Description: File Transfer');
671
        header('Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
672
        header('Content-Disposition: attachment; filename=report.xls');
673
        header('Content-Transfer-Encoding: binary');
674
        header('Expires: 0');
675
        header('Cache-Control: must-revalidate');
676
        header('Pragma: public');
677
 
678
        $writer = new Xlsx($spreadsheet);
679
 
680
        ob_clean();
681
        flush();
682
        $writer->save('php://output');
683
        return;
684
    }
685
}