From 09d0381bc8379061bd64ad1e5e35c9ba0c2a9d7a Mon Sep 17 00:00:00 2001 From: Kusab Date: Thu, 12 Jan 2023 13:57:52 +0100 Subject: [PATCH 1/2] Allow to add Cell instances as data collection values Apply StyleCI patch Fix Codacy Static Code Analysis Issues - The method exportOrDownload() has an NPath complexity of 300. The configured NPath comp1lexity threshold is 200. - The method exportOrDownload() has a Cyclomatic Complexity of 13. The configured cyclomatic complexity threshold is 10. - Avoid using static access to class '\OpenSpout\Common\Entity\Row' in method 'writeHeader'. Fix test with 'ubuntu-latest, 8.1, lowest' build Extract Cell and Row creations to single methods Remove static Cell creation from Exportable trait Eat this Codacy Static Code Analyzer Enough joking , Codacy --- src/Exportable.php | 241 ++++++++++++++++++-------------------- src/FastExcel.php | 2 +- tests/StyledExcelTest.php | 116 ++++++++++++++++++ 3 files changed, 229 insertions(+), 130 deletions(-) create mode 100644 tests/StyledExcelTest.php diff --git a/src/Exportable.php b/src/Exportable.php index 2854ff3..66c4399 100644 --- a/src/Exportable.php +++ b/src/Exportable.php @@ -6,10 +6,9 @@ use Illuminate\Support\Collection; use Illuminate\Support\Str; use InvalidArgumentException; +use OpenSpout\Common\Entity\Cell; use OpenSpout\Common\Entity\Row; use OpenSpout\Common\Entity\Style\Style; -use OpenSpout\Writer\Common\Creator\WriterEntityFactory; -use OpenSpout\Writer\XLSX\Writer; /** * Trait Exportable. @@ -20,31 +19,21 @@ */ trait Exportable { - /** - * @var Style - */ - private $header_style; - private $rows_style; - - /** - * @param \OpenSpout\Reader\ReaderInterface|\OpenSpout\Writer\WriterInterface $reader_or_writer - * - * @return mixed - */ - abstract protected function setOptions(&$reader_or_writer); + private ?Style $header_style = null; + private ?Style $rows_style = null; /** * @param string $path * @param callable|null $callback * - * @throws \OpenSpout\Common\Exception\InvalidArgumentException * @throws \OpenSpout\Common\Exception\UnsupportedTypeException * @throws \OpenSpout\Writer\Exception\WriterNotOpenedException * @throws \OpenSpout\Common\Exception\IOException + * @throws \OpenSpout\Common\Exception\InvalidArgumentException * * @return string */ - public function export($path, callable $callback = null) + public function export(string $path, callable $callback = null) { self::exportOrDownload($path, 'openToFile', $callback); @@ -55,10 +44,10 @@ public function export($path, callable $callback = null) * @param $path * @param callable|null $callback * - * @throws \OpenSpout\Common\Exception\InvalidArgumentException * @throws \OpenSpout\Common\Exception\UnsupportedTypeException * @throws \OpenSpout\Writer\Exception\WriterNotOpenedException * @throws \OpenSpout\Common\Exception\IOException + * @throws \OpenSpout\Common\Exception\InvalidArgumentException * * @return \Symfony\Component\HttpFoundation\StreamedResponse|string */ @@ -74,6 +63,37 @@ public function download($path, callable $callback = null) return ''; } + /** + * @param Style $style + * + * @return Exportable + */ + public function headerStyle(Style $style) + { + $this->header_style = $style; + + return $this; + } + + /** + * @param Style $style + * + * @return Exportable + */ + public function rowsStyle(Style $style) + { + $this->rows_style = $style; + + return $this; + } + + /** + * @param \OpenSpout\Reader\ReaderInterface|\OpenSpout\Writer\WriterInterface $reader_or_writer + * + * @return mixed + */ + abstract protected function setOptions(&$reader_or_writer); + /** * @param $path * @param string $function @@ -85,27 +105,15 @@ public function download($path, callable $callback = null) * @throws \OpenSpout\Writer\Exception\WriterNotOpenedException * @throws \OpenSpout\Common\Exception\SpoutException */ - private function exportOrDownload($path, $function, callable $callback = null) + private function exportOrDownload($path, string $function, callable $callback = null) { - if (Str::endsWith($path, 'csv')) { - $options = new \OpenSpout\Writer\CSV\Options(); - $writer = new \OpenSpout\Writer\CSV\Writer($options); - } elseif (Str::endsWith($path, 'ods')) { - $options = new \OpenSpout\Writer\ODS\Options(); - $writer = new \OpenSpout\Writer\ODS\Writer($options); - } else { - $options = new \OpenSpout\Writer\XLSX\Options(); - $writer = new \OpenSpout\Writer\XLSX\Writer($options); - } - - $this->setOptions($options); - /* @var \OpenSpout\Writer\WriterInterface $writer */ + /* @var \OpenSpout\Writer\WriterInterface $writer */ + $writer = $this->prepareWriter($path); + $this->setOptions($writer); $writer->$function($path); - $has_sheets = ($writer instanceof \OpenSpout\Writer\XLSX\Writer || $writer instanceof \OpenSpout\Writer\ODS\Writer); - // It can export one sheet (Collection) or N sheets (SheetCollection) - $data = $this->transpose ? $this->transposeData() : ($this->data instanceof SheetCollection ? $this->data : collect([$this->data])); + $data = $this->prepareDataForExport(); foreach ($data as $key => $collection) { if ($collection instanceof Collection) { @@ -120,19 +128,48 @@ private function exportOrDownload($path, $function, callable $callback = null) if (is_string($key)) { $writer->getCurrentSheet()->setName($key); } - if ($has_sheets && $data->keys()->last() !== $key) { + if ($this->hasSheets($writer) && $data->keys()->last() !== $key) { $writer->addNewSheetAndMakeItCurrent(); } } $writer->close(); } + private function prepareWriter($path): \OpenSpout\Writer\WriterInterface + { + if (Str::endsWith($path, 'csv')) { + $writer = new \OpenSpout\Writer\CSV\Writer(new \OpenSpout\Writer\CSV\Options()); + } elseif (Str::endsWith($path, 'ods')) { + $writer = new \OpenSpout\Writer\ODS\Writer(new \OpenSpout\Writer\ODS\Options()); + } else { + $writer = new \OpenSpout\Writer\XLSX\Writer(new \OpenSpout\Writer\XLSX\Options()); + } + + return $writer; + } + + private function hasSheets(\OpenSpout\Writer\WriterInterface $writer): bool + { + return $writer instanceof \OpenSpout\Writer\XLSX\Writer || $writer instanceof \OpenSpout\Writer\ODS\Writer; + } + + private function prepareDataForExport(): SheetCollection|array|Generator|Collection|null + { + return $this->transpose + ? $this->transposeData() + : ( + $this->data instanceof SheetCollection + ? $this->data + : collect([$this->data]) + ); + } + /** * Transpose data from rows to columns. * * @return SheetCollection */ - private function transposeData() + private function transposeData(): SheetCollection { $data = $this->data instanceof SheetCollection ? $this->data : collect([$this->data]); $transposedData = []; @@ -164,39 +201,22 @@ private function writeRowsFromCollection($writer, Collection $collection, ?calla return $callback($value); }); } + // Prepare collection (i.e remove non-string) - $this->prepareCollection($collection); + // and transform into collection of row cells collections + //$this->transformCollection($collection); + // Add header row. if ($this->with_header) { - $this->writeHeader($writer, $collection->first()); - } - - // createRowFromArray works only with arrays - if (!is_array($collection->first())) { - $collection = $collection->map(function ($value) { - return $value->toArray(); - }); + $this->writeHeader($writer, $this->transformRow($collection->first())); } - // is_array($first_row) ? $first_row : $first_row->toArray()) - $all_rows = $collection->map(function ($value) { - return Row::fromValues($value); - })->toArray(); - if ($this->rows_style) { - $this->addRowsWithStyle($writer, $all_rows, $this->rows_style); - } else { - $writer->addRows($all_rows); - } - } - - private function addRowsWithStyle($writer, $all_rows, $rows_style) - { - $styled_rows = []; - // Style rows one by one - foreach ($all_rows as $row) { - $styled_rows[] = Row::fromValues($row->toArray(), $rows_style); - } - $writer->addRows($styled_rows); + // Add rows + $writer->addRows( + $collection->map(function ($row) { + return $this->createRow($this->transformRow($row), $this->rows_style); + })->toArray() + ); } private function writeRowsFromGenerator($writer, Generator $generator, ?callable $callback = null) @@ -208,14 +228,17 @@ private function writeRowsFromGenerator($writer, Generator $generator, ?callable } // Prepare row (i.e remove non-string) - $item = $this->transformRow($item); + // and transform to collection of Cells + $row_cells = $this->transformRow($item); // Add header row. if ($this->with_header && $key === 0) { - $this->writeHeader($writer, $item); + $this->writeHeader($writer, $row_cells); } // Write rows (one by one). - $writer->addRow(Row::fromValues($item->toArray(), $this->rows_style)); + $writer->addRow( + $this->createRow($row_cells, $this->rows_style) + ); } } @@ -229,82 +252,42 @@ private function writeRowsFromArray($writer, array $array, ?callable $callback = } } - private function writeHeader($writer, $first_row) + private function writeHeader($writer, array $first_row) { - if ($first_row === null) { - return; - } - - $keys = array_keys(is_array($first_row) ? $first_row : $first_row->toArray()); - $writer->addRow(Row::fromValues($keys)); -// $writer->addRow(WriterEntityFactory::createRowFromArray($keys, $this->header_style)); + $writer->addRow( + $this->createRow($this->transformRow(array_keys($first_row)), $this->header_style) + ); } /** - * Prepare collection by removing non string if required. - */ - protected function prepareCollection(Collection $collection) - { - $need_conversion = false; - $first_row = $collection->first(); - - if (!$first_row) { - return; - } - - foreach ($first_row as $item) { - if (!is_string($item)) { - $need_conversion = true; - } - } - if ($need_conversion) { - $this->transform($collection); - } - } - - /** - * Transform the collection. + * Transform one row (i.e remove non-string). + * into array of Cells. */ - private function transform(Collection $collection) + private function transformRow($data): array { - $collection->transform(function ($data) { - return $this->transformRow($data); - }); + return collect($data) + ->filter(function ($value) { + return $this->isValidValue($value); + })->map(function ($value) { + return $this->createCell($value, null); + })->toArray(); } - /** - * Transform one row (i.e remove non-string). - */ - private function transformRow($data) + private function isValidValue($value): bool { - return collect($data)->map(function ($value) { - return is_null($value) ? (string) $value : $value; - })->filter(function ($value) { - return is_string($value) || is_int($value) || is_float($value); - }); + return is_string($value) || is_int($value) || is_float($value) + || is_null($value) || $value instanceof Cell; } - /** - * @param Style $style - * - * @return Exportable - */ - public function headerStyle(Style $style) + private function createCell($value, ?Style $style): Cell { - $this->header_style = $style; - - return $this; + return ($value instanceof Cell) + ? $value + : Cell::fromValue($value, $style); } - /** - * @param Style $style - * - * @return Exportable - */ - public function rowsStyle(Style $style) + private function createRow(array $row_cells, ?Style $row_style): Row { - $this->rows_style = $style; - - return $this; + return new Row($row_cells, $row_style); } } diff --git a/src/FastExcel.php b/src/FastExcel.php index c09fdd7..2fb80be 100644 --- a/src/FastExcel.php +++ b/src/FastExcel.php @@ -170,7 +170,7 @@ public function configureReaderUsing(?callable $callback = null) } /** - * Configure the underlying Spout Reader using a callback. + * Configure the underlying Spout Writer using a callback. * * @param callable|null $callback * diff --git a/tests/StyledExcelTest.php b/tests/StyledExcelTest.php new file mode 100644 index 0000000..e783cc3 --- /dev/null +++ b/tests/StyledExcelTest.php @@ -0,0 +1,116 @@ +collection(); + + $style = (new Style()) + ->setFontItalic() + ->setFontSize(15) + ->setFontColor(Color::BLUE) + ->setShouldWrapText() + ->setBackgroundColor(Color::YELLOW); + + $file = __DIR__.'/test-row-style.xlsx'; + + (new FastExcel(clone $original_collection)) + ->rowsStyle($style) + ->export($file); + + $this->assertEquals($original_collection, (new FastExcel())->import($file)); + + unlink($file); + } + + /** + * @throws \OpenSpout\Common\Exception\IOException + * @throws \OpenSpout\Writer\Exception\WriterNotOpenedException + * @throws \OpenSpout\Reader\Exception\ReaderNotOpenedException + * @throws \OpenSpout\Common\Exception\UnsupportedTypeException + * @throws \OpenSpout\Common\Exception\InvalidArgumentException + */ + public function testExportWithCellsStyle() + { + $original_collection = $this->collection(); + + $collection = $this->applyCellStyles(clone $original_collection); + + $file = __DIR__.'/test-cells-style.xlsx'; + + (new FastExcel($collection))->export($file); + + $this->assertEquals($original_collection, (new FastExcel())->import($file)); + + unlink($file); + } + + /** + * @throws \OpenSpout\Common\Exception\InvalidArgumentException + */ + protected function applyCellStyles(Collection $data): Collection + { + $styles = [ + [ + 'col1' => (new Style()) + ->setFontBold() + ->setFontColor(Color::GREEN) + ->setShouldWrapText() + ->setBackgroundColor(Color::YELLOW), + + 'col2' => (new Style()) + ->setFontColor(Color::DARK_BLUE) + ->setBackgroundColor(Color::LIGHT_GREEN), + ], + + [ + 'col1' => (new Style()) + ->setCellAlignment(CellAlignment::CENTER) + ->setBackgroundColor(Color::YELLOW), + + 'col2' => (new Style()) + ->setCellAlignment(CellAlignment::RIGHT) + ->setFontColor(Color::DARK_BLUE), + ], + + [ + 'col1' => (new Style()) + ->setBackgroundColor(Color::ORANGE), + + 'col2' => (new Style()) + ->setCellAlignment(CellAlignment::RIGHT) + ->setFontColor(Color::DARK_BLUE) + ->setBackgroundColor(Color::LIGHT_GREEN), + ], + + ]; + + $data->transform(function ($row, $row_index) use ($styles) { + return collect($row)->transform(function ($value, $col_index) use ($row_index, $styles) { + return Cell::fromValue($value, $styles[$row_index][$col_index] ?? null); + }); + }); + + return $data; + } +} From 521d6da7ef03888d2cfa9e350b3cbe333761a6d0 Mon Sep 17 00:00:00 2001 From: Kusab Date: Tue, 14 Feb 2023 15:16:10 +0100 Subject: [PATCH 2/2] Fix issue 310 test error --- src/Exportable.php | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/Exportable.php b/src/Exportable.php index 66c4399..e2f1ec7 100644 --- a/src/Exportable.php +++ b/src/Exportable.php @@ -107,9 +107,10 @@ abstract protected function setOptions(&$reader_or_writer); */ private function exportOrDownload($path, string $function, callable $callback = null) { - /* @var \OpenSpout\Writer\WriterInterface $writer */ + /* @var \OpenSpout\Writer\WriterInterface $writer */ $writer = $this->prepareWriter($path); - $this->setOptions($writer); + $options = $writer->getOptions(); + $this->setOptions($options); $writer->$function($path); // It can export one sheet (Collection) or N sheets (SheetCollection)