PhpSpreadsheet/tests/PhpSpreadsheetTests/Functional/ConditionalTextTest.php

108 lines
4.4 KiB
PHP
Raw Permalink Normal View History

Save Excel 2010+ Functions Properly For functions introduced in Excel 2010 and beyond, Excel saves them in formulas with the xlfn_ prefix. PhpSpreadsheet does not do this; as a result, when a spreadsheet so created is opened, the cells which use the new functions display a #NAME? error. This the cause of bug report 1246: https://github.com/PHPOffice/PhpSpreadsheet/issues/1246 This change corrects that problem when the Xlsx writer encounters a 2010+ formula for a cell or a conditional style. A new class Writer/Xlsx/Xlfn, with 2 static methods, is introduced to facilitate this change. As part of the testing for this, I found some additional problems. When an unknown function name is used, Excel generates a #NAME? error. However, when an unknown function is used in PhpSpreadsheet: - if there are no parameters, it returns #VALUE!, which is wrong - if there are parameters, it throws an exception, which is horrible Both of these situations will now return #NAME? Tests have been added for these situations. The MODE (and MODE.SNGL) function is not quite in alignment with Excel. MODE(3, 3, 4, 4) returns 3 in both Excel and PhpSpreadsheet. However, MODE(4, 3, 3, 4) returns 4 in Excel, but 3 in PhpSpreadsheet. Both situations will now match Excel's result. Also, Excel allows its parameters for MODE to be an array, but PhpSpreadsheet did not; it now will. There had not been any tests for MODE. Now there are. The SHEET and SHEETS functions were introduced in Excel 2013, but were not introduced in PhpSpreadsheet. They are now introduced as DUMMY functions so that they can be parsed appropriately. Finally, in common with the "rate" changes for which I am creating a pull request at the same time as this one: samples/Basic/13_CalculationCyclicFormulae PhpUnit started reporting an error like "too much regression". The test deals with an infinite cyclic formula, and allowed the calculation engine to run for 100 cycles. The actual number of cycles seems irrelevant for the purpose of this test. I changed it to 15, and PhpUnit no longer complains.
2020-05-03 14:06:12 +00:00
<?php
namespace PhpOffice\PhpSpreadsheetTests\Functional;
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Style\Conditional;
use PhpOffice\PhpSpreadsheet\Style\Fill;
class ConditionalTextTest extends AbstractFunctional
{
const COLOR_GREEN = 'FF99FF66';
const COLOR_RED = 'FFFF5050';
const COLOR_BLUE = 'FF5050FF';
const COLOR_YELLOW = 'FFFFFF50';
public function testConditionalText(): void
{
$format = 'Xlsx';
$spreadsheet = new Spreadsheet();
$conditionalStyles = [];
// if text contains 'anywhere' - red background
$condition0 = new Conditional();
$condition0->setConditionType(Conditional::CONDITION_CONTAINSTEXT);
$condition0->setOperatorType(Conditional::CONDITION_CONTAINSTEXT);
$condition0->setText('anywhere');
$condition0->getStyle()->getFill()
->setFillType(Fill::FILL_SOLID)
->getEndColor()->setARGB(self::COLOR_RED);
array_push($conditionalStyles, $condition0);
// if text contains 'Left' on left - green background
$condition1 = new Conditional();
$condition1->setConditionType(Conditional::CONDITION_CONTAINSTEXT);
$condition1->setOperatorType(Conditional::OPERATOR_BEGINSWITH);
$condition1->setText('Left');
$condition1->getStyle()->getFill()
->setFillType(Fill::FILL_SOLID)
->getEndColor()->setARGB(self::COLOR_GREEN);
array_push($conditionalStyles, $condition1);
// if text contains 'right' on right - blue background
$condition2 = new Conditional();
$condition2->setConditionType(Conditional::CONDITION_CONTAINSTEXT);
$condition2->setOperatorType(Conditional::OPERATOR_ENDSWITH);
$condition2->setText('right');
$condition2->getStyle()->getFill()
->setFillType(Fill::FILL_SOLID)
->getEndColor()->setARGB(self::COLOR_BLUE);
array_push($conditionalStyles, $condition2);
// if text contains no spaces - yellow background
$condition3 = new Conditional();
$condition3->setConditionType(Conditional::CONDITION_CONTAINSTEXT);
$condition3->setOperatorType(Conditional::OPERATOR_NOTCONTAINS);
$condition3->setText(' ');
$condition3->getStyle()->getFill()
->setFillType(Fill::FILL_SOLID)
->getEndColor()->setARGB(self::COLOR_YELLOW);
array_push($conditionalStyles, $condition3);
$sheet = $spreadsheet->getActiveSheet();
$sheet->setCellValue('B1', 'This should match anywhere, right?');
$sheet->setCellValue('B2', 'This should match nowhere, right?');
$sheet->setCellValue('B3', 'Left match');
$sheet->setCellValue('B4', 'Match on right');
$sheet->setCellValue('B5', 'nospaces');
$xpCoordinate = 'B1:B5';
$spreadsheet->getActiveSheet()->setConditionalStyles($xpCoordinate, $conditionalStyles);
$sheet->getColumnDimension('B')->setAutoSize(true);
$reloadedSpreadsheet = $this->writeAndReload($spreadsheet, $format);
// see if we successfully written conditional text elements
$newConditionalStyles = $reloadedSpreadsheet->getActiveSheet()->getConditionalStyles($xpCoordinate);
$cnt = count($conditionalStyles);
for ($i = 0; $i < $cnt; ++$i) {
self::assertEquals(
$conditionalStyles[$i]->getConditionType(),
$newConditionalStyles[$i]->getConditionType(),
"Failure on condition type $i"
);
self::assertEquals(
$conditionalStyles[$i]->getOperatorType(),
$newConditionalStyles[$i]->getOperatorType(),
"Failure on operator type $i"
);
self::assertEquals(
$conditionalStyles[$i]->getText(),
$newConditionalStyles[$i]->getText(),
"Failure on text $i"
);
$filCond = $conditionalStyles[$i]->getStyle()->getFill();
$newCond = $newConditionalStyles[$i]->getStyle()->getFill();
self::assertEquals(
$filCond->getFillType(),
$newCond->getFillType(),
"Failure on fill type $i"
);
self::assertEquals(
$filCond->getEndColor()->getARGB(),
$newCond->getEndColor()->getARGB(),
"Failure on end color $i"
);
}
}
}