2017-09-22 05:49:38 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace PhpOffice\PhpSpreadsheetTests\Reader;
|
|
|
|
|
2019-06-30 21:42:25 +00:00
|
|
|
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
|
2019-06-10 14:44:55 +00:00
|
|
|
use PhpOffice\PhpSpreadsheet\Document\Properties;
|
2017-12-17 07:34:40 +00:00
|
|
|
use PhpOffice\PhpSpreadsheet\Reader\Xlsx;
|
2019-05-30 08:38:04 +00:00
|
|
|
use PhpOffice\PhpSpreadsheet\Shared\File;
|
2019-06-30 21:42:25 +00:00
|
|
|
use PhpOffice\PhpSpreadsheet\Style\Conditional;
|
|
|
|
use PhpOffice\PhpSpreadsheet\Style\Style;
|
|
|
|
use PhpOffice\PhpSpreadsheet\Worksheet\AutoFilter;
|
|
|
|
use PhpOffice\PhpSpreadsheet\Worksheet\PageSetup;
|
2017-11-08 15:48:01 +00:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2017-09-22 05:49:38 +00:00
|
|
|
|
2017-11-08 15:48:01 +00:00
|
|
|
class XlsxTest extends TestCase
|
2017-09-22 05:49:38 +00:00
|
|
|
{
|
2019-06-30 21:42:25 +00:00
|
|
|
public function testLoadXlsxWorkbookProperties()
|
2019-06-10 14:44:55 +00:00
|
|
|
{
|
2019-06-30 21:42:25 +00:00
|
|
|
$customPropertySet = [
|
|
|
|
'Publisher' => ['type' => Properties::PROPERTY_TYPE_STRING, 'value' => 'PHPOffice Suite'],
|
|
|
|
'Tested' => ['type' => Properties::PROPERTY_TYPE_BOOLEAN, 'value' => true],
|
|
|
|
'Counter' => ['type' => Properties::PROPERTY_TYPE_INTEGER, 'value' => 15],
|
|
|
|
'Rate' => ['type' => Properties::PROPERTY_TYPE_FLOAT, 'value' => 1.15],
|
|
|
|
'Refactor Date' => ['type' => Properties::PROPERTY_TYPE_DATE, 'value' => '2019-06-10'],
|
|
|
|
];
|
|
|
|
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/propertyTest.xlsx';
|
2019-06-10 14:44:55 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$properties = $spreadsheet->getProperties();
|
|
|
|
// Core Properties
|
2019-06-30 21:42:25 +00:00
|
|
|
$this->assertSame('Mark Baker', $properties->getCreator());
|
|
|
|
$this->assertSame('Unit Testing', $properties->getTitle());
|
|
|
|
$this->assertSame('Property Test', $properties->getSubject());
|
|
|
|
|
2019-06-10 14:44:55 +00:00
|
|
|
// Extended Properties
|
2019-06-30 21:42:25 +00:00
|
|
|
$this->assertSame('PHPOffice', $properties->getCompany());
|
|
|
|
$this->assertSame('The Big Boss', $properties->getManager());
|
|
|
|
|
2019-06-10 14:44:55 +00:00
|
|
|
// Custom Properties
|
|
|
|
$customProperties = $properties->getCustomProperties();
|
2019-07-25 17:15:53 +00:00
|
|
|
self::assertIsArray($customProperties);
|
2019-06-10 14:44:55 +00:00
|
|
|
$customProperties = array_flip($customProperties);
|
|
|
|
$this->assertArrayHasKey('Publisher', $customProperties);
|
2019-06-30 21:42:25 +00:00
|
|
|
|
|
|
|
foreach ($customPropertySet as $propertyName => $testData) {
|
|
|
|
$this->assertTrue($properties->isCustomPropertySet($propertyName));
|
|
|
|
$this->assertSame($testData['type'], $properties->getCustomPropertyType($propertyName));
|
|
|
|
if ($properties->getCustomPropertyType($propertyName) == Properties::PROPERTY_TYPE_DATE) {
|
|
|
|
$this->assertSame($testData['value'], date('Y-m-d', $properties->getCustomPropertyValue($propertyName)));
|
|
|
|
} else {
|
|
|
|
$this->assertSame($testData['value'], $properties->getCustomPropertyValue($propertyName));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadXlsxRowColumnAttributes()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/rowColumnAttributeTest.xlsx';
|
2019-06-30 21:42:25 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
for ($row = 1; $row <= 4; ++$row) {
|
|
|
|
$this->assertEquals($row * 5 + 10, floor($worksheet->getRowDimension($row)->getRowHeight()));
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->assertFalse($worksheet->getRowDimension(5)->getVisible());
|
|
|
|
|
|
|
|
for ($column = 1; $column <= 4; ++$column) {
|
|
|
|
$columnAddress = Coordinate::stringFromColumnIndex($column);
|
|
|
|
$this->assertEquals(
|
|
|
|
$column * 2 + 2,
|
|
|
|
floor($worksheet->getColumnDimension($columnAddress)->getWidth())
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->assertFalse($worksheet->getColumnDimension('E')->getVisible());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadXlsxWithStyles()
|
|
|
|
{
|
|
|
|
$expectedColours = [
|
|
|
|
1 => ['A' => 'C00000', 'C' => 'FF0000', 'E' => 'FFC000'],
|
2020-02-14 13:32:41 +00:00
|
|
|
3 => ['A' => '7030A0', 'C' => 'FFFFFF', 'E' => 'FFFF00'],
|
|
|
|
5 => ['A' => '002060', 'C' => 'FFFFFF', 'E' => '92D050'],
|
2019-06-30 21:42:25 +00:00
|
|
|
7 => ['A' => '0070C0', 'C' => '00B0F0', 'E' => '00B050'],
|
|
|
|
];
|
|
|
|
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/stylesTest.xlsx';
|
2019-06-30 21:42:25 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
for ($row = 1; $row <= 8; $row += 2) {
|
|
|
|
for ($column = 'A'; $column !== 'G'; ++$column, ++$column) {
|
|
|
|
$this->assertEquals(
|
|
|
|
$expectedColours[$row][$column],
|
|
|
|
$worksheet->getStyle($column . $row)->getFill()->getStartColor()->getRGB()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadXlsxAutofilter()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/autofilterTest.xlsx';
|
2019-06-30 21:42:25 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
|
|
|
|
$autofilter = $worksheet->getAutoFilter();
|
|
|
|
$this->assertInstanceOf(AutoFilter::class, $autofilter);
|
|
|
|
$this->assertEquals('A1:D57', $autofilter->getRange());
|
|
|
|
$this->assertEquals(
|
|
|
|
AutoFilter\Column::AUTOFILTER_FILTERTYPE_FILTER,
|
|
|
|
$autofilter->getColumn('A')->getFilterType()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadXlsxPageSetup()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/pageSetupTest.xlsx';
|
2019-06-30 21:42:25 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
|
|
|
|
$pageMargins = $worksheet->getPageMargins();
|
|
|
|
// Convert from inches to cm for testing
|
|
|
|
$this->assertEquals(2.5, $pageMargins->getTop() * 2.54);
|
|
|
|
$this->assertEquals(3.3, $pageMargins->getLeft() * 2.54);
|
|
|
|
$this->assertEquals(3.3, $pageMargins->getRight() * 2.54);
|
|
|
|
$this->assertEquals(1.3, $pageMargins->getHeader() * 2.54);
|
|
|
|
|
|
|
|
$this->assertEquals(PageSetup::PAPERSIZE_A4, $worksheet->getPageSetup()->getPaperSize());
|
|
|
|
$this->assertEquals(['A10', 'A20', 'A30', 'A40', 'A50'], array_keys($worksheet->getBreaks()));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadXlsxConditionalFormatting()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/conditionalFormattingTest.xlsx';
|
2019-06-30 21:42:25 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
|
|
|
|
$conditionalStyle = $worksheet->getCell('B2')->getStyle()->getConditionalStyles();
|
|
|
|
|
|
|
|
$this->assertNotEmpty($conditionalStyle);
|
|
|
|
$conditionalRule = $conditionalStyle[0];
|
|
|
|
$this->assertNotEmpty($conditionalRule->getConditions());
|
|
|
|
$this->assertEquals(Conditional::CONDITION_CELLIS, $conditionalRule->getConditionType());
|
|
|
|
$this->assertEquals(Conditional::OPERATOR_BETWEEN, $conditionalRule->getOperatorType());
|
|
|
|
$this->assertEquals(['200', '400'], $conditionalRule->getConditions());
|
|
|
|
$this->assertInstanceOf(Style::class, $conditionalRule->getStyle());
|
2019-06-10 14:44:55 +00:00
|
|
|
}
|
|
|
|
|
2019-07-08 17:55:14 +00:00
|
|
|
public function testLoadXlsxDataValidation()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/dataValidationTest.xlsx';
|
2019-07-08 17:55:14 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$spreadsheet = $reader->load($filename);
|
|
|
|
|
|
|
|
$worksheet = $spreadsheet->getActiveSheet();
|
|
|
|
|
|
|
|
$this->assertTrue($worksheet->getCell('B3')->hasDataValidation());
|
|
|
|
}
|
|
|
|
|
2017-09-22 05:49:38 +00:00
|
|
|
/**
|
|
|
|
* Test load Xlsx file without cell reference.
|
2019-06-30 21:42:25 +00:00
|
|
|
*
|
|
|
|
* @doesNotPerformAssertions
|
2017-09-22 05:49:38 +00:00
|
|
|
*/
|
|
|
|
public function testLoadXlsxWithoutCellReference()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/without_cell_reference.xlsx';
|
2017-12-17 07:34:40 +00:00
|
|
|
$reader = new Xlsx();
|
2017-09-22 05:49:38 +00:00
|
|
|
$reader->load($filename);
|
|
|
|
}
|
2018-11-19 13:48:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Test load Xlsx file and use a read filter.
|
|
|
|
*/
|
|
|
|
public function testLoadWithReadFilter()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/without_cell_reference.xlsx';
|
2018-11-19 13:48:30 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$reader->setReadFilter(new OddColumnReadFilter());
|
|
|
|
$data = $reader->load($filename)->getActiveSheet()->toArray();
|
|
|
|
$ref = [1.0, null, 3.0, null, 5.0, null, 7.0, null, 9.0, null];
|
|
|
|
|
|
|
|
for ($i = 0; $i < 10; ++$i) {
|
|
|
|
$this->assertEquals($ref, \array_slice($data[$i], 0, 10, true));
|
|
|
|
}
|
|
|
|
}
|
2019-05-30 10:23:25 +00:00
|
|
|
|
2019-05-30 09:42:00 +00:00
|
|
|
/**
|
|
|
|
* Test load Xlsx file with drawing having double attributes.
|
2019-06-30 21:42:25 +00:00
|
|
|
*
|
|
|
|
* @doesNotPerformAssertions
|
2019-05-30 09:42:00 +00:00
|
|
|
*/
|
|
|
|
public function testLoadXlsxWithDoubleAttrDrawing()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/double_attr_drawing.xlsx';
|
2019-05-30 09:42:00 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$reader->load($filename);
|
|
|
|
}
|
2019-05-30 08:38:04 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Test correct save and load xlsx files with empty drawings.
|
|
|
|
* Such files can be generated by Google Sheets.
|
|
|
|
*/
|
|
|
|
public function testLoadSaveWithEmptyDrawings()
|
|
|
|
{
|
2020-05-17 09:35:55 +00:00
|
|
|
$filename = 'tests/data/Reader/XLSX/empty_drawing.xlsx';
|
2019-05-30 08:38:04 +00:00
|
|
|
$reader = new Xlsx();
|
|
|
|
$excel = $reader->load($filename);
|
|
|
|
$resultFilename = tempnam(File::sysGetTempDir(), 'phpspreadsheet-test');
|
|
|
|
$writer = new \PhpOffice\PhpSpreadsheet\Writer\Xlsx($excel);
|
|
|
|
$writer->save($resultFilename);
|
|
|
|
$excel = $reader->load($resultFilename);
|
|
|
|
// Fake assert. The only thing we need is to ensure the file is loaded without exception
|
|
|
|
$this->assertNotNull($excel);
|
|
|
|
}
|
2020-02-04 12:19:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Test if all whitespace is removed from a style definition string.
|
|
|
|
* This is needed to parse it into properties with the correct keys.
|
|
|
|
*
|
|
|
|
* @param $string
|
|
|
|
* @dataProvider providerStripsWhiteSpaceFromStyleString
|
|
|
|
*/
|
|
|
|
public function testStripsWhiteSpaceFromStyleString($string)
|
|
|
|
{
|
|
|
|
$string = Xlsx::stripWhiteSpaceFromStyleString($string);
|
|
|
|
$this->assertEquals(preg_match('/\s/', $string), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function providerStripsWhiteSpaceFromStyleString()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
['position:absolute;margin-left:424.5pt;margin-top:169.5pt;width:67.5pt;
|
|
|
|
height:13.5pt;z-index:5;mso-wrap-style:tight'],
|
|
|
|
['position:absolute;margin-left:424.5pt;margin-top:169.5pt;width:67.5pt;
|
|
|
|
height:13.5pt;z-index:5;mso-wrap-style:tight'],
|
|
|
|
['position:absolute; margin-left:424.5pt; margin-top:169.5pt; width:67.5pt;
|
|
|
|
height:13.5pt;z-index:5;mso-wrap-style:tight'],
|
|
|
|
];
|
|
|
|
}
|
2017-09-22 05:49:38 +00:00
|
|
|
}
|